@@ -11,41 +11,50 @@ import (
11
11
)
12
12
13
13
const (
14
+ // ErrFormatterNotFound is returned when the Command for a Formatter is not available.
14
15
ErrFormatterNotFound = errors .ConstError ("formatter not found" )
15
16
)
16
17
18
+ // Formatter represents a command which should be applied to a filesystem.
17
19
type Formatter struct {
18
- Name string
19
- Command string
20
- Options []string
20
+ // Command is the command invoke when applying this Formatter.
21
+ Command string
22
+ // Options are an optional list of args to be passed to Command.
23
+ Options []string
24
+ // Includes is a list of glob patterns used to determine whether this Formatter should be applied against a path.
21
25
Includes []string
26
+ // Excludes is an optional list of glob patterns used to exclude certain files from this Formatter.
22
27
Excludes []string
23
- Before []string
24
28
25
- log * log.Logger
29
+ name string
30
+ log * log.Logger
26
31
27
- // globs for matching against paths
32
+ // internal compiled versions of Includes and Excludes.
28
33
includes []glob.Glob
29
34
excludes []glob.Glob
30
35
36
+ // inbox is used to accept new paths for formatting.
31
37
inbox chan string
32
38
39
+ // Entries from inbox are batched according to batchSize and stored in batch for processing when the batchSize has
40
+ // been reached or Close is invoked.
33
41
batch []string
34
42
batchSize int
35
43
}
36
44
37
45
func (f * Formatter ) Init (name string ) error {
38
- f .Name = name
46
+ // capture the name from the config file
47
+ f .name = name
39
48
40
49
// test if the formatter is available
41
50
if err := exec .Command (f .Command , "--help" ).Run (); err != nil {
42
51
return ErrFormatterNotFound
43
52
}
44
53
54
+ // initialise internal state
45
55
f .log = log .WithPrefix ("format | " + name )
46
- f .inbox = make (chan string , 1024 )
47
-
48
56
f .batchSize = 1024
57
+ f .inbox = make (chan string , f .batchSize )
49
58
f .batch = make ([]string , f .batchSize )
50
59
f .batch = f .batch [:0 ]
51
60
@@ -54,7 +63,7 @@ func (f *Formatter) Init(name string) error {
54
63
for _ , pattern := range f .Includes {
55
64
g , err := glob .Compile ("**/" + pattern )
56
65
if err != nil {
57
- return errors .Annotatef (err , "failed to compile include pattern '%v' for formatter '%v'" , pattern , f .Name )
66
+ return errors .Annotatef (err , "failed to compile include pattern '%v' for formatter '%v'" , pattern , f .name )
58
67
}
59
68
f .includes = append (f .includes , g )
60
69
}
@@ -64,7 +73,7 @@ func (f *Formatter) Init(name string) error {
64
73
for _ , pattern := range f .Excludes {
65
74
g , err := glob .Compile ("**/" + pattern )
66
75
if err != nil {
67
- return errors .Annotatef (err , "failed to compile exclude pattern '%v' for formatter '%v'" , pattern , f .Name )
76
+ return errors .Annotatef (err , "failed to compile exclude pattern '%v' for formatter '%v'" , pattern , f .name )
68
77
}
69
78
f .excludes = append (f .excludes , g )
70
79
}
@@ -73,6 +82,8 @@ func (f *Formatter) Init(name string) error {
73
82
return nil
74
83
}
75
84
85
+ // Wants is used to test if a Formatter wants path based on it's configured Includes and Excludes patterns.
86
+ // Returns true if the Formatter should be applied to path, false otherwise.
76
87
func (f * Formatter ) Wants (path string ) bool {
77
88
match := ! PathMatches (path , f .excludes ) && PathMatches (path , f .includes )
78
89
if match {
@@ -81,24 +92,31 @@ func (f *Formatter) Wants(path string) bool {
81
92
return match
82
93
}
83
94
95
+ // Put add path into this Formatter's inbox for processing.
84
96
func (f * Formatter ) Put (path string ) {
85
97
f .inbox <- path
86
98
}
87
99
100
+ // Run is the main processing loop for this Formatter.
101
+ // It accepts a context which is used to lookup certain dependencies and for cancellation.
88
102
func (f * Formatter ) Run (ctx context.Context ) (err error ) {
89
103
LOOP:
104
+ // keep processing until ctx has been cancelled or inbox has been closed
90
105
for {
91
106
select {
107
+
92
108
case <- ctx .Done ():
109
+ // ctx has been cancelled
93
110
err = ctx .Err ()
94
111
break LOOP
95
112
96
113
case path , ok := <- f .inbox :
114
+ // check if the inbox has been closed
97
115
if ! ok {
98
116
break LOOP
99
117
}
100
118
101
- // add to the current batch
119
+ // add path to the current batch
102
120
f .batch = append (f .batch , path )
103
121
104
122
if len (f .batch ) == f .batchSize {
@@ -110,14 +128,17 @@ LOOP:
110
128
}
111
129
}
112
130
131
+ // check if LOOP was exited due to an error
113
132
if err != nil {
114
133
return
115
134
}
116
135
117
- // final flush
136
+ // processing any lingering batch
118
137
return f .apply (ctx )
119
138
}
120
139
140
+ // apply executes Command against the latest batch of paths.
141
+ // It accepts a context which is used to lookup certain dependencies and for cancellation.
121
142
func (f * Formatter ) apply (ctx context.Context ) error {
122
143
// empty check
123
144
if len (f .batch ) == 0 {
@@ -132,6 +153,7 @@ func (f *Formatter) apply(ctx context.Context) error {
132
153
args = append (args , path )
133
154
}
134
155
156
+ // execute
135
157
start := time .Now ()
136
158
cmd := exec .CommandContext (ctx , f .Command , args ... )
137
159
@@ -142,15 +164,9 @@ func (f *Formatter) apply(ctx context.Context) error {
142
164
143
165
f .log .Infof ("%v files processed in %v" , len (f .batch ), time .Now ().Sub (start ))
144
166
145
- // mark completed or forward on
146
- if len (f .Before ) == 0 {
147
- for _ , path := range f .batch {
148
- MarkFormatComplete (ctx , path )
149
- }
150
- } else {
151
- for _ , path := range f .batch {
152
- ForwardPath (ctx , path , f .Before )
153
- }
167
+ // mark each path in this batch as completed
168
+ for _ , path := range f .batch {
169
+ MarkFormatComplete (ctx , path )
154
170
}
155
171
156
172
// reset batch
@@ -159,6 +175,7 @@ func (f *Formatter) apply(ctx context.Context) error {
159
175
return nil
160
176
}
161
177
178
+ // Close is used to indicate that a Formatter should process any remaining paths and then stop it's processing loop.
162
179
func (f * Formatter ) Close () {
163
180
close (f .inbox )
164
181
}
0 commit comments