| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362 |
- package job
- import (
- "context"
- "errors"
- "fmt"
- "os"
- "path/filepath"
- "reflect"
- "strings"
- "time"
- "git.bazzel.dev/bmallen/helios/pkg/state"
- "git.bazzel.dev/bmallen/helios/pkg/symbols"
- "github.com/go-git/go-git/v6"
- "github.com/go-git/go-git/v6/plumbing"
- "github.com/go-git/go-git/v6/plumbing/transport/http"
- "github.com/otiai10/copy"
- "github.com/sirupsen/logrus"
- "github.com/traefik/yaegi/interp"
- "github.com/traefik/yaegi/stdlib"
- "golang.org/x/mod/modfile"
- "gopkg.in/yaml.v3"
- )
- type Job struct {
- state *state.State
- sourceDir string
- files []string
- source string
- ref string
- user string
- password string
- globals map[string]any
- timeout time.Duration
- autoimport bool
- autocleanup bool
- module string
- errors []error
- log *logrus.Logger
- logs []string
- workspaceCreated bool
- workspacedir string
- i *interp.Interpreter
- }
- func New() *Job {
- j := &Job{
- state: state.New(),
- timeout: 10 * time.Second,
- log: logrus.New(),
- globals: map[string]any{},
- workspacedir: "workspace",
- }
- j.log.AddHook(j)
- return j
- }
- func NewWithState(state *state.State) *Job {
- j := New()
- j.state = state
- return j
- }
- func (j *Job) Levels() []logrus.Level {
- return logrus.AllLevels
- }
- func (j *Job) Fire(e *logrus.Entry) error {
- // j.logs = append(j.logs, fmt.Sprintf("%s: %s", e.Level.String(), e.Message))
- j.logs = append(j.logs, e.Message)
- return nil
- }
- func (j *Job) error(err error) *Job {
- if err != nil {
- j.log.Error(err)
- if j.errors == nil {
- j.errors = []error{}
- }
- j.errors = append(j.errors, err)
- }
- return j
- }
- func (j *Job) SourceDir(dir string) *Job {
- j.sourceDir = dir
- return j
- }
- func (j *Job) AutoCleanup() *Job {
- j.autocleanup = true
- return j
- }
- func (j *Job) Module(module string) *Job {
- j.module = module
- return j
- }
- func (j *Job) Source(url, ref string) *Job {
- if j.source != "" {
- return j.error(errors.New("source has already been set"))
- }
- j.source = url
- j.ref = url
- err := os.MkdirAll(filepath.Join(j.workspacedir, j.ID()), 0770)
- if err != nil {
- return j.error(err)
- }
- j.log.Info("Cloning...")
- _, err = git.PlainClone(
- filepath.Join(j.workspacedir, j.ID(), "repo"),
- &git.CloneOptions{
- URL: j.source,
- ReferenceName: plumbing.ReferenceName(ref),
- Progress: j.log.WriterLevel(logrus.DebugLevel),
- Auth: &http.BasicAuth{
- Username: j.user, // yes, this can be anything except an empty string
- Password: j.password,
- },
- })
- j.error(err)
- j.sourceDir = filepath.Join(j.workspacedir, j.ID(), "repo")
- return j
- }
- func (j *Job) User(user string) *Job {
- j.user = user
- return j
- }
- func (j *Job) Password(password string) *Job {
- j.password = password
- return j
- }
- func (j *Job) Timeout(timeout time.Duration) *Job {
- j.timeout = timeout
- return j
- }
- func (j *Job) AutoImport() *Job {
- j.autoimport = true
- return j
- }
- func (j *Job) Run(f string) *Job {
- if j.sourceDir == "" {
- return j.error(errors.New("must provide source"))
- }
- start := time.Now()
- workdir := filepath.Join(j.workspacedir, j.state.ID(), "src", j.module)
- if !j.workspaceCreated {
- if j.module == "" {
- modfiledata, err := os.ReadFile(filepath.Join(j.sourceDir, "go.mod"))
- if err != nil {
- return j.error(err)
- }
- fff, err := modfile.Parse("go.mod", modfiledata, nil)
- if err != nil {
- return j.error(err)
- }
- // for _, r := range fff.Require {
- // j.log.Info(" ", r.Mod.String())
- // }
- j.module = fff.Module.Mod.Path
- }
- j.log.Info("Creating workspace...")
- err := os.MkdirAll(filepath.Join(j.workspacedir, j.state.ID(), "src"), 0770)
- if err != nil {
- return j.error(err)
- }
- workdir = filepath.Join(j.workspacedir, j.state.ID(), "src", j.module)
- err = copy.Copy(j.sourceDir, workdir)
- if err != nil {
- return j.error(err)
- }
- j.workspaceCreated = true
- }
- if j.autocleanup {
- defer func() {
- }()
- }
- j.log.Info("Setting up new interpreter...")
- workspace, err := filepath.Abs(filepath.Join(j.workspacedir, j.state.ID()))
- if err != nil {
- return j.error(err)
- }
- if j.autocleanup {
- defer func() {
- err := os.RemoveAll(workspace)
- if err != nil {
- j.error(err)
- }
- }()
- }
- j.i = interp.New(interp.Options{
- GoPath: workspace,
- Unrestricted: true,
- Stdin: nil,
- Stdout: j.log.WriterLevel(logrus.InfoLevel),
- Stderr: j.log.WriterLevel(logrus.WarnLevel),
- })
- err = j.i.Use(stdlib.Symbols)
- if err != nil {
- return j.error(err)
- }
- err = j.i.Use(symbols.Symbols)
- if err != nil {
- return j.error(err)
- }
- err = j.i.Use(j.state.Symbols())
- if err != nil {
- return j.error(err)
- }
- if j.autoimport {
- j.i.ImportUsed()
- }
- j.log.Info("Starting job...")
- err = os.Chdir(workdir)
- if err != nil {
- return j.error(err)
- }
- j.files = append(j.files, f)
- if strings.HasSuffix(f, ".go") {
- data, err := os.ReadFile(f)
- if err != nil {
- return j.error(err)
- }
- ctx, _ := context.WithTimeout(context.Background(), j.timeout)
- _, err = j.i.EvalWithContext(ctx, string(data))
- if err != nil {
- j.error(err)
- }
- } else {
- ctx, _ := context.WithTimeout(context.Background(), j.timeout)
- _, err := j.i.EvalPathWithContext(ctx, filepath.Join(j.module, f))
- if err != nil {
- j.error(err)
- }
- }
- for k, v := range j.i.Globals() {
- if v.CanInterface() {
- if v.Kind() == reflect.Func {
- continue
- }
- j.globals[k] = v.Interface()
- }
- }
- j.log.Info("Job finished in: ", time.Now().Sub(start))
- return j
- }
- func (j *Job) Globals() map[string]any {
- return j.globals
- }
- func (j *Job) State() *state.State {
- return j.state
- }
- func (j *Job) ID() string {
- return j.state.ID()
- }
- func (j *Job) Error() error {
- if j.errors != nil {
- err := []string{}
- if len(j.errors) > 1 {
- err = append(err, fmt.Sprintf("!!! %d: errors in stack !!!", len(j.errors)))
- }
- for i, e := range j.errors {
- err = append(err, fmt.Sprintf("%d: %s", i, e.Error()))
- }
- return errors.New(strings.Join(err, "\n"))
- }
- return nil
- }
- type Report struct {
- Module string
- ID string
- Source string
- Files []string
- Globals map[string]any
- State map[string]any
- Errors []string
- Logs []string
- }
- func (j *Job) Report() *Job {
- report := Report{
- ID: j.ID(),
- Source: j.source,
- Files: j.files,
- Module: j.module,
- Globals: map[string]any{},
- State: map[string]any{},
- Logs: j.logs,
- }
- for k, v := range j.Globals() {
- report.Globals[k] = v
- }
- for _, v := range j.errors {
- report.Errors = append(report.Errors, v.Error())
- }
- j.state.Range(func(key string, value any) bool {
- report.State[key] = value
- return true
- })
- out, err := yaml.Marshal(report)
- if err != nil {
- j.log.Error(err)
- } else {
- fmt.Print(string(out))
- }
- return j
- }
|