From ec5a394c630391ea92c2d2af998ed332539b3228 Mon Sep 17 00:00:00 2001 From: Kurounin Date: Thu, 31 Aug 2023 13:11:26 +0300 Subject: [PATCH] feat: add refresh-only flag for plan and apply methods (#402) --- CHANGELOG.md | 4 ++++ tfexec/apply.go | 16 ++++++++++++++++ tfexec/apply_test.go | 20 ++++++++++++++++++++ tfexec/options.go | 8 ++++++++ tfexec/plan.go | 16 ++++++++++++++++ tfexec/plan_test.go | 19 +++++++++++++++++++ tfexec/version.go | 1 + 7 files changed, 84 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index b83a3cf3..443e17fb 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,6 +4,10 @@ BUG FIXES: - Fix bug in which the `TF_WORKSPACE` env var was set to an empty string, instead of being unset as intended. [GH-388] +ENHANCEMENTS: + +- tfexec: Add `-refresh-only` to `(Terraform).Apply()` and `(Terraform).Plan()` methods ([#402](https://github.com/hashicorp/terraform-exec/pull/402)) + # 0.18.1 (March 01, 2023) BUG FIXES: diff --git a/tfexec/apply.go b/tfexec/apply.go index a13ff85d..2c5a6d07 100644 --- a/tfexec/apply.go +++ b/tfexec/apply.go @@ -22,6 +22,7 @@ type applyConfig struct { parallelism int reattachInfo ReattachInfo refresh bool + refreshOnly bool replaceAddrs []string state string stateOut string @@ -80,6 +81,10 @@ func (opt *RefreshOption) configureApply(conf *applyConfig) { conf.refresh = opt.refresh } +func (opt *RefreshOnlyOption) configureApply(conf *applyConfig) { + conf.refreshOnly = opt.refreshOnly +} + func (opt *ReplaceOption) configureApply(conf *applyConfig) { conf.replaceAddrs = append(conf.replaceAddrs, opt.address) } @@ -187,6 +192,17 @@ func (tf *Terraform) buildApplyArgs(ctx context.Context, c applyConfig) ([]strin args = append(args, "-parallelism="+fmt.Sprint(c.parallelism)) args = append(args, "-refresh="+strconv.FormatBool(c.refresh)) + if c.refreshOnly { + err := tf.compatible(ctx, tf0_15_4, nil) + if err != nil { + return nil, fmt.Errorf("refresh-only option was introduced in Terraform 0.15.4: %w", err) + } + if !c.refresh { + return nil, fmt.Errorf("you cannot use refresh=false in refresh-only planning mode") + } + args = append(args, "-refresh-only") + } + // string slice opts: split into separate args if c.replaceAddrs != nil { err := tf.compatible(ctx, tf0_15_2, nil) diff --git a/tfexec/apply_test.go b/tfexec/apply_test.go index c226a1f6..8e2ddeed 100644 --- a/tfexec/apply_test.go +++ b/tfexec/apply_test.go @@ -69,6 +69,26 @@ func TestApplyCmd(t *testing.T) { "testfile", }, nil, applyCmd) }) + + t.Run("refresh-only operation", func(t *testing.T) { + applyCmd, err := tf.applyCmd(context.Background(), + RefreshOnly(true), + ) + if err != nil { + t.Fatal(err) + } + + assertCmd(t, []string{ + "apply", + "-no-color", + "-auto-approve", + "-input=false", + "-lock=true", + "-parallelism=10", + "-refresh=true", + "-refresh-only", + }, nil, applyCmd) + }) } func TestApplyJSONCmd(t *testing.T) { diff --git a/tfexec/options.go b/tfexec/options.go index 5cccde3e..5f04680b 100644 --- a/tfexec/options.go +++ b/tfexec/options.go @@ -327,6 +327,14 @@ func Refresh(refresh bool) *RefreshOption { return &RefreshOption{refresh} } +type RefreshOnlyOption struct { + refreshOnly bool +} + +func RefreshOnly(refreshOnly bool) *RefreshOnlyOption { + return &RefreshOnlyOption{refreshOnly} +} + type ReplaceOption struct { address string } diff --git a/tfexec/plan.go b/tfexec/plan.go index a4aacfbd..946ce8d0 100644 --- a/tfexec/plan.go +++ b/tfexec/plan.go @@ -20,6 +20,7 @@ type planConfig struct { parallelism int reattachInfo ReattachInfo refresh bool + refreshOnly bool replaceAddrs []string state string targets []string @@ -68,6 +69,10 @@ func (opt *RefreshOption) configurePlan(conf *planConfig) { conf.refresh = opt.refresh } +func (opt *RefreshOnlyOption) configurePlan(conf *planConfig) { + conf.refreshOnly = opt.refreshOnly +} + func (opt *ReplaceOption) configurePlan(conf *planConfig) { conf.replaceAddrs = append(conf.replaceAddrs, opt.address) } @@ -202,6 +207,17 @@ func (tf *Terraform) buildPlanArgs(ctx context.Context, c planConfig) ([]string, args = append(args, "-parallelism="+fmt.Sprint(c.parallelism)) args = append(args, "-refresh="+strconv.FormatBool(c.refresh)) + if c.refreshOnly { + err := tf.compatible(ctx, tf0_15_4, nil) + if err != nil { + return nil, fmt.Errorf("refresh-only option was introduced in Terraform 0.15.4: %w", err) + } + if !c.refresh { + return nil, fmt.Errorf("you cannot use refresh=false in refresh-only planning mode") + } + args = append(args, "-refresh-only") + } + // unary flags: pass if true if c.replaceAddrs != nil { err := tf.compatible(ctx, tf0_15_2, nil) diff --git a/tfexec/plan_test.go b/tfexec/plan_test.go index 8abfb8e7..b0c404ef 100644 --- a/tfexec/plan_test.go +++ b/tfexec/plan_test.go @@ -82,6 +82,25 @@ func TestPlanCmd(t *testing.T) { "earth", }, nil, planCmd) }) + + t.Run("run a refresh-only plan", func(t *testing.T) { + planCmd, err := tf.planCmd(context.Background(), RefreshOnly(true)) + if err != nil { + t.Fatal(err) + } + + assertCmd(t, []string{ + "plan", + "-no-color", + "-input=false", + "-detailed-exitcode", + "-lock-timeout=0s", + "-lock=true", + "-parallelism=10", + "-refresh=true", + "-refresh-only", + }, nil, planCmd) + }) } func TestPlanJSONCmd(t *testing.T) { diff --git a/tfexec/version.go b/tfexec/version.go index 62332122..4ba4f6ea 100644 --- a/tfexec/version.go +++ b/tfexec/version.go @@ -29,6 +29,7 @@ var ( tf0_15_0 = version.Must(version.NewVersion("0.15.0")) tf0_15_2 = version.Must(version.NewVersion("0.15.2")) tf0_15_3 = version.Must(version.NewVersion("0.15.3")) + tf0_15_4 = version.Must(version.NewVersion("0.15.4")) tf1_1_0 = version.Must(version.NewVersion("1.1.0")) tf1_4_0 = version.Must(version.NewVersion("1.4.0")) tf1_6_0 = version.Must(version.NewVersion("1.6.0"))