From e036f0837604d7b29f88cd93ed999637256cab1e Mon Sep 17 00:00:00 2001 From: Brian Flad Date: Wed, 13 Apr 2022 10:25:03 -0400 Subject: [PATCH 1/4] tfexec: Add (Terraform).SetLog() method Reference: https://github.com/hashicorp/terraform-exec/issues/290 --- tfexec/cmd.go | 3 +-- tfexec/terraform.go | 30 +++++++++++++++++++++++++++--- 2 files changed, 28 insertions(+), 5 deletions(-) diff --git a/tfexec/cmd.go b/tfexec/cmd.go index 83abd22d..21243624 100644 --- a/tfexec/cmd.go +++ b/tfexec/cmd.go @@ -149,8 +149,7 @@ func (tf *Terraform) buildEnv(mergeEnv map[string]string) []string { env[logPathEnvVar] = "" } else { env[logPathEnvVar] = tf.logPath - // Log levels other than TRACE are currently unreliable, the CLI recommends using TRACE only. - env[logEnvVar] = "TRACE" + env[logEnvVar] = tf.log } // constant automation override env vars diff --git a/tfexec/terraform.go b/tfexec/terraform.go index 2ad143a4..a743ef30 100644 --- a/tfexec/terraform.go +++ b/tfexec/terraform.go @@ -48,9 +48,14 @@ type Terraform struct { skipProviderVerify bool env map[string]string - stdout io.Writer - stderr io.Writer - logger printfer + stdout io.Writer + stderr io.Writer + logger printfer + + // TF_LOG environment variable, defaults to TRACE if logPath is set. + log string + + // TF_LOG_PATH environment variable logPath string versionLock sync.Mutex @@ -122,10 +127,29 @@ func (tf *Terraform) SetStderr(w io.Writer) { tf.stderr = w } +// SetLog sets the TF_LOG environment variable for Terraform CLI execution. +// This must be combined with a call to SetLogPath to take effect. +// +// This is only compatible with Terraform CLI 0.15.0 or later as setting the +// log level was unreliable in earlier versions. It will default to TRACE for +// those earlier versions when SetLogPath is called. +func (tf *Terraform) SetLog(log string) error { + err := tf.compatible(context.Background(), nil, tf0_15_0) + if err != nil { + return err + } + tf.log = log + return nil +} + // SetLogPath sets the TF_LOG_PATH environment variable for Terraform CLI // execution. func (tf *Terraform) SetLogPath(path string) error { tf.logPath = path + // Prevent setting the log path without enabling logging + if tf.log == "" { + tf.log = "TRACE" + } return nil } From 47e6c290999dff0d5249ab9c076d9b63e2b047fc Mon Sep 17 00:00:00 2001 From: Brian Flad Date: Tue, 17 May 2022 19:25:06 -0500 Subject: [PATCH 2/4] tfexec: Add testing for SetLog --- tfexec/terraform_test.go | 130 +++++++++++++++++++++++++++++++++++++++ 1 file changed, 130 insertions(+) diff --git a/tfexec/terraform_test.go b/tfexec/terraform_test.go index d8f4ab4c..da158d49 100644 --- a/tfexec/terraform_test.go +++ b/tfexec/terraform_test.go @@ -5,6 +5,7 @@ import ( "errors" "io/ioutil" "os" + "path/filepath" "testing" "github.com/hashicorp/terraform-exec/tfexec/internal/testutil" @@ -61,6 +62,135 @@ func TestSetEnv(t *testing.T) { } } +func TestSetLog(t *testing.T) { + td := t.TempDir() + + tf, err := NewTerraform(td, tfVersion(t, testutil.Latest012)) + if err != nil { + t.Fatal(err) + } + + // Required so all testing environment variables are not copied. + err = tf.SetEnv(map[string]string{ + "CLEARENV": "1", + }) + + if err != nil { + t.Fatalf("unexpected SetEnv error: %s", err) + } + + t.Run("case 1: SetLog TRACE no SetLogPath", func(t *testing.T) { + err := tf.SetLog("TRACE") + + if err != nil { + t.Fatalf("unexpected SetLog error: %s", err) + } + + initCmd, err := tf.initCmd(context.Background()) + + if err != nil { + t.Fatalf("unexpected command error: %s", err) + } + + assertCmd(t, []string{ + "init", + "-no-color", + "-force-copy", + "-input=false", + "-lock-timeout=0s", + "-backend=true", + "-get=true", + "-upgrade=false", + "-lock=true", + "-get-plugins=true", + "-verify-plugins=true", + }, map[string]string{ + "CLEARENV": "1", + "TF_LOG": "", + }, initCmd) + }) + + t.Run("case 2: SetLog TRACE and SetLogPath", func(t *testing.T) { + tfLogPath := filepath.Join(td, "test.log") + + err := tf.SetLog("TRACE") + + if err != nil { + t.Fatalf("unexpected SetLog error: %s", err) + } + + err = tf.SetLogPath(tfLogPath) + + if err != nil { + t.Fatalf("unexpected SetLogPath error: %s", err) + } + + initCmd, err := tf.initCmd(context.Background()) + + if err != nil { + t.Fatalf("unexpected command error: %s", err) + } + + assertCmd(t, []string{ + "init", + "-no-color", + "-force-copy", + "-input=false", + "-lock-timeout=0s", + "-backend=true", + "-get=true", + "-upgrade=false", + "-lock=true", + "-get-plugins=true", + "-verify-plugins=true", + }, map[string]string{ + "CLEARENV": "1", + "TF_LOG": "TRACE", + "TF_LOG_PATH": tfLogPath, + }, initCmd) + }) + + t.Run("case 3: SetLog DEBUG and SetLogPath", func(t *testing.T) { + tfLogPath := filepath.Join(td, "test.log") + + err := tf.SetLog("DEBUG") + + if err != nil { + t.Fatalf("unexpected SetLog error: %s", err) + } + + err = tf.SetLogPath(tfLogPath) + + if err != nil { + t.Fatalf("unexpected SetLogPath error: %s", err) + } + + initCmd, err := tf.initCmd(context.Background()) + + if err != nil { + t.Fatalf("unexpected command error: %s", err) + } + + assertCmd(t, []string{ + "init", + "-no-color", + "-force-copy", + "-input=false", + "-lock-timeout=0s", + "-backend=true", + "-get=true", + "-upgrade=false", + "-lock=true", + "-get-plugins=true", + "-verify-plugins=true", + }, map[string]string{ + "CLEARENV": "1", + "TF_LOG": "DEBUG", + "TF_LOG_PATH": tfLogPath, + }, initCmd) + }) +} + func TestCheckpointDisablePropagation(t *testing.T) { td := t.TempDir() From fc291d2aa52bd6a0c04d8cd63369dcb6109f1f41 Mon Sep 17 00:00:00 2001 From: Brian Flad Date: Tue, 17 May 2022 19:29:41 -0500 Subject: [PATCH 3/4] tfexec: Add SetLogPath testing --- tfexec/terraform_test.go | 78 ++++++++++++++++++++++++++++++++++++++++ 1 file changed, 78 insertions(+) diff --git a/tfexec/terraform_test.go b/tfexec/terraform_test.go index da158d49..6933cab6 100644 --- a/tfexec/terraform_test.go +++ b/tfexec/terraform_test.go @@ -191,6 +191,84 @@ func TestSetLog(t *testing.T) { }) } +func TestSetLogPath(t *testing.T) { + td := t.TempDir() + + tf, err := NewTerraform(td, tfVersion(t, testutil.Latest012)) + if err != nil { + t.Fatal(err) + } + + // Required so all testing environment variables are not copied. + err = tf.SetEnv(map[string]string{ + "CLEARENV": "1", + }) + + if err != nil { + t.Fatalf("unexpected SetEnv error: %s", err) + } + + t.Run("case 1: No SetLogPath", func(t *testing.T) { + initCmd, err := tf.initCmd(context.Background()) + + if err != nil { + t.Fatalf("unexpected command error: %s", err) + } + + assertCmd(t, []string{ + "init", + "-no-color", + "-force-copy", + "-input=false", + "-lock-timeout=0s", + "-backend=true", + "-get=true", + "-upgrade=false", + "-lock=true", + "-get-plugins=true", + "-verify-plugins=true", + }, map[string]string{ + "CLEARENV": "1", + "TF_LOG": "", + "TF_LOG_PATH": "", + }, initCmd) + }) + + t.Run("case 2: SetLogPath sets TF_LOG and TF_LOG_PATH", func(t *testing.T) { + tfLogPath := filepath.Join(td, "test.log") + + err = tf.SetLogPath(tfLogPath) + + if err != nil { + t.Fatalf("unexpected SetLogPath error: %s", err) + } + + initCmd, err := tf.initCmd(context.Background()) + + if err != nil { + t.Fatalf("unexpected command error: %s", err) + } + + assertCmd(t, []string{ + "init", + "-no-color", + "-force-copy", + "-input=false", + "-lock-timeout=0s", + "-backend=true", + "-get=true", + "-upgrade=false", + "-lock=true", + "-get-plugins=true", + "-verify-plugins=true", + }, map[string]string{ + "CLEARENV": "1", + "TF_LOG": "TRACE", + "TF_LOG_PATH": tfLogPath, + }, initCmd) + }) +} + func TestCheckpointDisablePropagation(t *testing.T) { td := t.TempDir() From 952b45aa71a2a10b08e8f3df7ce6e6502f609622 Mon Sep 17 00:00:00 2001 From: Brian Flad Date: Tue, 17 May 2022 19:50:18 -0500 Subject: [PATCH 4/4] tfexec: Ensure SetLog version compatibility is 0.15 minimum, testing updates for darwin/arm64 --- tfexec/terraform.go | 2 +- tfexec/terraform_test.go | 57 +++++++++++++++++++++------------------- 2 files changed, 31 insertions(+), 28 deletions(-) diff --git a/tfexec/terraform.go b/tfexec/terraform.go index a743ef30..835e7e06 100644 --- a/tfexec/terraform.go +++ b/tfexec/terraform.go @@ -134,7 +134,7 @@ func (tf *Terraform) SetStderr(w io.Writer) { // log level was unreliable in earlier versions. It will default to TRACE for // those earlier versions when SetLogPath is called. func (tf *Terraform) SetLog(log string) error { - err := tf.compatible(context.Background(), nil, tf0_15_0) + err := tf.compatible(context.Background(), tf0_15_0, nil) if err != nil { return err } diff --git a/tfexec/terraform_test.go b/tfexec/terraform_test.go index 6933cab6..d6521b4c 100644 --- a/tfexec/terraform_test.go +++ b/tfexec/terraform_test.go @@ -6,6 +6,7 @@ import ( "io/ioutil" "os" "path/filepath" + "runtime" "testing" "github.com/hashicorp/terraform-exec/tfexec/internal/testutil" @@ -65,9 +66,10 @@ func TestSetEnv(t *testing.T) { func TestSetLog(t *testing.T) { td := t.TempDir() - tf, err := NewTerraform(td, tfVersion(t, testutil.Latest012)) + tf, err := NewTerraform(td, tfVersion(t, testutil.Latest_v1)) + if err != nil { - t.Fatal(err) + t.Fatalf("unexpected NewTerraform error: %s", err) } // Required so all testing environment variables are not copied. @@ -79,7 +81,27 @@ func TestSetLog(t *testing.T) { t.Fatalf("unexpected SetEnv error: %s", err) } - t.Run("case 1: SetLog TRACE no SetLogPath", func(t *testing.T) { + t.Run("case 1: SetLog <= 0.15 error", func(t *testing.T) { + if runtime.GOOS == "darwin" && runtime.GOARCH == "arm64" { + t.Skip("Terraform for darwin/arm64 is not available until v1") + } + + td012 := t.TempDir() + + tf012, err := NewTerraform(td012, tfVersion(t, testutil.Latest012)) + + if err != nil { + t.Fatalf("unexpected NewTerraform error: %s", err) + } + + err = tf012.SetLog("TRACE") + + if err == nil { + t.Fatal("expected SetLog error, got none") + } + }) + + t.Run("case 2: SetLog TRACE no SetLogPath", func(t *testing.T) { err := tf.SetLog("TRACE") if err != nil { @@ -97,20 +119,16 @@ func TestSetLog(t *testing.T) { "-no-color", "-force-copy", "-input=false", - "-lock-timeout=0s", "-backend=true", "-get=true", "-upgrade=false", - "-lock=true", - "-get-plugins=true", - "-verify-plugins=true", }, map[string]string{ "CLEARENV": "1", "TF_LOG": "", }, initCmd) }) - t.Run("case 2: SetLog TRACE and SetLogPath", func(t *testing.T) { + t.Run("case 3: SetLog TRACE and SetLogPath", func(t *testing.T) { tfLogPath := filepath.Join(td, "test.log") err := tf.SetLog("TRACE") @@ -136,13 +154,9 @@ func TestSetLog(t *testing.T) { "-no-color", "-force-copy", "-input=false", - "-lock-timeout=0s", "-backend=true", "-get=true", "-upgrade=false", - "-lock=true", - "-get-plugins=true", - "-verify-plugins=true", }, map[string]string{ "CLEARENV": "1", "TF_LOG": "TRACE", @@ -150,7 +164,7 @@ func TestSetLog(t *testing.T) { }, initCmd) }) - t.Run("case 3: SetLog DEBUG and SetLogPath", func(t *testing.T) { + t.Run("case 4: SetLog DEBUG and SetLogPath", func(t *testing.T) { tfLogPath := filepath.Join(td, "test.log") err := tf.SetLog("DEBUG") @@ -176,13 +190,9 @@ func TestSetLog(t *testing.T) { "-no-color", "-force-copy", "-input=false", - "-lock-timeout=0s", "-backend=true", "-get=true", "-upgrade=false", - "-lock=true", - "-get-plugins=true", - "-verify-plugins=true", }, map[string]string{ "CLEARENV": "1", "TF_LOG": "DEBUG", @@ -194,9 +204,10 @@ func TestSetLog(t *testing.T) { func TestSetLogPath(t *testing.T) { td := t.TempDir() - tf, err := NewTerraform(td, tfVersion(t, testutil.Latest012)) + tf, err := NewTerraform(td, tfVersion(t, testutil.Latest_v1)) + if err != nil { - t.Fatal(err) + t.Fatalf("unexpected NewTerraform error: %s", err) } // Required so all testing environment variables are not copied. @@ -220,13 +231,9 @@ func TestSetLogPath(t *testing.T) { "-no-color", "-force-copy", "-input=false", - "-lock-timeout=0s", "-backend=true", "-get=true", "-upgrade=false", - "-lock=true", - "-get-plugins=true", - "-verify-plugins=true", }, map[string]string{ "CLEARENV": "1", "TF_LOG": "", @@ -254,13 +261,9 @@ func TestSetLogPath(t *testing.T) { "-no-color", "-force-copy", "-input=false", - "-lock-timeout=0s", "-backend=true", "-get=true", "-upgrade=false", - "-lock=true", - "-get-plugins=true", - "-verify-plugins=true", }, map[string]string{ "CLEARENV": "1", "TF_LOG": "TRACE",