2017-02-27 16:44:08 -03:00
|
|
|
package task
|
|
|
|
|
|
|
|
import (
|
2017-07-16 16:15:29 -03:00
|
|
|
"context"
|
2017-09-16 11:44:13 -03:00
|
|
|
"fmt"
|
2017-02-27 16:44:08 -03:00
|
|
|
|
2018-11-04 21:23:35 -02:00
|
|
|
"github.com/go-task/task/v2/internal/execext"
|
2019-06-15 22:47:15 -03:00
|
|
|
"github.com/go-task/task/v2/internal/logger"
|
2018-11-04 21:23:35 -02:00
|
|
|
"github.com/go-task/task/v2/internal/status"
|
|
|
|
"github.com/go-task/task/v2/internal/taskfile"
|
2017-02-27 16:44:08 -03:00
|
|
|
)
|
|
|
|
|
2017-12-26 21:43:52 -02:00
|
|
|
// Status returns an error if any the of given tasks is not up-to-date
|
2019-02-09 10:16:13 -02:00
|
|
|
func (e *Executor) Status(ctx context.Context, calls ...taskfile.Call) error {
|
2017-12-26 21:43:52 -02:00
|
|
|
for _, call := range calls {
|
2018-10-06 17:55:23 -03:00
|
|
|
t, err := e.CompiledTask(call)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2017-12-26 21:43:52 -02:00
|
|
|
}
|
2019-02-09 10:41:19 -02:00
|
|
|
isUpToDate, err := e.isTaskUpToDate(ctx, t)
|
2017-12-26 21:43:52 -02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !isUpToDate {
|
|
|
|
return fmt.Errorf(`task: Task "%s" is not up-to-date`, t.Task)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-02-09 10:41:19 -02:00
|
|
|
func (e *Executor) isTaskUpToDate(ctx context.Context, t *taskfile.Task) (bool, error) {
|
2017-07-16 16:15:29 -03:00
|
|
|
if len(t.Status) > 0 {
|
2019-02-09 10:41:19 -02:00
|
|
|
return e.isTaskUpToDateStatus(ctx, t)
|
2017-07-16 16:15:29 -03:00
|
|
|
}
|
2017-09-16 11:44:13 -03:00
|
|
|
|
2019-02-09 10:41:19 -02:00
|
|
|
checker, err := e.getStatusChecker(t)
|
2017-09-16 11:44:13 -03:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return checker.IsUpToDate()
|
|
|
|
}
|
|
|
|
|
2019-02-09 10:41:19 -02:00
|
|
|
func (e *Executor) statusOnError(t *taskfile.Task) error {
|
|
|
|
checker, err := e.getStatusChecker(t)
|
2017-09-16 11:44:13 -03:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return checker.OnError()
|
|
|
|
}
|
|
|
|
|
2019-02-09 10:41:19 -02:00
|
|
|
func (e *Executor) getStatusChecker(t *taskfile.Task) (status.Checker, error) {
|
2019-09-08 22:26:24 -03:00
|
|
|
method := t.Method
|
|
|
|
if method == "" {
|
|
|
|
method = e.Taskfile.Method
|
|
|
|
}
|
|
|
|
switch method {
|
2019-09-08 22:51:56 -03:00
|
|
|
case "timestamp":
|
2017-09-16 11:44:13 -03:00
|
|
|
return &status.Timestamp{
|
|
|
|
Dir: t.Dir,
|
|
|
|
Sources: t.Sources,
|
|
|
|
Generates: t.Generates,
|
|
|
|
}, nil
|
|
|
|
case "checksum":
|
|
|
|
return &status.Checksum{
|
2019-08-01 12:51:27 +09:00
|
|
|
Dir: t.Dir,
|
|
|
|
Task: t.Task,
|
|
|
|
Sources: t.Sources,
|
|
|
|
Generates: t.Generates,
|
|
|
|
Dry: e.Dry,
|
2017-09-16 11:44:13 -03:00
|
|
|
}, nil
|
|
|
|
case "none":
|
|
|
|
return status.None{}, nil
|
|
|
|
default:
|
2019-09-08 22:26:24 -03:00
|
|
|
return nil, fmt.Errorf(`task: invalid method "%s"`, method)
|
2017-09-16 11:44:13 -03:00
|
|
|
}
|
2017-07-16 16:15:29 -03:00
|
|
|
}
|
|
|
|
|
2019-02-09 10:41:19 -02:00
|
|
|
func (e *Executor) isTaskUpToDateStatus(ctx context.Context, t *taskfile.Task) (bool, error) {
|
2017-07-16 16:15:29 -03:00
|
|
|
for _, s := range t.Status {
|
2018-09-01 11:02:23 -03:00
|
|
|
err := execext.RunCommand(ctx, &execext.RunCommandOptions{
|
2017-07-16 16:15:29 -03:00
|
|
|
Command: s,
|
2017-07-30 19:45:01 -03:00
|
|
|
Dir: t.Dir,
|
2018-02-17 14:22:18 -02:00
|
|
|
Env: getEnviron(t),
|
2017-07-16 16:15:29 -03:00
|
|
|
})
|
|
|
|
if err != nil {
|
2019-06-15 22:47:15 -03:00
|
|
|
e.Logger.VerboseOutf(logger.Yellow, "task: status command %s exited non-zero: %s", s, err)
|
2017-07-16 16:15:29 -03:00
|
|
|
return false, nil
|
|
|
|
}
|
2019-06-15 22:47:15 -03:00
|
|
|
e.Logger.VerboseOutf(logger.Yellow, "task: status command %s exited zero", s)
|
2017-07-16 16:15:29 -03:00
|
|
|
}
|
|
|
|
return true, nil
|
|
|
|
}
|