2017-03-19 15:18:18 -03:00
|
|
|
package task
|
|
|
|
|
|
|
|
import (
|
2022-12-17 07:31:00 -06:00
|
|
|
"context"
|
|
|
|
"encoding/json"
|
2017-03-19 15:18:18 -03:00
|
|
|
"fmt"
|
2021-07-01 16:05:46 -05:00
|
|
|
"io"
|
|
|
|
"os"
|
|
|
|
"strings"
|
2017-03-19 15:18:18 -03:00
|
|
|
"text/tabwriter"
|
2018-02-17 14:22:18 -02:00
|
|
|
|
2023-03-17 12:34:06 +00:00
|
|
|
"golang.org/x/sync/errgroup"
|
|
|
|
|
2022-12-17 07:31:00 -06:00
|
|
|
"github.com/go-task/task/v3/internal/editors"
|
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/logger"
|
2023-04-06 12:07:57 +01:00
|
|
|
"github.com/go-task/task/v3/internal/sort"
|
2023-12-29 20:32:03 +00:00
|
|
|
"github.com/go-task/task/v3/taskfile/ast"
|
2017-03-19 15:18:18 -03:00
|
|
|
)
|
|
|
|
|
2022-12-17 07:31:00 -06:00
|
|
|
// ListOptions collects list-related options
|
|
|
|
type ListOptions struct {
|
|
|
|
ListOnlyTasksWithDescriptions bool
|
|
|
|
ListAllTasks bool
|
|
|
|
FormatTaskListAsJSON bool
|
2023-11-15 20:31:02 -05:00
|
|
|
NoStatus bool
|
2022-12-17 07:31:00 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewListOptions creates a new ListOptions instance
|
2023-11-15 20:31:02 -05:00
|
|
|
func NewListOptions(list, listAll, listAsJson, noStatus bool) ListOptions {
|
2022-12-17 07:31:00 -06:00
|
|
|
return ListOptions{
|
|
|
|
ListOnlyTasksWithDescriptions: list,
|
|
|
|
ListAllTasks: listAll,
|
|
|
|
FormatTaskListAsJSON: listAsJson,
|
2023-11-15 20:31:02 -05:00
|
|
|
NoStatus: noStatus,
|
2022-12-17 07:31:00 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ShouldListTasks returns true if one of the options to list tasks has been set to true
|
|
|
|
func (o ListOptions) ShouldListTasks() bool {
|
|
|
|
return o.ListOnlyTasksWithDescriptions || o.ListAllTasks
|
|
|
|
}
|
|
|
|
|
|
|
|
// Validate validates that the collection of list-related options are in a valid configuration
|
|
|
|
func (o ListOptions) Validate() error {
|
|
|
|
if o.ListOnlyTasksWithDescriptions && o.ListAllTasks {
|
|
|
|
return fmt.Errorf("task: cannot use --list and --list-all at the same time")
|
|
|
|
}
|
|
|
|
if o.FormatTaskListAsJSON && !o.ShouldListTasks() {
|
|
|
|
return fmt.Errorf("task: --json only applies to --list or --list-all")
|
|
|
|
}
|
2023-11-15 20:31:02 -05:00
|
|
|
if o.NoStatus && !o.FormatTaskListAsJSON {
|
|
|
|
return fmt.Errorf("task: --no-status only applies to --json with --list or --list-all")
|
|
|
|
}
|
2022-12-17 07:31:00 -06:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Filters returns the slice of FilterFunc which filters a list
|
2023-12-29 20:32:03 +00:00
|
|
|
// of ast.Task according to the given ListOptions
|
2022-12-17 07:31:00 -06:00
|
|
|
func (o ListOptions) Filters() []FilterFunc {
|
2023-01-14 13:45:52 -06:00
|
|
|
filters := []FilterFunc{FilterOutInternal}
|
2022-12-17 07:31:00 -06:00
|
|
|
|
|
|
|
if o.ListOnlyTasksWithDescriptions {
|
2023-01-14 13:45:52 -06:00
|
|
|
filters = append(filters, FilterOutNoDesc)
|
2022-12-17 07:31:00 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return filters
|
|
|
|
}
|
|
|
|
|
2022-11-02 14:38:26 +00:00
|
|
|
// ListTasks prints a list of tasks.
|
|
|
|
// Tasks that match the given filters will be excluded from the list.
|
2022-12-17 07:31:00 -06:00
|
|
|
// The function returns a boolean indicating whether tasks were found
|
|
|
|
// and an error if one was encountered while preparing the output.
|
|
|
|
func (e *Executor) ListTasks(o ListOptions) (bool, error) {
|
2023-01-14 13:45:52 -06:00
|
|
|
tasks, err := e.GetTaskList(o.Filters()...)
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
2022-12-17 07:31:00 -06:00
|
|
|
if o.FormatTaskListAsJSON {
|
2023-11-15 20:31:02 -05:00
|
|
|
output, err := e.ToEditorOutput(tasks, o.NoStatus)
|
2022-12-17 07:31:00 -06:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
encoder := json.NewEncoder(e.Stdout)
|
|
|
|
encoder.SetIndent("", " ")
|
|
|
|
if err := encoder.Encode(output); err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return len(tasks) > 0, nil
|
|
|
|
}
|
2017-03-19 15:18:18 -03:00
|
|
|
if len(tasks) == 0 {
|
2022-12-17 07:31:00 -06:00
|
|
|
if o.ListOnlyTasksWithDescriptions {
|
2023-04-27 01:20:06 +01:00
|
|
|
e.Logger.Outf(logger.Yellow, "task: No tasks with description available. Try --list-all to list all tasks\n")
|
2022-12-17 07:31:00 -06:00
|
|
|
} else if o.ListAllTasks {
|
2023-04-27 01:20:06 +01:00
|
|
|
e.Logger.Outf(logger.Yellow, "task: No tasks available\n")
|
2022-12-17 07:31:00 -06:00
|
|
|
}
|
|
|
|
return false, nil
|
2017-03-19 15:18:18 -03:00
|
|
|
}
|
2023-04-27 01:20:06 +01:00
|
|
|
e.Logger.Outf(logger.Default, "task: Available tasks for this project:\n")
|
2017-03-19 15:18:18 -03:00
|
|
|
|
|
|
|
// Format in tab-separated columns with a tab stop of 8.
|
2022-09-29 17:13:12 +00:00
|
|
|
w := tabwriter.NewWriter(e.Stdout, 0, 8, 6, ' ', 0)
|
2017-03-19 15:18:18 -03:00
|
|
|
for _, task := range tasks {
|
2023-05-06 23:04:20 +01:00
|
|
|
e.Logger.FOutf(w, logger.Yellow, "* ")
|
|
|
|
e.Logger.FOutf(w, logger.Green, task.Task)
|
|
|
|
e.Logger.FOutf(w, logger.Default, ": \t%s", task.Desc)
|
2022-10-14 19:28:05 -03:00
|
|
|
if len(task.Aliases) > 0 {
|
2023-05-06 23:04:20 +01:00
|
|
|
e.Logger.FOutf(w, logger.Cyan, "\t(aliases: %s)", strings.Join(task.Aliases, ", "))
|
2022-10-14 19:28:05 -03:00
|
|
|
}
|
2023-05-06 23:04:20 +01:00
|
|
|
_, _ = fmt.Fprint(w, "\n")
|
2017-03-19 15:18:18 -03:00
|
|
|
}
|
2022-12-17 07:31:00 -06:00
|
|
|
if err := w.Flush(); err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
return true, nil
|
2017-03-19 15:18:18 -03:00
|
|
|
}
|
2021-07-01 16:05:46 -05:00
|
|
|
|
2022-10-02 18:49:38 +03:00
|
|
|
// ListTaskNames prints only the task names in a Taskfile.
|
2022-03-21 12:59:25 -05:00
|
|
|
// Only tasks with a non-empty description are printed if allTasks is false.
|
|
|
|
// Otherwise, all task names are printed.
|
2024-02-19 21:15:51 +00:00
|
|
|
func (e *Executor) ListTaskNames(allTasks bool) error {
|
2024-02-19 21:01:40 +00:00
|
|
|
// if called from cmd/task.go, e.Logger has not yet been initialized
|
|
|
|
if e.Logger == nil {
|
|
|
|
e.setupLogger()
|
|
|
|
}
|
2021-07-01 16:05:46 -05:00
|
|
|
// if called from cmd/task.go, e.Taskfile has not yet been parsed
|
2022-03-31 21:37:06 -03:00
|
|
|
if e.Taskfile == nil {
|
|
|
|
if err := e.readTaskfile(); err != nil {
|
2024-02-19 21:15:51 +00:00
|
|
|
return err
|
2022-03-31 21:37:06 -03:00
|
|
|
}
|
2021-07-01 16:05:46 -05:00
|
|
|
}
|
2022-03-21 12:59:25 -05:00
|
|
|
// use stdout if no output defined
|
2021-07-01 16:05:46 -05:00
|
|
|
var w io.Writer = os.Stdout
|
2022-03-21 12:59:25 -05:00
|
|
|
if e.Stdout != nil {
|
2021-07-01 16:05:46 -05:00
|
|
|
w = e.Stdout
|
|
|
|
}
|
2023-04-06 12:07:57 +01:00
|
|
|
|
|
|
|
// Get the list of tasks and sort them
|
|
|
|
tasks := e.Taskfile.Tasks.Values()
|
|
|
|
|
|
|
|
// Sort the tasks
|
|
|
|
if e.TaskSorter == nil {
|
|
|
|
e.TaskSorter = &sort.AlphaNumericWithRootTasksFirst{}
|
|
|
|
}
|
|
|
|
e.TaskSorter.Sort(tasks)
|
|
|
|
|
|
|
|
// Create a list of task names
|
|
|
|
taskNames := make([]string, 0, e.Taskfile.Tasks.Len())
|
|
|
|
for _, task := range tasks {
|
|
|
|
if (allTasks || task.Desc != "") && !task.Internal {
|
|
|
|
taskNames = append(taskNames, strings.TrimRight(task.Task, ":"))
|
|
|
|
for _, alias := range task.Aliases {
|
|
|
|
taskNames = append(taskNames, strings.TrimRight(alias, ":"))
|
2022-10-31 17:16:12 +00:00
|
|
|
}
|
2022-03-21 12:59:25 -05:00
|
|
|
}
|
2021-07-01 16:05:46 -05:00
|
|
|
}
|
2023-04-06 12:07:57 +01:00
|
|
|
for _, t := range taskNames {
|
2022-03-21 12:59:25 -05:00
|
|
|
fmt.Fprintln(w, t)
|
2021-07-01 16:05:46 -05:00
|
|
|
}
|
2024-02-19 21:15:51 +00:00
|
|
|
return nil
|
2021-07-01 16:05:46 -05:00
|
|
|
}
|
2022-12-17 07:31:00 -06:00
|
|
|
|
2023-12-29 20:32:03 +00:00
|
|
|
func (e *Executor) ToEditorOutput(tasks []*ast.Task, noStatus bool) (*editors.Taskfile, error) {
|
2023-03-17 12:34:06 +00:00
|
|
|
o := &editors.Taskfile{
|
|
|
|
Tasks: make([]editors.Task, len(tasks)),
|
|
|
|
Location: e.Taskfile.Location,
|
2022-12-17 07:31:00 -06:00
|
|
|
}
|
2023-03-17 12:34:06 +00:00
|
|
|
var g errgroup.Group
|
|
|
|
for i := range tasks {
|
|
|
|
task := tasks[i]
|
|
|
|
j := i
|
2023-12-18 09:45:41 +00:00
|
|
|
aliases := []string{}
|
|
|
|
if len(task.Aliases) > 0 {
|
|
|
|
aliases = task.Aliases
|
|
|
|
}
|
2023-11-15 22:38:53 -03:00
|
|
|
g.Go(func() error {
|
|
|
|
o.Tasks[j] = editors.Task{
|
|
|
|
Name: task.Name(),
|
|
|
|
Desc: task.Desc,
|
|
|
|
Summary: task.Summary,
|
2023-12-18 09:45:41 +00:00
|
|
|
Aliases: aliases,
|
2023-11-15 22:38:53 -03:00
|
|
|
UpToDate: false,
|
|
|
|
Location: &editors.Location{
|
|
|
|
Line: task.Location.Line,
|
|
|
|
Column: task.Location.Column,
|
|
|
|
Taskfile: task.Location.Taskfile,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
if noStatus {
|
2023-11-15 20:31:02 -05:00
|
|
|
return nil
|
2023-11-15 22:38:53 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get the fingerprinting method to use
|
|
|
|
method := e.Taskfile.Method
|
|
|
|
if task.Method != "" {
|
|
|
|
method = task.Method
|
|
|
|
}
|
|
|
|
upToDate, err := fingerprint.IsTaskUpToDate(context.Background(), task,
|
|
|
|
fingerprint.WithMethod(method),
|
|
|
|
fingerprint.WithTempDir(e.TempDir),
|
|
|
|
fingerprint.WithDry(e.Dry),
|
|
|
|
fingerprint.WithLogger(e.Logger),
|
|
|
|
)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
o.Tasks[j].UpToDate = upToDate
|
|
|
|
|
|
|
|
return nil
|
|
|
|
})
|
2022-12-17 07:31:00 -06:00
|
|
|
}
|
2023-03-17 12:34:06 +00:00
|
|
|
return o, g.Wait()
|
2022-12-17 07:31:00 -06:00
|
|
|
}
|