aboutsummaryrefslogtreecommitdiff
path: root/object/signed/commit/parse.go
blob: fa498093321f08c7a7a1a4854c4f9dffc66fe61b (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
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
package signedcommit

import (
	"bytes"

	objectid "codeberg.org/lindenii/furgit/object/id"
)

// Parse parses one raw commit object body for signature extraction.
//
// The returned Commit remains valid only while body remains unchanged.
//
// Labels: Deps-Borrowed, Life-Parent.
func Parse(body []byte) (*Commit, error) {
	commit := &Commit{
		body:       body,
		signatures: make(map[objectid.Algorithm][]byteRange),
	}

	payloadStart := 0
	i := 0

	for i < len(body) {
		lineStart := i

		rel := bytes.IndexByte(body[i:], '\n')
		next := len(body)

		lineEnd := len(body)
		if rel >= 0 {
			lineEnd = i + rel
			next = lineEnd + 1
		}

		line := body[lineStart:lineEnd]
		i = next

		if len(line) == 0 {
			commit.appendPayloadRange(payloadStart, len(body))

			return commit, nil
		}

		if line[0] == ' ' {
			continue
		}

		if !bytes.HasPrefix(line, []byte("gpgsig")) {
			continue
		}

		commit.appendPayloadRange(payloadStart, lineStart)

		key, valueStart, found := bytes.Cut(line, []byte{' '})
		if found {
			if algo, ok := objectid.ParseSignatureHeaderName(string(key)); ok {
				commit.signatures[algo] = append(commit.signatures[algo], byteRange{
					start: lineEnd - len(valueStart),
					end:   next,
				})
			}
		}

		for i < len(body) {
			rel := bytes.IndexByte(body[i:], '\n')
			next = len(body)

			lineEnd = len(body)
			if rel >= 0 {
				lineEnd = i + rel
				next = lineEnd + 1
			}

			contStart := i

			cont := body[contStart:lineEnd]
			if len(cont) == 0 || cont[0] != ' ' {
				break
			}

			if found {
				if algo, ok := objectid.ParseSignatureHeaderName(string(key)); ok {
					commit.signatures[algo] = append(commit.signatures[algo], byteRange{
						start: contStart + 1,
						end:   next,
					})
				}
			}

			i = next
		}

		payloadStart = i
	}

	commit.appendPayloadRange(payloadStart, len(body))

	return commit, nil
}

func (commit *Commit) appendPayloadRange(start, end int) {
	if start >= end {
		return
	}

	commit.payload = append(commit.payload, byteRange{start: start, end: end})
}