blob: 86508d41e5f30911bc139ecbaff827b61509fc06 [file] [log] [blame]
package tfexec
import (
"context"
"fmt"
"os/exec"
"strconv"
)
type applyConfig struct {
backup string
dirOrPlan string
lock bool
// LockTimeout must be a string with time unit, e.g. '10s'
lockTimeout string
parallelism int
refresh bool
state string
stateOut string
targets []string
// Vars: each var must be supplied as a single string, e.g. 'foo=bar'
vars []string
varFiles []string
}
var defaultApplyOptions = applyConfig{
lock: true,
parallelism: 10,
refresh: true,
}
// ApplyOption represents options used in the Apply method.
type ApplyOption interface {
configureApply(*applyConfig)
}
func (opt *ParallelismOption) configureApply(conf *applyConfig) {
conf.parallelism = opt.parallelism
}
func (opt *BackupOption) configureApply(conf *applyConfig) {
conf.backup = opt.path
}
func (opt *TargetOption) configureApply(conf *applyConfig) {
conf.targets = append(conf.targets, opt.target)
}
func (opt *LockTimeoutOption) configureApply(conf *applyConfig) {
conf.lockTimeout = opt.timeout
}
func (opt *StateOption) configureApply(conf *applyConfig) {
conf.state = opt.path
}
func (opt *StateOutOption) configureApply(conf *applyConfig) {
conf.stateOut = opt.path
}
func (opt *VarFileOption) configureApply(conf *applyConfig) {
conf.varFiles = append(conf.varFiles, opt.path)
}
func (opt *LockOption) configureApply(conf *applyConfig) {
conf.lock = opt.lock
}
func (opt *RefreshOption) configureApply(conf *applyConfig) {
conf.refresh = opt.refresh
}
func (opt *VarOption) configureApply(conf *applyConfig) {
conf.vars = append(conf.vars, opt.assignment)
}
func (opt *DirOrPlanOption) configureApply(conf *applyConfig) {
conf.dirOrPlan = opt.path
}
func (tf *Terraform) Apply(ctx context.Context, opts ...ApplyOption) error {
return tf.runTerraformCmd(tf.applyCmd(ctx, opts...))
}
func (tf *Terraform) applyCmd(ctx context.Context, opts ...ApplyOption) *exec.Cmd {
c := defaultApplyOptions
for _, o := range opts {
o.configureApply(&c)
}
args := []string{"apply", "-no-color", "-auto-approve", "-input=false"}
// string opts: only pass if set
if c.backup != "" {
args = append(args, "-backup="+c.backup)
}
if c.lockTimeout != "" {
args = append(args, "-lock-timeout="+c.lockTimeout)
}
if c.state != "" {
args = append(args, "-state="+c.state)
}
if c.stateOut != "" {
args = append(args, "-state-out="+c.stateOut)
}
for _, vf := range c.varFiles {
args = append(args, "-var-file="+vf)
}
// boolean and numerical opts: always pass
args = append(args, "-lock="+strconv.FormatBool(c.lock))
args = append(args, "-parallelism="+fmt.Sprint(c.parallelism))
args = append(args, "-refresh="+strconv.FormatBool(c.refresh))
// string slice opts: split into separate args
if c.targets != nil {
for _, ta := range c.targets {
args = append(args, "-target="+ta)
}
}
if c.vars != nil {
for _, v := range c.vars {
args = append(args, "-var", v)
}
}
// string argument: pass if set
if c.dirOrPlan != "" {
args = append(args, c.dirOrPlan)
}
return tf.buildTerraformCmd(ctx, args...)
}