Skip to content

Commit 2e77e12

Browse files
committed
feat: optimize git walker
Signed-off-by: Brian McGee <[email protected]>
1 parent 8b92dca commit 2e77e12

File tree

1 file changed

+52
-10
lines changed

1 file changed

+52
-10
lines changed

walk/git.go

+52-10
Original file line numberDiff line numberDiff line change
@@ -2,13 +2,12 @@ package walk
22

33
import (
44
"context"
5-
"errors"
65
"fmt"
76
"io/fs"
7+
"os"
88
"path/filepath"
99

1010
"github.com/charmbracelet/log"
11-
"github.com/go-git/go-git/v5/plumbing/format/index"
1211

1312
"github.com/go-git/go-git/v5"
1413
)
@@ -39,9 +38,57 @@ func (g *gitWalker) Walk(ctx context.Context, fn WalkFunc) error {
3938
return fmt.Errorf("failed to open git index: %w", err)
4039
}
4140

41+
// cache in-memory whether a path is present in the git index
42+
var cache map[string]bool
43+
4244
for path := range g.paths {
4345

44-
err = filepath.Walk(path, func(path string, info fs.FileInfo, err error) error {
46+
if path == g.root {
47+
// we can just iterate the index entries
48+
for _, entry := range idx.Entries {
49+
select {
50+
case <-ctx.Done():
51+
return ctx.Err()
52+
default:
53+
path := filepath.Join(g.root, entry.Name)
54+
55+
// stat the file
56+
info, err := os.Lstat(path)
57+
58+
file := File{
59+
Path: path,
60+
RelPath: relPathFn(path),
61+
Info: info,
62+
}
63+
64+
if err = fn(&file, err); err != nil {
65+
return err
66+
}
67+
}
68+
}
69+
continue
70+
}
71+
72+
// otherwise we ensure the git index entries are cached and then check if they are in the git index
73+
if cache == nil {
74+
cache = make(map[string]bool)
75+
for _, entry := range idx.Entries {
76+
cache[entry.Name] = true
77+
}
78+
}
79+
80+
relPath, err := filepath.Rel(g.root, path)
81+
if err != nil {
82+
return fmt.Errorf("failed to find relative path for %v: %w", path, err)
83+
}
84+
85+
_, ok := cache[relPath]
86+
if !(path == g.root || ok) {
87+
log.Debugf("path %v not found in git index, skipping", path)
88+
continue
89+
}
90+
91+
return filepath.Walk(path, func(path string, info fs.FileInfo, err error) error {
4592
if info.IsDir() {
4693
return nil
4794
}
@@ -51,9 +98,8 @@ func (g *gitWalker) Walk(ctx context.Context, fn WalkFunc) error {
5198
return err
5299
}
53100

54-
if _, err = idx.Entry(relPath); errors.Is(err, index.ErrEntryNotFound) {
55-
// we skip this path as it's not staged
56-
log.Debugf("Path not found in git index, skipping: %v, %v", relPath, path)
101+
if _, ok := cache[relPath]; !ok {
102+
log.Debugf("path %v not found in git index, skipping", path)
57103
return nil
58104
}
59105

@@ -65,10 +111,6 @@ func (g *gitWalker) Walk(ctx context.Context, fn WalkFunc) error {
65111

66112
return fn(&file, err)
67113
})
68-
if err != nil {
69-
return err
70-
}
71-
72114
}
73115

74116
return nil

0 commit comments

Comments
 (0)