/*
goredo -- djb's redo implementation on pure Go
-Copyright (C) 2020-2021 Sergey Matveev <stargrave@stargrave.org>
+Copyright (C) 2020-2023 Sergey Matveev <stargrave@stargrave.org>
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
"flag"
"fmt"
"io"
+ "io/fs"
"os"
"path"
- "path/filepath"
"sort"
"strconv"
"strings"
func parseBuildLogRec(dir, tgt string) (map[string][]string, error) {
fd, err := os.Open(path.Join(dir, RedoDir, tgt+LogRecSuffix))
if err != nil {
- return nil, err
+ return nil, ErrLine(err)
}
r := recfile.NewReader(bufio.NewReader(fd))
rec, err := r.NextMapWithSlice()
fd.Close()
- return rec, err
+ return rec, ErrLine(err)
}
func depthPrefix(depth int) string {
}
func showBuildLogSub(sub *BuildLogJob, depth int) error {
- abs, err := filepath.Abs(path.Join(sub.dir, sub.tgt))
- if err != nil {
- return err
- }
+ abs := mustAbs(path.Join(sub.dir, sub.tgt))
if _, ok := buildLogSeen[abs]; ok {
return nil
}
}
durationSec, durationNsec, err := durationToInts(sub.rec["Duration"][0])
if err != nil {
- return err
+ return ErrLine(err)
}
if sub.exitCode > 0 {
fmt.Printf(
func showBuildLogCmd(m map[string][]string, depth int) error {
started, err := tai64n.Decode(m["Started"][0])
if err != nil {
- return err
+ return ErrLine(err)
}
dp := depthPrefix(depth)
fmt.Printf(
}
finished, err := tai64n.Decode(m["Finished"][0])
if err != nil {
- return err
+ return ErrLine(err)
}
durationSec, durationNsec, err := durationToInts(m["Duration"][0])
if err != nil {
- return err
+ return ErrLine(err)
}
fmt.Printf(
"%s%sStarted:\t%s\n%s%sFinished:\t%s\n%s%sDuration:\t%d.%ds\n\n",
}
fd, err := os.Open(path.Join(dirNormalized, RedoDir, tgtNormalized+LogSuffix))
if err != nil {
- return err
+ return ErrLine(err)
}
if !*flagBuildLogRecursive {
w := bufio.NewWriter(os.Stdout)
fd.Close()
if err != nil {
w.Flush()
- return err
+ return ErrLine(err)
}
- return w.Flush()
+ return ErrLine(w.Flush())
}
defer fd.Close()
subs := make([]*BuildLogJob, 0, len(buildLogRec["Ifchange"]))
}
rec, err := parseBuildLogRec(subDir, subTgt)
if err != nil {
- if os.IsNotExist(err) {
+ if errors.Is(err, fs.ErrNotExist) {
continue
}
return err
}
started, err := tai64n.Decode(rec["Started"][0])
if err != nil {
- return err
+ return ErrLine(err)
}
var exitCode int
if len(rec["ExitCode"]) > 0 {
exitCode, err = strconv.Atoi(rec["ExitCode"][0])
if err != nil {
- return err
+ return ErrLine(err)
}
}
subs = append(subs, &BuildLogJob{
for {
if !scanner.Scan() {
if err = scanner.Err(); err != nil {
- return err
+ return ErrLine(err)
}
break
}
text = scanner.Text()
if text[0] != '@' {
- return errors.New("unexpected non-TAI64Ned string")
+ return ErrLine(errors.New("unexpected non-TAI64Ned string"))
}
sep = strings.IndexByte(text, byte(' '))
if sep == -1 {
}
t, err = tai64n.Decode(text[1:sep])
if err != nil {
- return err
+ return ErrLine(err)
}
for sub != nil && t.After(sub.started) {
if err = showBuildLogSub(sub, depth); err != nil {