@@ -2,13 +2,12 @@ package walk
2
2
3
3
import (
4
4
"context"
5
- "errors"
6
5
"fmt"
7
6
"io/fs"
7
+ "os"
8
8
"path/filepath"
9
9
10
10
"github.com/charmbracelet/log"
11
- "github.com/go-git/go-git/v5/plumbing/format/index"
12
11
13
12
"github.com/go-git/go-git/v5"
14
13
)
@@ -39,9 +38,57 @@ func (g *gitWalker) Walk(ctx context.Context, fn WalkFunc) error {
39
38
return fmt .Errorf ("failed to open git index: %w" , err )
40
39
}
41
40
41
+ // cache in-memory whether a path is present in the git index
42
+ var cache map [string ]bool
43
+
42
44
for path := range g .paths {
43
45
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 {
45
92
if info .IsDir () {
46
93
return nil
47
94
}
@@ -51,9 +98,8 @@ func (g *gitWalker) Walk(ctx context.Context, fn WalkFunc) error {
51
98
return err
52
99
}
53
100
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 )
57
103
return nil
58
104
}
59
105
@@ -65,10 +111,6 @@ func (g *gitWalker) Walk(ctx context.Context, fn WalkFunc) error {
65
111
66
112
return fn (& file , err )
67
113
})
68
- if err != nil {
69
- return err
70
- }
71
-
72
114
}
73
115
74
116
return nil
0 commit comments