mirror of
https://github.com/go-task/task.git
synced 2024-12-12 10:45:49 +02:00
247c2586c2
* refactor: ast package * feat: read -> taskfile * refactor: taskfile.Taskfile -> taskfile.Read * refactor: move merge function back into taskfile package * refactor: rename taskfile.go to read.go
152 lines
3.4 KiB
Go
152 lines
3.4 KiB
Go
package fingerprint
|
|
|
|
import (
|
|
"os"
|
|
"path/filepath"
|
|
"time"
|
|
|
|
"github.com/go-task/task/v3/taskfile/ast"
|
|
)
|
|
|
|
// TimestampChecker checks if any source change compared with the generated files,
|
|
// using file modifications timestamps.
|
|
type TimestampChecker struct {
|
|
tempDir string
|
|
dry bool
|
|
}
|
|
|
|
func NewTimestampChecker(tempDir string, dry bool) *TimestampChecker {
|
|
return &TimestampChecker{
|
|
tempDir: tempDir,
|
|
dry: dry,
|
|
}
|
|
}
|
|
|
|
// IsUpToDate implements the Checker interface
|
|
func (checker *TimestampChecker) IsUpToDate(t *ast.Task) (bool, error) {
|
|
if len(t.Sources) == 0 {
|
|
return false, nil
|
|
}
|
|
|
|
sources, err := Globs(t.Dir, t.Sources)
|
|
if err != nil {
|
|
return false, nil
|
|
}
|
|
generates, err := Globs(t.Dir, t.Generates)
|
|
if err != nil {
|
|
return false, nil
|
|
}
|
|
|
|
timestampFile := checker.timestampFilePath(t)
|
|
|
|
// If the file exists, add the file path to the generates.
|
|
// If the generate file is old, the task will be executed.
|
|
_, err = os.Stat(timestampFile)
|
|
if err == nil {
|
|
generates = append(generates, timestampFile)
|
|
} else {
|
|
// Create the timestamp file for the next execution when the file does not exist.
|
|
if !checker.dry {
|
|
if err := os.MkdirAll(filepath.Dir(timestampFile), 0o755); err != nil {
|
|
return false, err
|
|
}
|
|
f, err := os.Create(timestampFile)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
f.Close()
|
|
}
|
|
}
|
|
|
|
taskTime := time.Now()
|
|
|
|
// Compare the time of the generates and sources. If the generates are old, the task will be executed.
|
|
|
|
// Get the max time of the generates.
|
|
generateMaxTime, err := getMaxTime(generates...)
|
|
if err != nil || generateMaxTime.IsZero() {
|
|
return false, nil
|
|
}
|
|
|
|
// Check if any of the source files is newer than the max time of the generates.
|
|
shouldUpdate, err := anyFileNewerThan(sources, generateMaxTime)
|
|
if err != nil {
|
|
return false, nil
|
|
}
|
|
|
|
// Modify the metadata of the file to the the current time.
|
|
if !checker.dry {
|
|
if err := os.Chtimes(timestampFile, taskTime, taskTime); err != nil {
|
|
return false, err
|
|
}
|
|
}
|
|
|
|
return !shouldUpdate, nil
|
|
}
|
|
|
|
func (checker *TimestampChecker) Kind() string {
|
|
return "timestamp"
|
|
}
|
|
|
|
// Value implements the Checker Interface
|
|
func (checker *TimestampChecker) Value(t *ast.Task) (any, error) {
|
|
sources, err := Globs(t.Dir, t.Sources)
|
|
if err != nil {
|
|
return time.Now(), err
|
|
}
|
|
|
|
sourcesMaxTime, err := getMaxTime(sources...)
|
|
if err != nil {
|
|
return time.Now(), err
|
|
}
|
|
|
|
if sourcesMaxTime.IsZero() {
|
|
return time.Unix(0, 0), nil
|
|
}
|
|
|
|
return sourcesMaxTime, nil
|
|
}
|
|
|
|
func getMaxTime(files ...string) (time.Time, error) {
|
|
var t time.Time
|
|
for _, f := range files {
|
|
info, err := os.Stat(f)
|
|
if err != nil {
|
|
return time.Time{}, err
|
|
}
|
|
t = maxTime(t, info.ModTime())
|
|
}
|
|
return t, nil
|
|
}
|
|
|
|
func maxTime(a, b time.Time) time.Time {
|
|
if a.After(b) {
|
|
return a
|
|
}
|
|
return b
|
|
}
|
|
|
|
// If the modification time of any of the files is newer than the the given time, returns true.
|
|
// This function is lazy, as it stops when it finds a file newer than the given time.
|
|
func anyFileNewerThan(files []string, givenTime time.Time) (bool, error) {
|
|
for _, f := range files {
|
|
info, err := os.Stat(f)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
if info.ModTime().After(givenTime) {
|
|
return true, nil
|
|
}
|
|
}
|
|
return false, nil
|
|
}
|
|
|
|
// OnError implements the Checker interface
|
|
func (*TimestampChecker) OnError(t *ast.Task) error {
|
|
return nil
|
|
}
|
|
|
|
func (checker *TimestampChecker) timestampFilePath(t *ast.Task) string {
|
|
return filepath.Join(checker.tempDir, "timestamp", normalizeFilename(t.Task))
|
|
}
|