2017-03-02 10:46:20 +01:00
|
|
|
package task
|
|
|
|
|
|
|
|
import (
|
2022-12-06 00:25:16 +00:00
|
|
|
"os"
|
2019-06-11 11:49:37 -07:00
|
|
|
"strings"
|
2017-03-02 11:28:34 +01:00
|
|
|
|
2022-12-06 00:25:16 +00:00
|
|
|
"github.com/joho/godotenv"
|
|
|
|
|
2020-08-16 15:48:19 -03:00
|
|
|
"github.com/go-task/task/v3/internal/execext"
|
2022-08-06 18:19:07 -03:00
|
|
|
"github.com/go-task/task/v3/internal/filepathext"
|
2023-03-10 18:27:30 +00:00
|
|
|
"github.com/go-task/task/v3/internal/fingerprint"
|
2020-08-16 15:48:19 -03:00
|
|
|
"github.com/go-task/task/v3/internal/templater"
|
2020-08-19 10:59:58 +02:00
|
|
|
"github.com/go-task/task/v3/taskfile"
|
2017-03-02 10:46:20 +01:00
|
|
|
)
|
|
|
|
|
2017-09-03 12:48:06 +02:00
|
|
|
// CompiledTask returns a copy of a task, but replacing variables in almost all
|
|
|
|
// properties using the Go template package.
|
2018-02-17 14:22:18 -02:00
|
|
|
func (e *Executor) CompiledTask(call taskfile.Call) (*taskfile.Task, error) {
|
2021-01-12 12:03:04 -03:00
|
|
|
return e.compiledTask(call, true)
|
|
|
|
}
|
|
|
|
|
|
|
|
// FastCompiledTask is like CompiledTask, but it skippes dynamic variables.
|
|
|
|
func (e *Executor) FastCompiledTask(call taskfile.Call) (*taskfile.Task, error) {
|
|
|
|
return e.compiledTask(call, false)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *Executor) compiledTask(call taskfile.Call, evaluateShVars bool) (*taskfile.Task, error) {
|
2022-10-01 22:39:44 +00:00
|
|
|
origTask, err := e.GetTask(call)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2017-08-16 13:04:58 +02:00
|
|
|
}
|
|
|
|
|
2021-01-12 12:03:04 -03:00
|
|
|
var vars *taskfile.Vars
|
|
|
|
if evaluateShVars {
|
|
|
|
vars, err = e.Compiler.GetVariables(origTask, call)
|
|
|
|
} else {
|
|
|
|
vars, err = e.Compiler.FastGetVariables(origTask, call)
|
|
|
|
}
|
2017-11-02 10:25:50 -02:00
|
|
|
if err != nil {
|
2017-08-16 13:04:58 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
2019-06-11 11:49:37 -07:00
|
|
|
|
2023-02-08 10:21:43 +00:00
|
|
|
r := templater.Templater{Vars: vars, RemoveNoValue: e.Taskfile.Version.Compare(taskfile.V3) >= 0}
|
2017-07-20 09:05:37 +02:00
|
|
|
|
2018-02-17 14:22:18 -02:00
|
|
|
new := taskfile.Task{
|
2022-03-19 18:41:03 -03:00
|
|
|
Task: origTask.Task,
|
|
|
|
Label: r.Replace(origTask.Label),
|
|
|
|
Desc: r.Replace(origTask.Desc),
|
|
|
|
Summary: r.Replace(origTask.Summary),
|
2022-10-01 22:39:44 +00:00
|
|
|
Aliases: origTask.Aliases,
|
2022-03-19 18:41:03 -03:00
|
|
|
Sources: r.ReplaceSlice(origTask.Sources),
|
|
|
|
Generates: r.ReplaceSlice(origTask.Generates),
|
|
|
|
Dir: r.Replace(origTask.Dir),
|
2023-01-14 13:41:56 -06:00
|
|
|
Set: origTask.Set,
|
|
|
|
Shopt: origTask.Shopt,
|
2022-03-19 18:41:03 -03:00
|
|
|
Vars: nil,
|
|
|
|
Env: nil,
|
2022-12-06 00:25:16 +00:00
|
|
|
Dotenv: r.ReplaceSlice(origTask.Dotenv),
|
2022-03-19 18:41:03 -03:00
|
|
|
Silent: origTask.Silent,
|
|
|
|
Interactive: origTask.Interactive,
|
2022-07-22 02:15:35 +00:00
|
|
|
Internal: origTask.Internal,
|
2022-03-19 18:41:03 -03:00
|
|
|
Method: r.Replace(origTask.Method),
|
|
|
|
Prefix: r.Replace(origTask.Prefix),
|
|
|
|
IgnoreError: origTask.IgnoreError,
|
|
|
|
Run: r.Replace(origTask.Run),
|
|
|
|
IncludeVars: origTask.IncludeVars,
|
|
|
|
IncludedTaskfileVars: origTask.IncludedTaskfileVars,
|
2023-01-07 11:38:35 +11:00
|
|
|
Platforms: origTask.Platforms,
|
2023-03-17 12:34:06 +00:00
|
|
|
Location: origTask.Location,
|
2017-07-16 16:09:55 -03:00
|
|
|
}
|
2018-12-24 15:19:53 -02:00
|
|
|
new.Dir, err = execext.Expand(new.Dir)
|
2017-11-02 10:25:50 -02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-08-06 18:19:07 -03:00
|
|
|
if e.Dir != "" {
|
|
|
|
new.Dir = filepathext.SmartJoin(e.Dir, new.Dir)
|
2017-08-16 13:04:58 +02:00
|
|
|
}
|
2018-04-22 15:41:53 -03:00
|
|
|
if new.Prefix == "" {
|
|
|
|
new.Prefix = new.Task
|
|
|
|
}
|
2019-01-02 12:05:40 -02:00
|
|
|
|
2022-12-06 00:25:16 +00:00
|
|
|
dotenvEnvs := &taskfile.Vars{}
|
|
|
|
if len(new.Dotenv) > 0 {
|
|
|
|
for _, dotEnvPath := range new.Dotenv {
|
|
|
|
dotEnvPath = filepathext.SmartJoin(new.Dir, dotEnvPath)
|
|
|
|
if _, err := os.Stat(dotEnvPath); os.IsNotExist(err) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
envs, err := godotenv.Read(dotEnvPath)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
for key, value := range envs {
|
2023-04-06 12:07:57 +01:00
|
|
|
if ok := dotenvEnvs.Exists(key); !ok {
|
2022-12-06 00:25:16 +00:00
|
|
|
dotenvEnvs.Set(key, taskfile.Var{Static: value})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-29 16:54:59 -03:00
|
|
|
new.Env = &taskfile.Vars{}
|
|
|
|
new.Env.Merge(r.ReplaceVars(e.Taskfile.Env))
|
2022-12-06 00:25:16 +00:00
|
|
|
new.Env.Merge(r.ReplaceVars(dotenvEnvs))
|
2020-03-29 16:54:59 -03:00
|
|
|
new.Env.Merge(r.ReplaceVars(origTask.Env))
|
2021-01-12 12:03:04 -03:00
|
|
|
if evaluateShVars {
|
|
|
|
err = new.Env.Range(func(k string, v taskfile.Var) error {
|
|
|
|
static, err := e.Compiler.HandleDynamicVar(v, new.Dir)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
new.Env.Set(k, taskfile.Var{Static: static})
|
|
|
|
return nil
|
|
|
|
})
|
2017-08-16 13:04:58 +02:00
|
|
|
if err != nil {
|
2021-01-12 12:03:04 -03:00
|
|
|
return nil, err
|
2017-08-16 13:04:58 +02:00
|
|
|
}
|
2017-07-30 19:45:01 -03:00
|
|
|
}
|
2017-07-16 16:09:55 -03:00
|
|
|
|
2017-08-16 13:04:58 +02:00
|
|
|
if len(origTask.Cmds) > 0 {
|
2021-09-15 00:01:33 +09:00
|
|
|
new.Cmds = make([]*taskfile.Cmd, 0, len(origTask.Cmds))
|
|
|
|
for _, cmd := range origTask.Cmds {
|
|
|
|
if cmd == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
new.Cmds = append(new.Cmds, &taskfile.Cmd{
|
2018-07-10 10:44:58 +02:00
|
|
|
Cmd: r.Replace(cmd.Cmd),
|
2023-01-14 13:41:56 -06:00
|
|
|
Silent: cmd.Silent,
|
|
|
|
Task: r.Replace(cmd.Task),
|
|
|
|
Set: cmd.Set,
|
|
|
|
Shopt: cmd.Shopt,
|
2018-07-10 10:44:58 +02:00
|
|
|
Vars: r.ReplaceVars(cmd.Vars),
|
|
|
|
IgnoreError: cmd.IgnoreError,
|
2021-12-15 00:03:37 -05:00
|
|
|
Defer: cmd.Defer,
|
2023-01-07 11:38:35 +11:00
|
|
|
Platforms: cmd.Platforms,
|
2021-09-15 00:01:33 +09:00
|
|
|
})
|
2017-07-16 16:09:55 -03:00
|
|
|
}
|
|
|
|
}
|
2017-08-16 13:04:58 +02:00
|
|
|
if len(origTask.Deps) > 0 {
|
2021-09-15 00:01:33 +09:00
|
|
|
new.Deps = make([]*taskfile.Dep, 0, len(origTask.Deps))
|
|
|
|
for _, dep := range origTask.Deps {
|
|
|
|
if dep == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
new.Deps = append(new.Deps, &taskfile.Dep{
|
2018-02-17 16:12:41 -02:00
|
|
|
Task: r.Replace(dep.Task),
|
|
|
|
Vars: r.ReplaceVars(dep.Vars),
|
2021-09-15 00:01:33 +09:00
|
|
|
})
|
2017-07-16 16:09:55 -03:00
|
|
|
}
|
|
|
|
}
|
2019-05-28 13:02:59 -07:00
|
|
|
|
|
|
|
if len(origTask.Preconditions) > 0 {
|
2021-09-15 00:01:33 +09:00
|
|
|
new.Preconditions = make([]*taskfile.Precondition, 0, len(origTask.Preconditions))
|
|
|
|
for _, precond := range origTask.Preconditions {
|
|
|
|
if precond == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
new.Preconditions = append(new.Preconditions, &taskfile.Precondition{
|
2019-05-28 13:02:59 -07:00
|
|
|
Sh: r.Replace(precond.Sh),
|
|
|
|
Msg: r.Replace(precond.Msg),
|
2021-09-15 00:01:33 +09:00
|
|
|
})
|
2019-05-17 13:13:47 -07:00
|
|
|
}
|
|
|
|
}
|
2017-07-16 16:09:55 -03:00
|
|
|
|
2019-08-25 10:30:00 -07:00
|
|
|
if len(origTask.Status) > 0 {
|
2023-03-10 18:27:30 +00:00
|
|
|
timestampChecker := fingerprint.NewTimestampChecker(e.TempDir, e.Dry)
|
|
|
|
checksumChecker := fingerprint.NewChecksumChecker(e.TempDir, e.Dry)
|
|
|
|
|
|
|
|
for _, checker := range []fingerprint.SourcesCheckable{timestampChecker, checksumChecker} {
|
|
|
|
value, err := checker.Value(&new)
|
2019-09-14 18:04:41 -03:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-03-29 16:54:59 -03:00
|
|
|
vars.Set(strings.ToUpper(checker.Kind()), taskfile.Var{Live: value})
|
2019-08-25 10:30:00 -07:00
|
|
|
}
|
|
|
|
|
2019-08-25 13:16:59 -07:00
|
|
|
// Adding new variables, requires us to refresh the templaters
|
|
|
|
// cache of the the values manually
|
2019-09-14 17:54:41 -03:00
|
|
|
r.ResetCache()
|
2019-08-25 10:30:00 -07:00
|
|
|
|
|
|
|
new.Status = r.ReplaceSlice(origTask.Status)
|
|
|
|
}
|
|
|
|
|
2018-02-17 16:12:41 -02:00
|
|
|
return &new, r.Err()
|
2017-09-03 12:48:06 +02:00
|
|
|
}
|