Skip to content

Commit 8b92dca

Browse files
committed
feat: streaming processing of paths from stdin and paths argument
Signed-off-by: Brian McGee <[email protected]>
1 parent f1f56d1 commit 8b92dca

File tree

5 files changed

+84
-92
lines changed

5 files changed

+84
-92
lines changed

cli/format.go

+44-11
Original file line numberDiff line numberDiff line change
@@ -207,10 +207,28 @@ func updateCache(ctx context.Context) func() error {
207207

208208
func walkFilesystem(ctx context.Context) func() error {
209209
return func() error {
210-
paths := Cli.Paths
210+
eg, ctx := errgroup.WithContext(ctx)
211+
pathsCh := make(chan string, BatchSize)
211212

212-
// we read paths from stdin if the cli flag has been set and no paths were provided as cli args
213-
if len(paths) == 0 && Cli.Stdin {
213+
walkPaths := func() error {
214+
defer close(pathsCh)
215+
216+
var idx int
217+
for idx < len(Cli.Paths) {
218+
select {
219+
case <-ctx.Done():
220+
return ctx.Err()
221+
default:
222+
pathsCh <- Cli.Paths[idx]
223+
idx += 1
224+
}
225+
}
226+
227+
return nil
228+
}
229+
230+
walkStdin := func() error {
231+
defer close(pathsCh)
214232

215233
// determine the current working directory
216234
cwd, err := os.Getwd()
@@ -220,20 +238,35 @@ func walkFilesystem(ctx context.Context) func() error {
220238

221239
// read in all the paths
222240
scanner := bufio.NewScanner(os.Stdin)
241+
223242
for scanner.Scan() {
224-
path := scanner.Text()
225-
if !strings.HasPrefix(path, "/") {
226-
// append the cwd
227-
path = filepath.Join(cwd, path)
243+
select {
244+
case <-ctx.Done():
245+
return ctx.Err()
246+
default:
247+
path := scanner.Text()
248+
if !strings.HasPrefix(path, "/") {
249+
// append the cwd
250+
path = filepath.Join(cwd, path)
251+
}
252+
pathsCh <- path
228253
}
229-
230-
// append the fully qualified path to our paths list
231-
paths = append(paths, path)
232254
}
255+
return nil
256+
}
257+
258+
if len(Cli.Paths) > 0 {
259+
eg.Go(walkPaths)
260+
} else if Cli.Stdin {
261+
eg.Go(walkStdin)
262+
} else {
263+
// no explicit paths to process, so we only need to process root
264+
pathsCh <- Cli.TreeRoot
265+
close(pathsCh)
233266
}
234267

235268
// create a filesystem walker
236-
walker, err := walk.New(Cli.Walk, Cli.TreeRoot, paths)
269+
walker, err := walk.New(Cli.Walk, Cli.TreeRoot, pathsCh)
237270
if err != nil {
238271
return fmt.Errorf("failed to create walker: %w", err)
239272
}

cli/format_test.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -488,7 +488,7 @@ func TestPathsArg(t *testing.T) {
488488
// specify some explicit paths
489489
_, err = cmd(t, "-C", tempDir, "-c", "elm/elm.json", "haskell/Nested/Foo.hs")
490490
as.NoError(err)
491-
assertStats(t, as, 4, 4, 4, 0)
491+
assertStats(t, as, 2, 2, 2, 0)
492492

493493
// specify a bad path
494494
_, err = cmd(t, "-C", tempDir, "-c", "elm/elm.json", "haskell/Nested/Bar.hs")
@@ -548,7 +548,7 @@ go/main.go
548548

549549
_, err = cmd(t, "-C", tempDir, "--stdin")
550550
as.NoError(err)
551-
assertStats(t, as, 6, 6, 6, 0)
551+
assertStats(t, as, 3, 3, 3, 0)
552552
}
553553

554554
func TestDeterministicOrderingInPipeline(t *testing.T) {

walk/filesystem.go

+5-21
Original file line numberDiff line numberDiff line change
@@ -3,13 +3,12 @@ package walk
33
import (
44
"context"
55
"io/fs"
6-
"os"
76
"path/filepath"
87
)
98

109
type filesystemWalker struct {
11-
root string
12-
paths []string
10+
root string
11+
pathsCh chan string
1312
}
1413

1514
func (f filesystemWalker) Root() string {
@@ -35,30 +34,15 @@ func (f filesystemWalker) Walk(_ context.Context, fn WalkFunc) error {
3534
return fn(&file, err)
3635
}
3736

38-
if len(f.paths) == 0 {
39-
return filepath.Walk(f.root, walkFn)
40-
}
41-
42-
for _, path := range f.paths {
43-
info, err := os.Stat(path)
44-
if err = filepath.Walk(path, walkFn); err != nil {
45-
return err
46-
}
47-
48-
file := File{
49-
Path: path,
50-
RelPath: relPathFn(path),
51-
Info: info,
52-
}
53-
54-
if err = fn(&file, err); err != nil {
37+
for path := range f.pathsCh {
38+
if err := filepath.Walk(path, walkFn); err != nil {
5539
return err
5640
}
5741
}
5842

5943
return nil
6044
}
6145

62-
func NewFilesystem(root string, paths []string) (Walker, error) {
46+
func NewFilesystem(root string, paths chan string) (Walker, error) {
6347
return filesystemWalker{root, paths}, nil
6448
}

walk/git.go

+26-51
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,6 @@ import (
55
"errors"
66
"fmt"
77
"io/fs"
8-
"os"
98
"path/filepath"
109

1110
"github.com/charmbracelet/log"
@@ -16,7 +15,7 @@ import (
1615

1716
type gitWalker struct {
1817
root string
19-
paths []string
18+
paths chan string
2019
repo *git.Repository
2120
}
2221

@@ -40,66 +39,42 @@ func (g *gitWalker) Walk(ctx context.Context, fn WalkFunc) error {
4039
return fmt.Errorf("failed to open git index: %w", err)
4140
}
4241

43-
if len(g.paths) > 0 {
44-
for _, path := range g.paths {
45-
46-
err = filepath.Walk(path, func(path string, info fs.FileInfo, err error) error {
47-
if info.IsDir() {
48-
return nil
49-
}
50-
51-
relPath, err := filepath.Rel(g.root, path)
52-
if err != nil {
53-
return err
54-
}
55-
56-
if _, err = idx.Entry(relPath); errors.Is(err, index.ErrEntryNotFound) {
57-
// we skip this path as it's not staged
58-
log.Debugf("Path not found in git index, skipping: %v, %v", relPath, path)
59-
return nil
60-
}
61-
62-
file := File{
63-
Path: path,
64-
RelPath: relPathFn(path),
65-
Info: info,
66-
}
67-
68-
return fn(&file, err)
69-
})
42+
for path := range g.paths {
43+
44+
err = filepath.Walk(path, func(path string, info fs.FileInfo, err error) error {
45+
if info.IsDir() {
46+
return nil
47+
}
48+
49+
relPath, err := filepath.Rel(g.root, path)
7050
if err != nil {
7151
return err
7252
}
7353

74-
}
75-
} else {
76-
for _, entry := range idx.Entries {
77-
select {
78-
case <-ctx.Done():
79-
return ctx.Err()
80-
default:
81-
path := filepath.Join(g.root, entry.Name)
82-
83-
// stat the file
84-
info, err := os.Lstat(path)
85-
86-
file := File{
87-
Path: path,
88-
RelPath: relPathFn(path),
89-
Info: info,
90-
}
91-
92-
if err = fn(&file, err); err != nil {
93-
return err
94-
}
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)
57+
return nil
9558
}
59+
60+
file := File{
61+
Path: path,
62+
RelPath: relPathFn(path),
63+
Info: info,
64+
}
65+
66+
return fn(&file, err)
67+
})
68+
if err != nil {
69+
return err
9670
}
71+
9772
}
9873

9974
return nil
10075
}
10176

102-
func NewGit(root string, paths []string) (Walker, error) {
77+
func NewGit(root string, paths chan string) (Walker, error) {
10378
repo, err := git.PlainOpen(root)
10479
if err != nil {
10580
return nil, fmt.Errorf("failed to open git repo: %w", err)

walk/walker.go

+7-7
Original file line numberDiff line numberDiff line change
@@ -31,24 +31,24 @@ type Walker interface {
3131
Walk(ctx context.Context, fn WalkFunc) error
3232
}
3333

34-
func New(walkerType Type, root string, paths []string) (Walker, error) {
34+
func New(walkerType Type, root string, pathsCh chan string) (Walker, error) {
3535
switch walkerType {
3636
case Git:
37-
return NewGit(root, paths)
37+
return NewGit(root, pathsCh)
3838
case Auto:
39-
return Detect(root, paths)
39+
return Detect(root, pathsCh)
4040
case Filesystem:
41-
return NewFilesystem(root, paths)
41+
return NewFilesystem(root, pathsCh)
4242
default:
4343
return nil, fmt.Errorf("unknown walker type: %v", walkerType)
4444
}
4545
}
4646

47-
func Detect(root string, paths []string) (Walker, error) {
47+
func Detect(root string, pathsCh chan string) (Walker, error) {
4848
// for now, we keep it simple and try git first, filesystem second
49-
w, err := NewGit(root, paths)
49+
w, err := NewGit(root, pathsCh)
5050
if err == nil {
5151
return w, err
5252
}
53-
return NewFilesystem(root, paths)
53+
return NewFilesystem(root, pathsCh)
5454
}

0 commit comments

Comments
 (0)