2020-02-04 18:38:41 -06:00
|
|
|
package runner
|
|
|
|
|
|
|
|
import (
|
2020-02-11 11:10:35 -06:00
|
|
|
"context"
|
2020-02-17 12:25:28 -06:00
|
|
|
"fmt"
|
2020-02-04 18:38:41 -06:00
|
|
|
"io/ioutil"
|
|
|
|
|
|
|
|
"github.com/nektos/act/pkg/common"
|
|
|
|
"github.com/nektos/act/pkg/model"
|
|
|
|
log "github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Runner provides capabilities to run GitHub actions
|
|
|
|
type Runner interface {
|
2020-02-07 00:17:58 -06:00
|
|
|
NewPlanExecutor(plan *model.Plan) common.Executor
|
2020-02-14 02:41:20 -06:00
|
|
|
NewRunExecutor(run *model.Run, matrix map[string]interface{}) common.Executor
|
2020-02-04 18:38:41 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Config contains the config for a new runner
|
|
|
|
type Config struct {
|
2020-02-17 23:51:49 -06:00
|
|
|
Workdir string // path to working directory
|
|
|
|
EventName string // name of event to run
|
|
|
|
EventPath string // path to JSON file to use for event.json in containers
|
|
|
|
ReuseContainers bool // reuse containers to maintain state
|
|
|
|
ForcePull bool // force pulling of the image, if already present
|
|
|
|
LogOutput bool // log the output from docker run
|
|
|
|
Secrets map[string]string // list of secrets
|
2020-02-19 21:16:40 -06:00
|
|
|
Platforms map[string]string // list of platforms
|
2020-02-04 18:38:41 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
type runnerImpl struct {
|
|
|
|
config *Config
|
|
|
|
eventJSON string
|
|
|
|
}
|
|
|
|
|
2020-02-07 00:17:58 -06:00
|
|
|
// New Creates a new Runner
|
|
|
|
func New(runnerConfig *Config) (Runner, error) {
|
2020-02-04 18:38:41 -06:00
|
|
|
runner := &runnerImpl{
|
|
|
|
config: runnerConfig,
|
|
|
|
}
|
|
|
|
|
|
|
|
runner.eventJSON = "{}"
|
2020-02-07 00:17:58 -06:00
|
|
|
if runnerConfig.EventPath != "" {
|
2020-02-04 18:38:41 -06:00
|
|
|
log.Debugf("Reading event.json from %s", runner.config.EventPath)
|
|
|
|
eventJSONBytes, err := ioutil.ReadFile(runner.config.EventPath)
|
|
|
|
if err != nil {
|
2020-02-07 00:17:58 -06:00
|
|
|
return nil, err
|
2020-02-04 18:38:41 -06:00
|
|
|
}
|
|
|
|
runner.eventJSON = string(eventJSONBytes)
|
|
|
|
}
|
2020-02-07 00:17:58 -06:00
|
|
|
return runner, nil
|
2020-02-04 18:38:41 -06:00
|
|
|
}
|
|
|
|
|
2020-02-07 00:17:58 -06:00
|
|
|
func (runner *runnerImpl) NewPlanExecutor(plan *model.Plan) common.Executor {
|
2020-02-17 12:30:52 -06:00
|
|
|
maxJobNameLen := plan.MaxRunNameLen()
|
2020-02-17 12:25:28 -06:00
|
|
|
|
2020-02-04 18:38:41 -06:00
|
|
|
pipeline := make([]common.Executor, 0)
|
|
|
|
for _, stage := range plan.Stages {
|
|
|
|
stageExecutor := make([]common.Executor, 0)
|
|
|
|
for _, run := range stage.Runs {
|
2020-02-17 12:11:16 -06:00
|
|
|
job := run.Job()
|
|
|
|
matrixes := make([]map[string]interface{}, 0)
|
|
|
|
if job.Strategy != nil {
|
|
|
|
includes := make([]map[string]interface{}, 0)
|
|
|
|
for _, v := range job.Strategy.Matrix["include"] {
|
|
|
|
includes = append(includes, v.(map[string]interface{}))
|
|
|
|
}
|
|
|
|
delete(job.Strategy.Matrix, "include")
|
|
|
|
|
|
|
|
excludes := make([]map[string]interface{}, 0)
|
|
|
|
for _, v := range job.Strategy.Matrix["exclude"] {
|
|
|
|
excludes = append(excludes, v.(map[string]interface{}))
|
|
|
|
}
|
|
|
|
delete(job.Strategy.Matrix, "exclude")
|
|
|
|
|
|
|
|
matrixProduct := common.CartesianProduct(job.Strategy.Matrix)
|
|
|
|
|
|
|
|
MATRIX:
|
|
|
|
for _, matrix := range matrixProduct {
|
|
|
|
for _, exclude := range excludes {
|
|
|
|
if commonKeysMatch(matrix, exclude) {
|
|
|
|
log.Debugf("Skipping matrix '%v' due to exclude '%v'", matrix, exclude)
|
|
|
|
continue MATRIX
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, include := range includes {
|
|
|
|
if commonKeysMatch(matrix, include) {
|
|
|
|
log.Debugf("Setting add'l values on matrix '%v' due to include '%v'", matrix, include)
|
|
|
|
for k, v := range include {
|
|
|
|
matrix[k] = v
|
|
|
|
}
|
|
|
|
}
|
2020-02-14 02:41:20 -06:00
|
|
|
}
|
2020-02-17 12:11:16 -06:00
|
|
|
matrixes = append(matrixes, matrix)
|
2020-02-14 02:41:20 -06:00
|
|
|
}
|
2020-02-17 12:11:16 -06:00
|
|
|
|
|
|
|
} else {
|
|
|
|
matrixes = append(matrixes, make(map[string]interface{}))
|
2020-02-14 02:41:20 -06:00
|
|
|
}
|
|
|
|
|
2020-02-17 12:25:28 -06:00
|
|
|
jobName := fmt.Sprintf("%-*s", maxJobNameLen, run.String())
|
2020-02-17 12:11:16 -06:00
|
|
|
for _, matrix := range matrixes {
|
2020-02-17 12:25:28 -06:00
|
|
|
m := matrix
|
|
|
|
runExecutor := runner.NewRunExecutor(run, matrix)
|
|
|
|
stageExecutor = append(stageExecutor, func(ctx context.Context) error {
|
|
|
|
ctx = WithJobLogger(ctx, jobName)
|
|
|
|
if len(m) > 0 {
|
|
|
|
common.Logger(ctx).Infof("\U0001F9EA Matrix: %v", m)
|
|
|
|
}
|
|
|
|
return runExecutor(ctx)
|
|
|
|
})
|
2020-02-17 12:11:16 -06:00
|
|
|
}
|
2020-02-04 18:38:41 -06:00
|
|
|
}
|
|
|
|
pipeline = append(pipeline, common.NewParallelExecutor(stageExecutor...))
|
|
|
|
}
|
|
|
|
|
2020-02-07 00:17:58 -06:00
|
|
|
return common.NewPipelineExecutor(pipeline...)
|
2020-02-04 18:38:41 -06:00
|
|
|
}
|
|
|
|
|
2020-02-17 12:11:16 -06:00
|
|
|
func commonKeysMatch(a map[string]interface{}, b map[string]interface{}) bool {
|
|
|
|
for aKey, aVal := range a {
|
|
|
|
if bVal, ok := b[aKey]; ok && aVal != bVal {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
2020-02-14 02:41:20 -06:00
|
|
|
func (runner *runnerImpl) NewRunExecutor(run *model.Run, matrix map[string]interface{}) common.Executor {
|
2020-02-07 00:17:58 -06:00
|
|
|
rc := new(RunContext)
|
|
|
|
rc.Config = runner.config
|
|
|
|
rc.Run = run
|
|
|
|
rc.EventJSON = runner.eventJSON
|
2020-02-14 02:41:20 -06:00
|
|
|
rc.StepResults = make(map[string]*stepResult)
|
|
|
|
rc.Matrix = matrix
|
2020-02-17 12:11:16 -06:00
|
|
|
rc.ExprEval = rc.NewExpressionEvaluator()
|
2020-02-17 12:25:28 -06:00
|
|
|
return rc.Executor()
|
2020-02-04 18:38:41 -06:00
|
|
|
}
|