From 7a8a23612d514efc6eb298e46b69722bb88340d8 Mon Sep 17 00:00:00 2001 From: Mathieu Cantin Date: Thu, 14 Jan 2021 22:07:27 -0500 Subject: [PATCH 1/5] fix(output): Remove Refreshing state... from output Since Terraform 0.14.0 there are no separator between refreshing plan and the plan. --- server/events/runtime/apply_step_runner.go | 16 ++--- server/events/runtime/plan_step_runner.go | 49 +++++++++----- .../events/runtime/plan_step_runner_test.go | 65 +++++++++++++++++++ 3 files changed, 103 insertions(+), 27 deletions(-) diff --git a/server/events/runtime/apply_step_runner.go b/server/events/runtime/apply_step_runner.go index 1f8cb44640..a749ce4f87 100644 --- a/server/events/runtime/apply_step_runner.go +++ b/server/events/runtime/apply_step_runner.go @@ -170,7 +170,7 @@ func (a *ApplyStepRunner) runRemoteApply( ctx.Log.Debug("remote apply is waiting for confirmation") // Check if the plan is as expected. - planChangedErr = a.remotePlanChanged(string(planfileBytes), strings.Join(lines, "\n")) + planChangedErr = a.remotePlanChanged(string(planfileBytes), strings.Join(lines, "\n"), tfVersion) if planChangedErr != nil { ctx.Log.Err("plan generated during apply does not match expected plan, aborting") inCh <- "no\n" @@ -206,19 +206,15 @@ func (a *ApplyStepRunner) runRemoteApply( // the one we're about to apply in the apply phase. // If the plans don't match, it returns an error with a diff of the two plans // that can be printed to the pull request. -func (a *ApplyStepRunner) remotePlanChanged(planfileContents string, applyOut string) error { - // The plan is between the refresh separator... - planStartIdx := strings.Index(applyOut, refreshSeparator) - if planStartIdx < 0 { - return fmt.Errorf("Couldn't find refresh separator when parsing apply output:\n%q", applyOut) - } +func (a *ApplyStepRunner) remotePlanChanged(planfileContents string, applyOut string, tfVersion *version.Version) error { + output := StripRefreshingFromPlanOutput(applyOut, tfVersion) - // ...and the prompt to execute the plan. - planEndIdx := strings.Index(applyOut, "Do you want to perform these actions in workspace \"") + // The output stop to execute the plan. + planEndIdx := strings.Index(output, "Do you want to perform these actions in workspace \"") if planEndIdx < 0 { return fmt.Errorf("Couldn't find plan end when parsing apply output:\n%q", applyOut) } - currPlan := strings.TrimSpace(applyOut[planStartIdx+len(refreshSeparator) : planEndIdx]) + currPlan := strings.TrimSpace(output[: planEndIdx]) // Ensure we strip the remoteOpsHeader from the plan contents so the // comparison is fair. We add this header in the plan phase so we can diff --git a/server/events/runtime/plan_step_runner.go b/server/events/runtime/plan_step_runner.go index 3796070add..35b13d9946 100644 --- a/server/events/runtime/plan_step_runner.go +++ b/server/events/runtime/plan_step_runner.go @@ -15,8 +15,7 @@ import ( const ( defaultWorkspace = "default" - // refreshSeparator is what separates the refresh stage from the calculated - // plan during a terraform plan. + refreshKeyword = "Refreshing state..." refreshSeparator = "------------------------------------------------------------------------\n" ) @@ -55,7 +54,7 @@ func (p *PlanStepRunner) Run(ctx models.ProjectCommandContext, extraArgs []strin if err != nil { return output, err } - return p.fmtPlanOutput(output), nil + return p.fmtPlanOutput(output, tfVersion), nil } // isRemoteOpsErr returns true if there was an error caused due to this @@ -89,11 +88,7 @@ func (p *PlanStepRunner) remotePlan(ctx models.ProjectCommandContext, extraArgs // plan. To ensure that what gets applied is the plan we printed to the PR, // during the apply phase, we diff the output we stored in the fake // planfile with the pending apply output. - planOutput := output - sepIdx := strings.Index(planOutput, refreshSeparator) - if sepIdx > -1 { - planOutput = planOutput[sepIdx+len(refreshSeparator):] - } + planOutput := StripRefreshingFromPlanOutput(output, tfVersion) // We also prepend our own remote ops header to the file so during apply we // know this is a remote apply. @@ -102,7 +97,7 @@ func (p *PlanStepRunner) remotePlan(ctx models.ProjectCommandContext, extraArgs return output, errors.Wrap(err, "unable to create planfile for remote ops") } - return p.fmtPlanOutput(output), nil + return p.fmtPlanOutput(output, tfVersion), nil } // switchWorkspace changes the terraform workspace if necessary and will create @@ -228,14 +223,8 @@ func (p *PlanStepRunner) flatten(slices [][]string) []string { // "- aws_security_group_rule.allow_all" // We do it for +, ~ and -. // It also removes the "Refreshing..." preamble. -func (p *PlanStepRunner) fmtPlanOutput(output string) string { - // Plan output contains a lot of "Refreshing..." lines followed by a - // separator. We want to remove everything before that separator. - sepIdx := strings.Index(output, refreshSeparator) - if sepIdx > -1 { - output = output[sepIdx+len(refreshSeparator):] - } - +func (p *PlanStepRunner) fmtPlanOutput(output string, tfVersion *version.Version) string { + output = StripRefreshingFromPlanOutput(output, tfVersion) output = plusDiffRegex.ReplaceAllString(output, "+") output = tildeDiffRegex.ReplaceAllString(output, "~") return minusDiffRegex.ReplaceAllString(output, "-") @@ -299,6 +288,32 @@ func (p *PlanStepRunner) runRemotePlan( return output, err } +func StripRefreshingFromPlanOutput(output string, tfVersion *version.Version) string { + if tfVersion.GreaterThanOrEqual(version.Must(version.NewVersion("0.14.0"))) { + // Plan output contains a lot of "Refreshing..." lines, remove it + lines := strings.Split(output, "\n") + finalIndex := 0 + for i, line := range lines { + if strings.Contains(line, refreshKeyword) { + finalIndex = i + } + } + + if finalIndex != 0 { + output = strings.Join(lines[finalIndex + 1:], "\n") + } + return output + } else { + // Plan output contains a lot of "Refreshing..." lines followed by a + // separator. We want to remove everything before that separator. + sepIdx := strings.Index(output, refreshSeparator) + if sepIdx > -1 { + output = output[sepIdx+len(refreshSeparator):] + } + return output + } +} + // remoteOpsErr01114 is the error terraform plan will return if this project is // using TFE remote operations in TF 0.11.14. var remoteOpsErr01114 = `Error: Saving a generated plan is currently not supported! diff --git a/server/events/runtime/plan_step_runner_test.go b/server/events/runtime/plan_step_runner_test.go index 5ef5809a1a..1830196dde 100644 --- a/server/events/runtime/plan_step_runner_test.go +++ b/server/events/runtime/plan_step_runner_test.go @@ -814,6 +814,71 @@ Plan: 0 to add, 0 to change, 1 to destroy.`, string(bytes)) } } +// Test striping output method +func TestStripRefreshingFromPlanOutput(t *testing.T) { + tfVersion_0135, _ := version.NewVersion("0.13.5") + tfVersion_0140, _ := version.NewVersion("0.14.0") + cases := []struct { + out string + tfVersion *version.Version + }{ + { + remotePlanOutput, + tfVersion_0135, + }, + { + `Running plan in the remote backend. Output will stream here. Pressing Ctrl-C +will stop streaming the logs, but will not stop the plan running remotely. + +Preparing the remote plan... + +To view this run in a browser, visit: +https://app.terraform.io/app/lkysow-enterprises/atlantis-tfe-test/runs/run-is4oVvJfrkud1KvE + +Waiting for the plan to start... + +Terraform v0.14.0 + +Configuring remote state backend... +Initializing Terraform configuration... +2019/02/20 22:40:52 [DEBUG] Using modified User-Agent: Terraform/0.14.0TFE/202eeff +Refreshing Terraform state in-memory prior to plan... +The refreshed state will be used to calculate this plan, but will not be +persisted to local or remote state storage. + +null_resource.hi: Refreshing state... (ID: 217661332516885645) +null_resource.hi[1]: Refreshing state... (ID: 6064510335076839362) + +An execution plan has been generated and is shown below. +Resource actions are indicated with the following symbols: + - destroy + +Terraform will perform the following actions: + + - null_resource.hi[1] + + +Plan: 0 to add, 0 to change, 1 to destroy.`, + tfVersion_0140, + }, + } + + for _, c := range cases { + output := runtime.StripRefreshingFromPlanOutput(c.out, c.tfVersion) + Equals(t, ` +An execution plan has been generated and is shown below. +Resource actions are indicated with the following symbols: + - destroy + +Terraform will perform the following actions: + + - null_resource.hi[1] + + +Plan: 0 to add, 0 to change, 1 to destroy.`, output) + } +} + type remotePlanMock struct { // LinesToSend will be sent on the channel. LinesToSend string From d1d50564a9a64bf81bb20cbab4c9922c0fef1f8c Mon Sep 17 00:00:00 2001 From: Mathieu Cantin Date: Thu, 25 Feb 2021 09:38:18 -0500 Subject: [PATCH 2/5] Fix comment typo --- server/events/runtime/apply_step_runner.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/server/events/runtime/apply_step_runner.go b/server/events/runtime/apply_step_runner.go index a749ce4f87..8e87d3fe82 100644 --- a/server/events/runtime/apply_step_runner.go +++ b/server/events/runtime/apply_step_runner.go @@ -98,7 +98,7 @@ func (a *ApplyStepRunner) cleanRemoteApplyOutput(out string) string { applyStartText := ` Terraform will perform the actions described above. Only 'yes' will be accepted to approve. - Enter a value: + Enter a value: ` applyStartIdx := strings.Index(out, applyStartText) if applyStartIdx < 0 { @@ -209,7 +209,7 @@ func (a *ApplyStepRunner) runRemoteApply( func (a *ApplyStepRunner) remotePlanChanged(planfileContents string, applyOut string, tfVersion *version.Version) error { output := StripRefreshingFromPlanOutput(applyOut, tfVersion) - // The output stop to execute the plan. + // Strip plan output after the prompt to execute the plan. planEndIdx := strings.Index(output, "Do you want to perform these actions in workspace \"") if planEndIdx < 0 { return fmt.Errorf("Couldn't find plan end when parsing apply output:\n%q", applyOut) From f7abecc8de7aaa7c656cb2e8dd70211647cf652a Mon Sep 17 00:00:00 2001 From: Mathieu Cantin Date: Thu, 25 Feb 2021 12:46:02 -0500 Subject: [PATCH 3/5] Revert editor change (training whitespace) --- server/events/runtime/apply_step_runner.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/server/events/runtime/apply_step_runner.go b/server/events/runtime/apply_step_runner.go index 8e87d3fe82..85749b14e4 100644 --- a/server/events/runtime/apply_step_runner.go +++ b/server/events/runtime/apply_step_runner.go @@ -98,7 +98,7 @@ func (a *ApplyStepRunner) cleanRemoteApplyOutput(out string) string { applyStartText := ` Terraform will perform the actions described above. Only 'yes' will be accepted to approve. - Enter a value: + Enter a value: ` applyStartIdx := strings.Index(out, applyStartText) if applyStartIdx < 0 { From b5ab221cb0c4b490517ac766b7a1218524574869 Mon Sep 17 00:00:00 2001 From: Mathieu Cantin Date: Thu, 25 Feb 2021 13:05:25 -0500 Subject: [PATCH 4/5] Fix gofmt --- server/events/runtime/apply_step_runner.go | 2 +- server/events/runtime/plan_step_runner.go | 4 ++-- server/events/runtime/plan_step_runner_test.go | 2 +- 3 files changed, 4 insertions(+), 4 deletions(-) diff --git a/server/events/runtime/apply_step_runner.go b/server/events/runtime/apply_step_runner.go index 85749b14e4..d07810b9eb 100644 --- a/server/events/runtime/apply_step_runner.go +++ b/server/events/runtime/apply_step_runner.go @@ -214,7 +214,7 @@ func (a *ApplyStepRunner) remotePlanChanged(planfileContents string, applyOut st if planEndIdx < 0 { return fmt.Errorf("Couldn't find plan end when parsing apply output:\n%q", applyOut) } - currPlan := strings.TrimSpace(output[: planEndIdx]) + currPlan := strings.TrimSpace(output[:planEndIdx]) // Ensure we strip the remoteOpsHeader from the plan contents so the // comparison is fair. We add this header in the plan phase so we can diff --git a/server/events/runtime/plan_step_runner.go b/server/events/runtime/plan_step_runner.go index 35b13d9946..123b596770 100644 --- a/server/events/runtime/plan_step_runner.go +++ b/server/events/runtime/plan_step_runner.go @@ -15,7 +15,7 @@ import ( const ( defaultWorkspace = "default" - refreshKeyword = "Refreshing state..." + refreshKeyword = "Refreshing state..." refreshSeparator = "------------------------------------------------------------------------\n" ) @@ -300,7 +300,7 @@ func StripRefreshingFromPlanOutput(output string, tfVersion *version.Version) st } if finalIndex != 0 { - output = strings.Join(lines[finalIndex + 1:], "\n") + output = strings.Join(lines[finalIndex+1:], "\n") } return output } else { diff --git a/server/events/runtime/plan_step_runner_test.go b/server/events/runtime/plan_step_runner_test.go index 1830196dde..2f1aed7312 100644 --- a/server/events/runtime/plan_step_runner_test.go +++ b/server/events/runtime/plan_step_runner_test.go @@ -819,7 +819,7 @@ func TestStripRefreshingFromPlanOutput(t *testing.T) { tfVersion_0135, _ := version.NewVersion("0.13.5") tfVersion_0140, _ := version.NewVersion("0.14.0") cases := []struct { - out string + out string tfVersion *version.Version }{ { From e1a884e158fe02fa38b2ae6fc11674459108d2e2 Mon Sep 17 00:00:00 2001 From: Mathieu Cantin Date: Thu, 25 Feb 2021 13:12:23 -0500 Subject: [PATCH 5/5] Fix gofmt --- server/events/runtime/plan_step_runner.go | 3 +-- server/events/runtime/plan_step_runner_test.go | 8 ++++---- 2 files changed, 5 insertions(+), 6 deletions(-) diff --git a/server/events/runtime/plan_step_runner.go b/server/events/runtime/plan_step_runner.go index 123b596770..f55ee76871 100644 --- a/server/events/runtime/plan_step_runner.go +++ b/server/events/runtime/plan_step_runner.go @@ -302,7 +302,6 @@ func StripRefreshingFromPlanOutput(output string, tfVersion *version.Version) st if finalIndex != 0 { output = strings.Join(lines[finalIndex+1:], "\n") } - return output } else { // Plan output contains a lot of "Refreshing..." lines followed by a // separator. We want to remove everything before that separator. @@ -310,8 +309,8 @@ func StripRefreshingFromPlanOutput(output string, tfVersion *version.Version) st if sepIdx > -1 { output = output[sepIdx+len(refreshSeparator):] } - return output } + return output } // remoteOpsErr01114 is the error terraform plan will return if this project is diff --git a/server/events/runtime/plan_step_runner_test.go b/server/events/runtime/plan_step_runner_test.go index 2f1aed7312..740a895d03 100644 --- a/server/events/runtime/plan_step_runner_test.go +++ b/server/events/runtime/plan_step_runner_test.go @@ -816,15 +816,15 @@ Plan: 0 to add, 0 to change, 1 to destroy.`, string(bytes)) // Test striping output method func TestStripRefreshingFromPlanOutput(t *testing.T) { - tfVersion_0135, _ := version.NewVersion("0.13.5") - tfVersion_0140, _ := version.NewVersion("0.14.0") + tfVersion0135, _ := version.NewVersion("0.13.5") + tfVersion0140, _ := version.NewVersion("0.14.0") cases := []struct { out string tfVersion *version.Version }{ { remotePlanOutput, - tfVersion_0135, + tfVersion0135, }, { `Running plan in the remote backend. Output will stream here. Pressing Ctrl-C @@ -859,7 +859,7 @@ Terraform will perform the following actions: Plan: 0 to add, 0 to change, 1 to destroy.`, - tfVersion_0140, + tfVersion0140, }, }