Skip to content

Commit 30e1ecd

Browse files
committed
feat: improve error handling in format Run method
When reviewing #511, I noticed there was some shadowing of variables and edge cases which were not being handled gracefully. Signed-off-by: Brian McGee <[email protected]>
1 parent 45881a4 commit 30e1ecd

File tree

2 files changed

+46
-21
lines changed

2 files changed

+46
-21
lines changed

cmd/format/format.go

+44-19
Original file line numberDiff line numberDiff line change
@@ -167,45 +167,70 @@ func Run(v *viper.Viper, statz *stats.Stats, cmd *cobra.Command, paths []string)
167167
// start traversing
168168
files := make([]*walk.File, BatchSize)
169169

170+
var (
171+
n int
172+
readErr, formatErr error
173+
)
174+
170175
for {
171176
// read the next batch
172-
readCtx, cancel := context.WithTimeout(ctx, 1*time.Second)
173-
n, err := walker.Read(readCtx, files)
177+
readCtx, cancelRead := context.WithTimeout(ctx, 1*time.Second)
178+
179+
n, readErr = walker.Read(readCtx, files)
180+
log.Debugf("read %d files", n)
174181

175182
// ensure context is cancelled to release resources
176-
cancel()
183+
cancelRead()
177184

178-
// format
179-
if err := formatter.Apply(ctx, files[:n]); err != nil {
180-
return fmt.Errorf("formatting failure: %w", err)
185+
// format any files that were read before processing the read error
186+
if formatErr = formatter.Apply(ctx, files[:n]); formatErr != nil {
187+
break
181188
}
182189

183-
if errors.Is(err, io.EOF) {
184-
// we have finished traversing
190+
// stop reading files if there was a read error
191+
if readErr != nil {
185192
break
186-
} else if err != nil {
187-
// something went wrong
188-
return fmt.Errorf("failed to read files: %w", err)
189193
}
190194
}
191195

192-
// finalize formatting
193-
formatErr := formatter.Close(ctx)
196+
// finalize formatting (there could be formatting tasks in-flight)
197+
formatCloseErr := formatter.Close(ctx)
194198

195199
// close the walker, ensuring any pending file release hooks finish
196-
if err = walker.Close(); err != nil {
197-
return fmt.Errorf("failed to close walker: %w", err)
198-
}
200+
walkerCloseErr := walker.Close()
199201

200202
// print stats to stderr
201203
if !cfg.Quiet {
202204
statz.PrintToStderr()
203205
}
204206

207+
// process errors
208+
209+
//nolint:gocritic
210+
if errors.Is(readErr, io.EOF) {
211+
// nothing more to read, reset the error and break out of the read loop
212+
log.Debugf("no more files to read")
213+
} else if errors.Is(readErr, context.DeadlineExceeded) {
214+
// the read timed-out
215+
return errors.New("timeout reading files")
216+
} else if readErr != nil {
217+
// something unexpected happened
218+
return fmt.Errorf("failed to read files: %w", readErr)
219+
}
220+
205221
if formatErr != nil {
206-
// return an error if any formatting failures were detected
207-
return formatErr //nolint:wrapcheck
208-
} else if cfg.FailOnChange && statz.Value(stats.Changed) != 0 {
222+
return fmt.Errorf("failed to format files: %w", formatErr)
223+
}
224+
225+
if formatCloseErr != nil {
226+
return fmt.Errorf("failed to finalise formatting: %w", formatCloseErr)
227+
}
228+
229+
if walkerCloseErr != nil {
230+
return fmt.Errorf("failed to close walker: %w", walkerCloseErr)
231+
}
232+
233+
if cfg.FailOnChange && statz.Value(stats.Changed) != 0 {
209234
// if fail on change has been enabled, check that no files were actually changed, throwing an error if so
210235
return ErrFailOnChange
211236
}

cmd/root_test.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -1433,12 +1433,12 @@ func TestGit(t *testing.T) {
14331433
}),
14341434
)
14351435

1436-
// try with a path not in the git index, e.g. it is skipped
1436+
// try with a path not in the git index
14371437
_, err := os.Create(filepath.Join(tempDir, "foo.txt"))
14381438
as.NoError(err)
14391439

14401440
treefmt(t,
1441-
withArgs("haskell", "foo.txt"),
1441+
withArgs("haskell", "foo.txt", "-vv"),
14421442
withConfig(configPath, cfg),
14431443
withNoError(t),
14441444
withStats(t, map[stats.Type]int{

0 commit comments

Comments
 (0)