2017-09-16 16:44:13 +02:00
|
|
|
package status
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/md5"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2017-09-30 19:40:11 +02:00
|
|
|
"regexp"
|
2017-09-16 16:44:13 +02:00
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Checksum validades if a task is up to date by calculating its source
|
|
|
|
// files checksum
|
|
|
|
type Checksum struct {
|
2019-08-01 05:51:27 +02:00
|
|
|
Dir string
|
|
|
|
Task string
|
|
|
|
Sources []string
|
|
|
|
Generates []string
|
|
|
|
Dry bool
|
2017-09-16 16:44:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// IsUpToDate implements the Checker interface
|
|
|
|
func (c *Checksum) IsUpToDate() (bool, error) {
|
2017-10-01 20:05:09 +02:00
|
|
|
checksumFile := c.checksumFilePath()
|
2017-09-16 16:44:13 +02:00
|
|
|
|
|
|
|
data, _ := ioutil.ReadFile(checksumFile)
|
|
|
|
oldMd5 := strings.TrimSpace(string(data))
|
|
|
|
|
2019-08-21 06:33:12 +02:00
|
|
|
sources, err := globs(c.Dir, c.Sources)
|
2017-09-16 16:44:13 +02:00
|
|
|
if err != nil {
|
2019-08-21 06:42:19 +02:00
|
|
|
return false, err
|
2019-08-01 05:51:27 +02:00
|
|
|
}
|
2019-08-21 06:42:19 +02:00
|
|
|
|
2017-09-16 16:44:13 +02:00
|
|
|
newMd5, err := c.checksum(sources...)
|
|
|
|
if err != nil {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
|
2019-02-09 14:41:19 +02:00
|
|
|
if !c.Dry {
|
|
|
|
_ = os.MkdirAll(filepath.Join(c.Dir, ".task", "checksum"), 0755)
|
|
|
|
if err = ioutil.WriteFile(checksumFile, []byte(newMd5+"\n"), 0644); err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
2017-09-16 16:44:13 +02:00
|
|
|
}
|
2019-08-21 06:35:16 +02:00
|
|
|
|
|
|
|
if len(c.Generates) != 0 {
|
|
|
|
// For each specified 'generates' field, check whether the files actually exist.
|
|
|
|
for _, g := range c.Generates {
|
|
|
|
generates, err := glob(c.Dir, g)
|
|
|
|
if err != nil {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
if len(generates) == 0 {
|
|
|
|
return false, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-16 16:44:13 +02:00
|
|
|
return oldMd5 == newMd5, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Checksum) checksum(files ...string) (string, error) {
|
|
|
|
h := md5.New()
|
|
|
|
|
|
|
|
for _, f := range files {
|
2019-08-21 06:34:58 +02:00
|
|
|
// also sum the filename, so checksum changes for renaming a file
|
|
|
|
if _, err := io.Copy(h, strings.NewReader(filepath.Base(f))); err != nil {
|
2017-09-16 16:44:13 +02:00
|
|
|
return "", err
|
|
|
|
}
|
2019-08-21 06:34:58 +02:00
|
|
|
f, err := os.Open(f)
|
2017-09-19 19:03:24 +02:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
2017-11-02 14:31:34 +02:00
|
|
|
if _, err = io.Copy(h, f); err != nil {
|
2017-09-16 16:44:13 +02:00
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return fmt.Sprintf("%x", h.Sum(nil)), nil
|
|
|
|
}
|
|
|
|
|
2017-10-01 20:05:09 +02:00
|
|
|
// OnError implements the Checker interface
|
|
|
|
func (c *Checksum) OnError() error {
|
|
|
|
return os.Remove(c.checksumFilePath())
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Checksum) checksumFilePath() string {
|
2017-11-02 14:37:02 +02:00
|
|
|
return filepath.Join(c.Dir, ".task", "checksum", c.normalizeFilename(c.Task))
|
2017-10-01 20:05:09 +02:00
|
|
|
}
|
|
|
|
|
2017-09-30 19:40:11 +02:00
|
|
|
var checksumFilenameRegexp = regexp.MustCompile("[^A-z0-9]")
|
|
|
|
|
|
|
|
// replaces invalid caracters on filenames with "-"
|
|
|
|
func (*Checksum) normalizeFilename(f string) string {
|
|
|
|
return checksumFilenameRegexp.ReplaceAllString(f, "-")
|
|
|
|
}
|