1
0
mirror of https://github.com/go-task/task.git synced 2025-01-18 04:59:01 +02:00
task/task.go

495 lines
12 KiB
Go
Raw Normal View History

2017-02-27 09:48:50 -03:00
package task
2017-02-26 20:43:50 -03:00
import (
"bufio"
"context"
"fmt"
"io"
2017-02-26 20:43:50 -03:00
"os"
"runtime"
"strings"
2019-06-15 21:12:54 -03:00
"sync"
"sync/atomic"
2022-12-31 10:48:49 -06:00
"time"
2017-02-26 20:43:50 -03:00
2023-04-15 21:22:25 +01:00
"github.com/go-task/task/v3/errors"
"github.com/go-task/task/v3/internal/compiler"
"github.com/go-task/task/v3/internal/env"
"github.com/go-task/task/v3/internal/execext"
"github.com/go-task/task/v3/internal/fingerprint"
"github.com/go-task/task/v3/internal/logger"
"github.com/go-task/task/v3/internal/output"
"github.com/go-task/task/v3/internal/slicesext"
"github.com/go-task/task/v3/internal/sort"
"github.com/go-task/task/v3/internal/summary"
"github.com/go-task/task/v3/internal/templater"
"github.com/go-task/task/v3/internal/term"
"github.com/go-task/task/v3/taskfile"
2017-03-12 17:18:59 -03:00
"github.com/sajari/fuzzy"
"golang.org/x/exp/slices"
"golang.org/x/sync/errgroup"
2017-02-26 20:43:50 -03:00
)
const (
// MaximumTaskCall is the max number of times a task can be called.
// This exists to prevent infinite loops on cyclic dependencies
MaximumTaskCall = 100
)
2017-02-26 21:18:53 -03:00
func shouldPromptContinue(input string) bool {
input = strings.ToLower(strings.TrimSpace(input))
return slices.Contains([]string{"y", "yes"}, input)
}
// Executor executes a Taskfile
type Executor struct {
Taskfile *taskfile.Taskfile
2020-06-12 12:09:53 -06:00
Dir string
TempDir string
2020-06-12 12:09:53 -06:00
Entrypoint string
Force bool
ForceAll bool
2020-06-12 12:09:53 -06:00
Watch bool
Verbose bool
Silent bool
AssumeYes bool
2020-06-12 12:09:53 -06:00
Dry bool
Summary bool
Parallel bool
Color bool
Concurrency int
2022-12-31 10:48:49 -06:00
Interval time.Duration
AssumesTerm bool
Stdin io.Reader
Stdout io.Writer
Stderr io.Writer
2019-02-09 10:15:38 -02:00
Logger *logger.Logger
Compiler compiler.Compiler
Output output.Output
2022-02-19 19:31:27 -03:00
OutputStyle taskfile.Output
TaskSorter sort.TaskSorter
taskvars *taskfile.Vars
fuzzyModel *fuzzy.Model
2017-07-05 21:03:59 -03:00
2020-06-12 12:09:53 -06:00
concurrencySemaphore chan struct{}
taskCallCount map[string]*int32
mkdirMutexMap map[string]*sync.Mutex
2021-07-28 14:39:00 -06:00
executionHashes map[string]context.Context
executionHashesMutex sync.Mutex
}
2017-02-26 20:43:50 -03:00
2017-02-28 09:50:40 -03:00
// Run runs Task
2019-02-09 10:16:13 -02:00
func (e *Executor) Run(ctx context.Context, calls ...taskfile.Call) error {
// check if given tasks exist
for _, call := range calls {
task, err := e.GetTask(call)
if err != nil {
2023-04-15 21:22:25 +01:00
if _, ok := err.(*errors.TaskNotFoundError); ok {
2022-12-17 10:35:30 -03:00
if _, err := e.ListTasks(ListOptions{ListOnlyTasksWithDescriptions: true}); err != nil {
return err
}
}
return err
}
if task.Internal {
2023-04-15 21:22:25 +01:00
if _, ok := err.(*errors.TaskNotFoundError); ok {
2022-12-17 10:35:30 -03:00
if _, err := e.ListTasks(ListOptions{ListOnlyTasksWithDescriptions: true}); err != nil {
return err
}
}
2023-04-15 21:22:25 +01:00
return &errors.TaskInternalError{TaskName: call.Task}
}
}
2019-02-24 15:33:09 +01:00
if e.Summary {
for i, c := range calls {
compiledTask, err := e.FastCompiledTask(c)
if err != nil {
return nil
}
summary.PrintSpaceBetweenSummaries(e.Logger, i)
summary.PrintTask(e.Logger, compiledTask)
}
2019-02-24 09:24:57 +01:00
return nil
}
if e.Watch {
return e.watchTasks(calls...)
}
g, ctx := errgroup.WithContext(ctx)
for _, c := range calls {
c := c
if e.Parallel {
g.Go(func() error { return e.RunTask(ctx, c) })
} else {
if err := e.RunTask(ctx, c); err != nil {
return err
}
}
}
return g.Wait()
}
2017-02-28 09:50:40 -03:00
// RunTask runs a task by its name
func (e *Executor) RunTask(ctx context.Context, call taskfile.Call) error {
t, err := e.CompiledTask(call)
if err != nil {
return err
}
if !e.Watch && atomic.AddInt32(e.taskCallCount[t.Task], 1) >= MaximumTaskCall {
2023-04-15 21:22:25 +01:00
return &errors.TaskCalledTooManyTimesError{TaskName: t.Task}
}
2020-06-12 12:09:53 -06:00
release := e.acquireConcurrencyLimit()
defer release()
if t.Prompt != "" && !e.AssumeYes {
if !e.AssumesTerm && !term.IsTerminal() {
return &errors.TaskCancelledNoTerminalError{TaskName: call.Task}
}
e.Logger.Outf(logger.Yellow, "task: %q [y/N]\n", t.Prompt)
reader := bufio.NewReader(e.Stdin)
userInput, err := reader.ReadString('\n')
if err != nil {
return err
}
userInput = strings.ToLower(strings.TrimSpace(userInput))
if !shouldPromptContinue(userInput) {
return &errors.TaskCancelledByUserError{TaskName: call.Task}
}
}
2021-07-28 14:39:00 -06:00
return e.startExecution(ctx, t, func(ctx context.Context) error {
if !shouldRunOnCurrentPlatform(t.Platforms) {
e.Logger.VerboseOutf(logger.Yellow, `task: %q not for current platform - ignored\n`, call.Task)
return nil
}
e.Logger.VerboseErrf(logger.Magenta, "task: %q started\n", call.Task)
2021-07-28 14:39:00 -06:00
if err := e.runDeps(ctx, t); err != nil {
return err
}
2019-05-17 13:13:47 -07:00
skipFingerprinting := e.ForceAll || (call.Direct && e.Force)
if !skipFingerprinting {
if err := ctx.Err(); err != nil {
return err
}
2021-07-28 14:39:00 -06:00
preCondMet, err := e.areTaskPreconditionsMet(ctx, t)
if err != nil {
return err
}
2019-05-17 13:13:47 -07:00
// Get the fingerprinting method to use
method := e.Taskfile.Method
if t.Method != "" {
method = t.Method
}
upToDate, err := fingerprint.IsTaskUpToDate(ctx, t,
fingerprint.WithMethod(method),
fingerprint.WithTempDir(e.TempDir),
fingerprint.WithDry(e.Dry),
fingerprint.WithLogger(e.Logger),
)
2021-07-28 14:39:00 -06:00
if err != nil {
return err
}
if upToDate && preCondMet {
if e.Verbose || (!call.Silent && !t.Silent && !e.Taskfile.Silent && !e.Silent) {
e.Logger.Errf(logger.Magenta, "task: Task %q is up to date\n", t.Name())
2021-07-28 14:39:00 -06:00
}
return nil
}
}
2017-03-05 10:15:49 +01:00
2021-07-28 14:39:00 -06:00
if err := e.mkdir(t); err != nil {
e.Logger.Errf(logger.Red, "task: cannot make directory %q: %v\n", t.Dir, err)
2021-07-28 14:39:00 -06:00
}
2021-07-28 14:39:00 -06:00
for i := range t.Cmds {
2021-12-15 00:03:37 -05:00
if t.Cmds[i].Defer {
defer e.runDeferred(t, call, i)
continue
}
2021-07-28 14:39:00 -06:00
if err := e.runCommand(ctx, t, call, i); err != nil {
if err2 := e.statusOnError(t); err2 != nil {
e.Logger.VerboseErrf(logger.Yellow, "task: error cleaning status on error: %v\n", err2)
2021-07-28 14:39:00 -06:00
}
2018-08-05 12:53:42 -03:00
2021-07-28 14:39:00 -06:00
if execext.IsExitError(err) && t.IgnoreError {
e.Logger.VerboseErrf(logger.Yellow, "task: task error ignored: %v\n", err)
2021-07-28 14:39:00 -06:00
continue
}
2018-08-05 12:53:42 -03:00
2023-04-15 21:22:25 +01:00
return &errors.TaskRunError{TaskName: t.Task, Err: err}
2021-07-28 14:39:00 -06:00
}
2017-02-26 20:43:50 -03:00
}
e.Logger.VerboseErrf(logger.Magenta, "task: %q finished\n", call.Task)
2021-07-28 14:39:00 -06:00
return nil
})
2017-02-26 20:43:50 -03:00
}
2019-06-15 21:12:54 -03:00
func (e *Executor) mkdir(t *taskfile.Task) error {
if t.Dir == "" {
return nil
}
mutex := e.mkdirMutexMap[t.Task]
mutex.Lock()
defer mutex.Unlock()
if _, err := os.Stat(t.Dir); os.IsNotExist(err) {
2022-08-17 19:37:58 +02:00
if err := os.MkdirAll(t.Dir, 0o755); err != nil {
2019-06-15 21:12:54 -03:00
return err
}
}
return nil
}
func (e *Executor) runDeps(ctx context.Context, t *taskfile.Task) error {
g, ctx := errgroup.WithContext(ctx)
2017-03-15 20:19:29 -03:00
2020-06-12 12:09:53 -06:00
reacquire := e.releaseConcurrencyLimit()
defer reacquire()
2017-03-15 20:19:29 -03:00
for _, d := range t.Deps {
2017-07-02 15:30:50 -03:00
d := d
g.Go(func() error {
err := e.RunTask(ctx, taskfile.Call{Task: d.Task, Vars: d.Vars, Silent: d.Silent})
2019-05-17 13:13:47 -07:00
if err != nil {
2019-05-28 13:02:59 -07:00
return err
2019-05-17 13:13:47 -07:00
}
return nil
})
2017-03-15 20:19:29 -03:00
}
2017-07-08 15:08:44 -03:00
return g.Wait()
2017-03-15 20:19:29 -03:00
}
2021-12-15 00:03:37 -05:00
func (e *Executor) runDeferred(t *taskfile.Task, call taskfile.Call, i int) {
ctx, cancel := context.WithCancel(context.Background())
2021-12-15 00:03:37 -05:00
defer cancel()
2021-12-15 00:03:37 -05:00
if err := e.runCommand(ctx, t, call, i); err != nil {
e.Logger.VerboseErrf(logger.Yellow, "task: ignored error in deferred cmd: %s\n", err.Error())
2021-12-15 00:03:37 -05:00
}
}
func (e *Executor) runCommand(ctx context.Context, t *taskfile.Task, call taskfile.Call, i int) error {
2017-07-02 15:30:50 -03:00
cmd := t.Cmds[i]
2018-06-24 10:29:46 -03:00
switch {
case cmd.Task != "":
2020-06-12 12:09:53 -06:00
reacquire := e.releaseConcurrencyLimit()
defer reacquire()
err := e.RunTask(ctx, taskfile.Call{Task: cmd.Task, Vars: cmd.Vars, Silent: cmd.Silent})
2019-05-17 13:13:47 -07:00
if err != nil {
2019-05-28 13:02:59 -07:00
return err
2019-05-17 13:13:47 -07:00
}
return nil
2018-06-24 10:29:46 -03:00
case cmd.Cmd != "":
if !shouldRunOnCurrentPlatform(cmd.Platforms) {
e.Logger.VerboseOutf(logger.Yellow, "task: [%s] %s not for current platform - ignored\n", t.Name(), cmd.Cmd)
return nil
}
if e.Verbose || (!call.Silent && !cmd.Silent && !t.Silent && !e.Taskfile.Silent && !e.Silent) {
e.Logger.Errf(logger.Green, "task: [%s] %s\n", t.Name(), cmd.Cmd)
2018-06-24 10:29:46 -03:00
}
2017-03-25 15:26:42 -03:00
2018-08-05 11:28:02 -03:00
if e.Dry {
return nil
}
outputWrapper := e.Output
if t.Interactive {
outputWrapper = output.Interleaved{}
}
vars, err := e.Compiler.FastGetVariables(t, call)
outputTemplater := &templater.Templater{Vars: vars, RemoveNoValue: true}
if err != nil {
return fmt.Errorf("task: failed to get variables: %w", err)
}
stdOut, stdErr, close := outputWrapper.WrapWriter(e.Stdout, e.Stderr, t.Prefix, outputTemplater)
2018-06-24 10:29:46 -03:00
err = execext.RunCommand(ctx, &execext.RunCommandOptions{
Command: cmd.Cmd,
Dir: t.Dir,
Env: env.Get(t),
PosixOpts: slicesext.UniqueJoin(e.Taskfile.Set, t.Set, cmd.Set),
BashOpts: slicesext.UniqueJoin(e.Taskfile.Shopt, t.Shopt, cmd.Shopt),
Stdin: e.Stdin,
Stdout: stdOut,
Stderr: stdErr,
2018-06-24 10:29:46 -03:00
})
2023-03-08 22:37:04 -03:00
if closeErr := close(err); closeErr != nil {
e.Logger.Errf(logger.Red, "task: unable to close writer: %v\n", closeErr)
2023-03-08 22:37:04 -03:00
}
2018-09-01 11:02:23 -03:00
if execext.IsExitError(err) && cmd.IgnoreError {
e.Logger.VerboseErrf(logger.Yellow, "task: [%s] command error ignored: %v\n", t.Name(), err)
return nil
}
return err
2018-06-24 10:29:46 -03:00
default:
return nil
2017-09-16 14:05:07 -03:00
}
}
2021-07-28 14:39:00 -06:00
func (e *Executor) startExecution(ctx context.Context, t *taskfile.Task, execute func(ctx context.Context) error) error {
2020-08-17 13:25:17 -06:00
h, err := e.GetHash(t)
if err != nil {
2021-07-28 14:39:00 -06:00
return err
2020-08-17 13:25:17 -06:00
}
if h == "" {
2021-07-28 14:39:00 -06:00
return execute(ctx)
2020-08-17 13:25:17 -06:00
}
2021-07-28 14:39:00 -06:00
e.executionHashesMutex.Lock()
if otherExecutionCtx, ok := e.executionHashes[h]; ok {
e.executionHashesMutex.Unlock()
e.Logger.VerboseErrf(logger.Magenta, "task: skipping execution of task: %s\n", h)
// Release our execution slot to avoid blocking other tasks while we wait
reacquire := e.releaseConcurrencyLimit()
defer reacquire()
2021-07-28 14:39:00 -06:00
<-otherExecutionCtx.Done()
return nil
2020-08-17 13:25:17 -06:00
}
2021-07-28 14:39:00 -06:00
ctx, cancel := context.WithCancel(ctx)
defer cancel()
2021-07-28 14:39:00 -06:00
e.executionHashes[h] = ctx
e.executionHashesMutex.Unlock()
2021-07-28 14:39:00 -06:00
return execute(ctx)
2020-08-17 13:25:17 -06:00
}
// GetTask will return the task with the name matching the given call from the taskfile.
// If no task is found, it will search for tasks with a matching alias.
// If multiple tasks contain the same alias or no matches are found an error is returned.
func (e *Executor) GetTask(call taskfile.Call) (*taskfile.Task, error) {
// Search for a matching task
matchingTask := e.Taskfile.Tasks.Get(call.Task)
if matchingTask != nil {
return matchingTask, nil
}
// If didn't find one, search for a task with a matching alias
var aliasedTasks []string
for _, task := range e.Taskfile.Tasks.Values() {
if slices.Contains(task.Aliases, call.Task) {
aliasedTasks = append(aliasedTasks, task.Task)
matchingTask = task
}
}
// If we found multiple tasks
if len(aliasedTasks) > 1 {
2023-04-15 21:22:25 +01:00
return nil, &errors.TaskNameConflictError{
AliasName: call.Task,
TaskNames: aliasedTasks,
}
}
// If we found no tasks
if len(aliasedTasks) == 0 {
didYouMean := ""
if e.fuzzyModel != nil {
didYouMean = e.fuzzyModel.SpellCheck(call.Task)
}
2023-04-15 21:22:25 +01:00
return nil, &errors.TaskNotFoundError{
TaskName: call.Task,
DidYouMean: didYouMean,
}
}
return matchingTask, nil
}
2023-01-14 13:45:52 -06:00
type FilterFunc func(task *taskfile.Task) bool
2023-01-14 13:45:52 -06:00
func (e *Executor) GetTaskList(filters ...FilterFunc) ([]*taskfile.Task, error) {
tasks := make([]*taskfile.Task, 0, e.Taskfile.Tasks.Len())
2023-01-14 13:45:52 -06:00
// Create an error group to wait for each task to be compiled
var g errgroup.Group
// Filter tasks based on the given filter functions
for _, task := range e.Taskfile.Tasks.Values() {
var shouldFilter bool
for _, filter := range filters {
if filter(task) {
shouldFilter = true
2023-01-14 13:45:52 -06:00
}
}
if !shouldFilter {
tasks = append(tasks, task)
}
}
2023-01-14 13:45:52 -06:00
// Compile the list of tasks
for i := range tasks {
task := tasks[i]
g.Go(func() error {
2023-01-14 13:45:52 -06:00
compiledTask, err := e.FastCompiledTask(taskfile.Call{Task: task.Task})
if err == nil {
task = compiledTask
}
task = compiledTask
2023-01-14 13:45:52 -06:00
return nil
})
}
2023-01-14 13:45:52 -06:00
// Wait for all the go routines to finish
if err := g.Wait(); err != nil {
return nil, err
}
// Sort the tasks
if e.TaskSorter == nil {
e.TaskSorter = &sort.AlphaNumericWithRootTasksFirst{}
}
e.TaskSorter.Sort(tasks)
2023-01-14 13:45:52 -06:00
return tasks, nil
}
// FilterOutNoDesc removes all tasks that do not contain a description.
2023-01-14 13:45:52 -06:00
func FilterOutNoDesc(task *taskfile.Task) bool {
return task.Desc == ""
}
// FilterOutInternal removes all tasks that are marked as internal.
2023-01-14 13:45:52 -06:00
func FilterOutInternal(task *taskfile.Task) bool {
return task.Internal
}
func shouldRunOnCurrentPlatform(platforms []*taskfile.Platform) bool {
if len(platforms) == 0 {
return true
}
for _, p := range platforms {
if (p.OS == "" || p.OS == runtime.GOOS) && (p.Arch == "" || p.Arch == runtime.GOARCH) {
return true
}
}
return false
}