181 lines
4.8 KiB
Go
181 lines
4.8 KiB
Go
package repostatus
|
|
|
|
// does processing on the go.mod and go.sum files
|
|
|
|
import (
|
|
"bufio"
|
|
"errors"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"go.wit.com/log"
|
|
)
|
|
|
|
// Detect a 'Primative' package. Sets the isPrimative flag
|
|
// will return true if the repo is truly not dependent on _anything_ else
|
|
// like spew or lib/widget
|
|
// it assumes go mod ran init and tidy ran without error
|
|
func (rs *RepoStatus) isPrimativeGoMod() (bool, error) {
|
|
// go mod init & go mod tidy ran without errors
|
|
log.Log(REPO, "isPrimativeGoMod()", rs.realPath.String())
|
|
tmp := filepath.Join(rs.realPath.String(), "go.mod")
|
|
gomod, err := os.Open(tmp)
|
|
if err != nil {
|
|
log.Log(REPO, "missing go.mod", rs.realPath.String())
|
|
rs.goConfig = nil
|
|
return false, err
|
|
}
|
|
defer gomod.Close()
|
|
|
|
scanner := bufio.NewScanner(gomod)
|
|
for scanner.Scan() {
|
|
line := strings.TrimSpace(scanner.Text())
|
|
|
|
parts := strings.Split(line, " ")
|
|
log.Log(REPO, " gomod:", parts)
|
|
if len(parts) >= 1 {
|
|
log.Log(REPO, " gomod: part[0] =", parts[0])
|
|
if parts[0] == "require" {
|
|
log.Log(REPO, " should return false here")
|
|
return false, errors.New("go.mod file is not primative")
|
|
}
|
|
|
|
}
|
|
}
|
|
return true, nil
|
|
}
|
|
|
|
// readGoMod reads and parses the go.sum file
|
|
// saves the config information in *Repo.goConfig
|
|
func (rs *RepoStatus) parseGoSum() (bool, error) {
|
|
tmp := filepath.Join(rs.realPath.String(), "go.sum")
|
|
gosum, err := os.Open(tmp)
|
|
if err != nil {
|
|
log.Log(REPO, "missing go.sum", rs.realPath.String())
|
|
rs.goConfig = nil
|
|
return false, err
|
|
}
|
|
defer gosum.Close()
|
|
|
|
var deps GoConfig
|
|
deps = make(GoConfig)
|
|
|
|
scanner := bufio.NewScanner(gosum)
|
|
log.Log(REPO, "gosum:", tmp)
|
|
for scanner.Scan() {
|
|
line := strings.TrimSpace(scanner.Text())
|
|
|
|
parts := strings.Split(line, " ")
|
|
if len(parts) == 3 {
|
|
godep := strings.TrimSpace(parts[0])
|
|
version := strings.TrimSpace(parts[1])
|
|
if strings.HasSuffix(version, "/go.mod") {
|
|
version = strings.TrimSuffix(version, "/go.mod")
|
|
}
|
|
currentversion, ok := deps[godep]
|
|
if ok {
|
|
// only use the first value found in the file?
|
|
// this shouldn't have been possible. this function should
|
|
// only be called from MakeRedomod()
|
|
// todo: make go things a seperate package so this function
|
|
// isn't exported?
|
|
if version != currentversion {
|
|
log.Log(REPOWARN, "\tgo.sum ", godep, "had both", version, currentversion)
|
|
}
|
|
} else {
|
|
deps[godep] = version
|
|
log.Log(REPO, "\t", godep, "=", version)
|
|
}
|
|
} else {
|
|
// I've never seen this happen yet
|
|
return false, errors.New("go.sum invalid: " + line)
|
|
}
|
|
}
|
|
|
|
if err := scanner.Err(); err != nil {
|
|
rs.goConfig = nil
|
|
return false, err
|
|
}
|
|
|
|
rs.goConfig = deps
|
|
return true, nil
|
|
}
|
|
|
|
func (rs *RepoStatus) GoConfig() map[string]string {
|
|
return rs.goConfig
|
|
}
|
|
|
|
// for now, even check cmd.Exit
|
|
func (rs *RepoStatus) strictRun(cmd []string) (bool, error) {
|
|
r := rs.Run(cmd)
|
|
if r.Error != nil {
|
|
log.Log(REPO, "go mod init failed err:", r.Error)
|
|
return false, r.Error
|
|
}
|
|
if r.Exit != 0 {
|
|
log.Log(REPO, "go mod init exit =", r.Exit)
|
|
return false, r.Error
|
|
}
|
|
return true, nil
|
|
}
|
|
|
|
// poor name perhaps. It's because in most of these
|
|
// repos you can also type "make redomod" to do the same thing
|
|
// since it's a Makefile task that is also useful to be able to run
|
|
// from the command line
|
|
func (rs *RepoStatus) MakeRedomod() (bool, error) {
|
|
if rs.ReadOnly() {
|
|
log.Log(REPO, "will not go mod redo read only repos", rs.String())
|
|
return false, errors.New(rs.GoPath() + " is read-only ")
|
|
}
|
|
|
|
// unset the go development ENV var to generate release files
|
|
os.Unsetenv("GO111MODULE")
|
|
if ok, err := rs.strictRun([]string{"rm", "-f", "go.mod", "go.sum"}); !ok {
|
|
log.Log(REPO, "rm go.mod go.sum failed", err)
|
|
return ok, err
|
|
}
|
|
if ok, err := rs.strictRun([]string{"go", "mod", "init", rs.GoPath()}); !ok {
|
|
log.Log(REPO, "go mod init failed", err)
|
|
return ok, err
|
|
}
|
|
if ok, err := rs.strictRun([]string{"go", "mod", "tidy"}); !ok {
|
|
log.Log(REPO, "go mod tidy failed", err)
|
|
return ok, err
|
|
}
|
|
if ok, err := rs.strictRun([]string{"go", "mod", "edit", "-go=1.20"}); !ok {
|
|
log.Log(REPO, "go mod edit failed", err)
|
|
return ok, err
|
|
}
|
|
log.Log(REPO, "MakeRedomod() worked", rs.GoPath())
|
|
|
|
if rs.Exists("go.sum") {
|
|
// return the attempt to parse go.mod & go.sum
|
|
return rs.parseGoSum()
|
|
}
|
|
rs.goConfig = nil
|
|
rs.primitive.SetText("false")
|
|
|
|
ok, err := rs.isPrimativeGoMod()
|
|
if err != nil {
|
|
// this means this repo does not depend on any other package
|
|
log.Info("PRIMATIVE repo:", rs.String(), "err =", err)
|
|
return false, err
|
|
}
|
|
if ok {
|
|
// this means the repo is primitive so there is no go.sum
|
|
rs.primitive.SetText("true")
|
|
return true, nil
|
|
}
|
|
// this should never happen
|
|
return false, errors.New("MakeRedomod() logic failed")
|
|
}
|
|
|
|
func (rs *RepoStatus) IsReleased() bool {
|
|
if rs.GetTargetVersion() == rs.GetCurrentVersion() {
|
|
return true
|
|
}
|
|
return false
|
|
}
|