2022-12-09 07:57:30 +00:00
|
|
|
package terraform
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
|
|
|
|
"github.com/databricks/bricks/bundle"
|
2023-04-18 14:55:06 +00:00
|
|
|
"github.com/databricks/bricks/libs/cmdio"
|
2023-04-19 13:14:26 +00:00
|
|
|
"github.com/fatih/color"
|
2022-12-09 07:57:30 +00:00
|
|
|
"github.com/hashicorp/terraform-exec/tfexec"
|
|
|
|
)
|
|
|
|
|
2023-04-19 13:14:26 +00:00
|
|
|
type ApplyGoal string
|
|
|
|
|
|
|
|
var (
|
|
|
|
ApplyDeploy = ApplyGoal("deploy")
|
|
|
|
ApplyDestroy = ApplyGoal("destroy")
|
|
|
|
)
|
|
|
|
|
|
|
|
type apply struct {
|
|
|
|
goal ApplyGoal
|
|
|
|
}
|
2022-12-09 07:57:30 +00:00
|
|
|
|
|
|
|
func (w *apply) Name() string {
|
|
|
|
return "terraform.Apply"
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *apply) Apply(ctx context.Context, b *bundle.Bundle) ([]bundle.Mutator, error) {
|
2023-04-19 13:14:26 +00:00
|
|
|
// return early if plan is empty
|
|
|
|
if b.Plan.IsEmpty {
|
|
|
|
if w.goal == ApplyDeploy {
|
|
|
|
cmdio.LogString(ctx, "No resource changes to apply. Skipping deploy!")
|
|
|
|
}
|
|
|
|
if w.goal == ApplyDestroy {
|
|
|
|
cmdio.LogString(ctx, "No resources to destroy in plan. Skipping destroy!")
|
|
|
|
}
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2022-12-15 16:30:33 +00:00
|
|
|
tf := b.Terraform
|
|
|
|
if tf == nil {
|
|
|
|
return nil, fmt.Errorf("terraform not initialized")
|
2022-12-09 07:57:30 +00:00
|
|
|
}
|
|
|
|
|
2022-12-15 16:30:33 +00:00
|
|
|
err := tf.Init(ctx, tfexec.Upgrade(true))
|
2022-12-09 07:57:30 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("terraform init: %w", err)
|
|
|
|
}
|
|
|
|
|
2023-04-19 13:14:26 +00:00
|
|
|
// Ask for confirmation, if needed
|
|
|
|
if !b.Plan.ConfirmApply {
|
|
|
|
red := color.New(color.FgRed).SprintFunc()
|
|
|
|
yellow := color.New(color.FgYellow).SprintFunc()
|
|
|
|
if b.Plan.IsReplacingResource {
|
|
|
|
cmdio.LogString(ctx, fmt.Sprintf("One or more resources will be %s. Any previous metadata associated might be lost.", yellow("replaced")))
|
|
|
|
}
|
|
|
|
if b.Plan.IsDeletingResource {
|
|
|
|
cmdio.LogString(ctx, fmt.Sprintf("One or more resources will be permanently %s", red("destroyed")))
|
|
|
|
}
|
|
|
|
b.Plan.ConfirmApply, err = cmdio.Ask(ctx, "Proceed with apply? [y/n]: ")
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// return if confirmation was not provided
|
|
|
|
if !b.Plan.ConfirmApply {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
if b.Plan.Path == "" {
|
|
|
|
return nil, fmt.Errorf("no plan found")
|
|
|
|
}
|
|
|
|
|
2023-04-19 14:02:30 +00:00
|
|
|
if w.goal == ApplyDeploy {
|
|
|
|
cmdio.LogString(ctx, "\nStarting resource deployment")
|
|
|
|
}
|
|
|
|
if w.goal == ApplyDestroy {
|
|
|
|
cmdio.LogString(ctx, "\nStarting resource destruction")
|
|
|
|
}
|
2023-04-19 13:14:26 +00:00
|
|
|
|
2023-04-19 14:02:30 +00:00
|
|
|
// Apply terraform according to the computed plan
|
2023-04-19 13:14:26 +00:00
|
|
|
err = tf.Apply(ctx, tfexec.DirOrPlan(b.Plan.Path))
|
2022-12-09 07:57:30 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("terraform apply: %w", err)
|
|
|
|
}
|
|
|
|
|
2023-04-19 13:14:26 +00:00
|
|
|
if w.goal == ApplyDeploy {
|
|
|
|
cmdio.LogString(ctx, "Successfully deployed resources!")
|
|
|
|
}
|
|
|
|
if w.goal == ApplyDestroy {
|
|
|
|
cmdio.LogString(ctx, "Successfully destroyed resources!")
|
|
|
|
}
|
2022-12-09 07:57:30 +00:00
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply returns a [bundle.Mutator] that runs the equivalent of `terraform apply`
|
|
|
|
// from the bundle's ephemeral working directory for Terraform.
|
2023-04-19 13:14:26 +00:00
|
|
|
func Apply(goal ApplyGoal) bundle.Mutator {
|
|
|
|
return &apply{
|
|
|
|
goal: goal,
|
|
|
|
}
|
2022-12-09 07:57:30 +00:00
|
|
|
}
|