Skip to content

Commit 8aeee1d

Browse files
committed
fix: exit with error if any formatters fail
We try to apply formatters to all files on a best-effort basis, continuing if formatting any particular batch of files fails. This fix ensures that if any formatting errors occur, the process will exit with an error. Closes #450 Signed-off-by: Brian McGee <[email protected]>
1 parent c8a96da commit 8aeee1d

File tree

3 files changed

+35
-25
lines changed

3 files changed

+35
-25
lines changed

cmd/format/format.go

+18-7
Original file line numberDiff line numberDiff line change
@@ -32,7 +32,10 @@ const (
3232
BatchSize = 1024
3333
)
3434

35-
var ErrFailOnChange = errors.New("unexpected changes detected, --fail-on-change is enabled")
35+
var (
36+
ErrFailOnChange = errors.New("unexpected changes detected, --fail-on-change is enabled")
37+
ErrFormattingFailures = errors.New("formatting failures detected")
38+
)
3639

3740
func Run(v *viper.Viper, statz *stats.Stats, cmd *cobra.Command, paths []string) error {
3841
cmd.SilenceUsage = true
@@ -404,6 +407,8 @@ func postProcessing(
404407
formattedCh chan *format.Task,
405408
) func() error {
406409
return func() error {
410+
var formattingFailures bool // track if there were any formatting failures
411+
407412
LOOP:
408413
for {
409414
select {
@@ -420,8 +425,9 @@ func postProcessing(
420425
// grab the underlying file reference
421426
file := task.File
422427

423-
// check if there were any errors processing the file
424428
if len(task.Errors) > 0 {
429+
formattingFailures = true
430+
425431
// release the file, passing the first task error
426432
// note: task errors are related to the batch in which a task was applied
427433
// this does not necessarily indicate this file had a problem being formatted, but this approach
@@ -471,16 +477,21 @@ func postProcessing(
471477
}
472478
}
473479

474-
// if fail on change has been enabled, check that no files were actually changed, throwing an error if so
475-
if cfg.FailOnChange && statz.Value(stats.Changed) != 0 {
476-
return ErrFailOnChange
477-
}
478-
479480
// print stats to stdout unless we are processing stdin and printing the results to stdout
480481
if !cfg.Stdin {
481482
statz.Print()
482483
}
483484

485+
// return an error if any formatting failures were detected
486+
if formattingFailures {
487+
return ErrFormattingFailures
488+
}
489+
490+
// if fail on change has been enabled, check that no files were actually changed, throwing an error if so
491+
if cfg.FailOnChange && statz.Value(stats.Changed) != 0 {
492+
return ErrFailOnChange
493+
}
494+
484495
return nil
485496
}
486497
}

cmd/root_test.go

+16-17
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ import (
1414

1515
"github.com/charmbracelet/log"
1616
"github.com/numtide/treefmt/cmd"
17-
format2 "github.com/numtide/treefmt/cmd/format"
17+
formatCmd "github.com/numtide/treefmt/cmd/format"
1818
"github.com/numtide/treefmt/config"
1919
"github.com/numtide/treefmt/format"
2020
"github.com/numtide/treefmt/stats"
@@ -482,7 +482,7 @@ func TestCache(t *testing.T) {
482482

483483
// running should match but not format anything
484484
_, statz, err = treefmt(t, "--config-file", configPath, "--tree-root", tempDir)
485-
as.NoError(err)
485+
as.ErrorIs(err, formatCmd.ErrFormattingFailures)
486486

487487
assertStats(t, as, statz, map[stats.Type]int{
488488
stats.Traversed: 32,
@@ -492,8 +492,8 @@ func TestCache(t *testing.T) {
492492
})
493493

494494
// running again should provide the same result
495-
_, statz, err = treefmt(t, "--config-file", configPath, "--tree-root", tempDir, "-vv")
496-
as.NoError(err)
495+
_, statz, err = treefmt(t, "--config-file", configPath, "--tree-root", tempDir)
496+
as.ErrorIs(err, formatCmd.ErrFormattingFailures)
497497

498498
assertStats(t, as, statz, map[stats.Type]int{
499499
stats.Traversed: 32,
@@ -588,13 +588,13 @@ func TestFailOnChange(t *testing.T) {
588588

589589
test.WriteConfig(t, configPath, cfg)
590590
_, _, err := treefmt(t, "--fail-on-change", "--config-file", configPath, "--tree-root", tempDir)
591-
as.ErrorIs(err, format2.ErrFailOnChange)
591+
as.ErrorIs(err, formatCmd.ErrFailOnChange)
592592

593593
// test with no cache
594594
t.Setenv("TREEFMT_FAIL_ON_CHANGE", "true")
595595
test.WriteConfig(t, configPath, cfg)
596596
_, _, err = treefmt(t, "--config-file", configPath, "--tree-root", tempDir, "--no-cache")
597-
as.ErrorIs(err, format2.ErrFailOnChange)
597+
as.ErrorIs(err, formatCmd.ErrFailOnChange)
598598
}
599599

600600
func TestBustCacheOnFormatterChange(t *testing.T) {
@@ -1027,7 +1027,7 @@ func TestStdin(t *testing.T) {
10271027
// we get an error about the missing filename parameter.
10281028
out, _, err := treefmt(t, "-C", tempDir, "--allow-missing-formatter", "--stdin")
10291029
as.EqualError(err, "exactly one path should be specified when using the --stdin flag")
1030-
as.Equal("", string(out))
1030+
as.Equal("Error: exactly one path should be specified when using the --stdin flag\n", string(out))
10311031

10321032
// now pass along the filename parameter
10331033
os.Stdin = test.TempFile(t, "", "stdin", &contents)
@@ -1051,7 +1051,7 @@ func TestStdin(t *testing.T) {
10511051

10521052
out, _, err = treefmt(t, "-C", tempDir, "--allow-missing-formatter", "--stdin", "../test.nix")
10531053
as.Errorf(err, "path ../test.nix not inside the tree root %s", tempDir)
1054-
as.Equal("", string(out))
1054+
as.Contains(string(out), "Error: path ../test.nix not inside the tree root")
10551055

10561056
// try some markdown instead
10571057
contents = `
@@ -1281,21 +1281,20 @@ func treefmt(t *testing.T, args ...string) ([]byte, *stats.Stats, error) {
12811281
time.Sleep(time.Until(waitUntil))
12821282
}()
12831283

1284-
if err := root.Execute(); err != nil {
1285-
return nil, nil, err
1286-
}
1284+
// execute the command
1285+
cmdErr := root.Execute()
12871286

12881287
// reset and read the temporary output
1289-
if _, err := tempOut.Seek(0, 0); err != nil {
1290-
return nil, nil, fmt.Errorf("failed to reset temp output for reading: %w", err)
1288+
if _, resetErr := tempOut.Seek(0, 0); resetErr != nil {
1289+
t.Fatal(fmt.Errorf("failed to reset temp output for reading: %w", resetErr))
12911290
}
12921291

1293-
out, err := io.ReadAll(tempOut)
1294-
if err != nil {
1295-
return nil, nil, fmt.Errorf("failed to read temp output: %w", err)
1292+
out, readErr := io.ReadAll(tempOut)
1293+
if readErr != nil {
1294+
t.Fatal(fmt.Errorf("failed to read temp output: %w", readErr))
12961295
}
12971296

1298-
return out, statz, nil
1297+
return out, statz, cmdErr
12991298
}
13001299

13011300
func assertStats(

format/formatter.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -77,7 +77,7 @@ func (f *Formatter) Apply(ctx context.Context, tasks []*Task) error {
7777
f.log.Errorf("failed to apply with options '%v': %s", f.config.Options, err)
7878

7979
if len(out) > 0 {
80-
_, _ = fmt.Fprintf(os.Stderr, "%s error:\n%s\n", f.name, out)
80+
_, _ = fmt.Fprintf(os.Stderr, "\n%s\n", out)
8181
}
8282

8383
return fmt.Errorf("formatter '%s' with options '%v' failed to apply: %w", f.config.Command, f.config.Options, err)

0 commit comments

Comments
 (0)