You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
terraformDummyRepo2/vendor/github.com/hashicorp/terraform-plugin-test/working_dir.go

378 lines
11 KiB
Go

package tftest
import (
"fmt"
"io/ioutil"
"os"
"path/filepath"
tfjson "github.com/hashicorp/terraform-json"
)
// WorkingDir represents a distinct working directory that can be used for
// running tests. Each test should construct its own WorkingDir by calling
// NewWorkingDir or RequireNewWorkingDir on its package's singleton
// tftest.Helper.
type WorkingDir struct {
h *Helper
// baseDir is the root of the working directory tree
baseDir string
// baseArgs is arguments that should be appended to all commands
baseArgs []string
// configDir contains the singular config file generated for each test
configDir string
}
// Close deletes the directories and files created to represent the receiving
// working directory. After this method is called, the working directory object
// is invalid and may no longer be used.
func (wd *WorkingDir) Close() error {
return os.RemoveAll(wd.baseDir)
}
// SetConfig sets a new configuration for the working directory.
//
// This must be called at least once before any call to Init, Plan, Apply, or
// Destroy to establish the configuration. Any previously-set configuration is
// discarded and any saved plan is cleared.
func (wd *WorkingDir) SetConfig(cfg string) error {
// Each call to SetConfig creates a new directory under our baseDir.
// We create them within so that our final cleanup step will delete them
// automatically without any additional tracking.
configDir, err := ioutil.TempDir(wd.baseDir, "config")
if err != nil {
return err
}
configFilename := filepath.Join(configDir, "terraform_plugin_test.tf")
err = ioutil.WriteFile(configFilename, []byte(cfg), 0700)
if err != nil {
return err
}
wd.configDir = configDir
// Changing configuration invalidates any saved plan.
err = wd.ClearPlan()
if err != nil {
return err
}
return nil
}
// RequireSetConfig is a variant of SetConfig that will fail the test via the
// given TestControl if the configuration cannot be set.
func (wd *WorkingDir) RequireSetConfig(t TestControl, cfg string) {
t.Helper()
if err := wd.SetConfig(cfg); err != nil {
t := testingT{t}
t.Fatalf("failed to set config: %s", err)
}
}
// ClearState deletes any Terraform state present in the working directory.
//
// Any remote objects tracked by the state are not destroyed first, so this
// will leave them dangling in the remote system.
func (wd *WorkingDir) ClearState() error {
err := os.Remove(filepath.Join(wd.baseDir, "terraform.tfstate"))
if os.IsNotExist(err) {
return nil
}
return err
}
// RequireClearState is a variant of ClearState that will fail the test via the
// given TestControl if the state cannot be cleared.
func (wd *WorkingDir) RequireClearState(t TestControl) {
t.Helper()
if err := wd.ClearState(); err != nil {
t := testingT{t}
t.Fatalf("failed to clear state: %s", err)
}
}
// ClearPlan deletes any saved plan present in the working directory.
func (wd *WorkingDir) ClearPlan() error {
err := os.Remove(wd.planFilename())
if os.IsNotExist(err) {
return nil
}
return err
}
// RequireClearPlan is a variant of ClearPlan that will fail the test via the
// given TestControl if the plan cannot be cleared.
func (wd *WorkingDir) RequireClearPlan(t TestControl) {
t.Helper()
if err := wd.ClearPlan(); err != nil {
t := testingT{t}
t.Fatalf("failed to clear plan: %s", err)
}
}
func (wd *WorkingDir) init(pluginDir string) error {
args := []string{"init"}
args = append(args, wd.baseArgs...)
return wd.runTerraform(args...)
}
// Init runs "terraform init" for the given working directory, forcing Terraform
// to use the current version of the plugin under test.
func (wd *WorkingDir) Init() error {
if wd.configDir == "" {
return fmt.Errorf("must call SetConfig before Init")
}
return wd.init(wd.h.PluginDir())
}
// RequireInit is a variant of Init that will fail the test via the given
// TestControl if init fails.
func (wd *WorkingDir) RequireInit(t TestControl) {
t.Helper()
if err := wd.Init(); err != nil {
t := testingT{t}
t.Fatalf("init failed: %s", err)
}
}
// InitPrevious runs "terraform init" for the given working directory, forcing
// Terraform to use the previous version of the plugin under test.
//
// This method will panic if no previous plugin version is available. Use
// HasPreviousVersion or RequirePreviousVersion on the test helper singleton
// to check this first.
func (wd *WorkingDir) InitPrevious() error {
if wd.configDir == "" {
return fmt.Errorf("must call SetConfig before InitPrevious")
}
return wd.init(wd.h.PreviousPluginDir())
}
// RequireInitPrevious is a variant of InitPrevious that will fail the test
// via the given TestControl if init fails.
func (wd *WorkingDir) RequireInitPrevious(t TestControl) {
t.Helper()
if err := wd.InitPrevious(); err != nil {
t := testingT{t}
t.Fatalf("init failed: %s", err)
}
}
func (wd *WorkingDir) planFilename() string {
return filepath.Join(wd.baseDir, "tfplan")
}
// CreatePlan runs "terraform plan" to create a saved plan file, which if successful
// will then be used for the next call to Apply.
func (wd *WorkingDir) CreatePlan() error {
args := []string{"plan", "-refresh=false"}
args = append(args, wd.baseArgs...)
args = append(args, "-out=tfplan", wd.configDir)
return wd.runTerraform(args...)
}
// RequireCreatePlan is a variant of CreatePlan that will fail the test via
// the given TestControl if plan creation fails.
func (wd *WorkingDir) RequireCreatePlan(t TestControl) {
t.Helper()
if err := wd.CreatePlan(); err != nil {
t := testingT{t}
t.Fatalf("failed to create plan: %s", err)
}
}
// Apply runs "terraform apply". If CreatePlan has previously completed
// successfully and the saved plan has not been cleared in the meantime then
// this will apply the saved plan. Otherwise, it will implicitly create a new
// plan and apply it.
func (wd *WorkingDir) Apply() error {
args := []string{"apply", "-refresh=false"}
args = append(args, wd.baseArgs...)
if wd.HasSavedPlan() {
args = append(args, "tfplan")
} else {
args = append(args, "-auto-approve")
args = append(args, wd.configDir)
}
return wd.runTerraform(args...)
}
// RequireApply is a variant of Apply that will fail the test via
// the given TestControl if the apply operation fails.
func (wd *WorkingDir) RequireApply(t TestControl) {
t.Helper()
if err := wd.Apply(); err != nil {
t := testingT{t}
t.Fatalf("failed to apply: %s", err)
}
}
// Destroy runs "terraform destroy". It does not consider or modify any saved
// plan, and is primarily for cleaning up at the end of a test run.
//
// If destroy fails then remote objects might still exist, and continue to
// exist after a particular test is concluded.
func (wd *WorkingDir) Destroy() error {
args := []string{"destroy", "-refresh=false"}
args = append(args, wd.baseArgs...)
args = append(args, "-auto-approve", wd.configDir)
return wd.runTerraform(args...)
}
// RequireDestroy is a variant of Destroy that will fail the test via
// the given TestControl if the destroy operation fails.
//
// If destroy fails then remote objects might still exist, and continue to
// exist after a particular test is concluded.
func (wd *WorkingDir) RequireDestroy(t TestControl) {
t.Helper()
if err := wd.Destroy(); err != nil {
t := testingT{t}
t.Logf("WARNING: destroy failed, so remote objects may still exist and be subject to billing")
t.Fatalf("failed to destroy: %s", err)
}
}
// HasSavedPlan returns true if there is a saved plan in the working directory. If
// so, a subsequent call to Apply will apply that saved plan.
func (wd *WorkingDir) HasSavedPlan() bool {
_, err := os.Stat(wd.planFilename())
return err == nil
}
// SavedPlan returns an object describing the current saved plan file, if any.
//
// If no plan is saved or if the plan file cannot be read, SavedPlan returns
// an error.
func (wd *WorkingDir) SavedPlan() (*tfjson.Plan, error) {
if !wd.HasSavedPlan() {
return nil, fmt.Errorf("there is no current saved plan")
}
var ret tfjson.Plan
args := []string{"show"}
args = append(args, wd.baseArgs...)
args = append(args, "-json", wd.planFilename())
err := wd.runTerraformJSON(&ret, args...)
if err != nil {
return nil, err
}
return &ret, nil
}
// RequireSavedPlan is a variant of SavedPlan that will fail the test via
// the given TestControl if the plan cannot be read.
func (wd *WorkingDir) RequireSavedPlan(t TestControl) *tfjson.Plan {
t.Helper()
ret, err := wd.SavedPlan()
if err != nil {
t := testingT{t}
t.Fatalf("failed to read saved plan: %s", err)
}
return ret
}
// State returns an object describing the current state.
//
// If the state cannot be read, State returns an error.
func (wd *WorkingDir) State() (*tfjson.State, error) {
var ret tfjson.State
args := []string{"show"}
args = append(args, wd.baseArgs...)
args = append(args, "-json")
err := wd.runTerraformJSON(&ret, args...)
if err != nil {
return nil, err
}
return &ret, nil
}
// RequireState is a variant of State that will fail the test via
// the given TestControl if the state cannot be read.
func (wd *WorkingDir) RequireState(t TestControl) *tfjson.State {
t.Helper()
ret, err := wd.State()
if err != nil {
t := testingT{t}
t.Fatalf("failed to read state plan: %s", err)
}
return ret
}
// Import runs terraform import
func (wd *WorkingDir) Import(resource, id string) error {
args := []string{"import"}
args = append(args, wd.baseArgs...)
args = append(args, "-config="+wd.configDir, resource, id)
return wd.runTerraform(args...)
}
// RequireImport is a variant of Import that will fail the test via
// the given TestControl if the import is non successful.
func (wd *WorkingDir) RequireImport(t TestControl, resource, id string) {
t.Helper()
if err := wd.Import(resource, id); err != nil {
t := testingT{t}
t.Fatalf("failed to import: %s", err)
}
}
// Refresh runs terraform refresh
func (wd *WorkingDir) Refresh() error {
args := []string{"refresh"}
args = append(args, wd.baseArgs...)
args = append(args, "-state="+filepath.Join(wd.baseDir, "terraform.tfstate"))
args = append(args, wd.configDir)
return wd.runTerraform(args...)
}
// RequireRefresh is a variant of Refresh that will fail the test via
// the given TestControl if the refresh is non successful.
func (wd *WorkingDir) RequireRefresh(t TestControl) {
t.Helper()
if err := wd.Refresh(); err != nil {
t := testingT{t}
t.Fatalf("failed to refresh: %s", err)
}
}
// Schemas returns an object describing the provider schemas.
//
// If the schemas cannot be read, Schemas returns an error.
func (wd *WorkingDir) Schemas() (*tfjson.ProviderSchemas, error) {
args := []string{"providers", wd.configDir, "schema"}
var ret tfjson.ProviderSchemas
err := wd.runTerraformJSON(&ret, args...)
if err != nil {
return nil, err
}
return &ret, nil
}
// RequireSchemas is a variant of Schemas that will fail the test via
// the given TestControl if the schemas cannot be read.
func (wd *WorkingDir) RequireSchemas(t TestControl) *tfjson.ProviderSchemas {
t.Helper()
ret, err := wd.Schemas()
if err != nil {
t := testingT{t}
t.Fatalf("failed to read schemas: %s", err)
}
return ret
}