Skip to content

Commit

Permalink
Introduce autofix API
Browse files Browse the repository at this point in the history
  • Loading branch information
wata727 committed May 27, 2023
1 parent b8f431f commit e764c34
Show file tree
Hide file tree
Showing 19 changed files with 3,251 additions and 479 deletions.
35 changes: 34 additions & 1 deletion helper/runner.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,8 +21,10 @@ type Runner struct {
Issues Issues

files map[string]*hcl.File
sources map[string][]byte
config Config
variables map[string]*Variable
fixer *tflint.Fixer
}

// Variable is an implementation of variables in Terraform language
Expand Down Expand Up @@ -318,6 +320,30 @@ func (r *Runner) EmitIssue(rule tflint.Rule, message string, location hcl.Range)
return nil
}

// EmitIssueWithFix adds an issue and invoke fix.
func (r *Runner) EmitIssueWithFix(rule tflint.Rule, message string, location hcl.Range, fixFunc func(f *tflint.Fixer) error) error {
if err := fixFunc(r.fixer); err != nil {
return err
}
return r.EmitIssue(rule, message, location)
}

// HasChanges returns true if the fixer has changes.
func (r *Runner) HasChanges() bool {
return r.fixer.HasChanges()
}

// ApplyChanges applies changes to the fixer.
func (r *Runner) ApplyChanges() error {
r.fixer.ApplyChanges()
return nil
}

// Changes returns changes by the fixer.
func (r *Runner) Changes() map[string][]byte {
return r.fixer.Changes()
}

// EnsureNoError is a method that simply runs a function if there is no error.
//
// Deprecated: Use EvaluateExpr with a function callback. e.g. EvaluateExpr(expr, func (val T) error {}, ...)
Expand All @@ -331,7 +357,12 @@ func (r *Runner) EnsureNoError(err error, proc func() error) error {
// NewLocalRunner initialises a new test runner.
// Internal use only.
func NewLocalRunner(files map[string]*hcl.File, issues Issues) *Runner {
return &Runner{files: map[string]*hcl.File{}, variables: map[string]*Variable{}, Issues: issues}
return &Runner{
files: map[string]*hcl.File{},
sources: map[string][]byte{},
variables: map[string]*Variable{},
Issues: issues,
}
}

// AddLocalFile adds a new file to the current mapped files.
Expand All @@ -342,6 +373,7 @@ func (r *Runner) AddLocalFile(name string, file *hcl.File) bool {
}

r.files[name] = file
r.sources[name] = file.Bytes
return true
}

Expand All @@ -365,6 +397,7 @@ func (r *Runner) initFromFiles() error {
}
}
}
r.fixer = tflint.NewFixer(r.sources)

return nil
}
Expand Down
31 changes: 25 additions & 6 deletions helper/testing.go
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,8 @@ import (
// TestRunner returns a mock Runner for testing.
// You can pass the map of file names and their contents in the second argument.
func TestRunner(t *testing.T, files map[string]string) *Runner {
t.Helper()

runner := NewLocalRunner(map[string]*hcl.File{}, Issues{})
parser := hclparse.NewParser()

Expand Down Expand Up @@ -50,25 +52,42 @@ func TestRunner(t *testing.T, files map[string]string) *Runner {
}

// AssertIssues is an assertion helper for comparing issues.
func AssertIssues(t *testing.T, expected Issues, actual Issues) {
func AssertIssues(t *testing.T, want Issues, got Issues) {
t.Helper()

opts := []cmp.Option{
// Byte field will be ignored because it's not important in tests such as positions
cmpopts.IgnoreFields(hcl.Pos{}, "Byte"),
ruleComparer(),
}
if !cmp.Equal(expected, actual, opts...) {
t.Fatalf("Expected issues are not matched:\n %s\n", cmp.Diff(expected, actual, opts...))
if diff := cmp.Diff(want, got, opts...); diff != "" {
t.Fatalf("Expected issues are not matched:\n %s\n", diff)
}
}

// AssertIssuesWithoutRange is an assertion helper for comparing issues except for range.
func AssertIssuesWithoutRange(t *testing.T, expected Issues, actual Issues) {
func AssertIssuesWithoutRange(t *testing.T, want Issues, got Issues) {
t.Helper()

opts := []cmp.Option{
cmpopts.IgnoreFields(Issue{}, "Range"),
ruleComparer(),
}
if !cmp.Equal(expected, actual, opts...) {
t.Fatalf("Expected issues are not matched:\n %s\n", cmp.Diff(expected, actual, opts...))
if diff := cmp.Diff(want, got, opts...); diff != "" {
t.Fatalf("Expected issues are not matched:\n %s\n", diff)
}
}

// AssertChanges is an assertion helper for comparing autofix changes.
func AssertChanges(t *testing.T, want map[string]string, got map[string][]byte) {
t.Helper()

sources := make(map[string]string)
for name, src := range got {
sources[name] = string(src)
}
if diff := cmp.Diff(want, sources); diff != "" {
t.Fatalf("Expected changes are not matched:\n %s\n", diff)
}
}

Expand Down
1 change: 1 addition & 0 deletions plugin/fromproto/fromproto.go
Original file line number Diff line number Diff line change
Expand Up @@ -262,6 +262,7 @@ func Config(config *proto.ApplyGlobalConfig_Config) *tflint.Config {
Rules: rules,
DisabledByDefault: config.DisabledByDefault,
Only: config.Only,
Fix: config.Fix,
}
}

Expand Down
6 changes: 5 additions & 1 deletion plugin/host2plugin/host2plugin_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -575,7 +575,11 @@ func (s *mockServer) EvaluateExpr(expr hcl.Expression, opts tflint.EvaluateExprO
return cty.Value{}, nil
}

func (s *mockServer) EmitIssue(rule tflint.Rule, message string, location hcl.Range) error {
func (s *mockServer) EmitIssue(rule tflint.Rule, message string, location hcl.Range, fixable bool) (bool, error) {
return true, nil
}

func (s *mockServer) ApplyChanges(sources map[string][]byte) error {
return nil
}

Expand Down
8 changes: 7 additions & 1 deletion plugin/host2plugin/server.go
Original file line number Diff line number Diff line change
Expand Up @@ -116,7 +116,13 @@ func (s *GRPCServer) Check(ctx context.Context, req *proto.Check_Request) (*prot
}
defer conn.Close()

runner, err := s.impl.NewRunner(&plugin2host.GRPCClient{Client: proto.NewRunnerClient(conn)})
client := proto.NewRunnerClient(conn)
resp, err := client.GetFiles(ctx, &proto.GetFiles_Request{})
if err != nil {
return nil, toproto.Error(codes.FailedPrecondition, err)
}

runner, err := s.impl.NewRunner(&plugin2host.GRPCClient{Client: client, Fixer: tflint.NewFixer(resp.Files), FixEnabled: s.impl.FixEnabled()})
if err != nil {
return nil, toproto.Error(codes.FailedPrecondition, err)
}
Expand Down
42 changes: 41 additions & 1 deletion plugin/plugin2host/client.go
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,9 @@ import (

// GRPCClient is a plugin-side implementation. Plugin can send requests through the client to host's gRPC server.
type GRPCClient struct {
Client proto.RunnerClient
Client proto.RunnerClient
Fixer *tflint.Fixer
FixEnabled bool
}

var _ tflint.Runner = &GRPCClient{}
Expand Down Expand Up @@ -423,6 +425,44 @@ func (c *GRPCClient) EmitIssue(rule tflint.Rule, message string, location hcl.Ra
return nil
}

// EmitIssueWithFix emits the issue with the passed rule, message, location.
// If the fix is applied, the fixFunc is invoked.
// An emitted issue is always marked as fixable.
func (c *GRPCClient) EmitIssueWithFix(rule tflint.Rule, message string, location hcl.Range, fixFunc func(f *tflint.Fixer) error) error {
resp, err := c.Client.EmitIssue(context.Background(), &proto.EmitIssue_Request{Rule: toproto.Rule(rule), Message: message, Range: toproto.Range(location), Fixable: true})
if err != nil {
return fromproto.Error(err)
}

if c.FixEnabled && resp.Applied {
path, err := c.GetModulePath()
if err != nil {
return fromproto.Error(err)
}
// If the issue is not in the root module, skip the fix.
if !path.IsRoot() {
return nil
}
return fixFunc(c.Fixer)
}
return nil
}

// HasChanges returns true if the fixer has changes
func (c *GRPCClient) HasChanges() bool {
return c.Fixer.HasChanges()
}

// ApplyChanges applies the changes in the fixer to the server
func (c *GRPCClient) ApplyChanges() error {
_, err := c.Client.ApplyChanges(context.Background(), &proto.ApplyChanges_Request{Changes: c.Fixer.Changes()})
if err != nil {
return fromproto.Error(err)
}
c.Fixer.ApplyChanges()
return nil
}

// EnsureNoError is a helper for error handling. Depending on the type of error generated by EvaluateExpr,
// determine whether to exit, skip, or continue. If it is continued, the passed function will be executed.
//
Expand Down
32 changes: 18 additions & 14 deletions plugin/plugin2host/plugin2host_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,7 @@ type mockServerImpl struct {
getFiles func() map[string][]byte
getRuleConfigContent func(string, *hclext.BodySchema) (*hclext.BodyContent, map[string][]byte, error)
evaluateExpr func(hcl.Expression, tflint.EvaluateExprOption) (cty.Value, error)
emitIssue func(tflint.Rule, string, hcl.Range) error
emitIssue func(tflint.Rule, string, hcl.Range, bool) (bool, error)
}

func newMockServer(impl mockServerImpl) *mockServer {
Expand Down Expand Up @@ -99,10 +99,14 @@ func (s *mockServer) EvaluateExpr(expr hcl.Expression, opts tflint.EvaluateExprO
return cty.Value{}, nil
}

func (s *mockServer) EmitIssue(rule tflint.Rule, message string, location hcl.Range) error {
func (s *mockServer) EmitIssue(rule tflint.Rule, message string, location hcl.Range, fixable bool) (bool, error) {
if s.impl.emitIssue != nil {
return s.impl.emitIssue(rule, message, location)
return s.impl.emitIssue(rule, message, location, fixable)
}
return true, nil
}

func (s *mockServer) ApplyChanges(sources map[string][]byte) error {
return nil
}

Expand Down Expand Up @@ -2359,39 +2363,39 @@ func TestEmitIssue(t *testing.T) {
tests := []struct {
Name string
Args func() (tflint.Rule, string, hcl.Range)
ServerImpl func(tflint.Rule, string, hcl.Range) error
ServerImpl func(tflint.Rule, string, hcl.Range, bool) (bool, error)
ErrCheck func(error) bool
}{
{
Name: "emit issue",
Args: func() (tflint.Rule, string, hcl.Range) {
return &Rule{}, "this is test", hcl.Range{Filename: "test.tf", Start: hcl.Pos{Line: 2, Column: 2}, End: hcl.Pos{Line: 2, Column: 10}}
},
ServerImpl: func(rule tflint.Rule, message string, location hcl.Range) error {
ServerImpl: func(rule tflint.Rule, message string, location hcl.Range, fixable bool) (bool, error) {
if rule.Name() != "test_rule" {
return fmt.Errorf("rule.Name() should be test_rule, but %s", rule.Name())
return false, fmt.Errorf("rule.Name() should be test_rule, but %s", rule.Name())
}
if rule.Enabled() != true {
return fmt.Errorf("rule.Enabled() should be true, but %t", rule.Enabled())
return false, fmt.Errorf("rule.Enabled() should be true, but %t", rule.Enabled())
}
if rule.Severity() != tflint.ERROR {
return fmt.Errorf("rule.Severity() should be ERROR, but %s", rule.Severity())
return false, fmt.Errorf("rule.Severity() should be ERROR, but %s", rule.Severity())
}
if rule.Link() != "https://example.com" {
return fmt.Errorf("rule.Link() should be https://example.com, but %s", rule.Link())
return false, fmt.Errorf("rule.Link() should be https://example.com, but %s", rule.Link())
}
if message != "this is test" {
return fmt.Errorf("message should be `this is test`, but %s", message)
return false, fmt.Errorf("message should be `this is test`, but %s", message)
}
want := hcl.Range{
Filename: "test.tf",
Start: hcl.Pos{Line: 2, Column: 2},
End: hcl.Pos{Line: 2, Column: 10},
}
if diff := cmp.Diff(location, want); diff != "" {
return fmt.Errorf("diff: %s", diff)
return false, fmt.Errorf("diff: %s", diff)
}
return nil
return true, nil
},
ErrCheck: neverHappend,
},
Expand All @@ -2400,8 +2404,8 @@ func TestEmitIssue(t *testing.T) {
Args: func() (tflint.Rule, string, hcl.Range) {
return &Rule{}, "this is test", hcl.Range{Filename: "test.tf", Start: hcl.Pos{Line: 2, Column: 2}, End: hcl.Pos{Line: 2, Column: 10}}
},
ServerImpl: func(tflint.Rule, string, hcl.Range) error {
return errors.New("unexpected error")
ServerImpl: func(tflint.Rule, string, hcl.Range, bool) (bool, error) {
return false, errors.New("unexpected error")
},
ErrCheck: func(err error) bool {
return err == nil || err.Error() != "unexpected error"
Expand Down
16 changes: 13 additions & 3 deletions plugin/plugin2host/server.go
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,8 @@ type Server interface {
GetFiles(tflint.ModuleCtxType) map[string][]byte
GetRuleConfigContent(string, *hclext.BodySchema) (*hclext.BodyContent, map[string][]byte, error)
EvaluateExpr(hcl.Expression, tflint.EvaluateExprOption) (cty.Value, error)
EmitIssue(rule tflint.Rule, message string, location hcl.Range) error
EmitIssue(rule tflint.Rule, message string, location hcl.Range, fixable bool) (bool, error)
ApplyChanges(map[string][]byte) error
}

// GetOriginalwd gets the original working directory.
Expand Down Expand Up @@ -172,9 +173,18 @@ func (s *GRPCServer) EmitIssue(ctx context.Context, req *proto.EmitIssue_Request
return nil, status.Error(codes.InvalidArgument, "range should not be null")
}

err := s.Impl.EmitIssue(fromproto.Rule(req.Rule), req.Message, fromproto.Range(req.Range))
applied, err := s.Impl.EmitIssue(fromproto.Rule(req.Rule), req.Message, fromproto.Range(req.Range), req.Fixable)
if err != nil {
return nil, toproto.Error(codes.FailedPrecondition, err)
}
return &proto.EmitIssue_Response{}, nil
return &proto.EmitIssue_Response{Applied: applied}, nil
}

// ApplyChanges applies the passed changes.
func (s *GRPCServer) ApplyChanges(ctx context.Context, req *proto.ApplyChanges_Request) (*proto.ApplyChanges_Response, error) {
err := s.Impl.ApplyChanges(req.Changes)
if err != nil {
return nil, toproto.Error(codes.InvalidArgument, err)
}
return &proto.ApplyChanges_Response{}, nil
}
Loading

0 comments on commit e764c34

Please sign in to comment.