2023-12-29 22:32:03 +02:00
|
|
|
package taskfile
|
2023-09-02 22:24:01 +02:00
|
|
|
|
|
|
|
import (
|
2023-09-12 23:42:54 +02:00
|
|
|
"context"
|
|
|
|
"io"
|
2023-09-02 22:24:01 +02:00
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2024-02-13 03:07:00 +02:00
|
|
|
"strings"
|
2023-09-02 22:24:01 +02:00
|
|
|
|
2024-02-13 03:07:00 +02:00
|
|
|
"github.com/go-task/task/v3/internal/execext"
|
2023-09-02 22:24:01 +02:00
|
|
|
"github.com/go-task/task/v3/internal/filepathext"
|
2024-02-13 03:07:00 +02:00
|
|
|
"github.com/go-task/task/v3/internal/logger"
|
2023-09-02 22:24:01 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
// A FileNode is a node that reads a taskfile from the local filesystem.
|
|
|
|
type FileNode struct {
|
2023-09-12 23:42:54 +02:00
|
|
|
*BaseNode
|
2023-09-02 22:24:01 +02:00
|
|
|
Entrypoint string
|
|
|
|
}
|
|
|
|
|
2024-02-13 03:07:00 +02:00
|
|
|
func NewFileNode(l *logger.Logger, entrypoint, dir string, opts ...NodeOption) (*FileNode, error) {
|
|
|
|
var err error
|
2024-03-04 20:00:28 +02:00
|
|
|
base := NewBaseNode(dir, opts...)
|
|
|
|
entrypoint, base.dir, err = resolveFileNodeEntrypointAndDir(l, entrypoint, base.dir)
|
2023-09-02 22:24:01 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &FileNode{
|
2023-09-12 23:42:54 +02:00
|
|
|
BaseNode: base,
|
2024-02-13 03:07:00 +02:00
|
|
|
Entrypoint: entrypoint,
|
2023-09-02 22:24:01 +02:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (node *FileNode) Location() string {
|
2024-02-13 03:07:00 +02:00
|
|
|
return node.Entrypoint
|
2023-09-02 22:24:01 +02:00
|
|
|
}
|
|
|
|
|
2023-09-12 23:42:54 +02:00
|
|
|
func (node *FileNode) Remote() bool {
|
|
|
|
return false
|
|
|
|
}
|
2023-09-02 22:24:01 +02:00
|
|
|
|
2023-09-12 23:42:54 +02:00
|
|
|
func (node *FileNode) Read(ctx context.Context) ([]byte, error) {
|
|
|
|
f, err := os.Open(node.Location())
|
2023-09-02 22:24:01 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
2023-09-12 23:42:54 +02:00
|
|
|
return io.ReadAll(f)
|
2023-09-02 22:24:01 +02:00
|
|
|
}
|
2024-01-25 14:36:31 +02:00
|
|
|
|
2024-02-13 03:07:00 +02:00
|
|
|
// resolveFileNodeEntrypointAndDir resolves checks the values of entrypoint and dir and
|
|
|
|
// populates them with default values if necessary.
|
|
|
|
func resolveFileNodeEntrypointAndDir(l *logger.Logger, entrypoint, dir string) (string, string, error) {
|
|
|
|
var err error
|
|
|
|
if entrypoint != "" {
|
|
|
|
entrypoint, err = Exists(l, entrypoint)
|
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
|
|
|
if dir == "" {
|
|
|
|
dir = filepath.Dir(entrypoint)
|
|
|
|
}
|
|
|
|
return entrypoint, dir, nil
|
|
|
|
}
|
|
|
|
if dir == "" {
|
|
|
|
dir, err = os.Getwd()
|
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
entrypoint, err = ExistsWalk(l, dir)
|
|
|
|
if err != nil {
|
|
|
|
return "", "", err
|
|
|
|
}
|
|
|
|
dir = filepath.Dir(entrypoint)
|
|
|
|
return entrypoint, dir, nil
|
|
|
|
}
|
|
|
|
|
2024-02-13 21:29:28 +02:00
|
|
|
func (node *FileNode) ResolveEntrypoint(entrypoint string) (string, error) {
|
2024-02-13 03:07:00 +02:00
|
|
|
// If the file is remote, we don't need to resolve the path
|
2024-02-13 21:28:42 +02:00
|
|
|
if strings.Contains(entrypoint, "://") {
|
|
|
|
return entrypoint, nil
|
2024-02-13 03:07:00 +02:00
|
|
|
}
|
2024-09-24 19:44:54 +02:00
|
|
|
if strings.HasPrefix(entrypoint, "git") {
|
|
|
|
return entrypoint, nil
|
|
|
|
}
|
2024-02-13 03:07:00 +02:00
|
|
|
|
2024-02-13 21:28:42 +02:00
|
|
|
path, err := execext.Expand(entrypoint)
|
2024-02-13 03:07:00 +02:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
if filepathext.IsAbs(path) {
|
|
|
|
return path, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE: Uses the directory of the entrypoint (Taskfile), not the current working directory
|
|
|
|
// This means that files are included relative to one another
|
|
|
|
entrypointDir := filepath.Dir(node.Entrypoint)
|
|
|
|
return filepathext.SmartJoin(entrypointDir, path), nil
|
|
|
|
}
|
|
|
|
|
2024-02-13 21:29:28 +02:00
|
|
|
func (node *FileNode) ResolveDir(dir string) (string, error) {
|
2024-02-13 21:28:42 +02:00
|
|
|
path, err := execext.Expand(dir)
|
2024-02-13 03:07:00 +02:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
if filepathext.IsAbs(path) {
|
|
|
|
return path, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NOTE: Uses the directory of the entrypoint (Taskfile), not the current working directory
|
|
|
|
// This means that files are included relative to one another
|
|
|
|
entrypointDir := filepath.Dir(node.Entrypoint)
|
|
|
|
return filepathext.SmartJoin(entrypointDir, path), nil
|
2024-01-25 14:36:31 +02:00
|
|
|
}
|
2024-06-28 18:07:43 +02:00
|
|
|
|
|
|
|
func (node *FileNode) FilenameAndLastDir() (string, string) {
|
|
|
|
return "", filepath.Base(node.Entrypoint)
|
|
|
|
}
|