"golang.org/x/sys/unix"
)
+const (
+ CmdNameGoredo = "goredo"
+ CmdNameRedo = "redo"
+ CmdNameRedoAffects = "redo-affects"
+ CmdNameRedoAlways = "redo-always"
+ CmdNameRedoCleanup = "redo-cleanup"
+ CmdNameRedoDot = "redo-dot"
+ CmdNameRedoIfchange = "redo-ifchange"
+ CmdNameRedoIfcreate = "redo-ifcreate"
+ CmdNameRedoLog = "redo-log"
+ CmdNameRedoOOD = "redo-ood"
+ CmdNameRedoSources = "redo-sources"
+ CmdNameRedoStamp = "redo-stamp"
+ CmdNameRedoTargets = "redo-targets"
+ CmdNameRedoWhichdo = "redo-whichdo"
+)
+
var (
Cwd string
BuildUUID string
+ IsTopRedo bool // is it the top redo instance
)
func mustSetenv(key, value string) {
return fd
}
+func CmdName() string {
+ return path.Base(os.Args[0])
+}
+
func main() {
version := flag.Bool("version", false, "print version")
warranty := flag.Bool("warranty", false, "print warranty information")
- symlinks := flag.Bool("symlinks", false, "create necessary symlinks in current directory")
+
+ var symlinks *bool
+ cmdName := CmdName()
+ if cmdName == "goredo" {
+ symlinks = flag.Bool("symlinks", false, "create necessary symlinks in current directory")
+ }
flag.Usage = func() { usage(os.Args[0]) }
flag.Parse()
fmt.Println("goredo", Version, "built with", runtime.Version())
return
}
- if *symlinks {
+ if cmdName == CmdNameGoredo && *symlinks {
rc := 0
for _, cmdName := range []string{
- "redo",
- "redo-always",
- "redo-cleanup",
- "redo-dot",
- "redo-ifchange",
- "redo-ifcreate",
- "redo-log",
- "redo-ood",
- "redo-sources",
- "redo-stamp",
- "redo-targets",
- "redo-whichdo",
+ CmdNameRedo,
+ CmdNameRedoAffects,
+ CmdNameRedoAlways,
+ CmdNameRedoCleanup,
+ CmdNameRedoDot,
+ CmdNameRedoIfchange,
+ CmdNameRedoIfcreate,
+ CmdNameRedoLog,
+ CmdNameRedoOOD,
+ CmdNameRedoSources,
+ CmdNameRedoStamp,
+ CmdNameRedoTargets,
+ CmdNameRedoWhichdo,
} {
fmt.Println(os.Args[0], "<-", cmdName)
if err := os.Symlink(os.Args[0], cmdName); err != nil {
}
}
DirPrefix = os.Getenv(EnvDirPrefix)
+ DepCwd = os.Getenv(EnvDepCwd)
- if *flagStderrKeep {
+ if flagStderrKeep != nil && *flagStderrKeep {
mustSetenv(EnvStderrKeep, "1")
}
- if *flagStderrSilent {
+ if flagStderrSilent != nil && *flagStderrSilent {
mustSetenv(EnvStderrSilent, "1")
}
- if *flagNoProgress {
+ if flagNoProgress != nil && *flagNoProgress {
mustSetenv(EnvNoProgress, "1")
}
- if *flagDebug {
+ if flagDebug != nil && *flagDebug {
mustSetenv(EnvDebug, "1")
}
- if *flagLogWait {
+ if flagLogWait != nil && *flagLogWait {
mustSetenv(EnvLogWait, "1")
}
- if *flagLogLock {
+ if flagLogLock != nil && *flagLogLock {
mustSetenv(EnvLogLock, "1")
}
- if *flagLogPid {
+ if flagLogPid != nil && *flagLogPid {
mustSetenv(EnvLogPid, "1")
}
- if *flagLogJS {
+ if flagLogJS != nil && *flagLogJS {
mustSetenv(EnvLogJS, "1")
}
StderrKeep = os.Getenv(EnvStderrKeep) == "1"
MyPid = os.Getpid()
}
var traced bool
- if *flagTraceAll {
+ if flagTraceAll != nil && *flagTraceAll {
mustSetenv(EnvTrace, "1")
}
if os.Getenv(EnvTrace) == "1" {
TracedAll = true
traced = true
- } else {
+ } else if flagTrace != nil {
traced = *flagTrace
}
panic(err)
}
unix.Flock(int(fdLock.Fd()), unix.LOCK_UN)
- OODTgts = make(map[string]struct{})
+ OODTgts = map[string]struct{}{}
for _, tgtRaw := range bytes.Split(tgtsRaw, []byte{0}) {
t := string(tgtRaw)
if t == "" {
tgts := flag.Args()
BuildUUID = os.Getenv(EnvBuildUUID)
+ tgtsWasEmpty := len(tgts) == 0
if BuildUUID == "" {
+ IsTopRedo = true
raw := new([16]byte)
if _, err = io.ReadFull(rand.Reader, raw[:]); err != nil {
log.Fatalln(err)
"%x-%x-%x-%x-%x",
raw[0:4], raw[4:6], raw[6:8], raw[8:10], raw[10:],
)
- if len(tgts) == 0 {
+ if tgtsWasEmpty {
tgts = []string{"all"}
}
}
- statusInit()
+ if cmdName == CmdNameRedo || cmdName == CmdNameRedoIfchange {
+ statusInit()
+ }
for i, tgt := range tgts {
if path.IsAbs(tgt) {
ok := true
err = nil
- cmdName := path.Base(os.Args[0])
trace(
CDebug, "[%s] run: %s %s cwd:%s dirprefix:%s",
BuildUUID, cmdName, tgts, Cwd, DirPrefix,
CmdSwitch:
switch cmdName {
- case "redo":
+ case CmdNameRedo:
for _, tgt := range tgts {
ok, err = ifchange([]string{tgt}, true, traced)
if err != nil || !ok {
break
}
}
- case "redo-ifchange":
+ case CmdNameRedoIfchange:
ok, err = ifchange(tgts, false, traced)
if err == nil {
err = writeDeps(fdDep, tgts)
}
- case "redo-ifcreate":
+ case CmdNameRedoIfcreate:
if fdDep == nil {
log.Fatalln("no", EnvDepFd)
}
break
}
}
- case "redo-always":
+ case CmdNameRedoAlways:
if fdDep == nil {
log.Fatalln("no", EnvDepFd)
}
err = always(fdDep)
- case "redo-cleanup":
+ case CmdNameRedoCleanup:
for _, what := range tgts {
err = cleanupWalker(Cwd, what)
if err != nil {
break
}
}
- case "redo-dot":
+ case CmdNameRedoDot:
err = dotPrint(tgts)
- case "redo-stamp":
+ case CmdNameRedoStamp:
if fdDep == nil {
log.Fatalln("no", EnvDepFd)
}
err = stamp(fdDep, os.Stdin)
- case "redo-log":
+ case CmdNameRedoLog:
if len(tgts) != 1 {
log.Fatalln("single target expected")
}
break
}
_, err = io.Copy(os.Stdout, fd)
- case "redo-whichdo":
+ case CmdNameRedoWhichdo:
if len(tgts) != 1 {
log.Fatalln("single target expected")
}
}
fmt.Println(rel)
}
- case "redo-targets":
- tgts, err = targetsWalker(Cwd)
+ case CmdNameRedoTargets:
+ if tgtsWasEmpty {
+ tgts = []string{Cwd}
+ }
+ tgts, err = targetsWalker(tgts)
+ if err != nil {
+ break
+ }
sort.Strings(tgts)
for _, tgt := range tgts {
fmt.Println(tgt)
}
- case "redo-ood":
- tgts, err = targetsWalker(Cwd)
+ case CmdNameRedoAffects:
+ if tgtsWasEmpty {
+ log.Fatalln("no targets specified")
+ }
+ var tgtsKnown []string
+ tgtsKnown, err = targetsWalker([]string{Cwd})
+ if err != nil {
+ break
+ }
+ deps := map[string]map[string]struct{}{}
+ for _, tgt := range tgtsKnown {
+ collectDeps(Cwd, tgt, 0, deps, true)
+ }
+ seen := map[string]struct{}{}
+ for _, tgt := range tgts {
+ collectWholeDeps(deps[tgt], deps, seen)
+ }
+ tgts := make([]string, 0, len(seen))
+ for dep := range seen {
+ tgts = append(tgts, dep)
+ }
+ sort.Strings(tgts)
+ for _, dep := range tgts {
+ fmt.Println(dep)
+ }
+ case CmdNameRedoOOD:
+ if tgtsWasEmpty {
+ tgts, err = targetsWalker([]string{Cwd})
+ if err != nil {
+ break
+ }
+ }
sort.Strings(tgts)
var ood bool
for _, tgt := range tgts {
fmt.Println(tgt)
}
}
- case "redo-sources":
- tgts, err = sourcesWalker()
+ case CmdNameRedoSources:
+ if tgtsWasEmpty {
+ tgts, err = targetsWalker([]string{Cwd})
+ if err != nil {
+ break
+ }
+ }
sort.Strings(tgts)
- for _, tgt := range tgts {
- fmt.Println(tgt)
+ var srcs []string
+ srcs, err = sourcesWalker(tgts)
+ sort.Strings(srcs)
+ for _, src := range srcs {
+ fmt.Println(src)
}
default:
log.Fatalln("unknown command", cmdName)