aboutsummaryrefslogtreecommitdiff
path: root/objectstore/loose/parse.go
blob: dfb420ba20a9c2f152d06e968f93dc1134d81b0e (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
50
51
52
53
54
55
package loose

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

	"codeberg.org/lindenii/furgit/internal/compress/zlib"
	objectheader "codeberg.org/lindenii/furgit/object/header"
	objecttype "codeberg.org/lindenii/furgit/object/type"
)

// 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
}