2020-05-16 20:45:41 +02:00
|
|
|
package v3
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
|
2020-08-16 20:48:19 +02:00
|
|
|
"github.com/go-task/task/v3/internal/compiler"
|
|
|
|
"github.com/go-task/task/v3/internal/execext"
|
2022-08-06 23:19:07 +02:00
|
|
|
"github.com/go-task/task/v3/internal/filepathext"
|
2020-08-16 20:48:19 +02:00
|
|
|
"github.com/go-task/task/v3/internal/logger"
|
|
|
|
"github.com/go-task/task/v3/internal/templater"
|
2023-02-20 03:28:17 +02:00
|
|
|
"github.com/go-task/task/v3/internal/version"
|
2020-08-19 10:59:58 +02:00
|
|
|
"github.com/go-task/task/v3/taskfile"
|
2020-05-16 20:45:41 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
var _ compiler.Compiler = &CompilerV3{}
|
|
|
|
|
|
|
|
type CompilerV3 struct {
|
2022-12-06 02:58:20 +02:00
|
|
|
Dir string
|
|
|
|
UserWorkingDir string
|
2020-05-16 20:45:41 +02:00
|
|
|
|
2021-01-12 16:09:46 +02:00
|
|
|
TaskfileEnv *taskfile.Vars
|
2020-05-16 20:45:41 +02:00
|
|
|
TaskfileVars *taskfile.Vars
|
|
|
|
|
|
|
|
Logger *logger.Logger
|
|
|
|
|
|
|
|
dynamicCache map[string]string
|
|
|
|
muDynamicCache sync.Mutex
|
|
|
|
}
|
|
|
|
|
2021-06-05 20:54:10 +02:00
|
|
|
func (c *CompilerV3) GetTaskfileVariables() (*taskfile.Vars, error) {
|
|
|
|
return c.getVariables(nil, nil, true)
|
|
|
|
}
|
|
|
|
|
2020-05-16 20:45:41 +02:00
|
|
|
func (c *CompilerV3) GetVariables(t *taskfile.Task, call taskfile.Call) (*taskfile.Vars, error) {
|
2021-06-05 20:54:10 +02:00
|
|
|
return c.getVariables(t, &call, true)
|
2021-01-12 17:03:04 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *CompilerV3) FastGetVariables(t *taskfile.Task, call taskfile.Call) (*taskfile.Vars, error) {
|
2021-06-05 20:54:10 +02:00
|
|
|
return c.getVariables(t, &call, false)
|
2021-01-12 17:03:04 +02:00
|
|
|
}
|
|
|
|
|
2021-06-05 20:54:10 +02:00
|
|
|
func (c *CompilerV3) getVariables(t *taskfile.Task, call *taskfile.Call, evaluateShVars bool) (*taskfile.Vars, error) {
|
2020-05-16 20:45:41 +02:00
|
|
|
result := compiler.GetEnviron()
|
2021-06-05 20:54:10 +02:00
|
|
|
if t != nil {
|
2022-09-03 23:14:54 +02:00
|
|
|
specialVars, err := c.getSpecialVars(t)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for k, v := range specialVars {
|
2023-11-28 20:18:28 +02:00
|
|
|
result.Set(k, taskfile.Var{Value: v})
|
2022-09-03 23:14:54 +02:00
|
|
|
}
|
2021-06-05 20:54:10 +02:00
|
|
|
}
|
2020-05-16 20:45:41 +02:00
|
|
|
|
2021-01-09 17:09:23 +02:00
|
|
|
getRangeFunc := func(dir string) func(k string, v taskfile.Var) error {
|
|
|
|
return func(k string, v taskfile.Var) error {
|
|
|
|
tr := templater.Templater{Vars: result, RemoveNoValue: true}
|
2023-11-29 18:21:21 +02:00
|
|
|
// Replace values
|
|
|
|
newVar := taskfile.Var{}
|
|
|
|
switch value := v.Value.(type) {
|
|
|
|
case string:
|
|
|
|
newVar.Value = tr.Replace(value)
|
|
|
|
default:
|
|
|
|
newVar.Value = value
|
2021-01-09 17:09:23 +02:00
|
|
|
}
|
2023-11-29 18:21:21 +02:00
|
|
|
newVar.Sh = tr.Replace(v.Sh)
|
|
|
|
newVar.Dir = v.Dir
|
2021-01-09 17:09:23 +02:00
|
|
|
if err := tr.Err(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-12-17 04:44:03 +02:00
|
|
|
// If the variable should not be evaluated, but is nil, set it to an empty string
|
|
|
|
// This stops empty interface errors when using the templater to replace values later
|
|
|
|
if !evaluateShVars && newVar.Value == nil {
|
|
|
|
result.Set(k, taskfile.Var{Value: ""})
|
|
|
|
return nil
|
|
|
|
}
|
2023-11-29 18:21:21 +02:00
|
|
|
// If the variable is not dynamic, we can set it and return
|
|
|
|
if !evaluateShVars || newVar.Value != nil || newVar.Sh == "" {
|
|
|
|
result.Set(k, taskfile.Var{Value: newVar.Value})
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// If the variable is dynamic, we need to resolve it first
|
|
|
|
static, err := c.HandleDynamicVar(newVar, dir)
|
2021-01-09 17:09:23 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-11-28 20:18:28 +02:00
|
|
|
result.Set(k, taskfile.Var{Value: static})
|
2021-01-09 17:09:23 +02:00
|
|
|
return nil
|
2020-05-16 20:45:41 +02:00
|
|
|
}
|
|
|
|
}
|
2021-01-09 17:09:23 +02:00
|
|
|
rangeFunc := getRangeFunc(c.Dir)
|
2020-05-16 20:45:41 +02:00
|
|
|
|
2022-03-19 23:41:03 +02:00
|
|
|
var taskRangeFunc func(k string, v taskfile.Var) error
|
|
|
|
if t != nil {
|
|
|
|
// NOTE(@andreynering): We're manually joining these paths here because
|
|
|
|
// this is the raw task, not the compiled one.
|
|
|
|
tr := templater.Templater{Vars: result, RemoveNoValue: true}
|
|
|
|
dir := tr.Replace(t.Dir)
|
|
|
|
if err := tr.Err(); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-08-06 23:19:07 +02:00
|
|
|
dir = filepathext.SmartJoin(c.Dir, dir)
|
2022-03-19 23:41:03 +02:00
|
|
|
taskRangeFunc = getRangeFunc(dir)
|
|
|
|
}
|
|
|
|
|
2021-01-12 16:09:46 +02:00
|
|
|
if err := c.TaskfileEnv.Range(rangeFunc); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-05-16 20:45:41 +02:00
|
|
|
if err := c.TaskfileVars.Range(rangeFunc); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-03-19 23:41:03 +02:00
|
|
|
if t != nil {
|
|
|
|
if err := t.IncludedTaskfileVars.Range(taskRangeFunc); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if err := t.IncludeVars.Range(rangeFunc); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
2021-06-05 20:54:10 +02:00
|
|
|
|
|
|
|
if t == nil || call == nil {
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
2020-05-16 20:45:41 +02:00
|
|
|
if err := call.Vars.Range(rangeFunc); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-01-09 17:09:23 +02:00
|
|
|
if err := t.Vars.Range(taskRangeFunc); err != nil {
|
2020-05-16 20:45:41 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
2021-01-07 16:17:38 +02:00
|
|
|
func (c *CompilerV3) HandleDynamicVar(v taskfile.Var, dir string) (string, error) {
|
2020-05-16 20:45:41 +02:00
|
|
|
c.muDynamicCache.Lock()
|
|
|
|
defer c.muDynamicCache.Unlock()
|
|
|
|
|
|
|
|
if c.dynamicCache == nil {
|
|
|
|
c.dynamicCache = make(map[string]string, 30)
|
|
|
|
}
|
|
|
|
if result, ok := c.dynamicCache[v.Sh]; ok {
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
2021-01-09 17:09:23 +02:00
|
|
|
// NOTE(@andreynering): If a var have a specific dir, use this instead
|
|
|
|
if v.Dir != "" {
|
|
|
|
dir = v.Dir
|
|
|
|
}
|
|
|
|
|
2020-05-16 20:45:41 +02:00
|
|
|
var stdout bytes.Buffer
|
|
|
|
opts := &execext.RunCommandOptions{
|
|
|
|
Command: v.Sh,
|
2021-01-07 16:17:38 +02:00
|
|
|
Dir: dir,
|
2020-05-16 20:45:41 +02:00
|
|
|
Stdout: &stdout,
|
|
|
|
Stderr: c.Logger.Stderr,
|
|
|
|
}
|
|
|
|
if err := execext.RunCommand(context.Background(), opts); err != nil {
|
2021-07-11 00:33:36 +02:00
|
|
|
return "", fmt.Errorf(`task: Command "%s" failed: %s`, opts.Command, err)
|
2020-05-16 20:45:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Trim a single trailing newline from the result to make most command
|
|
|
|
// output easier to use in shell commands.
|
2022-05-08 22:33:54 +02:00
|
|
|
result := strings.TrimSuffix(stdout.String(), "\r\n")
|
|
|
|
result = strings.TrimSuffix(result, "\n")
|
2020-05-16 20:45:41 +02:00
|
|
|
|
|
|
|
c.dynamicCache[v.Sh] = result
|
2023-04-27 02:20:06 +02:00
|
|
|
c.Logger.VerboseErrf(logger.Magenta, "task: dynamic variable: %q result: %q\n", v.Sh, result)
|
2020-05-16 20:45:41 +02:00
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|
2021-01-05 16:19:34 +02:00
|
|
|
|
|
|
|
// ResetCache clear the dymanic variables cache
|
|
|
|
func (c *CompilerV3) ResetCache() {
|
|
|
|
c.muDynamicCache.Lock()
|
|
|
|
defer c.muDynamicCache.Unlock()
|
|
|
|
|
|
|
|
c.dynamicCache = nil
|
|
|
|
}
|
2022-09-03 23:14:54 +02:00
|
|
|
|
|
|
|
func (c *CompilerV3) getSpecialVars(t *taskfile.Task) (map[string]string, error) {
|
|
|
|
taskfileDir, err := c.getTaskfileDir(t)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return map[string]string{
|
2022-12-06 02:58:20 +02:00
|
|
|
"TASK": t.Task,
|
|
|
|
"ROOT_DIR": c.Dir,
|
|
|
|
"TASKFILE_DIR": taskfileDir,
|
|
|
|
"USER_WORKING_DIR": c.UserWorkingDir,
|
2023-02-20 03:28:17 +02:00
|
|
|
"TASK_VERSION": version.GetVersion(),
|
2022-09-03 23:14:54 +02:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *CompilerV3) getTaskfileDir(t *taskfile.Task) (string, error) {
|
|
|
|
if t.IncludedTaskfile != nil {
|
|
|
|
return t.IncludedTaskfile.FullDirPath()
|
|
|
|
}
|
|
|
|
return c.Dir, nil
|
|
|
|
}
|