package runner import ( "context" "fmt" "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 { NewPlanExecutor(plan *model.Plan) common.Executor NewRunExecutor(run *model.Run, matrix map[string]interface{}) common.Executor } // Config contains the config for a new runner type Config struct { 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 Platforms map[string]string // list of platforms } type runnerImpl struct { config *Config eventJSON string } // New Creates a new Runner func New(runnerConfig *Config) (Runner, error) { runner := &runnerImpl{ config: runnerConfig, } runner.eventJSON = "{}" if runnerConfig.EventPath != "" { log.Debugf("Reading event.json from %s", runner.config.EventPath) eventJSONBytes, err := ioutil.ReadFile(runner.config.EventPath) if err != nil { return nil, err } runner.eventJSON = string(eventJSONBytes) } return runner, nil } func (runner *runnerImpl) NewPlanExecutor(plan *model.Plan) common.Executor { maxJobNameLen := plan.MaxRunNameLen() pipeline := make([]common.Executor, 0) for _, stage := range plan.Stages { stageExecutor := make([]common.Executor, 0) for _, run := range stage.Runs { 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 } } } matrixes = append(matrixes, matrix) } } else { matrixes = append(matrixes, make(map[string]interface{})) } jobName := fmt.Sprintf("%-*s", maxJobNameLen, run.String()) for _, matrix := range matrixes { 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) }) } } pipeline = append(pipeline, common.NewParallelExecutor(stageExecutor...)) } return common.NewPipelineExecutor(pipeline...) } 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 } func (runner *runnerImpl) NewRunExecutor(run *model.Run, matrix map[string]interface{}) common.Executor { rc := new(RunContext) rc.Config = runner.config rc.Run = run rc.EventJSON = runner.eventJSON rc.StepResults = make(map[string]*stepResult) rc.Matrix = matrix rc.ExprEval = rc.NewExpressionEvaluator() return rc.Executor() }