aboutsummaryrefslogtreecommitdiff
path: root/forged/internal/ipc/git2c/build.go
blob: 3d1b7a0a3d4c9be6a8b3bfd5cd4f138b81b184cd (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
108
109
110
111
112
113
114
115
116
117
118
119

// SPDX-License-Identifier: AGPL-3.0-only
// SPDX-FileCopyrightText: Copyright (c) 2025 Runxi Yu <https://runxiyu.org>

package git2c

import (
	"encoding/hex"
	"fmt"
	"path"
	"sort"
	"strings"
)

func (c *Client) BuildTreeRecursive(repoPath, baseTreeHex string, updates map[string]string) (string, error) {
	treeCache := make(map[string][]TreeEntryRaw)
	var walk func(prefix, hexid string) error
	walk = func(prefix, hexid string) error {
		ents, err := c.TreeListByOID(repoPath, hexid)
		if err != nil {
			return err
		}
		treeCache[prefix] = ents
		for _, e := range ents {
			if e.Mode == 40000 {
				sub := path.Join(prefix, e.Name)
				if err := walk(sub, e.OID); err != nil {
					return err
				}
			}
		}
		return nil
	}
	if err := walk("", baseTreeHex); err != nil {
		return "", err
	}

	for p, blob := range updates {
		parts := strings.Split(p, "/")
		dir := strings.Join(parts[:len(parts)-1], "/")
		name := parts[len(parts)-1]
		entries := treeCache[dir]
		found := false
		for i := range entries {
			if entries[i].Name == name {
				if blob == "" {
					entries = append(entries[:i], entries[i+1:]...)
				} else {
					entries[i].Mode = 0o100644
					entries[i].OID = blob
				}
				found = true
				break
			}
		}
		if !found && blob != "" {
			entries = append(entries, TreeEntryRaw{Mode: 0o100644, Name: name, OID: blob})
		}
		treeCache[dir] = entries
	}

	built := make(map[string]string)
	var build func(prefix string) (string, error)
	build = func(prefix string) (string, error) {
		entries := treeCache[prefix]
		for i := range entries {
			if entries[i].Mode == 0o40000 || entries[i].Mode == 40000 {
				sub := path.Join(prefix, entries[i].Name)
				var ok bool
				var oid string
				if oid, ok = built[sub]; !ok {
					var err error
					oid, err = build(sub)
					if err != nil {
						return "", err
					}
				}
				entries[i].Mode = 0o40000
				entries[i].OID = oid
			}
		}
		sort.Slice(entries, func(i, j int) bool {
			ni, nj := entries[i].Name, entries[j].Name
			if ni == nj {
				return entries[i].Mode != 0o40000 && entries[j].Mode == 0o40000
			}
			if strings.HasPrefix(nj, ni) && len(ni) < len(nj) {
				return entries[i].Mode != 0o40000
			}
			if strings.HasPrefix(ni, nj) && len(nj) < len(ni) {
				return entries[j].Mode == 0o40000
			}
			return ni < nj
		})
		wr := make([]TreeEntryRaw, 0, len(entries))
		for _, e := range entries {
			if e.OID == "" {
				continue
			}
			if e.Mode == 40000 {
				e.Mode = 0o40000
			}
			if _, err := hex.DecodeString(e.OID); err != nil {
				return "", fmt.Errorf("invalid OID hex for %s/%s: %w", prefix, e.Name, err)
			}
			wr = append(wr, TreeEntryRaw{Mode: e.Mode, Name: e.Name, OID: e.OID})
		}
		id, err := c.WriteTree(repoPath, wr)
		if err != nil {
			return "", err
		}
		built[prefix] = id
		return id, nil
	}
	root, err := build("")
	if err != nil {
		return "", err
	}
	return root, nil
}