@@ -149,96 +149,21 @@ func (f *Format) Run() (err error) {
149
149
f .filesCh = make (chan * walk.File , BatchSize * runtime .NumCPU ())
150
150
151
151
// create a channel for files that have been processed
152
+ f .formattedCh = make (chan * walk.File , cap (f .filesCh ))
153
+
154
+ // create a channel for files that have been formatted
152
155
f .processedCh = make (chan * walk.File , cap (f .filesCh ))
153
156
154
157
// start concurrent processing tasks in reverse order
155
- if ! f .NoCache {
156
- eg .Go (f .updateCache (ctx ))
157
- }
158
+ eg .Go (f .updateCache (ctx ))
159
+ eg .Go (f .detectFormatted (ctx ))
158
160
eg .Go (f .applyFormatters (ctx ))
159
161
eg .Go (f .walkFilesystem (ctx ))
160
162
161
163
// wait for everything to complete
162
164
return eg .Wait ()
163
165
}
164
166
165
- func (f * Format ) updateCache (ctx context.Context ) func () error {
166
- return func () error {
167
- // used to batch updates for more efficient txs
168
- batch := make ([]* walk.File , 0 , BatchSize )
169
-
170
- // apply a batch
171
- processBatch := func () error {
172
- if f .Stdin {
173
- // do nothing
174
- return nil
175
- }
176
- if err := cache .Update (batch ); err != nil {
177
- return err
178
- }
179
- batch = batch [:0 ]
180
- return nil
181
- }
182
-
183
- LOOP:
184
- for {
185
- select {
186
- // detect ctx cancellation
187
- case <- ctx .Done ():
188
- return ctx .Err ()
189
- // respond to processed files
190
- case file , ok := <- f .processedCh :
191
- if ! ok {
192
- // channel has been closed, no further files to process
193
- break LOOP
194
- }
195
-
196
- if f .Stdin {
197
- // dump file into stdout
198
- f , err := os .Open (file .Path )
199
- if err != nil {
200
- return fmt .Errorf ("failed to open %s: %w" , file .Path , err )
201
- }
202
- if _ , err = io .Copy (os .Stdout , f ); err != nil {
203
- return fmt .Errorf ("failed to copy %s to stdout: %w" , file .Path , err )
204
- }
205
- if err = os .Remove (f .Name ()); err != nil {
206
- return fmt .Errorf ("failed to remove temp file %s: %w" , file .Path , err )
207
- }
208
-
209
- stats .Add (stats .Formatted , 1 )
210
- continue
211
- }
212
-
213
- // append to batch and process if we have enough
214
- batch = append (batch , file )
215
- if len (batch ) == BatchSize {
216
- if err := processBatch (); err != nil {
217
- return err
218
- }
219
- }
220
- }
221
- }
222
-
223
- // final flush
224
- if err := processBatch (); err != nil {
225
- return err
226
- }
227
-
228
- // if fail on change has been enabled, check that no files were actually formatted, throwing an error if so
229
- if f .FailOnChange && stats .Value (stats .Formatted ) != 0 {
230
- return ErrFailOnChange
231
- }
232
-
233
- // print stats to stdout unless we are processing stdin and printing the results to stdout
234
- if ! f .Stdin {
235
- stats .Print ()
236
- }
237
-
238
- return nil
239
- }
240
- }
241
-
242
167
func (f * Format ) walkFilesystem (ctx context.Context ) func () error {
243
168
return func () error {
244
169
eg , ctx := errgroup .WithContext (ctx )
@@ -368,9 +293,9 @@ func (f *Format) applyFormatters(ctx context.Context) func() error {
368
293
}
369
294
}
370
295
371
- // pass each file to the processed channel
296
+ // pass each file to the formatted channel
372
297
for _ , task := range tasks {
373
- f .processedCh <- task .File
298
+ f .formattedCh <- task .File
374
299
}
375
300
376
301
return nil
@@ -392,7 +317,7 @@ func (f *Format) applyFormatters(ctx context.Context) func() error {
392
317
return func () error {
393
318
defer func () {
394
319
// close processed channel
395
- close (f .processedCh )
320
+ close (f .formattedCh )
396
321
}()
397
322
398
323
// iterate the files channel
@@ -402,7 +327,7 @@ func (f *Format) applyFormatters(ctx context.Context) func() error {
402
327
if format .PathMatches (file .RelPath , f .globalExcludes ) {
403
328
log .Debugf ("path matched global excludes: %s" , file .RelPath )
404
329
// mark it as processed and continue to the next
405
- f .processedCh <- file
330
+ f .formattedCh <- file
406
331
continue
407
332
}
408
333
@@ -421,7 +346,7 @@ func (f *Format) applyFormatters(ctx context.Context) func() error {
421
346
}
422
347
log .Logf (f .OnUnmatched , "no formatter for path: %s" , file .RelPath )
423
348
// mark it as processed and continue to the next
424
- f .processedCh <- file
349
+ f .formattedCh <- file
425
350
} else {
426
351
// record the match
427
352
stats .Add (stats .Matched , 1 )
@@ -444,6 +369,127 @@ func (f *Format) applyFormatters(ctx context.Context) func() error {
444
369
}
445
370
}
446
371
372
+ func (f * Format ) detectFormatted (ctx context.Context ) func () error {
373
+ return func () error {
374
+ defer func () {
375
+ // close formatted channel
376
+ close (f .processedCh )
377
+ }()
378
+
379
+ for {
380
+ select {
381
+
382
+ // detect ctx cancellation
383
+ case <- ctx .Done ():
384
+ return ctx .Err ()
385
+ // take the next file that has been processed
386
+ case file , ok := <- f .formattedCh :
387
+ if ! ok {
388
+ // channel has been closed, no further files to process
389
+ return nil
390
+ }
391
+
392
+ // look up current file info
393
+ currentInfo , err := os .Stat (file .Path )
394
+ if err != nil {
395
+ return fmt .Errorf ("failed to stat processed file: %w" , err )
396
+ }
397
+
398
+ // check if the file has changed
399
+ if ! (file .Info .ModTime () == currentInfo .ModTime () && file .Info .Size () == currentInfo .Size ()) {
400
+ // record the change
401
+ stats .Add (stats .Formatted , 1 )
402
+ // update the file info
403
+ file .Info = currentInfo
404
+ }
405
+
406
+ // mark as processed
407
+ f .processedCh <- file
408
+ }
409
+ }
410
+ }
411
+ }
412
+
413
+ func (f * Format ) updateCache (ctx context.Context ) func () error {
414
+ return func () error {
415
+ // used to batch updates for more efficient txs
416
+ batch := make ([]* walk.File , 0 , BatchSize )
417
+
418
+ // apply a batch
419
+ processBatch := func () error {
420
+ // if we are processing from stdin that means we are outputting to stdout, no caching involved
421
+ // if f.NoCache is set that means either the user explicitly disabled the cache or we failed to open on
422
+ if f .Stdin || f .NoCache {
423
+ // do nothing
424
+ return nil
425
+ }
426
+
427
+ // pass the batch to the cache for updating
428
+ if err := cache .Update (batch ); err != nil {
429
+ return err
430
+ }
431
+ batch = batch [:0 ]
432
+ return nil
433
+ }
434
+
435
+ LOOP:
436
+ for {
437
+ select {
438
+ // detect ctx cancellation
439
+ case <- ctx .Done ():
440
+ return ctx .Err ()
441
+ // respond to formatted files
442
+ case file , ok := <- f .processedCh :
443
+ if ! ok {
444
+ // channel has been closed, no further files to process
445
+ break LOOP
446
+ }
447
+
448
+ if f .Stdin {
449
+ // dump file into stdout
450
+ f , err := os .Open (file .Path )
451
+ if err != nil {
452
+ return fmt .Errorf ("failed to open %s: %w" , file .Path , err )
453
+ }
454
+ if _ , err = io .Copy (os .Stdout , f ); err != nil {
455
+ return fmt .Errorf ("failed to copy %s to stdout: %w" , file .Path , err )
456
+ }
457
+ if err = os .Remove (f .Name ()); err != nil {
458
+ return fmt .Errorf ("failed to remove temp file %s: %w" , file .Path , err )
459
+ }
460
+
461
+ continue
462
+ }
463
+
464
+ // append to batch and process if we have enough
465
+ batch = append (batch , file )
466
+ if len (batch ) == BatchSize {
467
+ if err := processBatch (); err != nil {
468
+ return err
469
+ }
470
+ }
471
+ }
472
+ }
473
+
474
+ // final flush
475
+ if err := processBatch (); err != nil {
476
+ return err
477
+ }
478
+
479
+ // if fail on change has been enabled, check that no files were actually formatted, throwing an error if so
480
+ if f .FailOnChange && stats .Value (stats .Formatted ) != 0 {
481
+ return ErrFailOnChange
482
+ }
483
+
484
+ // print stats to stdout unless we are processing stdin and printing the results to stdout
485
+ if ! f .Stdin {
486
+ stats .Print ()
487
+ }
488
+
489
+ return nil
490
+ }
491
+ }
492
+
447
493
func findUp (searchDir string , fileName string ) (path string , dir string , err error ) {
448
494
for _ , dir := range eachDir (searchDir ) {
449
495
path := filepath .Join (dir , fileName )
0 commit comments