2024-06-12 21:53:00 +02:00
|
|
|
package db_lib
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
|
|
|
"path"
|
|
|
|
"strings"
|
|
|
|
"time"
|
2024-10-21 13:03:12 +02:00
|
|
|
|
|
|
|
"github.com/ansible-semaphore/semaphore/db"
|
|
|
|
"github.com/ansible-semaphore/semaphore/pkg/task_logger"
|
|
|
|
"github.com/ansible-semaphore/semaphore/util"
|
2024-06-12 21:53:00 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
type TerraformApp struct {
|
|
|
|
Logger task_logger.Logger
|
|
|
|
Template db.Template
|
|
|
|
Repository db.Repository
|
2024-07-17 16:58:20 +02:00
|
|
|
Inventory db.Inventory
|
2024-06-12 21:53:00 +02:00
|
|
|
reader terraformReader
|
2024-07-17 17:29:12 +02:00
|
|
|
Name string
|
2024-06-17 20:37:45 +02:00
|
|
|
noChanges bool
|
2024-06-12 21:53:00 +02:00
|
|
|
}
|
|
|
|
|
2024-06-17 20:37:45 +02:00
|
|
|
type terraformReaderResult int
|
2024-06-12 21:53:00 +02:00
|
|
|
|
2024-06-17 20:37:45 +02:00
|
|
|
const (
|
|
|
|
terraformReaderConfirmed terraformReaderResult = iota
|
|
|
|
terraformReaderFailed
|
|
|
|
)
|
2024-06-12 21:53:00 +02:00
|
|
|
|
|
|
|
type terraformReader struct {
|
2024-06-17 20:37:45 +02:00
|
|
|
result *terraformReaderResult
|
2024-06-12 21:53:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (t *TerraformApp) makeCmd(command string, args []string, environmentVars *[]string) *exec.Cmd {
|
|
|
|
cmd := exec.Command(command, args...) //nolint: gas
|
|
|
|
cmd.Dir = t.GetFullPath()
|
|
|
|
|
2024-10-22 08:35:55 +02:00
|
|
|
cmd.Env = getEnvironmentVars()
|
2024-06-12 21:53:00 +02:00
|
|
|
cmd.Env = append(cmd.Env, fmt.Sprintf("HOME=%s", util.Config.TmpPath))
|
|
|
|
cmd.Env = append(cmd.Env, fmt.Sprintf("PWD=%s", cmd.Dir))
|
|
|
|
|
|
|
|
if environmentVars != nil {
|
|
|
|
cmd.Env = append(cmd.Env, *environmentVars...)
|
|
|
|
}
|
|
|
|
|
|
|
|
return cmd
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *TerraformApp) runCmd(command string, args []string) error {
|
|
|
|
cmd := t.makeCmd(command, args, nil)
|
|
|
|
t.Logger.LogCmd(cmd)
|
|
|
|
return cmd.Run()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *TerraformApp) GetFullPath() string {
|
|
|
|
return path.Join(t.Repository.GetFullPath(t.Template.ID), strings.TrimPrefix(t.Template.Playbook, "/"))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (t *TerraformApp) SetLogger(logger task_logger.Logger) task_logger.Logger {
|
2024-06-17 20:37:45 +02:00
|
|
|
t.Logger = logger
|
2024-06-12 21:53:00 +02:00
|
|
|
|
2024-06-17 20:37:45 +02:00
|
|
|
t.Logger.AddLogListener(func(new time.Time, msg string) {
|
|
|
|
if strings.Contains(msg, "No changes.") {
|
|
|
|
t.noChanges = true
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
t.Logger.AddStatusListener(func(status task_logger.TaskStatus) {
|
|
|
|
var result terraformReaderResult
|
|
|
|
|
|
|
|
switch status {
|
|
|
|
case task_logger.TaskConfirmed:
|
|
|
|
result = terraformReaderConfirmed
|
|
|
|
t.reader.result = &result
|
|
|
|
case task_logger.TaskFailStatus, task_logger.TaskStoppedStatus:
|
|
|
|
result = terraformReaderFailed
|
|
|
|
t.reader.result = &result
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
return logger
|
2024-06-12 21:53:00 +02:00
|
|
|
}
|
|
|
|
|
2024-10-10 13:36:59 +02:00
|
|
|
func (t *TerraformApp) init(environmentVars *[]string) error {
|
|
|
|
cmd := t.makeCmd(t.Name, []string{"init"}, environmentVars)
|
2024-06-17 20:37:45 +02:00
|
|
|
t.Logger.LogCmd(cmd)
|
|
|
|
err := cmd.Start()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2024-06-12 21:53:00 +02:00
|
|
|
}
|
2024-07-17 16:58:20 +02:00
|
|
|
|
2024-06-17 20:37:45 +02:00
|
|
|
return cmd.Wait()
|
|
|
|
}
|
2024-06-12 21:53:00 +02:00
|
|
|
|
2024-10-10 13:36:59 +02:00
|
|
|
func (t *TerraformApp) selectWorkspace(workspace string, environmentVars *[]string) error {
|
|
|
|
cmd := t.makeCmd(string(t.Name), []string{"workspace", "select", "-or-create=true", workspace}, environmentVars)
|
2024-07-17 16:58:20 +02:00
|
|
|
t.Logger.LogCmd(cmd)
|
|
|
|
err := cmd.Start()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return cmd.Wait()
|
|
|
|
}
|
|
|
|
|
2024-10-10 13:36:59 +02:00
|
|
|
func (t *TerraformApp) InstallRequirements(environmentVars *[]string) (err error) {
|
|
|
|
err = t.init(environmentVars)
|
2024-07-17 16:58:20 +02:00
|
|
|
if err != nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
workspace := "default"
|
|
|
|
|
|
|
|
if t.Inventory.Inventory != "" {
|
|
|
|
workspace = t.Inventory.Inventory
|
|
|
|
}
|
|
|
|
|
2024-10-10 13:36:59 +02:00
|
|
|
err = t.selectWorkspace(workspace, environmentVars)
|
2024-07-17 16:58:20 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2024-06-17 20:37:45 +02:00
|
|
|
func (t *TerraformApp) Plan(args []string, environmentVars *[]string, inputs map[string]string, cb func(*os.Process)) error {
|
|
|
|
args = append([]string{"plan"}, args...)
|
2024-07-17 17:29:12 +02:00
|
|
|
cmd := t.makeCmd(t.Name, args, environmentVars)
|
2024-06-12 21:53:00 +02:00
|
|
|
t.Logger.LogCmd(cmd)
|
2024-06-17 20:37:45 +02:00
|
|
|
cmd.Stdin = strings.NewReader("")
|
2024-06-12 21:53:00 +02:00
|
|
|
err := cmd.Start()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2024-06-17 20:37:45 +02:00
|
|
|
cb(cmd.Process)
|
2024-06-12 21:53:00 +02:00
|
|
|
return cmd.Wait()
|
|
|
|
}
|
|
|
|
|
2024-06-17 20:37:45 +02:00
|
|
|
func (t *TerraformApp) Apply(args []string, environmentVars *[]string, inputs map[string]string, cb func(*os.Process)) error {
|
|
|
|
args = append([]string{"apply", "-auto-approve"}, args...)
|
2024-07-17 17:29:12 +02:00
|
|
|
cmd := t.makeCmd(t.Name, args, environmentVars)
|
2024-06-12 21:53:00 +02:00
|
|
|
t.Logger.LogCmd(cmd)
|
2024-06-17 20:37:45 +02:00
|
|
|
cmd.Stdin = strings.NewReader("")
|
2024-06-12 21:53:00 +02:00
|
|
|
err := cmd.Start()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
cb(cmd.Process)
|
|
|
|
return cmd.Wait()
|
|
|
|
}
|
2024-06-17 20:37:45 +02:00
|
|
|
|
|
|
|
func (t *TerraformApp) Run(args []string, environmentVars *[]string, inputs map[string]string, cb func(*os.Process)) error {
|
|
|
|
err := t.Plan(args, environmentVars, inputs, cb)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if t.noChanges {
|
|
|
|
t.Logger.SetStatus(task_logger.TaskSuccessStatus)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
t.Logger.SetStatus(task_logger.TaskWaitingConfirmation)
|
|
|
|
|
|
|
|
for {
|
|
|
|
time.Sleep(time.Second * 3)
|
|
|
|
if t.reader.result != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
switch *t.reader.result {
|
|
|
|
case terraformReaderFailed:
|
|
|
|
return nil
|
|
|
|
case terraformReaderConfirmed:
|
|
|
|
t.Logger.SetStatus(task_logger.TaskRunningStatus)
|
|
|
|
return t.Apply(args, environmentVars, inputs, cb)
|
|
|
|
default:
|
|
|
|
return fmt.Errorf("unknown plan result")
|
|
|
|
}
|
|
|
|
}
|