Skip to content

Commit 0fbae06

Browse files
committed
feat: support reading paths from stdin
Signed-off-by: Brian McGee <[email protected]>
1 parent 4dd4c55 commit 0fbae06

11 files changed

+243
-51
lines changed

flake.lock

+16
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

flake.nix

+2
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,8 @@
1818
url = "github:nix-community/gomod2nix";
1919
inputs.nixpkgs.follows = "nixpkgs";
2020
};
21+
22+
nix-filter.url = "github:numtide/nix-filter";
2123
};
2224

2325
outputs = inputs @ {flake-parts, ...}:

internal/cache/cache.go

+2-7
Original file line numberDiff line numberDiff line change
@@ -173,7 +173,7 @@ func putEntry(bucket *bolt.Bucket, path string, entry *Entry) error {
173173

174174
// ChangeSet is used to walk a filesystem, starting at root, and outputting any new or changed paths using pathsCh.
175175
// It determines if a path is new or has changed by comparing against cache entries.
176-
func ChangeSet(ctx context.Context, root string, walkerType walk.Type, pathsCh chan<- string) error {
176+
func ChangeSet(ctx context.Context, walker walk.Walker, pathsCh chan<- string) error {
177177
var tx *bolt.Tx
178178
var bucket *bolt.Bucket
179179
var processed int
@@ -185,12 +185,7 @@ func ChangeSet(ctx context.Context, root string, walkerType walk.Type, pathsCh c
185185
}
186186
}()
187187

188-
w, err := walk.New(walkerType, root)
189-
if err != nil {
190-
return fmt.Errorf("%w: failed to create walker", err)
191-
}
192-
193-
return w.Walk(ctx, func(path string, info fs.FileInfo, err error) error {
188+
return walker.Walk(ctx, func(path string, info fs.FileInfo, err error) error {
194189
select {
195190
case <-ctx.Done():
196191
return ctx.Err()

internal/cli/cli.go

+10-11
Original file line numberDiff line numberDiff line change
@@ -6,30 +6,29 @@ import (
66
"github.com/charmbracelet/log"
77
)
88

9-
var Cli = Options{}
9+
var Cli = Format{}
1010

11-
type Options struct {
12-
AllowMissingFormatter bool `default:"false" help:"Do not exit with error if a configured formatter is missing."`
13-
WorkingDirectory kong.ChangeDirFlag `default:"." short:"C" help:"Run as if treefmt was started in the specified working directory instead of the current working directory."`
14-
ClearCache bool `short:"c" help:"Reset the evaluation cache. Use in case the cache is not precise enough."`
11+
type Format struct {
12+
AllowMissingFormatter bool `default:"false" help:"Do not exit with error if a configured formatter is missing"`
13+
WorkingDirectory kong.ChangeDirFlag `default:"." short:"C" help:"Run as if treefmt was started in the specified working directory instead of the current working directory"`
14+
ClearCache bool `short:"c" help:"Reset the evaluation cache. Use in case the cache is not precise enough"`
1515
ConfigFile string `type:"existingfile" default:"./treefmt.toml"`
1616
FailOnChange bool `help:"Exit with error if any changes were made. Useful for CI."`
1717
Formatters []string `help:"Specify formatters to apply. Defaults to all formatters."`
1818
TreeRoot string `type:"existingdir" default:"."`
1919
Walk walk.Type `enum:"auto,git,filesystem" default:"auto" help:"The method used to traverse the files within --tree-root. Currently supports 'auto', 'git' or 'filesystem'."`
2020
Verbosity int `name:"verbose" short:"v" type:"counter" default:"0" env:"LOG_LEVEL" help:"Set the verbosity of logs e.g. -vv."`
2121

22-
Format Format `cmd:"" default:"."`
22+
Paths []string `name:"paths" arg:"" type:"path" optional:"" help:"Paths to format. Defaults to formatting the whole tree."`
23+
Stdin bool `help:"Format the context passed in via stdin"`
2324
}
2425

25-
func (c *Options) Configure() {
26+
func (f *Format) Configure() {
2627
log.SetReportTimestamp(false)
2728

28-
if c.Verbosity == 0 {
29-
log.SetLevel(log.WarnLevel)
30-
} else if c.Verbosity == 1 {
29+
if f.Verbosity == 0 {
3130
log.SetLevel(log.InfoLevel)
32-
} else if c.Verbosity >= 2 {
31+
} else if f.Verbosity > 0 {
3332
log.SetLevel(log.DebugLevel)
3433
}
3534
}

internal/cli/format.go

+22-7
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
package cli
22

33
import (
4+
"bufio"
45
"context"
56
"errors"
67
"fmt"
@@ -10,6 +11,8 @@ import (
1011
"syscall"
1112
"time"
1213

14+
"git.numtide.com/numtide/treefmt/internal/walk"
15+
1316
"git.numtide.com/numtide/treefmt/internal/config"
1417

1518
"git.numtide.com/numtide/treefmt/internal/cache"
@@ -19,8 +22,6 @@ import (
1922
"golang.org/x/sync/errgroup"
2023
)
2124

22-
type Format struct{}
23-
2425
var ErrFailOnChange = errors.New("unexpected changes detected, --fail-on-change is enabled")
2526

2627
func (f *Format) Run() error {
@@ -201,7 +202,7 @@ func (f *Format) Run() error {
201202
return ErrFailOnChange
202203
}
203204

204-
fmt.Printf("%v files changed in %v", changes, time.Now().Sub(start))
205+
fmt.Printf("%v files changed in %v\n", changes, time.Now().Sub(start))
205206
return nil
206207
})
207208

@@ -235,10 +236,24 @@ func (f *Format) Run() error {
235236
return nil
236237
})
237238

238-
eg.Go(func() error {
239-
err := cache.ChangeSet(ctx, Cli.TreeRoot, Cli.Walk, pathsCh)
240-
close(pathsCh)
241-
return err
239+
eg.Go(func() (err error) {
240+
paths := Cli.Paths
241+
242+
if len(paths) == 0 && Cli.Stdin {
243+
// read in all the paths
244+
scanner := bufio.NewScanner(os.Stdin)
245+
for scanner.Scan() {
246+
paths = append(paths, scanner.Text())
247+
}
248+
}
249+
250+
walker, err := walk.New(Cli.Walk, Cli.TreeRoot, paths)
251+
if err != nil {
252+
return fmt.Errorf("%w: failed to create walker", err)
253+
}
254+
255+
defer close(pathsCh)
256+
return cache.ChangeSet(ctx, walker, pathsCh)
242257
})
243258

244259
// listen for shutdown and call cancel if required

internal/cli/format_test.go

+100
Original file line numberDiff line numberDiff line change
@@ -481,3 +481,103 @@ func TestOrderingFormatters(t *testing.T) {
481481
as.NoError(err)
482482
as.Contains(string(out), "8 files changed")
483483
}
484+
485+
func TestPathsArg(t *testing.T) {
486+
as := require.New(t)
487+
488+
// capture current cwd, so we can replace it after the test is finished
489+
cwd, err := os.Getwd()
490+
as.NoError(err)
491+
492+
t.Cleanup(func() {
493+
// return to the previous working directory
494+
as.NoError(os.Chdir(cwd))
495+
})
496+
497+
tempDir := test.TempExamples(t)
498+
configPath := filepath.Join(tempDir, "/treefmt.toml")
499+
500+
// change working directory to temp root
501+
as.NoError(os.Chdir(tempDir))
502+
503+
// basic config
504+
cfg := config.Config{
505+
Formatters: map[string]*config.Formatter{
506+
"echo": {
507+
Command: "echo",
508+
Includes: []string{"*"},
509+
},
510+
},
511+
}
512+
test.WriteConfig(t, configPath, cfg)
513+
514+
// without any path args
515+
out, err := cmd(t, "-C", tempDir)
516+
as.NoError(err)
517+
as.Contains(string(out), fmt.Sprintf("%d files changed", 29))
518+
519+
// specify some explicit paths
520+
out, err = cmd(t, "-C", tempDir, "-c", "elm/elm.json", "haskell/Nested/Foo.hs")
521+
as.NoError(err)
522+
as.Contains(string(out), fmt.Sprintf("%d files changed", 2))
523+
524+
// specify a bad path
525+
out, err = cmd(t, "-C", tempDir, "-c", "elm/elm.json", "haskell/Nested/Bar.hs")
526+
as.ErrorContains(err, "no such file or directory")
527+
}
528+
529+
func TestStdIn(t *testing.T) {
530+
as := require.New(t)
531+
532+
// capture current cwd, so we can replace it after the test is finished
533+
cwd, err := os.Getwd()
534+
as.NoError(err)
535+
536+
t.Cleanup(func() {
537+
// return to the previous working directory
538+
as.NoError(os.Chdir(cwd))
539+
})
540+
541+
tempDir := test.TempExamples(t)
542+
configPath := filepath.Join(tempDir, "/treefmt.toml")
543+
544+
// change working directory to temp root
545+
as.NoError(os.Chdir(tempDir))
546+
547+
// basic config
548+
cfg := config.Config{
549+
Formatters: map[string]*config.Formatter{
550+
"echo": {
551+
Command: "echo",
552+
Includes: []string{"*"},
553+
},
554+
},
555+
}
556+
test.WriteConfig(t, configPath, cfg)
557+
558+
// swap out stdin
559+
prevStdIn := os.Stdin
560+
stdin, err := os.CreateTemp("", "stdin")
561+
as.NoError(err)
562+
563+
os.Stdin = stdin
564+
565+
t.Cleanup(func() {
566+
os.Stdin = prevStdIn
567+
_ = os.Remove(stdin.Name())
568+
})
569+
570+
go func() {
571+
_, err := stdin.WriteString(`treefmt.toml
572+
elm/elm.json
573+
go/main.go
574+
`)
575+
as.NoError(err, "failed to write to stdin")
576+
as.NoError(stdin.Sync())
577+
_, _ = stdin.Seek(0, 0)
578+
}()
579+
580+
out, err := cmd(t, "-C", tempDir, "--stdin")
581+
as.NoError(err)
582+
as.Contains(string(out), fmt.Sprintf("%d files changed", 3))
583+
}

internal/walk/filesystem.go

+21-4
Original file line numberDiff line numberDiff line change
@@ -2,21 +2,38 @@ package walk
22

33
import (
44
"context"
5+
"os"
56
"path/filepath"
67
)
78

89
type filesystemWalker struct {
9-
root string
10+
root string
11+
paths []string
1012
}
1113

1214
func (f filesystemWalker) Root() string {
1315
return f.root
1416
}
1517

1618
func (f filesystemWalker) Walk(_ context.Context, fn filepath.WalkFunc) error {
17-
return filepath.Walk(f.root, fn)
19+
if len(f.paths) == 0 {
20+
return filepath.Walk(f.root, fn)
21+
}
22+
23+
for _, path := range f.paths {
24+
info, err := os.Stat(path)
25+
if err = filepath.Walk(path, fn); err != nil {
26+
return err
27+
}
28+
29+
if err = fn(path, info, err); err != nil {
30+
return err
31+
}
32+
}
33+
34+
return nil
1835
}
1936

20-
func NewFilesystem(root string) (Walker, error) {
21-
return filesystemWalker{root}, nil
37+
func NewFilesystem(root string, paths []string) (Walker, error) {
38+
return filesystemWalker{root, paths}, nil
2239
}

0 commit comments

Comments
 (0)