Skip to content

Commit b834fa3

Browse files
committed
feat: best-effort application of files to formatters
If a formatter errors out, continue with subsequent formatters regardless. Do not cache the result to ensure later invocations re-try the same files. Signed-off-by: Brian McGee <[email protected]>
1 parent 2feb7cf commit b834fa3

File tree

5 files changed

+47
-29
lines changed

5 files changed

+47
-29
lines changed

cli/cli.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -43,8 +43,8 @@ type Format struct {
4343
globalExcludes []glob.Glob
4444

4545
filesCh chan *walk.File
46-
formattedCh chan *walk.File
47-
processedCh chan *walk.File
46+
formattedCh chan *format.Task
47+
processedCh chan *format.Task
4848
}
4949

5050
func (f *Format) configureLogging() {

cli/format.go

+41-21
Original file line numberDiff line numberDiff line change
@@ -184,10 +184,10 @@ func (f *Format) Run() (err error) {
184184
f.filesCh = make(chan *walk.File, BatchSize*runtime.NumCPU())
185185

186186
// create a channel for files that have been formatted
187-
f.formattedCh = make(chan *walk.File, cap(f.filesCh))
187+
f.formattedCh = make(chan *format.Task, cap(f.filesCh))
188188

189189
// create a channel for files that have been processed
190-
f.processedCh = make(chan *walk.File, cap(f.filesCh))
190+
f.processedCh = make(chan *format.Task, cap(f.filesCh))
191191

192192
// start concurrent processing tasks in reverse order
193193
eg.Go(f.updateCache(ctx))
@@ -317,17 +317,21 @@ func (f *Format) applyFormatters(ctx context.Context) func() error {
317317

318318
// asynchronously apply the sequence formatters to the batch
319319
fg.Go(func() error {
320-
// iterate the formatters, applying them in sequence to the batch of tasks
321-
// we get the formatters list from the first task since they have all the same formatters list
322-
for _, f := range tasks[0].Formatters {
323-
if err := f.Apply(ctx, tasks); err != nil {
324-
return err
320+
// Iterate the formatters, applying them in sequence to the batch of tasks.
321+
// We get the formatter list from the first task since they have all the same formatters list.
322+
formatters := tasks[0].Formatters
323+
324+
var formatErrors []error
325+
for idx := range formatters {
326+
if err := formatters[idx].Apply(ctx, tasks); err != nil {
327+
formatErrors = append(formatErrors, err)
325328
}
326329
}
327330

328331
// pass each file to the formatted channel
329332
for _, task := range tasks {
330-
f.formattedCh <- task.File
333+
task.Errors = formatErrors
334+
f.formattedCh <- task
331335
}
332336

333337
return nil
@@ -359,7 +363,9 @@ func (f *Format) applyFormatters(ctx context.Context) func() error {
359363
if format.PathMatches(file.RelPath, f.globalExcludes) {
360364
log.Debugf("path matched global excludes: %s", file.RelPath)
361365
// mark it as processed and continue to the next
362-
f.formattedCh <- file
366+
f.formattedCh <- &format.Task{
367+
File: file,
368+
}
363369
continue
364370
}
365371

@@ -378,7 +384,9 @@ func (f *Format) applyFormatters(ctx context.Context) func() error {
378384
}
379385
log.Logf(f.OnUnmatched, "no formatter for path: %s", file.RelPath)
380386
// mark it as processed and continue to the next
381-
f.formattedCh <- file
387+
f.formattedCh <- &format.Task{
388+
File: file,
389+
}
382390
} else {
383391
// record the match
384392
stats.Add(stats.Matched, 1)
@@ -414,14 +422,15 @@ func (f *Format) detectFormatted(ctx context.Context) func() error {
414422
// detect ctx cancellation
415423
case <-ctx.Done():
416424
return ctx.Err()
417-
// take the next file that has been processed
418-
case file, ok := <-f.formattedCh:
425+
// take the next task that has been processed
426+
case task, ok := <-f.formattedCh:
419427
if !ok {
420428
// channel has been closed, no further files to process
421429
return nil
422430
}
423431

424432
// check if the file has changed
433+
file := task.File
425434
changed, newInfo, err := file.HasChanged()
426435
if err != nil {
427436
return err
@@ -451,7 +460,7 @@ func (f *Format) detectFormatted(ctx context.Context) func() error {
451460
}
452461

453462
// mark as processed
454-
f.processedCh <- file
463+
f.processedCh <- task
455464
}
456465
}
457466
}
@@ -460,12 +469,16 @@ func (f *Format) detectFormatted(ctx context.Context) func() error {
460469
func (f *Format) updateCache(ctx context.Context) func() error {
461470
return func() error {
462471
// used to batch updates for more efficient txs
463-
batch := make([]*walk.File, 0, BatchSize)
472+
batch := make([]*format.Task, 0, BatchSize)
464473

465474
// apply a batch
466475
processBatch := func() error {
467476
// pass the batch to the cache for updating
468-
if err := cache.Update(batch); err != nil {
477+
files := make([]*walk.File, len(batch))
478+
for idx := range batch {
479+
files[idx] = batch[idx].File
480+
}
481+
if err := cache.Update(files); err != nil {
469482
return err
470483
}
471484
batch = batch[:0]
@@ -486,12 +499,14 @@ func (f *Format) updateCache(ctx context.Context) func() error {
486499
case <-ctx.Done():
487500
return ctx.Err()
488501
// respond to formatted files
489-
case file, ok := <-f.processedCh:
502+
case task, ok := <-f.processedCh:
490503
if !ok {
491504
// channel has been closed, no further files to process
492505
break LOOP
493506
}
494507

508+
file := task.File
509+
495510
if f.Stdin {
496511
// dump file into stdout
497512
f, err := os.Open(file.Path)
@@ -508,11 +523,16 @@ func (f *Format) updateCache(ctx context.Context) func() error {
508523
continue
509524
}
510525

511-
// append to batch and process if we have enough
512-
batch = append(batch, file)
513-
if len(batch) == BatchSize {
514-
if err := processBatch(); err != nil {
515-
return err
526+
// Append to batch and process if we have enough.
527+
// We do not cache any files that were part of a pipeline in which one or more formatters failed.
528+
// This is to ensure those files are re-processed in later invocations after the user has potentially
529+
// resolved the issue, e.g. fixed a config problem.
530+
if len(task.Errors) == 0 {
531+
batch = append(batch, task)
532+
if len(batch) == BatchSize {
533+
if err := processBatch(); err != nil {
534+
return err
535+
}
516536
}
517537
}
518538
}

docs/contributing.md

-2
Original file line numberDiff line numberDiff line change
@@ -37,8 +37,6 @@ updated `nix/packages/treefmt/gomod2nix.toml`.
3737

3838
To sync it up, run `nix develop .#renovate -c gomod2nix:update`.
3939

40-
41-
4240
## Making changes
4341

4442
If you want to introduce changes to the project, please follow these steps:

format/formatter.go

+3-4
Original file line numberDiff line numberDiff line change
@@ -75,16 +75,15 @@ func (f *Formatter) Apply(ctx context.Context, tasks []*Task) error {
7575
f.log.Debugf("executing: %s", cmd.String())
7676

7777
if out, err := cmd.CombinedOutput(); err != nil {
78+
f.log.Errorf("failed to apply with options '%v': %s", f.config.Options, err)
7879
if len(out) > 0 {
7980
_, _ = fmt.Fprintf(os.Stderr, "%s error:\n%s\n", f.name, out)
8081
}
8182
return fmt.Errorf("formatter '%s' with options '%v' failed to apply: %w", f.config.Command, f.config.Options, err)
83+
} else {
84+
f.log.Infof("%v file(s) processed in %v", len(tasks), time.Since(start))
8285
}
8386

84-
//
85-
86-
f.log.Infof("%v file(s) processed in %v", len(tasks), time.Since(start))
87-
8887
return nil
8988
}
9089

format/task.go

+1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ type Task struct {
1111
File *walk.File
1212
Formatters []*Formatter
1313
BatchKey string
14+
Errors []error
1415
}
1516

1617
func NewTask(file *walk.File, formatters []*Formatter) Task {

0 commit comments

Comments
 (0)