aboutsummaryrefslogtreecommitdiff
path: root/objectstore/loose/parse.go
blob: 54bb23755f4e4e0ac268abe6ba8812e9a469835c (about) (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
package loose

import (
	"bufio"
	"errors"
	"io"
	"os"

	"codeberg.org/lindenii/furgit/internal/zlib"
	"codeberg.org/lindenii/furgit/objectheader"
	"codeberg.org/lindenii/furgit/objecttype"
)

// decodeAll inflates the full loose object payload from file.
func decodeAll(file *os.File) ([]byte, error) {
	zr, err := zlib.NewReader(file)
	if err != nil {
		return nil, err
	}
	defer func() { _ = zr.Close() }()
	return io.ReadAll(zr)
}

// parseRaw parses a loose object payload in "type size\0content" format.
func parseRaw(raw []byte) (objecttype.Type, []byte, error) {
	ty, size, headerLen, ok := objectheader.Parse(raw)
	if !ok {
		return objecttype.TypeInvalid, nil, errors.New("objectstore/loose: malformed object header")
	}
	content := raw[headerLen:]
	if int64(len(content)) != size {
		return objecttype.TypeInvalid, nil, errors.New("objectstore/loose: object header size/content mismatch")
	}
	return ty, content, nil
}

// readHeader reads and parses a loose object header from br, and returns
// the raw header bytes including the trailing NUL.
func readHeader(br *bufio.Reader) ([]byte, objecttype.Type, int64, error) {
	header, err := br.ReadSlice(0)
	if err != nil {
		return nil, objecttype.TypeInvalid, 0, err
	}
	ty, size, _, ok := objectheader.Parse(header)
	if !ok {
		return nil, objecttype.TypeInvalid, 0, errors.New("objectstore/loose: malformed object header")
	}
	return header, ty, size, nil
}