浏览代码

Update wrapper

Ben 1 周之前
父节点
当前提交
04a771665e
共有 8 个文件被更改,包括 427 次插入6 次删除
  1. 0 3
      .gitignore
  2. 3 0
      build.sh
  3. 二进制
      main
  4. 49 0
      main.go
  5. 362 0
      pkg/job/job.go
  6. 4 0
      test.sh
  7. 1 1
      tools/aws.go
  8. 8 2
      tools/terraform.go

+ 0 - 3
.gitignore

@@ -1,5 +1,2 @@
 workspace
 test
-pkg/job
-main.go
-main

+ 3 - 0
build.sh

@@ -0,0 +1,3 @@
+#!/bin/bash
+
+go build -o ~/bin/helios .

二进制
main


+ 49 - 0
main.go

@@ -0,0 +1,49 @@
+package main
+
+import (
+	"flag"
+	"log"
+
+	"git.bazzel.dev/bmallen/helios/pkg/job"
+)
+
+var (
+	srcdir      = flag.String("d", "", "source dir")
+	src         = flag.String("s", "", "source url")
+	ref         = flag.String("r", "", "source url ref")
+	f           = flag.String("f", "main.go", "file to run")
+	skipcleanup = flag.Bool("skipcleanup", false, "skip auto cleanup")
+	report      = flag.Bool("report", false, "output report")
+)
+
+func main() {
+	flag.Parse()
+
+	j := job.New()
+
+	if !*skipcleanup {
+		j.AutoCleanup()
+	}
+
+	if srcdir != nil {
+		if *srcdir != "" {
+			j.SourceDir(*srcdir)
+		}
+	}
+
+	if src != nil {
+		if *src != "" {
+			j.Source(*src, *ref)
+		}
+	}
+
+	j.Run(*f)
+
+	if *report {
+		j.Report()
+	}
+
+	if j.Error() != nil {
+		log.Fatal(j.Error())
+	}
+}

+ 362 - 0
pkg/job/job.go

@@ -0,0 +1,362 @@
+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
+}

+ 4 - 0
test.sh

@@ -0,0 +1,4 @@
+#!/bin/bash
+
+go run main.go -s https://git.bazzel.dev/bmallen/helios-job-example.git| yq
+

+ 1 - 1
tools/aws.go

@@ -19,7 +19,7 @@ func (t *aws) EC2() *ec2 { return &ec2{aws: t} }
 
 func (t *s3) Put(src, dst string) error  { return nil }
 func (t *s3) Get(src, dst string) error  { return nil }
-func (t *s3) List(src, dst string) error { return nil }
+func (t *s3) List(path string) error { return nil }
 
 func (t *ec2) Create() error    { return nil }
 func (t *ec2) Stop() error      { return nil }

+ 8 - 2
tools/terraform.go

@@ -41,8 +41,12 @@ func (_ tfLogParser) Write(p []byte) (n int, err error) {
 	if err != nil {
 		return 0, err
 	}
-
-	msg := []byte(fmt.Sprintf("[%s] %s\n", logEntry.Level(), logEntry.Message()))
+	var msg []byte
+	if logEntry.Level() == "" {
+		msg = []byte(fmt.Sprintf("%s\n", logEntry.Message()))
+	} else {
+		msg = []byte(fmt.Sprintf("[%s] %s\n", logEntry.Level(), logEntry.Message()))
+	}
 	log.Write(msg)
 
 	return len(msg), nil
@@ -77,6 +81,8 @@ func (t *terraform) Init() error {
 	}
 
 	if install {
+		log.Write([]byte(fmt.Sprintf("Installing Terraform version: %s", t.version)))
+
 		installer := &releases.ExactVersion{
 			Product: product.Terraform,
 			Version: version.Must(version.NewVersion(t.version)),