2017-03-02 10:46:20 +01:00
|
|
|
package task
|
|
|
|
|
|
|
|
import (
|
2017-03-05 16:49:44 -03:00
|
|
|
"bytes"
|
2017-03-08 19:03:17 -03:00
|
|
|
"errors"
|
2017-03-02 10:46:20 +01:00
|
|
|
"os"
|
2017-04-16 17:16:56 -03:00
|
|
|
"path/filepath"
|
2017-03-05 16:49:44 -03:00
|
|
|
"runtime"
|
2017-03-02 10:46:20 +01:00
|
|
|
"strings"
|
2017-03-05 16:49:44 -03:00
|
|
|
"text/template"
|
2017-03-02 11:28:34 +01:00
|
|
|
|
2017-03-12 17:18:59 -03:00
|
|
|
"github.com/go-task/task/execext"
|
|
|
|
|
2017-04-16 16:53:11 -03:00
|
|
|
"github.com/Masterminds/sprig"
|
2017-03-02 11:28:34 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2017-03-05 16:21:22 -03:00
|
|
|
// TaskvarsFilePath file containing additional variables
|
|
|
|
TaskvarsFilePath = "Taskvars"
|
2017-03-08 19:03:17 -03:00
|
|
|
// ErrMultilineResultCmd is returned when a command returns multiline result
|
|
|
|
ErrMultilineResultCmd = errors.New("Got multiline result from command")
|
2017-03-02 10:46:20 +01:00
|
|
|
)
|
|
|
|
|
2017-07-08 16:00:17 -03:00
|
|
|
var (
|
|
|
|
templateFuncs template.FuncMap
|
|
|
|
)
|
2017-04-16 16:53:11 -03:00
|
|
|
|
|
|
|
func init() {
|
|
|
|
taskFuncs := template.FuncMap{
|
|
|
|
"OS": func() string { return runtime.GOOS },
|
|
|
|
"ARCH": func() string { return runtime.GOARCH },
|
2017-04-22 15:46:29 -03:00
|
|
|
// historical reasons
|
|
|
|
"IsSH": func() bool { return true },
|
2017-04-16 17:16:56 -03:00
|
|
|
"FromSlash": func(path string) string {
|
|
|
|
return filepath.FromSlash(path)
|
|
|
|
},
|
|
|
|
"ToSlash": func(path string) string {
|
|
|
|
return filepath.ToSlash(path)
|
|
|
|
},
|
2017-05-27 10:52:22 -03:00
|
|
|
"ExeExt": func() string {
|
|
|
|
if runtime.GOOS == "windows" {
|
|
|
|
return ".exe"
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
},
|
2017-04-16 16:53:11 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
templateFuncs = sprig.TxtFuncMap()
|
|
|
|
for k, v := range taskFuncs {
|
|
|
|
templateFuncs[k] = v
|
|
|
|
}
|
2017-03-05 16:49:44 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// ReplaceVariables writes vars into initial string
|
2017-07-08 14:34:17 -03:00
|
|
|
func (e *Executor) ReplaceVariables(initial string, call Call) (string, error) {
|
2017-04-30 19:13:21 -03:00
|
|
|
templ, err := template.New("").Funcs(templateFuncs).Parse(initial)
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
2017-07-08 14:57:12 -03:00
|
|
|
var b bytes.Buffer
|
2017-07-08 16:00:17 -03:00
|
|
|
if err = templ.Execute(&b, call.Vars); err != nil {
|
2017-03-05 16:49:44 -03:00
|
|
|
return "", err
|
2017-03-02 10:46:20 +01:00
|
|
|
}
|
2017-03-05 16:49:44 -03:00
|
|
|
return b.String(), nil
|
2017-03-02 10:46:20 +01:00
|
|
|
}
|
|
|
|
|
2017-07-08 15:13:27 -03:00
|
|
|
// ReplaceSliceVariables writes vars into initial string slice
|
|
|
|
func (e *Executor) ReplaceSliceVariables(initials []string, call Call) ([]string, error) {
|
|
|
|
result := make([]string, len(initials))
|
|
|
|
for i, s := range initials {
|
|
|
|
var err error
|
|
|
|
result[i], err = e.ReplaceVariables(s, call)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *Executor) getVariables(call Call) (Vars, error) {
|
|
|
|
t := e.Tasks[call.Task]
|
|
|
|
|
|
|
|
result := make(Vars, len(t.Vars)+len(e.taskvars)+len(call.Vars))
|
2017-07-08 16:00:17 -03:00
|
|
|
merge := func(vars Vars, runTemplate bool) error {
|
2017-07-08 15:13:27 -03:00
|
|
|
for k, v := range vars {
|
2017-07-08 16:00:17 -03:00
|
|
|
if runTemplate {
|
|
|
|
var err error
|
|
|
|
v, err = e.ReplaceVariables(v, call)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-08 15:13:27 -03:00
|
|
|
v, err := e.handleDynamicVariableContent(v)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2017-07-08 16:00:17 -03:00
|
|
|
|
2017-07-08 15:13:27 -03:00
|
|
|
result[k] = v
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-07-08 16:00:17 -03:00
|
|
|
if err := merge(e.taskvars, false); err != nil {
|
2017-07-08 15:13:27 -03:00
|
|
|
return nil, err
|
|
|
|
}
|
2017-07-08 16:00:17 -03:00
|
|
|
if err := merge(t.Vars, true); err != nil {
|
2017-07-08 15:13:27 -03:00
|
|
|
return nil, err
|
|
|
|
}
|
2017-07-08 16:00:17 -03:00
|
|
|
if err := merge(getEnvironmentVariables(), false); err != nil {
|
2017-07-08 15:13:27 -03:00
|
|
|
return nil, err
|
|
|
|
}
|
2017-07-08 16:00:17 -03:00
|
|
|
if err := merge(call.Vars, false); err != nil {
|
2017-07-08 15:13:27 -03:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
2017-03-02 10:46:20 +01:00
|
|
|
// GetEnvironmentVariables returns environment variables as map
|
2017-07-08 14:46:49 -03:00
|
|
|
func getEnvironmentVariables() Vars {
|
2017-03-25 11:01:44 -03:00
|
|
|
var (
|
|
|
|
env = os.Environ()
|
2017-07-08 14:46:49 -03:00
|
|
|
m = make(Vars, len(env))
|
2017-03-25 11:01:44 -03:00
|
|
|
)
|
|
|
|
|
|
|
|
for _, e := range env {
|
|
|
|
keyVal := strings.SplitN(e, "=", 2)
|
|
|
|
key, val := keyVal[0], keyVal[1]
|
|
|
|
m[key] = val
|
2017-03-02 10:46:20 +01:00
|
|
|
}
|
2017-03-25 11:01:44 -03:00
|
|
|
return m
|
2017-03-02 10:46:20 +01:00
|
|
|
}
|
2017-07-08 15:13:27 -03:00
|
|
|
|
|
|
|
func (e *Executor) handleDynamicVariableContent(value string) (string, error) {
|
|
|
|
if !strings.HasPrefix(value, "$") {
|
|
|
|
return value, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
e.muDynamicCache.Lock()
|
|
|
|
defer e.muDynamicCache.Unlock()
|
|
|
|
if result, ok := e.dynamicCache[value]; ok {
|
|
|
|
return result, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
var stdout bytes.Buffer
|
|
|
|
opts := &execext.RunCommandOptions{
|
|
|
|
Command: strings.TrimPrefix(value, "$"),
|
|
|
|
Dir: e.Dir,
|
|
|
|
Stdout: &stdout,
|
|
|
|
Stderr: e.Stderr,
|
|
|
|
}
|
|
|
|
if err := execext.RunCommand(opts); err != nil {
|
|
|
|
return "", &dynamicVarError{cause: err, cmd: opts.Command}
|
|
|
|
}
|
|
|
|
|
|
|
|
result := strings.TrimSuffix(stdout.String(), "\n")
|
|
|
|
if strings.ContainsRune(result, '\n') {
|
|
|
|
return "", ErrMultilineResultCmd
|
|
|
|
}
|
|
|
|
|
|
|
|
result = strings.TrimSpace(result)
|
|
|
|
e.verbosePrintfln(`task: dynamic variable: "%s", result: "%s"`, value, result)
|
|
|
|
e.dynamicCache[value] = result
|
|
|
|
return result, nil
|
|
|
|
}
|