From 99940d7967db74d901b8f9cd4d135e1b53eeb3a1 Mon Sep 17 00:00:00 2001 From: Ryo Takaishi Date: Wed, 13 Sep 2023 09:33:41 +0900 Subject: [PATCH 1/4] refactor: add Renderer interface to use multiple rendering method --- cmd/terraform-j2md/main.go | 2 +- internal/terraform/plan.go | 82 ++++--------------- internal/terraform/unified_diff_renderer.go | 87 +++++++++++++++++++++ test/plan_test/plan_test.go | 6 +- 4 files changed, 105 insertions(+), 72 deletions(-) create mode 100644 internal/terraform/unified_diff_renderer.go diff --git a/cmd/terraform-j2md/main.go b/cmd/terraform-j2md/main.go index 772f9f3..78931dc 100644 --- a/cmd/terraform-j2md/main.go +++ b/cmd/terraform-j2md/main.go @@ -22,7 +22,7 @@ func main() { } func run() int { - planData, err := terraform.NewPlanData(os.Stdin) + planData, err := terraform.NewPlanData(os.Stdin, escapeHTML) if err != nil { fmt.Fprintf(os.Stderr, "cannot parse input as Terraform plan JSON: %v", err) return 1 diff --git a/internal/terraform/plan.go b/internal/terraform/plan.go index 0050539..b0eea98 100644 --- a/internal/terraform/plan.go +++ b/internal/terraform/plan.go @@ -1,17 +1,14 @@ package terraform import ( - "bytes" "encoding/json" "fmt" "github.com/hashicorp/terraform-json/sanitize" "github.com/reproio/terraform-j2md/internal/format" "io" - "strings" "text/template" tfjson "github.com/hashicorp/terraform-json" - "github.com/pmezard/go-difflib/difflib" ) const planTemplateBody = `### {{len .CreatedAddresses}} to add, {{len .UpdatedAddresses}} to change, {{len .DeletedAddresses}} to destroy, {{len .ReplacedAddresses}} to replace. @@ -36,7 +33,7 @@ const planTemplateBody = `### {{len .CreatedAddresses}} to add, {{len .UpdatedAd {{ range .ResourceChanges }} {{codeFence}}diff # {{.Header}} -{{.GetUnifiedDiffString}}{{codeFence}} +{{.Render}}{{codeFence}} {{end}} {{end}}` @@ -48,8 +45,10 @@ type PlanData struct { ReplacedAddresses []string ResourceChanges []ResourceChangeData } -type ResourceChangeData struct { - ResourceChange *tfjson.ResourceChange + +type ResourceChangeDataRenderer interface { + Render() (string, error) + Header() string } type Config struct { @@ -58,71 +57,17 @@ type Config struct { var config Config -func (r ResourceChangeData) GetUnifiedDiffString() (string, error) { - before, err := r.marshalChangeBefore() - if err != nil { - return "", fmt.Errorf("invalid resource changes (before): %w", err) - } - after, err := r.marshalChangeAfter() - if err != nil { - return "", fmt.Errorf("invalid resource changes (after) : %w", err) - } - // Try to parse JSON string in values - replacer := strings.NewReplacer(`\n`, "\n ", `\"`, "\"") - diff := difflib.UnifiedDiff{ - A: difflib.SplitLines(replacer.Replace(string(before))), - B: difflib.SplitLines(replacer.Replace(string(after))), - Context: 3, - } - diffText, err := difflib.GetUnifiedDiffString(diff) - if err != nil { - return "", fmt.Errorf("failed to create diff: %w", err) - } - return diffText, nil -} - -func (r ResourceChangeData) Header() string { - header := fmt.Sprintf("%s.%s %s", r.ResourceChange.Type, r.ResourceChange.Name, r.HeaderSuffix()) - - if r.ResourceChange.ModuleAddress == "" { - return header - } else { - return fmt.Sprintf("%s.%s", r.ResourceChange.ModuleAddress, header) - } -} - -func (r ResourceChangeData) marshalChangeBefore() ([]byte, error) { - return r.marshalChange(r.ResourceChange.Change.Before) -} - -func (r ResourceChangeData) marshalChangeAfter() ([]byte, error) { - return r.marshalChange(r.ResourceChange.Change.After) +type ResourceChangeData struct { + ResourceChange *tfjson.ResourceChange + Renderer ResourceChangeDataRenderer } -func (r ResourceChangeData) marshalChange(v any) ([]byte, error) { - var buffer bytes.Buffer - enc := json.NewEncoder(&buffer) - enc.SetIndent("", " ") - enc.SetEscapeHTML(config.EscapeHTML) - err := enc.Encode(v) - if err != nil { - return nil, err - } - return buffer.Bytes(), nil +func (r ResourceChangeData) Render() (string, error) { + return r.Renderer.Render() } -func (r ResourceChangeData) HeaderSuffix() string { - switch { - case r.ResourceChange.Change.Actions.Create(): - return "will be created" - case r.ResourceChange.Change.Actions.Update(): - return "will be updated in-place" - case r.ResourceChange.Change.Actions.Delete(): - return "will be destroyed" - case r.ResourceChange.Change.Actions.Replace(): - return "will be replaced" - } - return "" +func (r ResourceChangeData) Header() string { + return r.Renderer.Header() } func (plan *PlanData) Render(w io.Writer, escapeHTML bool) error { @@ -166,7 +111,7 @@ func processPlan(plan *tfjson.Plan) (*tfjson.Plan, error) { return plan, nil } -func NewPlanData(input io.Reader) (*PlanData, error) { +func NewPlanData(input io.Reader, escapeHTML bool) (*PlanData, error) { var err error var plan tfjson.Plan if err := json.NewDecoder(input).Decode(&plan); err != nil { @@ -196,6 +141,7 @@ func NewPlanData(input io.Reader) (*PlanData, error) { } planData.ResourceChanges = append(planData.ResourceChanges, ResourceChangeData{ ResourceChange: c, + Renderer: NewUnifiedDiffRenderer(c, escapeHTML), }) } return &planData, nil diff --git a/internal/terraform/unified_diff_renderer.go b/internal/terraform/unified_diff_renderer.go new file mode 100644 index 0000000..293511c --- /dev/null +++ b/internal/terraform/unified_diff_renderer.go @@ -0,0 +1,87 @@ +package terraform + +import ( + "bytes" + "encoding/json" + "fmt" + tfjson "github.com/hashicorp/terraform-json" + "github.com/pmezard/go-difflib/difflib" + "strings" +) + +type UnifiedDiffRenderer struct { + ResourceChange *tfjson.ResourceChange + EnableEscapeHTML bool +} + +func NewUnifiedDiffRenderer(resourceChange *tfjson.ResourceChange, enableEscapeHTML bool) *UnifiedDiffRenderer { + return &UnifiedDiffRenderer{ResourceChange: resourceChange, EnableEscapeHTML: enableEscapeHTML} +} + +func (r *UnifiedDiffRenderer) Render() (string, error) { + before, err := r.marshalChangeBefore() + if err != nil { + return "", fmt.Errorf("invalid resource changes (before): %w", err) + } + after, err := r.marshalChangeAfter() + if err != nil { + return "", fmt.Errorf("invalid resource changes (after) : %w", err) + } + // Try to parse JSON string in values + replacer := strings.NewReplacer(`\n`, "\n ", `\"`, "\"") + diff := difflib.UnifiedDiff{ + A: difflib.SplitLines(replacer.Replace(string(before))), + B: difflib.SplitLines(replacer.Replace(string(after))), + Context: 3, + } + diffText, err := difflib.GetUnifiedDiffString(diff) + if err != nil { + return "", fmt.Errorf("failed to create diff: %w", err) + } + + return diffText, nil +} + +func (r *UnifiedDiffRenderer) Header() string { + header := fmt.Sprintf("%s.%s %s", r.ResourceChange.Type, r.ResourceChange.Name, r.headerSuffix()) + + if r.ResourceChange.ModuleAddress == "" { + return header + } else { + return fmt.Sprintf("%s.%s", r.ResourceChange.ModuleAddress, header) + } +} + +func (r *UnifiedDiffRenderer) headerSuffix() string { + switch { + case r.ResourceChange.Change.Actions.Create(): + return "will be created" + case r.ResourceChange.Change.Actions.Update(): + return "will be updated in-place" + case r.ResourceChange.Change.Actions.Delete(): + return "will be destroyed" + case r.ResourceChange.Change.Actions.Replace(): + return "will be replaced" + } + return "" +} + +func (r *UnifiedDiffRenderer) marshalChangeBefore() ([]byte, error) { + return r.marshalChange(r.ResourceChange.Change.Before) +} + +func (r *UnifiedDiffRenderer) marshalChangeAfter() ([]byte, error) { + return r.marshalChange(r.ResourceChange.Change.After) +} + +func (r *UnifiedDiffRenderer) marshalChange(v any) ([]byte, error) { + var buffer bytes.Buffer + enc := json.NewEncoder(&buffer) + enc.SetIndent("", " ") + enc.SetEscapeHTML(r.EnableEscapeHTML) + err := enc.Encode(v) + if err != nil { + return nil, err + } + return buffer.Bytes(), nil +} diff --git a/test/plan_test/plan_test.go b/test/plan_test/plan_test.go index aa7fc11..0f6b375 100644 --- a/test/plan_test/plan_test.go +++ b/test/plan_test/plan_test.go @@ -38,7 +38,7 @@ func Test_newPlanData(t *testing.T) { } defer file.Close() - _, err = terraform.NewPlanData(file) + _, err = terraform.NewPlanData(file, false) if (err != nil) != tt.wantErr { t.Errorf("NewPlanData() error = %v, wantErr %v", err, tt.wantErr) return @@ -75,7 +75,7 @@ func Test_render(t *testing.T) { } defer file.Close() - plan, err := terraform.NewPlanData(file) + plan, err := terraform.NewPlanData(file, true) if err != nil { t.Errorf("cannot parse JSON as plan: %v", err) return @@ -119,7 +119,7 @@ func Test_render(t *testing.T) { } defer file.Close() - plan, err := terraform.NewPlanData(file) + plan, err := terraform.NewPlanData(file, false) if err != nil { t.Errorf("cannot parse JSON as plan: %v", err) return From 1d31e1bb497b153e2ad245f398a336aa10679969 Mon Sep 17 00:00:00 2001 From: Ryo Takaishi Date: Wed, 13 Sep 2023 09:20:04 +0900 Subject: [PATCH 2/4] feat: support moved block --- go.mod | 9 +- go.sum | 53 ++--- internal/terraform/moved_block_renderer.go | 64 ++++++ internal/terraform/plan.go | 18 ++ test/plan_test/plan_test.go | 1 + test/testdata/moved_block/expected.md | 13 ++ test/testdata/moved_block/main.tf | 25 +++ test/testdata/moved_block/plan.tfplan | Bin 0 -> 3632 bytes test/testdata/moved_block/show.json | 205 ++++++++++++++++++++ test/testdata/moved_block/terraform.tfstate | 53 +++++ 10 files changed, 395 insertions(+), 46 deletions(-) create mode 100644 internal/terraform/moved_block_renderer.go create mode 100644 test/testdata/moved_block/expected.md create mode 100644 test/testdata/moved_block/main.tf create mode 100644 test/testdata/moved_block/plan.tfplan create mode 100644 test/testdata/moved_block/show.json create mode 100644 test/testdata/moved_block/terraform.tfstate diff --git a/go.mod b/go.mod index 1e71984..1539407 100644 --- a/go.mod +++ b/go.mod @@ -3,14 +3,15 @@ module github.com/reproio/terraform-j2md go 1.18 require ( - github.com/hashicorp/terraform-json v0.14.0 + github.com/hashicorp/terraform-json v0.17.2-0.20230912071934-9901d28699bc github.com/pmezard/go-difflib v1.0.0 ) require ( - github.com/hashicorp/go-version v1.5.0 // indirect + github.com/apparentlymart/go-textseg/v15 v15.0.0 // indirect + github.com/hashicorp/go-version v1.6.0 // indirect github.com/mitchellh/copystructure v1.2.0 // indirect github.com/mitchellh/reflectwalk v1.0.2 // indirect - github.com/zclconf/go-cty v1.10.0 // indirect - golang.org/x/text v0.3.5 // indirect + github.com/zclconf/go-cty v1.14.0 // indirect + golang.org/x/text v0.11.0 // indirect ) diff --git a/go.sum b/go.sum index f3fb121..87da965 100644 --- a/go.sum +++ b/go.sum @@ -1,22 +1,11 @@ -github.com/apparentlymart/go-textseg v1.0.0/go.mod h1:z96Txxhf3xSFMPmb5X/1W05FF/Nj9VFpLOpjS5yuumk= -github.com/apparentlymart/go-textseg/v13 v13.0.0/go.mod h1:ZK2fH7c4NqDTLtiYLvIkEghdlcqw7yxLeM89kiTRPUo= -github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/apparentlymart/go-textseg/v15 v15.0.0 h1:uYvfpb3DyLSCGWnctWKGj857c6ew1u1fNQOlOtuGxQY= +github.com/apparentlymart/go-textseg/v15 v15.0.0/go.mod h1:K8XmNZdhEBkdlyDdvbmmsvpAG721bKi0joRfFdHIWJ4= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= -github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= -github.com/golang/protobuf v1.1.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= -github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= -github.com/golang/protobuf v1.3.4/go.mod h1:vzj43D7+SQXF/4pzW/hwtAqwc6iTitCiVSaWz5lYuqw= -github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= -github.com/google/go-cmp v0.5.8 h1:e6P7q2lk1O+qJJb4BtCQXlK8vWEO8V1ZeuEdJNOqZyg= -github.com/google/go-cmp v0.5.8/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= -github.com/hashicorp/go-version v1.5.0 h1:O293SZ2Eg+AAYijkVK3jR786Am1bhDEh2GHT0tIVE5E= -github.com/hashicorp/go-version v1.5.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= -github.com/hashicorp/terraform-json v0.14.0 h1:sh9iZ1Y8IFJLx+xQiKHGud6/TSUCM0N8e17dKDpqV7s= -github.com/hashicorp/terraform-json v0.14.0/go.mod h1:5A9HIWPkk4e5aeeXIBbkcOvaZbIYnAIkEyqP2pNSckM= -github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= -github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= -github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= -github.com/kylelemons/godebug v0.0.0-20170820004349-d65d576e9348/go.mod h1:B69LEHPfb2qLo0BaaOLcbitczOKLWTsrBG9LczfCD4k= +github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38= +github.com/hashicorp/go-version v1.6.0 h1:feTTfFNnjP967rlCxM/I9g701jU+RN74YKx2mOkIeek= +github.com/hashicorp/go-version v1.6.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= +github.com/hashicorp/terraform-json v0.17.2-0.20230912071934-9901d28699bc h1:ZtMfoibHiPAYJykA5nuHryaNoNDvfuREGWnIvukMb2Y= +github.com/hashicorp/terraform-json v0.17.2-0.20230912071934-9901d28699bc/go.mod h1:0a5tk65jPDbGo2lEMmvmwwvM0qCbOhW33hXtGrJQBgc= github.com/mitchellh/copystructure v1.2.0 h1:vpKXTN4ewci03Vljg/q9QvCGUDttBOGBIa15WveJJGw= github.com/mitchellh/copystructure v1.2.0/go.mod h1:qLl+cE2AmVv+CoeAwDPye/v+N2HKCj9FbZEVFJRxO9s= github.com/mitchellh/reflectwalk v1.0.2 h1:G2LzWKi524PWgd3mLHV8Y5k7s6XUvT0Gef6zxSIeXaQ= @@ -24,28 +13,8 @@ github.com/mitchellh/reflectwalk v1.0.2/go.mod h1:mSTlrgnPZtwu0c4WaC2kGObEpuNDbx github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= github.com/sebdah/goldie v1.0.0 h1:9GNhIat69MSlz/ndaBg48vl9dF5fI+NBB6kfOxgfkMc= -github.com/sebdah/goldie v1.0.0/go.mod h1:jXP4hmWywNEwZzhMuv2ccnqTSFpuq8iyQhtQdkkZBH4= -github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= -github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= -github.com/vmihailenco/msgpack v3.3.3+incompatible/go.mod h1:fy3FlTQTDXWkZ7Bh6AcGMlsjHatGryHQYUTf1ShIgkk= -github.com/vmihailenco/msgpack/v4 v4.3.12/go.mod h1:gborTTJjAo/GWTqqRjrLCn9pgNN+NXzzngzBKDPIqw4= -github.com/vmihailenco/tagparser v0.1.1/go.mod h1:OeAg3pn3UbLjkWt+rN9oFYB6u/cQgqMEUPoW2WPyhdI= -github.com/zclconf/go-cty v1.2.0/go.mod h1:hOPWgoHbaTUnI5k4D2ld+GRpFJSCe6bCM7m1q/N4PQ8= -github.com/zclconf/go-cty v1.10.0 h1:mp9ZXQeIcN8kAwuqorjH+Q+njbJKjLrvB2yIh4q7U+0= -github.com/zclconf/go-cty v1.10.0/go.mod h1:vVKLxnk3puL4qRAv72AO+W99LUD4da90g3uUAzyuvAk= +github.com/zclconf/go-cty v1.14.0 h1:/Xrd39K7DXbHzlisFP9c4pHao4yyf+/Ug9LEz+Y/yhc= +github.com/zclconf/go-cty v1.14.0/go.mod h1:VvMs5i0vgZdhYawQNq5kePSpLAoz8u1xvZgrPIxfnZE= github.com/zclconf/go-cty-debug v0.0.0-20191215020915-b22d67c1ba0b h1:FosyBZYxY34Wul7O/MSKey3txpPYyCqVO5ZyceuQJEI= -github.com/zclconf/go-cty-debug v0.0.0-20191215020915-b22d67c1ba0b/go.mod h1:ZRKQfBXbGkpdV6QMzT3rU1kSTAnfu1dO8dPKjYprgj8= -golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= -golang.org/x/net v0.0.0-20180811021610-c39426892332/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= -golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks= -golang.org/x/net v0.0.0-20200301022130-244492dfa37a/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= -golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= -golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= -golang.org/x/text v0.3.5 h1:i6eZZ+zk0SOf0xgBpEpPD18qWcJda6q1sxt3S0kzyUQ= -golang.org/x/text v0.3.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= -golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= -google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= -google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= -gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +golang.org/x/text v0.11.0 h1:LAntKIrcmeSKERyiOh0XMV39LXS8IE9UL2yP7+f5ij4= +golang.org/x/text v0.11.0/go.mod h1:TvPlkZtksWOMsz7fbANvkp4WM8x/WCo/om8BMLbz+aE= diff --git a/internal/terraform/moved_block_renderer.go b/internal/terraform/moved_block_renderer.go new file mode 100644 index 0000000..8c9b57f --- /dev/null +++ b/internal/terraform/moved_block_renderer.go @@ -0,0 +1,64 @@ +package terraform + +import ( + "bytes" + "fmt" + tfjson "github.com/hashicorp/terraform-json" + "text/template" +) + +const movedBlockTemplateBody = `resource "{{.ResourceChange.Type}}" "{{.ResourceChange.Name}}" { +{{.Attributes -}} +} +` + +// These attributes are important (https://github.com/hashicorp/terraform/blob/v1.5.6/internal/command/jsonformat/computed/renderers/block.go#L19-L23) +var importantAttributes = []string{ + "id", + "name", + "tags", +} + +type MovedBlockRenderer struct { + ResourceChange *tfjson.ResourceChange +} + +func NewMovedBlockRenderer(resourceChange *tfjson.ResourceChange) *MovedBlockRenderer { + return &MovedBlockRenderer{ResourceChange: resourceChange} +} + +func (r *MovedBlockRenderer) Render() (string, error) { + var buff bytes.Buffer + t, err := template.New("plan").Parse(movedBlockTemplateBody) + if err != nil { + return "", fmt.Errorf("invalid template text: %w", err) + } + + if err := t.Execute(&buff, r); err != nil { + return "", fmt.Errorf("failed to render template: %w", err) + } + return buff.String(), nil +} + +func (r *MovedBlockRenderer) Header() string { + return fmt.Sprintf("%s has moved to %s", r.ResourceChange.PreviousAddress, r.ResourceChange.Address) +} + +func (r *MovedBlockRenderer) Attributes() string { + var buff bytes.Buffer + for _, attr := range importantAttributes { + if v, ok := r.ResourceChange.Change.After.(map[string]interface{})[attr]; ok { + buff.WriteString(fmt.Sprintf(" %-*s = %s\n", 2, attr, r.value(v))) + } + } + return buff.String() +} + +func (r *MovedBlockRenderer) value(v any) string { + switch v.(type) { + case string: + return fmt.Sprintf("%q", v) + default: + return fmt.Sprintf("%v", v) + } +} diff --git a/internal/terraform/plan.go b/internal/terraform/plan.go index b0eea98..8ae57d7 100644 --- a/internal/terraform/plan.go +++ b/internal/terraform/plan.go @@ -28,6 +28,10 @@ const planTemplateBody = `### {{len .CreatedAddresses}} to add, {{len .UpdatedAd - replace{{ range .ReplacedAddresses }} - {{. -}} {{end}}{{end}} +{{- if .MovedAddresses}} +- moved{{ range .MovedAddresses }} + - {{. -}} +{{end}}{{end}} {{if .ResourceChanges -}}
Change details {{ range .ResourceChanges }} @@ -43,6 +47,7 @@ type PlanData struct { UpdatedAddresses []string DeletedAddresses []string ReplacedAddresses []string + MovedAddresses []string ResourceChanges []ResourceChangeData } @@ -125,6 +130,15 @@ func NewPlanData(input io.Reader, escapeHTML bool) (*PlanData, error) { planData := PlanData{} for _, c := range processedPlan.ResourceChanges { + if isMovedBlock(c) { + planData.MovedAddresses = append(planData.MovedAddresses, fmt.Sprintf("%s (from %s)", c.Address, c.PreviousAddress)) + planData.ResourceChanges = append(planData.ResourceChanges, ResourceChangeData{ + ResourceChange: c, + Renderer: NewMovedBlockRenderer(c), + }) + continue + } + if c.Change.Actions.NoOp() || c.Change.Actions.Read() { continue } @@ -146,3 +160,7 @@ func NewPlanData(input io.Reader, escapeHTML bool) (*PlanData, error) { } return &planData, nil } + +func isMovedBlock(rc *tfjson.ResourceChange) bool { + return rc.Change.Actions.NoOp() && rc.PreviousAddress != "" +} diff --git a/test/plan_test/plan_test.go b/test/plan_test/plan_test.go index 0f6b375..ce5b7a7 100644 --- a/test/plan_test/plan_test.go +++ b/test/plan_test/plan_test.go @@ -64,6 +64,7 @@ func Test_render(t *testing.T) { {name: "include_code_fence", wantErr: false}, {name: "include_module", wantErr: false}, {name: "known_after_apply", wantErr: false}, + {name: "moved_block", wantErr: false}, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { diff --git a/test/testdata/moved_block/expected.md b/test/testdata/moved_block/expected.md new file mode 100644 index 0000000..8e8ae76 --- /dev/null +++ b/test/testdata/moved_block/expected.md @@ -0,0 +1,13 @@ +### 0 to add, 0 to change, 0 to destroy, 0 to replace. +- moved + - random_id.test2 (from random_id.test) +
Change details + +````````diff +# random_id.test has moved to random_id.test2 +resource "random_id" "test2" { + id = "qD4MEwtJeTOwqg" +} +```````` + +
diff --git a/test/testdata/moved_block/main.tf b/test/testdata/moved_block/main.tf new file mode 100644 index 0000000..d9271e5 --- /dev/null +++ b/test/testdata/moved_block/main.tf @@ -0,0 +1,25 @@ +terraform { + required_providers { + env = { + source = "tchupp/env" + version = "0.0.2" + } + } +} + +provider "env" { + # Configuration options +} + +resource "env_variable" "test" { + name = random_id.test2.hex +} + +moved { + from = random_id.test + to = random_id.test2 +} + +resource "random_id" "test2" { + byte_length = 10 +} diff --git a/test/testdata/moved_block/plan.tfplan b/test/testdata/moved_block/plan.tfplan new file mode 100644 index 0000000000000000000000000000000000000000..82710995a0855014f67184bde69890b394dcdfd8 GIT binary patch literal 3632 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 literal 0 HcmV?d00001 diff --git a/test/testdata/moved_block/show.json b/test/testdata/moved_block/show.json new file mode 100644 index 0000000..4293a12 --- /dev/null +++ b/test/testdata/moved_block/show.json @@ -0,0 +1,205 @@ +{ + "format_version": "1.2", + "terraform_version": "1.5.3", + "planned_values": { + "root_module": { + "resources": [ + { + "address": "env_variable.test", + "mode": "managed", + "type": "env_variable", + "name": "test", + "provider_name": "registry.terraform.io/tchupp/env", + "schema_version": 0, + "values": { + "id": "a83e0c130b497933b0aa", + "name": "a83e0c130b497933b0aa", + "value": "" + }, + "sensitive_values": {} + }, + { + "address": "random_id.test2", + "mode": "managed", + "type": "random_id", + "name": "test2", + "provider_name": "registry.terraform.io/hashicorp/random", + "schema_version": 0, + "values": { + "b64_std": "qD4MEwtJeTOwqg==", + "b64_url": "qD4MEwtJeTOwqg", + "byte_length": 10, + "dec": "794502137306233594687658", + "hex": "a83e0c130b497933b0aa", + "id": "qD4MEwtJeTOwqg", + "keepers": null, + "prefix": null + }, + "sensitive_values": {} + } + ] + } + }, + "resource_changes": [ + { + "address": "env_variable.test", + "mode": "managed", + "type": "env_variable", + "name": "test", + "provider_name": "registry.terraform.io/tchupp/env", + "change": { + "actions": [ + "no-op" + ], + "before": { + "id": "a83e0c130b497933b0aa", + "name": "a83e0c130b497933b0aa", + "value": "" + }, + "after": { + "id": "a83e0c130b497933b0aa", + "name": "a83e0c130b497933b0aa", + "value": "" + }, + "after_unknown": {}, + "before_sensitive": { + "value": true + }, + "after_sensitive": { + "value": true + } + } + }, + { + "address": "random_id.test2", + "previous_address": "random_id.test", + "mode": "managed", + "type": "random_id", + "name": "test2", + "provider_name": "registry.terraform.io/hashicorp/random", + "change": { + "actions": [ + "no-op" + ], + "before": { + "b64_std": "qD4MEwtJeTOwqg==", + "b64_url": "qD4MEwtJeTOwqg", + "byte_length": 10, + "dec": "794502137306233594687658", + "hex": "a83e0c130b497933b0aa", + "id": "qD4MEwtJeTOwqg", + "keepers": null, + "prefix": null + }, + "after": { + "b64_std": "qD4MEwtJeTOwqg==", + "b64_url": "qD4MEwtJeTOwqg", + "byte_length": 10, + "dec": "794502137306233594687658", + "hex": "a83e0c130b497933b0aa", + "id": "qD4MEwtJeTOwqg", + "keepers": null, + "prefix": null + }, + "after_unknown": {}, + "before_sensitive": {}, + "after_sensitive": {} + } + } + ], + "prior_state": { + "format_version": "1.0", + "terraform_version": "1.5.3", + "values": { + "root_module": { + "resources": [ + { + "address": "env_variable.test", + "mode": "managed", + "type": "env_variable", + "name": "test", + "provider_name": "registry.terraform.io/tchupp/env", + "schema_version": 0, + "values": { + "id": "a83e0c130b497933b0aa", + "name": "a83e0c130b497933b0aa", + "value": "" + }, + "sensitive_values": { + "value": true + }, + "depends_on": [ + "random_id.test2" + ] + }, + { + "address": "random_id.test2", + "mode": "managed", + "type": "random_id", + "name": "test2", + "provider_name": "registry.terraform.io/hashicorp/random", + "schema_version": 0, + "values": { + "b64_std": "qD4MEwtJeTOwqg==", + "b64_url": "qD4MEwtJeTOwqg", + "byte_length": 10, + "dec": "794502137306233594687658", + "hex": "a83e0c130b497933b0aa", + "id": "qD4MEwtJeTOwqg", + "keepers": null, + "prefix": null + }, + "sensitive_values": {} + } + ] + } + } + }, + "configuration": { + "provider_config": { + "env": { + "name": "env", + "full_name": "registry.terraform.io/tchupp/env", + "version_constraint": "0.0.2" + }, + "random": { + "name": "random", + "full_name": "registry.terraform.io/hashicorp/random" + } + }, + "root_module": { + "resources": [ + { + "address": "env_variable.test", + "mode": "managed", + "type": "env_variable", + "name": "test", + "provider_config_key": "env", + "expressions": { + "name": { + "references": [ + "random_id.test2.hex", + "random_id.test2" + ] + } + }, + "schema_version": 0 + }, + { + "address": "random_id.test2", + "mode": "managed", + "type": "random_id", + "name": "test2", + "provider_config_key": "random", + "expressions": { + "byte_length": { + "constant_value": 10 + } + }, + "schema_version": 0 + } + ] + } + }, + "timestamp": "2023-08-29T08:27:04Z" +} diff --git a/test/testdata/moved_block/terraform.tfstate b/test/testdata/moved_block/terraform.tfstate new file mode 100644 index 0000000..d6a893b --- /dev/null +++ b/test/testdata/moved_block/terraform.tfstate @@ -0,0 +1,53 @@ +{ + "version": 4, + "terraform_version": "1.5.3", + "serial": 3, + "lineage": "dc23aa52-bb4e-e4f3-e6e1-fdd58d30b617", + "outputs": {}, + "resources": [ + { + "mode": "managed", + "type": "env_variable", + "name": "test", + "provider": "provider[\"registry.terraform.io/tchupp/env\"]", + "instances": [ + { + "schema_version": 0, + "attributes": { + "id": "a83e0c130b497933b0aa", + "name": "a83e0c130b497933b0aa", + "value": "" + }, + "sensitive_attributes": [], + "private": "bnVsbA==", + "dependencies": [ + "random_id.test" + ] + } + ] + }, + { + "mode": "managed", + "type": "random_id", + "name": "test", + "provider": "provider[\"registry.terraform.io/hashicorp/random\"]", + "instances": [ + { + "schema_version": 0, + "attributes": { + "b64_std": "qD4MEwtJeTOwqg==", + "b64_url": "qD4MEwtJeTOwqg", + "byte_length": 10, + "dec": "794502137306233594687658", + "hex": "a83e0c130b497933b0aa", + "id": "qD4MEwtJeTOwqg", + "keepers": null, + "prefix": null + }, + "sensitive_attributes": [] + } + ] + } + ], + "check_results": null +} From 24596a880ebe31f1fb8ff7924ba32fd4b48e6055 Mon Sep 17 00:00:00 2001 From: Ryo Takaishi Date: Wed, 13 Sep 2023 09:49:22 +0900 Subject: [PATCH 3/4] refactor: pass escapeHTML parameter to UnifiedDiffRenderer directly --- cmd/terraform-j2md/main.go | 2 +- internal/terraform/plan.go | 5 +---- test/plan_test/plan_test.go | 4 ++-- 3 files changed, 4 insertions(+), 7 deletions(-) diff --git a/cmd/terraform-j2md/main.go b/cmd/terraform-j2md/main.go index 78931dc..3f8deae 100644 --- a/cmd/terraform-j2md/main.go +++ b/cmd/terraform-j2md/main.go @@ -27,7 +27,7 @@ func run() int { fmt.Fprintf(os.Stderr, "cannot parse input as Terraform plan JSON: %v", err) return 1 } - if err = planData.Render(os.Stdout, escapeHTML); err != nil { + if err = planData.Render(os.Stdout); err != nil { fmt.Fprintf(os.Stderr, "cannot render: %v", err) return 1 } diff --git a/internal/terraform/plan.go b/internal/terraform/plan.go index 8ae57d7..220bd3c 100644 --- a/internal/terraform/plan.go +++ b/internal/terraform/plan.go @@ -60,8 +60,6 @@ type Config struct { EscapeHTML bool } -var config Config - type ResourceChangeData struct { ResourceChange *tfjson.ResourceChange Renderer ResourceChangeDataRenderer @@ -75,8 +73,7 @@ func (r ResourceChangeData) Header() string { return r.Renderer.Header() } -func (plan *PlanData) Render(w io.Writer, escapeHTML bool) error { - config.EscapeHTML = escapeHTML +func (plan *PlanData) Render(w io.Writer) error { funcMap := template.FuncMap{ "codeFence": func() string { return "````````" diff --git a/test/plan_test/plan_test.go b/test/plan_test/plan_test.go index ce5b7a7..e937b66 100644 --- a/test/plan_test/plan_test.go +++ b/test/plan_test/plan_test.go @@ -83,7 +83,7 @@ func Test_render(t *testing.T) { } got := bytes.Buffer{} - err = plan.Render(&got, true) + err = plan.Render(&got) if (err != nil) != tt.wantErr { t.Errorf("render() error = %v, wantErr %v", err, tt.wantErr) return @@ -127,7 +127,7 @@ func Test_render(t *testing.T) { } got := bytes.Buffer{} - err = plan.Render(&got, false) + err = plan.Render(&got) if (err != nil) != tt.wantErr { t.Errorf("render() error = %v, wantErr %v", err, tt.wantErr) return From d7167fdd22fbb2266fb2be16cfa200bfe9d7d1f7 Mon Sep 17 00:00:00 2001 From: Ryo Takaishi Date: Mon, 25 Sep 2023 14:55:34 +0900 Subject: [PATCH 4/4] refactor: remove unused struct --- internal/terraform/plan.go | 4 ---- 1 file changed, 4 deletions(-) diff --git a/internal/terraform/plan.go b/internal/terraform/plan.go index 220bd3c..2a500f6 100644 --- a/internal/terraform/plan.go +++ b/internal/terraform/plan.go @@ -56,10 +56,6 @@ type ResourceChangeDataRenderer interface { Header() string } -type Config struct { - EscapeHTML bool -} - type ResourceChangeData struct { ResourceChange *tfjson.ResourceChange Renderer ResourceChangeDataRenderer