From 3c7add2cf4154c54c42d348bc462e29198e69338 Mon Sep 17 00:00:00 2001 From: Runxi Yu Date: Thu, 19 Mar 2026 16:19:24 +0000 Subject: object/resolve: Object resolver --- object/resolve/path.go | 53 ++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 53 insertions(+) create mode 100644 object/resolve/path.go (limited to 'object/resolve/path.go') diff --git a/object/resolve/path.go b/object/resolve/path.go new file mode 100644 index 00000000..b3e2d642 --- /dev/null +++ b/object/resolve/path.go @@ -0,0 +1,53 @@ +package resolve + +import ( + "fmt" + + "codeberg.org/lindenii/furgit/object" + "codeberg.org/lindenii/furgit/objectid" +) + +// Path resolves parts within the tree identified by root and returns the final +// tree entry. +// +// The root object may be any tree-ish object accepted by PeelToTree. +// +// parts must contain at least one path segment. Intermediate path segments +// must resolve to tree entries. The final entry is returned without loading +// its object. +func (r *Resolver) Path(root objectid.ObjectID, parts [][]byte) (object.TreeEntry, error) { + if len(parts) == 0 { + return object.TreeEntry{}, fmt.Errorf("object/resolve: empty tree path") + } + + current, err := r.PeelToTree(root) + if err != nil { + return object.TreeEntry{}, err + } + + for i, part := range parts { + if len(part) == 0 { + return object.TreeEntry{}, fmt.Errorf("object/resolve: empty tree path segment") + } + + entry := current.Object().Entry(part) + if entry == nil { + return object.TreeEntry{}, fmt.Errorf("object/resolve: tree entry %q not found", part) + } + + if i == len(parts)-1 { + return *entry, nil + } + + if entry.Mode != object.FileModeDir { + return object.TreeEntry{}, fmt.Errorf("object/resolve: path segment %q is not a tree", part) + } + + current, err = r.ExactTree(entry.ID) + if err != nil { + return object.TreeEntry{}, err + } + } + + return object.TreeEntry{}, fmt.Errorf("object/resolve: tree entry not found") +} -- cgit v1.3.1-10-gc9f91