Skip to content

Commit b6405d0

Browse files
committed
feat: clean up and documentation
1 parent f322e1f commit b6405d0

File tree

6 files changed

+70
-67
lines changed

6 files changed

+70
-67
lines changed

internal/cache/cache.go

+23-6
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ import (
88
"io/fs"
99
"os"
1010
"path/filepath"
11+
"time"
1112

1213
"github.com/adrg/xdg"
1314
"github.com/juju/errors"
@@ -19,8 +20,19 @@ const (
1920
modifiedBucket = "modified"
2021
)
2122

23+
// Entry represents a cache entry, indicating the last size and modified time for a file path.
24+
type Entry struct {
25+
Size int64
26+
Modified time.Time
27+
}
28+
2229
var db *bolt.DB
2330

31+
// Open creates an instance of bolt.DB for a given treeRoot path.
32+
// If clean is true, Open will delete any existing data in the cache.
33+
//
34+
// The database will be located in `XDG_CACHE_DIR/treefmt/eval-cache/<id>.db`, where <id> is determined by hashing
35+
// the treeRoot path. This associates a given treeRoot with a given instance of the cache.
2436
func Open(treeRoot string, clean bool) (err error) {
2537
// determine a unique and consistent db name for the tree root
2638
h := sha1.New()
@@ -30,7 +42,7 @@ func Open(treeRoot string, clean bool) (err error) {
3042
name := base32.StdEncoding.EncodeToString(digest)
3143
path, err := xdg.CacheFile(fmt.Sprintf("treefmt/eval-cache/%v.db", name))
3244

33-
// bust the cache if specified
45+
// force a clean of the cache if specified
3446
if clean {
3547
err := os.Remove(path)
3648
if errors.Is(err, os.ErrNotExist) {
@@ -60,17 +72,19 @@ func Open(treeRoot string, clean bool) (err error) {
6072
return
6173
}
6274

75+
// Close closes any open instance of the cache.
6376
func Close() error {
6477
if db == nil {
6578
return nil
6679
}
6780
return db.Close()
6881
}
6982

70-
func getFileInfo(bucket *bolt.Bucket, path string) (*FileInfo, error) {
83+
// getEntry is a helper for reading cache entries from bolt.
84+
func getEntry(bucket *bolt.Bucket, path string) (*Entry, error) {
7185
b := bucket.Get([]byte(path))
7286
if b != nil {
73-
var cached FileInfo
87+
var cached Entry
7488
if err := msgpack.Unmarshal(b, &cached); err != nil {
7589
return nil, errors.Annotatef(err, "failed to unmarshal cache info for path '%v'", path)
7690
}
@@ -80,6 +94,8 @@ func getFileInfo(bucket *bolt.Bucket, path string) (*FileInfo, error) {
8094
}
8195
}
8296

97+
// ChangeSet is used to walk a filesystem, starting at root, and outputting any new or changed paths using pathsCh.
98+
// It determines if a path is new or has changed by comparing against cache entries.
8399
func ChangeSet(ctx context.Context, root string, pathsCh chan<- string) error {
84100
return db.Update(func(tx *bolt.Tx) error {
85101
bucket := tx.Bucket([]byte(modifiedBucket))
@@ -99,7 +115,7 @@ func ChangeSet(ctx context.Context, root string, pathsCh chan<- string) error {
99115
return nil
100116
}
101117

102-
cached, err := getFileInfo(bucket, path)
118+
cached, err := getEntry(bucket, path)
103119
if err != nil {
104120
return err
105121
}
@@ -118,6 +134,7 @@ func ChangeSet(ctx context.Context, root string, pathsCh chan<- string) error {
118134
})
119135
}
120136

137+
// Update is used to record updated cache information for the specified list of paths.
121138
func Update(paths []string) (int, error) {
122139
if len(paths) == 0 {
123140
return 0, nil
@@ -133,7 +150,7 @@ func Update(paths []string) (int, error) {
133150
continue
134151
}
135152

136-
cached, err := getFileInfo(bucket, path)
153+
cached, err := getEntry(bucket, path)
137154
if err != nil {
138155
return err
139156
}
@@ -150,7 +167,7 @@ func Update(paths []string) (int, error) {
150167
continue
151168
}
152169

153-
cacheInfo := FileInfo{
170+
cacheInfo := Entry{
154171
Size: pathInfo.Size(),
155172
Modified: pathInfo.ModTime(),
156173
}

internal/cache/types.go

-8
This file was deleted.

internal/format/config.go

+2
Original file line numberDiff line numberDiff line change
@@ -2,10 +2,12 @@ package format
22

33
import "github.com/BurntSushi/toml"
44

5+
// Config is used to represent the list of configured Formatters.
56
type Config struct {
67
Formatters map[string]*Formatter `toml:"formatter"`
78
}
89

10+
// ReadConfigFile reads from path and unmarshals toml into a Config instance.
911
func ReadConfigFile(path string) (cfg *Config, err error) {
1012
_, err = toml.DecodeFile(path, &cfg)
1113
return

internal/format/context.go

+6-10
Original file line numberDiff line numberDiff line change
@@ -9,28 +9,24 @@ const (
99
completedChKey = "completedCh"
1010
)
1111

12+
// RegisterFormatters is used to set a map of formatters in the provided context.
1213
func RegisterFormatters(ctx context.Context, formatters map[string]*Formatter) context.Context {
1314
return context.WithValue(ctx, formattersKey, formatters)
1415
}
1516

17+
// GetFormatters is used to retrieve a formatters map from the provided context.
1618
func GetFormatters(ctx context.Context) map[string]*Formatter {
1719
return ctx.Value(formattersKey).(map[string]*Formatter)
1820
}
1921

22+
// SetCompletedChannel is used to set a channel for indication processing completion in the provided context.
2023
func SetCompletedChannel(ctx context.Context, completedCh chan string) context.Context {
2124
return context.WithValue(ctx, completedChKey, completedCh)
2225
}
2326

27+
// MarkFormatComplete is used to indicate that all processing has finished for the provided path.
28+
// This is done by adding the path to the completion channel which should have already been set using
29+
// SetCompletedChannel.
2430
func MarkFormatComplete(ctx context.Context, path string) {
2531
ctx.Value(completedChKey).(chan string) <- path
2632
}
27-
28-
func ForwardPath(ctx context.Context, path string, names []string) {
29-
if len(names) == 0 {
30-
return
31-
}
32-
formatters := GetFormatters(ctx)
33-
for _, name := range names {
34-
formatters[name].Put(path)
35-
}
36-
}

internal/format/format.go

+39-22
Original file line numberDiff line numberDiff line change
@@ -11,41 +11,50 @@ import (
1111
)
1212

1313
const (
14+
// ErrFormatterNotFound is returned when the Command for a Formatter is not available.
1415
ErrFormatterNotFound = errors.ConstError("formatter not found")
1516
)
1617

18+
// Formatter represents a command which should be applied to a filesystem.
1719
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.
2125
Includes []string
26+
// Excludes is an optional list of glob patterns used to exclude certain files from this Formatter.
2227
Excludes []string
23-
Before []string
2428

25-
log *log.Logger
29+
name string
30+
log *log.Logger
2631

27-
// globs for matching against paths
32+
// internal compiled versions of Includes and Excludes.
2833
includes []glob.Glob
2934
excludes []glob.Glob
3035

36+
// inbox is used to accept new paths for formatting.
3137
inbox chan string
3238

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.
3341
batch []string
3442
batchSize int
3543
}
3644

3745
func (f *Formatter) Init(name string) error {
38-
f.Name = name
46+
// capture the name from the config file
47+
f.name = name
3948

4049
// test if the formatter is available
4150
if err := exec.Command(f.Command, "--help").Run(); err != nil {
4251
return ErrFormatterNotFound
4352
}
4453

54+
// initialise internal state
4555
f.log = log.WithPrefix("format | " + name)
46-
f.inbox = make(chan string, 1024)
47-
4856
f.batchSize = 1024
57+
f.inbox = make(chan string, f.batchSize)
4958
f.batch = make([]string, f.batchSize)
5059
f.batch = f.batch[:0]
5160

@@ -54,7 +63,7 @@ func (f *Formatter) Init(name string) error {
5463
for _, pattern := range f.Includes {
5564
g, err := glob.Compile("**/" + pattern)
5665
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)
5867
}
5968
f.includes = append(f.includes, g)
6069
}
@@ -64,7 +73,7 @@ func (f *Formatter) Init(name string) error {
6473
for _, pattern := range f.Excludes {
6574
g, err := glob.Compile("**/" + pattern)
6675
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)
6877
}
6978
f.excludes = append(f.excludes, g)
7079
}
@@ -73,6 +82,8 @@ func (f *Formatter) Init(name string) error {
7382
return nil
7483
}
7584

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.
7687
func (f *Formatter) Wants(path string) bool {
7788
match := !PathMatches(path, f.excludes) && PathMatches(path, f.includes)
7889
if match {
@@ -81,24 +92,31 @@ func (f *Formatter) Wants(path string) bool {
8192
return match
8293
}
8394

95+
// Put add path into this Formatter's inbox for processing.
8496
func (f *Formatter) Put(path string) {
8597
f.inbox <- path
8698
}
8799

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.
88102
func (f *Formatter) Run(ctx context.Context) (err error) {
89103
LOOP:
104+
// keep processing until ctx has been cancelled or inbox has been closed
90105
for {
91106
select {
107+
92108
case <-ctx.Done():
109+
// ctx has been cancelled
93110
err = ctx.Err()
94111
break LOOP
95112

96113
case path, ok := <-f.inbox:
114+
// check if the inbox has been closed
97115
if !ok {
98116
break LOOP
99117
}
100118

101-
// add to the current batch
119+
// add path to the current batch
102120
f.batch = append(f.batch, path)
103121

104122
if len(f.batch) == f.batchSize {
@@ -110,14 +128,17 @@ LOOP:
110128
}
111129
}
112130

131+
// check if LOOP was exited due to an error
113132
if err != nil {
114133
return
115134
}
116135

117-
// final flush
136+
// processing any lingering batch
118137
return f.apply(ctx)
119138
}
120139

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.
121142
func (f *Formatter) apply(ctx context.Context) error {
122143
// empty check
123144
if len(f.batch) == 0 {
@@ -132,6 +153,7 @@ func (f *Formatter) apply(ctx context.Context) error {
132153
args = append(args, path)
133154
}
134155

156+
// execute
135157
start := time.Now()
136158
cmd := exec.CommandContext(ctx, f.Command, args...)
137159

@@ -142,15 +164,9 @@ func (f *Formatter) apply(ctx context.Context) error {
142164

143165
f.log.Infof("%v files processed in %v", len(f.batch), time.Now().Sub(start))
144166

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)
154170
}
155171

156172
// reset batch
@@ -159,6 +175,7 @@ func (f *Formatter) apply(ctx context.Context) error {
159175
return nil
160176
}
161177

178+
// Close is used to indicate that a Formatter should process any remaining paths and then stop it's processing loop.
162179
func (f *Formatter) Close() {
163180
close(f.inbox)
164181
}

internal/log/writer.go

-21
This file was deleted.

0 commit comments

Comments
 (0)