Skip to content

Commit 4c45d2a

Browse files
brianmcgeeBrian McGee
authored and
Brian McGee
committed
feat: allow missing formatters (#6)
Closes #3 Reviewed-on: https://git.numtide.com/numtide/treefmt/pulls/6 Co-authored-by: Brian McGee <[email protected]> Co-committed-by: Brian McGee <[email protected]>
1 parent 298e5ac commit 4c45d2a

File tree

7 files changed

+104
-8
lines changed

7 files changed

+104
-8
lines changed

internal/cache/cache.go

+3
Original file line numberDiff line numberDiff line change
@@ -61,6 +61,9 @@ func Open(treeRoot string, clean bool) (err error) {
6161
}
6262

6363
func Close() error {
64+
if db == nil {
65+
return nil
66+
}
6467
return db.Close()
6568
}
6669

internal/cli/cli.go

+5-4
Original file line numberDiff line numberDiff line change
@@ -5,10 +5,11 @@ import "github.com/charmbracelet/log"
55
var Cli = Options{}
66

77
type Options struct {
8-
Verbosity int `name:"verbose" short:"v" type:"counter" default:"0" env:"LOG_LEVEL" help:"Set the verbosity of logs e.g. -vv"`
9-
ConfigFile string `type:"existingfile" default:"./treefmt.toml"`
10-
TreeRoot string `type:"existingdir" default:"."`
11-
ClearCache bool `short:"c" help:"Reset the evaluation cache. Use in case the cache is not precise enough"`
8+
AllowMissingFormatter bool `default:"false" help:"Do not exit with error if a configured formatter is missing"`
9+
ClearCache bool `short:"c" help:"Reset the evaluation cache. Use in case the cache is not precise enough"`
10+
ConfigFile string `type:"existingfile" default:"./treefmt.toml"`
11+
TreeRoot string `type:"existingdir" default:"."`
12+
Verbosity int `name:"verbose" short:"v" type:"counter" default:"0" env:"LOG_LEVEL" help:"Set the verbosity of logs e.g. -vv"`
1213

1314
Format Format `cmd:"" default:"."`
1415
}

internal/cli/format.go

+8-1
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,12 @@ func (f *Format) Run() error {
4444

4545
// init formatters
4646
for name, formatter := range cfg.Formatters {
47-
if err = formatter.Init(name); err != nil {
47+
err = formatter.Init(name)
48+
if err == format.ErrFormatterNotFound && Cli.AllowMissingFormatter {
49+
l.Debugf("formatter not found: %v", name)
50+
// remove this formatter
51+
delete(cfg.Formatters, name)
52+
} else if err != nil {
4853
return errors.Annotatef(err, "failed to initialise formatter: %v", name)
4954
}
5055
}
@@ -155,5 +160,7 @@ func (f *Format) Run() error {
155160
return cache.ChangeSet(ctx, Cli.TreeRoot, pathsCh)
156161
})
157162

163+
// shutdown.Listen(syscall.SIGINT, syscall.SIGTERM)
164+
158165
return eg.Wait()
159166
}

internal/cli/format_test.go

+68
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
package cli
2+
3+
import (
4+
"os"
5+
"testing"
6+
7+
"git.numtide.com/numtide/treefmt/internal/format"
8+
"github.com/BurntSushi/toml"
9+
"github.com/alecthomas/kong"
10+
"github.com/stretchr/testify/assert"
11+
"github.com/stretchr/testify/require"
12+
)
13+
14+
func writeConfig(t *testing.T, path string, cfg format.Config) {
15+
t.Helper()
16+
f, err := os.Create(path)
17+
if err != nil {
18+
t.Fatalf("failed to create a new config file: %v", err)
19+
}
20+
encoder := toml.NewEncoder(f)
21+
if err = encoder.Encode(cfg); err != nil {
22+
t.Fatalf("failed to write to config file: %v", err)
23+
}
24+
}
25+
26+
func newKong(t *testing.T, cli interface{}, options ...kong.Option) *kong.Kong {
27+
t.Helper()
28+
options = append([]kong.Option{
29+
kong.Name("test"),
30+
kong.Exit(func(int) {
31+
t.Helper()
32+
t.Fatalf("unexpected exit()")
33+
}),
34+
}, options...)
35+
parser, err := kong.New(cli, options...)
36+
assert.NoError(t, err)
37+
return parser
38+
}
39+
40+
func newCli(t *testing.T, args ...string) (*kong.Context, error) {
41+
t.Helper()
42+
p := newKong(t, &Cli)
43+
return p.Parse(args)
44+
}
45+
46+
func TestAllowMissingFormatter(t *testing.T) {
47+
as := require.New(t)
48+
49+
tempDir := t.TempDir()
50+
configPath := tempDir + "/treefmt.toml"
51+
52+
writeConfig(t, configPath, format.Config{
53+
Formatters: map[string]*format.Formatter{
54+
"foo-fmt": {
55+
Command: "foo-fmt",
56+
},
57+
},
58+
})
59+
60+
ctx, err := newCli(t, "--config-file", configPath, "--tree-root", tempDir)
61+
as.NoError(err)
62+
as.Error(ctx.Run(), format.ErrFormatterNotFound)
63+
64+
ctx, err = newCli(t, "--config-file", configPath, "--tree-root", tempDir, "--allow-missing-formatter")
65+
as.NoError(err)
66+
67+
as.NoError(ctx.Run())
68+
}

internal/format/config_test.go

+6-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ import (
66
"github.com/stretchr/testify/require"
77
)
88

9-
func TestConfig(t *testing.T) {
9+
func TestReadConfigFile(t *testing.T) {
1010
as := require.New(t)
1111

1212
cfg, err := ReadConfigFile("../../test/treefmt.toml")
@@ -119,4 +119,9 @@ shfmt -i 2 -s -w "$@"
119119
as.Equal([]string{"fmt"}, terraform.Options)
120120
as.Equal([]string{"*.tf"}, terraform.Includes)
121121
as.Nil(terraform.Excludes)
122+
123+
// missing
124+
foo, ok := cfg.Formatters["foo-fmt"]
125+
as.True(ok, "foo formatter not found")
126+
as.Equal("foo-fmt", foo.Command)
122127
}

internal/format/format.go

+10-1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,10 @@ import (
1010
"github.com/juju/errors"
1111
)
1212

13+
const (
14+
ErrFormatterNotFound = errors.ConstError("formatter not found")
15+
)
16+
1317
type Formatter struct {
1418
Name string
1519
Command string
@@ -32,8 +36,13 @@ type Formatter struct {
3236

3337
func (f *Formatter) Init(name string) error {
3438
f.Name = name
35-
f.log = log.WithPrefix("format | " + name)
3639

40+
// test if the formatter is available
41+
if err := exec.Command(f.Command, "--help").Run(); err != nil {
42+
return ErrFormatterNotFound
43+
}
44+
45+
f.log = log.WithPrefix("format | " + name)
3746
f.inbox = make(chan string, 1024)
3847

3948
f.batchSize = 1024

test/treefmt.toml

+4-1
Original file line numberDiff line numberDiff line change
@@ -79,4 +79,7 @@ includes = ["*.sh"]
7979
# see https://github.com/numtide/treefmt/issues/97
8080
command = "terraform"
8181
options = ["fmt"]
82-
includes = ["*.tf"]
82+
includes = ["*.tf"]
83+
84+
[formatter.foo-fmt]
85+
command = "foo-fmt"

0 commit comments

Comments
 (0)