diff --git a/precondition.go b/precondition.go index 1f25bd37f5..b1f22cc6ba 100644 --- a/precondition.go +++ b/precondition.go @@ -14,7 +14,7 @@ import ( var ErrPreconditionFailed = errors.New("task: precondition not met") func (e *Executor) areTaskPreconditionsMet(ctx context.Context, t *ast.Task) (bool, error) { - for _, p := range t.Preconditions { + for _, p := range append(t.Preconditions, e.Taskfile.Preconditions.Preconditions...) { err := execext.RunCommand(ctx, &execext.RunCommandOptions{ Command: p.Sh, Dir: t.Dir, diff --git a/task_test.go b/task_test.go index 8920460b32..6b06c39f14 100644 --- a/task_test.go +++ b/task_test.go @@ -443,10 +443,10 @@ func TestStatus(t *testing.T) { buff.Reset() } -func TestPrecondition(t *testing.T) { +func TestPreconditionLocal(t *testing.T) { t.Parallel() - const dir = "testdata/precondition" + const dir = "testdata/precondition/local" var buff bytes.Buffer e := &task.Executor{ @@ -486,6 +486,48 @@ func TestPrecondition(t *testing.T) { buff.Reset() } +func TestPreconditionGlobal(t *testing.T) { + t.Parallel() + + var buff bytes.Buffer + e := &task.Executor{ + Dir: "testdata/precondition/global", + Stdout: &buff, + Stderr: &buff, + } + + require.NoError(t, e.Setup()) + + // A global precondition that was not met + require.Error(t, e.Run(context.Background(), &ast.Call{Task: "impossible"})) + + if buff.String() != "task: 1 != 0 obviously!\n" { + t.Errorf("Wrong output message: %s", buff.String()) + } + buff.Reset() + + e = &task.Executor{ + Dir: "testdata/precondition/global/with_local", + Stdout: &buff, + Stderr: &buff, + } + + require.NoError(t, e.Setup()) + + // A global precondition that was met + require.NoError(t, e.Run(context.Background(), &ast.Call{Task: "foo"})) + if buff.String() != "" { + t.Errorf("Got Output when none was expected: %s", buff.String()) + } + + // A local precondition that was not met + require.Error(t, e.Run(context.Background(), &ast.Call{Task: "impossible"})) + + if buff.String() != "task: 1 != 0 obviously!\n" { + t.Errorf("Wrong output message: %s", buff.String()) + } +} + func TestGenerates(t *testing.T) { t.Parallel() diff --git a/taskfile/ast/precondition.go b/taskfile/ast/precondition.go index 275144c917..69ca5f004a 100644 --- a/taskfile/ast/precondition.go +++ b/taskfile/ast/precondition.go @@ -2,16 +2,56 @@ package ast import ( "fmt" - - "gopkg.in/yaml.v3" + "sync" "github.com/go-task/task/v3/errors" + "github.com/go-task/task/v3/internal/deepcopy" + + "gopkg.in/yaml.v3" ) // Precondition represents a precondition necessary for a task to run -type Precondition struct { - Sh string - Msg string +type ( + Preconditions struct { + Preconditions []*Precondition + mutex sync.RWMutex + } + + Precondition struct { + Sh string + Msg string + } +) + +func NewPreconditions() *Preconditions { + return &Preconditions{ + Preconditions: make([]*Precondition, 0), + } +} + +func (p *Preconditions) DeepCopy() *Preconditions { + if p == nil { + return nil + } + defer p.mutex.RUnlock() + p.mutex.RLock() + return &Preconditions{ + Preconditions: deepcopy.Slice(p.Preconditions), + } +} + +func (p *Preconditions) Merge(other *Preconditions) { + if p == nil || p.Preconditions == nil || other == nil { + return + } + + p.mutex.Lock() + defer p.mutex.Unlock() + + other.mutex.RLock() + defer other.mutex.RUnlock() + + p.Preconditions = append(p.Preconditions, deepcopy.Slice(other.Preconditions)...) } func (p *Precondition) DeepCopy() *Precondition { @@ -55,3 +95,15 @@ func (p *Precondition) UnmarshalYAML(node *yaml.Node) error { return errors.NewTaskfileDecodeError(nil, node).WithTypeMessage("precondition") } + +func (p *Preconditions) UnmarshalYAML(node *yaml.Node) error { + if p == nil || p.Preconditions == nil { + *p = *NewPreconditions() + } + + if err := node.Decode(&p.Preconditions); err != nil { + return errors.NewTaskfileDecodeError(err, node).WithTypeMessage("preconditions") + } + + return nil +} diff --git a/taskfile/ast/taskfile.go b/taskfile/ast/taskfile.go index 4aad932da7..988ae21613 100644 --- a/taskfile/ast/taskfile.go +++ b/taskfile/ast/taskfile.go @@ -20,20 +20,21 @@ var ErrIncludedTaskfilesCantHaveDotenvs = errors.New("task: Included Taskfiles c // Taskfile is the abstract syntax tree for a Taskfile type Taskfile struct { - Location string - Version *semver.Version - Output Output - Method string - Includes *Includes - Set []string - Shopt []string - Vars *Vars - Env *Vars - Tasks *Tasks - Silent bool - Dotenv []string - Run string - Interval time.Duration + Location string + Version *semver.Version + Output Output + Method string + Includes *Includes + Set []string + Shopt []string + Vars *Vars + Env *Vars + Preconditions *Preconditions + Tasks *Tasks + Silent bool + Dotenv []string + Run string + Interval time.Duration } // Merge merges the second Taskfile into the first @@ -59,8 +60,12 @@ func (t1 *Taskfile) Merge(t2 *Taskfile, include *Include) error { if t1.Tasks == nil { t1.Tasks = NewTasks() } + if t1.Preconditions == nil { + t1.Preconditions = NewPreconditions() + } t1.Vars.Merge(t2.Vars, include) t1.Env.Merge(t2.Env, include) + t1.Preconditions.Merge(t2.Preconditions) return t1.Tasks.Merge(t2.Tasks, include, t1.Vars) } @@ -68,19 +73,20 @@ func (tf *Taskfile) UnmarshalYAML(node *yaml.Node) error { switch node.Kind { case yaml.MappingNode: var taskfile struct { - Version *semver.Version - Output Output - Method string - Includes *Includes - Set []string - Shopt []string - Vars *Vars - Env *Vars - Tasks *Tasks - Silent bool - Dotenv []string - Run string - Interval time.Duration + Version *semver.Version + Output Output + Method string + Includes *Includes + Preconditions *Preconditions + Set []string + Shopt []string + Vars *Vars + Env *Vars + Tasks *Tasks + Silent bool + Dotenv []string + Run string + Interval time.Duration } if err := node.Decode(&taskfile); err != nil { return errors.NewTaskfileDecodeError(err, node) @@ -98,6 +104,7 @@ func (tf *Taskfile) UnmarshalYAML(node *yaml.Node) error { tf.Dotenv = taskfile.Dotenv tf.Run = taskfile.Run tf.Interval = taskfile.Interval + tf.Preconditions = taskfile.Preconditions if tf.Includes == nil { tf.Includes = NewIncludes() } @@ -110,6 +117,9 @@ func (tf *Taskfile) UnmarshalYAML(node *yaml.Node) error { if tf.Tasks == nil { tf.Tasks = NewTasks() } + if tf.Preconditions == nil { + tf.Preconditions = NewPreconditions() + } return nil } diff --git a/testdata/precondition/global/Taskfile.yml b/testdata/precondition/global/Taskfile.yml new file mode 100644 index 0000000000..cc2e5e5901 --- /dev/null +++ b/testdata/precondition/global/Taskfile.yml @@ -0,0 +1,9 @@ +version: '3' + +preconditions: + - sh: "[ 1 = 0 ]" + msg: "1 != 0 obviously!" + +tasks: + impossible: + cmd: echo "won't run" diff --git a/testdata/precondition/global/with_local/Taskfile.yml b/testdata/precondition/global/with_local/Taskfile.yml new file mode 100644 index 0000000000..d70d809f75 --- /dev/null +++ b/testdata/precondition/global/with_local/Taskfile.yml @@ -0,0 +1,12 @@ +version: '3' + +preconditions: + - test -f foo.txt + +tasks: + foo: + + impossible: + preconditions: + - sh: "[ 1 = 0 ]" + msg: "1 != 0 obviously!" diff --git a/testdata/precondition/foo.txt b/testdata/precondition/global/with_local/foo.txt similarity index 100% rename from testdata/precondition/foo.txt rename to testdata/precondition/global/with_local/foo.txt diff --git a/testdata/precondition/Taskfile.yml b/testdata/precondition/local/Taskfile.yml similarity index 100% rename from testdata/precondition/Taskfile.yml rename to testdata/precondition/local/Taskfile.yml diff --git a/testdata/precondition/local/foo.txt b/testdata/precondition/local/foo.txt new file mode 100644 index 0000000000..e69de29bb2 diff --git a/website/docs/usage.mdx b/website/docs/usage.mdx index d082b446c2..b56713df59 100644 --- a/website/docs/usage.mdx +++ b/website/docs/usage.mdx @@ -1019,6 +1019,21 @@ tasks: - echo "I will not run" ``` +They can be defined at two levels: + +- Global Level: Applies to all tasks. +- Task Level: Applies only to a specific task. + +```yaml +version: '3' + +preconditions: + - sh: 'exit 1' + +tasks: + task-will-fail: echo "I will not run" +``` + ### Limiting when tasks run If a task executed by multiple `cmds` or multiple `deps` you can control when it diff --git a/website/static/schema.json b/website/static/schema.json index da25a209f9..e4785c836b 100644 --- a/website/static/schema.json +++ b/website/static/schema.json @@ -678,6 +678,13 @@ "description": "A set of global environment variables.", "$ref": "#/definitions/env" }, + "preconditions": { + "description": "A list of commands to check if any task should run. If a condition is not met, the task will return an error.", + "type": "array", + "items": { + "$ref": "#/definitions/precondition" + } + }, "tasks": { "description": "A set of task definitions.", "$ref": "#/definitions/tasks"