-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtree.go
156 lines (119 loc) · 2.96 KB
/
tree.go
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
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
package git
import (
"bytes"
"errors"
"fmt"
"os"
"path/filepath"
"strconv"
"strings"
"git.codecrafters.io/0c40c1d7ba1ab4a0/object"
)
const (
ModeTree = 1 << 14
ModeBlob = 1 << 15
)
func (g *Git) LsTree(obj string, nameOnly bool) error {
path, err := g.findPath(obj)
if err != nil {
return fmt.Errorf("find object: %w", err)
}
typ, content, err := object.ReadFromFile(path)
if err != nil {
return fmt.Errorf("read file: %w", err)
}
if typ != "tree" {
return fmt.Errorf("not a tree object: %v", typ)
}
return g.lsTree(content, nameOnly)
}
func (g *Git) lsTree(content []byte, nameOnly bool) error {
for i := 0; i < len(content); {
space := i + bytes.IndexByte(content[i:], ' ')
if space < i { // IndexByte == -1
return errors.New("malformed tree")
}
mode := content[i:space]
zero := space + bytes.IndexByte(content[space:], '\000')
if zero < space { // IndexByte == -1
return errors.New("malformed tree")
}
name := content[space+1 : zero]
sum := content[zero+1 : zero+1+20]
if nameOnly {
fmt.Printf("%s\n", name)
} else {
modeInt, err := parseMode(string(mode))
if err != nil {
return fmt.Errorf("parse mode: %w", err)
}
typ := "????"
switch {
case modeInt&ModeTree != 0:
typ = "tree"
case modeInt&ModeBlob != 0:
typ = "blob"
}
fmt.Printf("%06o %s %x %s\n", modeInt, typ, sum, name)
}
i = zero + 1 + 20 // '\000' + 20 byte hash
}
return nil
}
func (g *Git) WriteTree(dir string, write bool) (object.Hash, error) {
files, err := os.ReadDir(dir)
if err != nil {
return object.Hash{}, fmt.Errorf("read dir: %w", err)
}
var table []byte
for _, f := range files {
if strings.HasPrefix(f.Name(), ".") {
continue
}
filePath := filepath.Join(dir, f.Name())
inf, err := f.Info()
if err != nil {
return object.Hash{}, fmt.Errorf("file info: %v: %w", filePath, err)
}
mode := inf.Mode()
if !mode.IsDir() && !mode.IsRegular() {
continue
}
var gitMode int
var sum object.Hash
if mode.IsDir() {
gitMode |= ModeTree
sum, err = g.WriteTree(filePath, write)
if err != nil {
return object.Hash{}, err
}
} else if mode.IsRegular() {
gitMode = ModeBlob | int(mode)&0xfff
sum, err = g.writeObject("blob", filePath, write)
if err != nil {
return object.Hash{}, fmt.Errorf("hash object %v: %w", filePath, err)
}
}
table = fmt.Appendf(table, "%06o %v\x00%s", gitMode, f.Name(), sum[:])
}
return g.hashObject("tree", table, write)
}
func (g *Git) hashObject(typ string, content []byte, write bool) (object.Hash, error) {
var buf bytes.Buffer
key, err := object.Write(&buf, typ, content)
if err != nil {
return object.Hash{}, fmt.Errorf("encode object")
}
if !write {
return key, nil
}
err = g.writeToStorage(key, buf.Bytes())
if err != nil {
return object.Hash{}, fmt.Errorf("write to storage: %w", err)
}
return key, nil
}
func parseMode(s string) (int, error) {
x, err := strconv.ParseInt(s, 8, 64)
return int(x), err
}