aboutsummaryrefslogtreecommitdiff
path: root/objectheader
diff options
context:
space:
mode:
authorGravatar Runxi Yu2026-02-21 00:42:00 +0800
committerGravatar Runxi Yu2026-02-21 00:42:00 +0800
commita03e6e14a1e807136b05f28072f37dcf8c030f6b (patch)
treec9e4069fbd959d05f684895a8bbd0ccbc311ea77 /objectheader
parenttestgit: Use objectid's SupportedAlgorithms (diff)
signatureNo signature
objectheader: Move out of internal
Diffstat (limited to 'objectheader')
-rw-r--r--objectheader/append.go27
-rw-r--r--objectheader/encode.go8
-rw-r--r--objectheader/parse.go40
3 files changed, 75 insertions, 0 deletions
diff --git a/objectheader/append.go b/objectheader/append.go
new file mode 100644
index 00000000..3965dc21
--- /dev/null
+++ b/objectheader/append.go
@@ -0,0 +1,27 @@
+package objectheader
+
+import (
+ "strconv"
+
+ "codeberg.org/lindenii/furgit/objecttype"
+)
+
+// Append appends a canonical loose-object header ("type size\\x00") to dst.
+func Append(dst []byte, ty objecttype.Type, size int64) ([]byte, bool) {
+ if size < 0 {
+ return nil, false
+ }
+ tyName, ok := objecttype.Name(ty)
+ if !ok {
+ return nil, false
+ }
+
+ sizeStr := strconv.FormatInt(size, 10)
+ out := make([]byte, 0, len(dst)+len(tyName)+len(sizeStr)+2)
+ out = append(out, dst...)
+ out = append(out, tyName...)
+ out = append(out, ' ')
+ out = append(out, sizeStr...)
+ out = append(out, 0)
+ return out, true
+}
diff --git a/objectheader/encode.go b/objectheader/encode.go
new file mode 100644
index 00000000..1cca968b
--- /dev/null
+++ b/objectheader/encode.go
@@ -0,0 +1,8 @@
+package objectheader
+
+import "codeberg.org/lindenii/furgit/objecttype"
+
+// Encode returns a canonical loose-object header ("type size\\x00").
+func Encode(ty objecttype.Type, size int64) ([]byte, bool) {
+ return Append(nil, ty, size)
+}
diff --git a/objectheader/parse.go b/objectheader/parse.go
new file mode 100644
index 00000000..c3d7d5dd
--- /dev/null
+++ b/objectheader/parse.go
@@ -0,0 +1,40 @@
+package objectheader
+
+import (
+ "bytes"
+ "strconv"
+
+ "codeberg.org/lindenii/furgit/objecttype"
+)
+
+// Parse parses a canonical loose-object header ("type size\\x00").
+// It returns the parsed type, size, bytes consumed (including trailing NUL),
+// and whether parsing succeeded.
+func Parse(data []byte) (objecttype.Type, int64, int, bool) {
+ space := bytes.IndexByte(data, ' ')
+ if space <= 0 {
+ return objecttype.TypeInvalid, 0, 0, false
+ }
+
+ nulRel := bytes.IndexByte(data[space+1:], 0)
+ if nulRel < 0 {
+ return objecttype.TypeInvalid, 0, 0, false
+ }
+ nul := space + 1 + nulRel
+
+ ty, ok := objecttype.ParseName(string(data[:space]))
+ if !ok {
+ return objecttype.TypeInvalid, 0, 0, false
+ }
+
+ sizeBytes := data[space+1 : nul]
+ if len(sizeBytes) == 0 {
+ return objecttype.TypeInvalid, 0, 0, false
+ }
+ size, err := strconv.ParseInt(string(sizeBytes), 10, 64)
+ if err != nil || size < 0 {
+ return objecttype.TypeInvalid, 0, 0, false
+ }
+
+ return ty, size, nul + 1, true
+}