fix: continue jobs + steps after failure (#840)
* fix: continue jobs + steps after failure To allow proper if expression handling on jobs and steps (like always, success, failure, ...) we need to continue running all executors in the prepared chain. To keep the error handling intact we add an occurred error to the go context and handle it later in the pipeline/chain. Also we add the job result to the needs context to give expressions access to it. The needs object, failure and success functions are split between run context (on jobs) and step context. Closes #442 Co-authored-by: Björn Brauer <zaubernerd@zaubernerd.de> * style: correct linter warnings Co-authored-by: Björn Brauer <zaubernerd@zaubernerd.de> * fix: job if value defaults to success() As described in the documentation, a default value of "success()" is applied when no "if" value is present on the job. https://docs.github.com/en/actions/learn-github-actions/expressions#job-status-check-functions Co-authored-by: Markus Wolf <mail@markus-wolf.de> * fix: check job needs recursively Ensure job result includes results of previous jobs Co-authored-by: Markus Wolf <markus.wolf@new-work.se> * test: add runner test for job status check functions Co-authored-by: Markus Wolf <markus.wolf@new-work.se> * test: add unit tests for run context if evaluation Co-authored-by: Björn Brauer <zaubernerd@zaubernerd.de> * refactor: move if expression evaluation Move if expression evaluation into own function (step context) to better support unit testing. Co-authored-by: Björn Brauer <zaubernerd@zaubernerd.de> * test: add unit tests for step context if evaluation Co-authored-by: Markus Wolf <markus.wolf@new-work.se> * fix: handle job error more resilient The job error is not stored in a context map instead of a context added value. Since context values are immutable an added value requires to keep the new context in all cases. This is fragile since it might slip unnoticed to other parts of the code. Storing the error of a job in the context map will make it more stable, since the map is always there and the context of the pipeline is stable for the whole run. * feat: steps should use a default if expression of success() * test: add integration test for if-expressions * chore: disable editorconfig-checker for yaml multiline string Co-authored-by: Björn Brauer <zaubernerd@zaubernerd.de> Co-authored-by: Björn Brauer <bjoern.brauer@new-work.se>
This commit is contained in:
parent
a545ceaec9
commit
1891c72ab1
13 changed files with 523 additions and 69 deletions
|
@ -136,13 +136,12 @@ func (e Executor) Then(then Executor) Executor {
|
||||||
case Warning:
|
case Warning:
|
||||||
log.Warning(err.Error())
|
log.Warning(err.Error())
|
||||||
default:
|
default:
|
||||||
log.Debugf("%+v", err)
|
SetJobError(ctx, err)
|
||||||
return err
|
|
||||||
}
|
}
|
||||||
|
} else if ctx.Err() != nil {
|
||||||
|
SetJobError(ctx, ctx.Err())
|
||||||
}
|
}
|
||||||
if ctx.Err() != nil {
|
|
||||||
return ctx.Err()
|
|
||||||
}
|
|
||||||
return then(ctx)
|
return then(ctx)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
30
pkg/common/job_error.go
Normal file
30
pkg/common/job_error.go
Normal file
|
@ -0,0 +1,30 @@
|
||||||
|
package common
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
)
|
||||||
|
|
||||||
|
type jobErrorContextKey string
|
||||||
|
|
||||||
|
const jobErrorContextKeyVal = jobErrorContextKey("job.error")
|
||||||
|
|
||||||
|
// JobError returns the job error for current context if any
|
||||||
|
func JobError(ctx context.Context) error {
|
||||||
|
val := ctx.Value(jobErrorContextKeyVal)
|
||||||
|
if val != nil {
|
||||||
|
if container, ok := val.(map[string]error); ok {
|
||||||
|
return container["error"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func SetJobError(ctx context.Context, err error) {
|
||||||
|
ctx.Value(jobErrorContextKeyVal).(map[string]error)["error"] = err
|
||||||
|
}
|
||||||
|
|
||||||
|
// WithJobErrorContainer adds a value to the context as a container for an error
|
||||||
|
func WithJobErrorContainer(ctx context.Context) context.Context {
|
||||||
|
container := map[string]error{}
|
||||||
|
return context.WithValue(ctx, jobErrorContextKeyVal, container)
|
||||||
|
}
|
|
@ -78,6 +78,9 @@ func FixIfStatement(content []byte, wr *Workflow) error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
if val == "" {
|
||||||
|
val = "success()"
|
||||||
|
}
|
||||||
jobs[j].Steps[i].If.Value = val
|
jobs[j].Steps[i].If.Value = val
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -69,6 +69,7 @@ type Job struct {
|
||||||
RawContainer yaml.Node `yaml:"container"`
|
RawContainer yaml.Node `yaml:"container"`
|
||||||
Defaults Defaults `yaml:"defaults"`
|
Defaults Defaults `yaml:"defaults"`
|
||||||
Outputs map[string]string `yaml:"outputs"`
|
Outputs map[string]string `yaml:"outputs"`
|
||||||
|
Result string
|
||||||
}
|
}
|
||||||
|
|
||||||
// Strategy for the job
|
// Strategy for the job
|
||||||
|
@ -433,6 +434,9 @@ func (w *Workflow) GetJob(jobID string) *Job {
|
||||||
if j.Name == "" {
|
if j.Name == "" {
|
||||||
j.Name = id
|
j.Name = id
|
||||||
}
|
}
|
||||||
|
if j.If.Value == "" {
|
||||||
|
j.If.Value = "success()"
|
||||||
|
}
|
||||||
return j
|
return j
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,6 +4,8 @@ import (
|
||||||
"crypto/sha256"
|
"crypto/sha256"
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
@ -25,6 +27,7 @@ func init() {
|
||||||
// NewExpressionEvaluator creates a new evaluator
|
// NewExpressionEvaluator creates a new evaluator
|
||||||
func (rc *RunContext) NewExpressionEvaluator() ExpressionEvaluator {
|
func (rc *RunContext) NewExpressionEvaluator() ExpressionEvaluator {
|
||||||
vm := rc.newVM()
|
vm := rc.newVM()
|
||||||
|
|
||||||
return &expressionEvaluator{
|
return &expressionEvaluator{
|
||||||
vm,
|
vm,
|
||||||
}
|
}
|
||||||
|
@ -36,6 +39,10 @@ func (sc *StepContext) NewExpressionEvaluator() ExpressionEvaluator {
|
||||||
configers := []func(*otto.Otto){
|
configers := []func(*otto.Otto){
|
||||||
sc.vmEnv(),
|
sc.vmEnv(),
|
||||||
sc.vmInputs(),
|
sc.vmInputs(),
|
||||||
|
|
||||||
|
sc.vmNeeds(),
|
||||||
|
sc.vmSuccess(),
|
||||||
|
sc.vmFailure(),
|
||||||
}
|
}
|
||||||
for _, configer := range configers {
|
for _, configer := range configers {
|
||||||
configer(vm)
|
configer(vm)
|
||||||
|
@ -373,14 +380,33 @@ func (rc *RunContext) vmHashFiles() func(*otto.Otto) {
|
||||||
func (rc *RunContext) vmSuccess() func(*otto.Otto) {
|
func (rc *RunContext) vmSuccess() func(*otto.Otto) {
|
||||||
return func(vm *otto.Otto) {
|
return func(vm *otto.Otto) {
|
||||||
_ = vm.Set("success", func() bool {
|
_ = vm.Set("success", func() bool {
|
||||||
return rc.getJobContext().Status == "success"
|
jobs := rc.Run.Workflow.Jobs
|
||||||
|
jobNeeds := rc.getNeedsTransitive(rc.Run.Job())
|
||||||
|
|
||||||
|
for _, needs := range jobNeeds {
|
||||||
|
if jobs[needs].Result != "success" {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (rc *RunContext) vmFailure() func(*otto.Otto) {
|
func (rc *RunContext) vmFailure() func(*otto.Otto) {
|
||||||
return func(vm *otto.Otto) {
|
return func(vm *otto.Otto) {
|
||||||
_ = vm.Set("failure", func() bool {
|
_ = vm.Set("failure", func() bool {
|
||||||
return rc.getJobContext().Status == "failure"
|
jobs := rc.Run.Workflow.Jobs
|
||||||
|
jobNeeds := rc.getNeedsTransitive(rc.Run.Job())
|
||||||
|
|
||||||
|
for _, needs := range jobNeeds {
|
||||||
|
if jobs[needs].Result == "failure" {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -440,9 +466,9 @@ func (sc *StepContext) vmInputs() func(*otto.Otto) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (rc *RunContext) vmNeeds() func(*otto.Otto) {
|
func (sc *StepContext) vmNeeds() func(*otto.Otto) {
|
||||||
jobs := rc.Run.Workflow.Jobs
|
jobs := sc.RunContext.Run.Workflow.Jobs
|
||||||
jobNeeds := rc.Run.Job().Needs()
|
jobNeeds := sc.RunContext.Run.Job().Needs()
|
||||||
|
|
||||||
using := make(map[string]map[string]map[string]string)
|
using := make(map[string]map[string]map[string]string)
|
||||||
for _, needs := range jobNeeds {
|
for _, needs := range jobNeeds {
|
||||||
|
@ -457,6 +483,70 @@ func (rc *RunContext) vmNeeds() func(*otto.Otto) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (sc *StepContext) vmSuccess() func(*otto.Otto) {
|
||||||
|
return func(vm *otto.Otto) {
|
||||||
|
_ = vm.Set("success", func() bool {
|
||||||
|
return sc.RunContext.getJobContext().Status == "success"
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (sc *StepContext) vmFailure() func(*otto.Otto) {
|
||||||
|
return func(vm *otto.Otto) {
|
||||||
|
_ = vm.Set("failure", func() bool {
|
||||||
|
return sc.RunContext.getJobContext().Status == "failure"
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
type vmNeedsStruct struct {
|
||||||
|
Outputs map[string]string `json:"outputs"`
|
||||||
|
Result string `json:"result"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func (rc *RunContext) vmNeeds() func(*otto.Otto) {
|
||||||
|
return func(vm *otto.Otto) {
|
||||||
|
needsFunc := func() otto.Value {
|
||||||
|
jobs := rc.Run.Workflow.Jobs
|
||||||
|
jobNeeds := rc.Run.Job().Needs()
|
||||||
|
|
||||||
|
using := make(map[string]vmNeedsStruct)
|
||||||
|
for _, needs := range jobNeeds {
|
||||||
|
using[needs] = vmNeedsStruct{
|
||||||
|
Outputs: jobs[needs].Outputs,
|
||||||
|
Result: jobs[needs].Result,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Debugf("context needs => %+v", using)
|
||||||
|
|
||||||
|
value, err := vm.ToValue(using)
|
||||||
|
if err != nil {
|
||||||
|
return vm.MakeTypeError(err.Error())
|
||||||
|
}
|
||||||
|
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
|
||||||
|
// Results might change after the Otto VM was created
|
||||||
|
// and initialized. To access the current state
|
||||||
|
// we can't just pass a copy to Otto - instead we
|
||||||
|
// created a 'live-binding'.
|
||||||
|
// Technical Note: We don't want to pollute the global
|
||||||
|
// js namespace (and add things github actions hasn't)
|
||||||
|
// we delete the helper function after installing it
|
||||||
|
// as a getter.
|
||||||
|
global, _ := vm.Run("this")
|
||||||
|
_ = global.Object().Set("__needs__", needsFunc)
|
||||||
|
_, _ = vm.Run(`
|
||||||
|
(function (global) {
|
||||||
|
Object.defineProperty(global, 'needs', { get: global.__needs__ });
|
||||||
|
delete global.__needs__;
|
||||||
|
})(this)
|
||||||
|
`)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (rc *RunContext) vmJob() func(*otto.Otto) {
|
func (rc *RunContext) vmJob() func(*otto.Otto) {
|
||||||
job := rc.getJobContext()
|
job := rc.getJobContext()
|
||||||
|
|
||||||
|
@ -518,3 +608,48 @@ func (rc *RunContext) vmMatrix() func(*otto.Otto) {
|
||||||
_ = vm.Set("matrix", rc.Matrix)
|
_ = vm.Set("matrix", rc.Matrix)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// EvalBool evaluates an expression against given evaluator
|
||||||
|
func EvalBool(evaluator ExpressionEvaluator, expr string) (bool, error) {
|
||||||
|
if splitPattern == nil {
|
||||||
|
splitPattern = regexp.MustCompile(fmt.Sprintf(`%s|%s|\S+`, expressionPattern.String(), operatorPattern.String()))
|
||||||
|
}
|
||||||
|
if strings.HasPrefix(strings.TrimSpace(expr), "!") {
|
||||||
|
return false, errors.New("expressions starting with ! must be wrapped in ${{ }}")
|
||||||
|
}
|
||||||
|
if expr != "" {
|
||||||
|
parts := splitPattern.FindAllString(expr, -1)
|
||||||
|
var evaluatedParts []string
|
||||||
|
for i, part := range parts {
|
||||||
|
if operatorPattern.MatchString(part) {
|
||||||
|
evaluatedParts = append(evaluatedParts, part)
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
interpolatedPart, isString := evaluator.InterpolateWithStringCheck(part)
|
||||||
|
|
||||||
|
// This peculiar transformation has to be done because the GitHub parser
|
||||||
|
// treats false returned from contexts as a string, not a boolean.
|
||||||
|
// Hence env.SOMETHING will be evaluated to true in an if: expression
|
||||||
|
// regardless if SOMETHING is set to false, true or any other string.
|
||||||
|
// It also handles some other weirdness that I found by trial and error.
|
||||||
|
if (expressionPattern.MatchString(part) && // it is an expression
|
||||||
|
!strings.Contains(part, "!")) && // but it's not negated
|
||||||
|
interpolatedPart == "false" && // and the interpolated string is false
|
||||||
|
(isString || previousOrNextPartIsAnOperator(i, parts)) { // and it's of type string or has an logical operator before or after
|
||||||
|
interpolatedPart = fmt.Sprintf("'%s'", interpolatedPart) // then we have to quote the false expression
|
||||||
|
}
|
||||||
|
|
||||||
|
evaluatedParts = append(evaluatedParts, interpolatedPart)
|
||||||
|
}
|
||||||
|
|
||||||
|
joined := strings.Join(evaluatedParts, " ")
|
||||||
|
v, _, err := evaluator.Evaluate(fmt.Sprintf("Boolean(%s)", joined))
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
log.Debugf("expression '%s' evaluated to '%s'", expr, v)
|
||||||
|
return v == "true", nil
|
||||||
|
}
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
|
82
pkg/runner/run_context.go
Executable file → Normal file
82
pkg/runner/run_context.go
Executable file → Normal file
|
@ -3,7 +3,6 @@ package runner
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"errors"
|
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
|
@ -288,7 +287,20 @@ func (rc *RunContext) Executor() common.Executor {
|
||||||
}
|
}
|
||||||
steps = append(steps, rc.newStepExecutor(step))
|
steps = append(steps, rc.newStepExecutor(step))
|
||||||
}
|
}
|
||||||
steps = append(steps, rc.stopJobContainer())
|
steps = append(steps, func(ctx context.Context) error {
|
||||||
|
err := rc.stopJobContainer()(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
rc.Run.Job().Result = "success"
|
||||||
|
jobError := common.JobError(ctx)
|
||||||
|
if jobError != nil {
|
||||||
|
rc.Run.Job().Result = "failure"
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
|
||||||
return common.NewPipelineExecutor(steps...).Finally(rc.interpolateOutputs()).Finally(func(ctx context.Context) error {
|
return common.NewPipelineExecutor(steps...).Finally(rc.interpolateOutputs()).Finally(func(ctx context.Context) error {
|
||||||
if rc.JobContainer != nil {
|
if rc.JobContainer != nil {
|
||||||
|
@ -310,15 +322,9 @@ func (rc *RunContext) newStepExecutor(step *model.Step) common.Executor {
|
||||||
Conclusion: stepStatusSuccess,
|
Conclusion: stepStatusSuccess,
|
||||||
Outputs: make(map[string]string),
|
Outputs: make(map[string]string),
|
||||||
}
|
}
|
||||||
runStep, err := rc.EvalBool(sc.Step.If.Value)
|
|
||||||
|
|
||||||
|
runStep, err := sc.isEnabled(ctx)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
common.Logger(ctx).Errorf(" \u274C Error in if: expression - %s", sc.Step)
|
|
||||||
exprEval, err := sc.setupEnv(ctx)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
rc.ExprEval = exprEval
|
|
||||||
rc.StepResults[rc.CurrentStep].Conclusion = stepStatusFailure
|
rc.StepResults[rc.CurrentStep].Conclusion = stepStatusFailure
|
||||||
rc.StepResults[rc.CurrentStep].Outcome = stepStatusFailure
|
rc.StepResults[rc.CurrentStep].Outcome = stepStatusFailure
|
||||||
return err
|
return err
|
||||||
|
@ -403,7 +409,7 @@ func (rc *RunContext) hostname() string {
|
||||||
func (rc *RunContext) isEnabled(ctx context.Context) bool {
|
func (rc *RunContext) isEnabled(ctx context.Context) bool {
|
||||||
job := rc.Run.Job()
|
job := rc.Run.Job()
|
||||||
l := common.Logger(ctx)
|
l := common.Logger(ctx)
|
||||||
runJob, err := rc.EvalBool(job.If.Value)
|
runJob, err := EvalBool(rc.ExprEval, job.If.Value)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
common.Logger(ctx).Errorf(" \u274C Error in if: expression - %s", job.Name)
|
common.Logger(ctx).Errorf(" \u274C Error in if: expression - %s", job.Name)
|
||||||
return false
|
return false
|
||||||
|
@ -430,51 +436,6 @@ func (rc *RunContext) isEnabled(ctx context.Context) bool {
|
||||||
|
|
||||||
var splitPattern *regexp.Regexp
|
var splitPattern *regexp.Regexp
|
||||||
|
|
||||||
// EvalBool evaluates an expression against current run context
|
|
||||||
func (rc *RunContext) EvalBool(expr string) (bool, error) {
|
|
||||||
if splitPattern == nil {
|
|
||||||
splitPattern = regexp.MustCompile(fmt.Sprintf(`%s|%s|\S+`, expressionPattern.String(), operatorPattern.String()))
|
|
||||||
}
|
|
||||||
if strings.HasPrefix(strings.TrimSpace(expr), "!") {
|
|
||||||
return false, errors.New("expressions starting with ! must be wrapped in ${{ }}")
|
|
||||||
}
|
|
||||||
if expr != "" {
|
|
||||||
parts := splitPattern.FindAllString(expr, -1)
|
|
||||||
var evaluatedParts []string
|
|
||||||
for i, part := range parts {
|
|
||||||
if operatorPattern.MatchString(part) {
|
|
||||||
evaluatedParts = append(evaluatedParts, part)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
interpolatedPart, isString := rc.ExprEval.InterpolateWithStringCheck(part)
|
|
||||||
|
|
||||||
// This peculiar transformation has to be done because the GitHub parser
|
|
||||||
// treats false returned from contexts as a string, not a boolean.
|
|
||||||
// Hence env.SOMETHING will be evaluated to true in an if: expression
|
|
||||||
// regardless if SOMETHING is set to false, true or any other string.
|
|
||||||
// It also handles some other weirdness that I found by trial and error.
|
|
||||||
if (expressionPattern.MatchString(part) && // it is an expression
|
|
||||||
!strings.Contains(part, "!")) && // but it's not negated
|
|
||||||
interpolatedPart == "false" && // and the interpolated string is false
|
|
||||||
(isString || previousOrNextPartIsAnOperator(i, parts)) { // and it's of type string or has an logical operator before or after
|
|
||||||
interpolatedPart = fmt.Sprintf("'%s'", interpolatedPart) // then we have to quote the false expression
|
|
||||||
}
|
|
||||||
|
|
||||||
evaluatedParts = append(evaluatedParts, interpolatedPart)
|
|
||||||
}
|
|
||||||
|
|
||||||
joined := strings.Join(evaluatedParts, " ")
|
|
||||||
v, _, err := rc.ExprEval.Evaluate(fmt.Sprintf("Boolean(%s)", joined))
|
|
||||||
if err != nil {
|
|
||||||
return false, err
|
|
||||||
}
|
|
||||||
log.Debugf("expression '%s' evaluated to '%s'", expr, v)
|
|
||||||
return v == "true", nil
|
|
||||||
}
|
|
||||||
return true, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func previousOrNextPartIsAnOperator(i int, parts []string) bool {
|
func previousOrNextPartIsAnOperator(i int, parts []string) bool {
|
||||||
operator := false
|
operator := false
|
||||||
if i > 0 {
|
if i > 0 {
|
||||||
|
@ -557,6 +518,17 @@ func (rc *RunContext) getStepsContext() map[string]*stepResult {
|
||||||
return rc.StepResults
|
return rc.StepResults
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (rc *RunContext) getNeedsTransitive(job *model.Job) []string {
|
||||||
|
needs := job.Needs()
|
||||||
|
|
||||||
|
for _, need := range needs {
|
||||||
|
parentNeeds := rc.getNeedsTransitive(rc.Run.Workflow.GetJob(need))
|
||||||
|
needs = append(needs, parentNeeds...)
|
||||||
|
}
|
||||||
|
|
||||||
|
return needs
|
||||||
|
}
|
||||||
|
|
||||||
type githubContext struct {
|
type githubContext struct {
|
||||||
Event map[string]interface{} `json:"event"`
|
Event map[string]interface{} `json:"event"`
|
||||||
EventPath string `json:"event_path"`
|
EventPath string `json:"event_path"`
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
package runner
|
package runner
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"context"
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
"regexp"
|
"regexp"
|
||||||
|
@ -153,7 +154,7 @@ func TestRunContext_EvalBool(t *testing.T) {
|
||||||
t.Run(table.in, func(t *testing.T) {
|
t.Run(table.in, func(t *testing.T) {
|
||||||
assertObject := assert.New(t)
|
assertObject := assert.New(t)
|
||||||
defer hook.Reset()
|
defer hook.Reset()
|
||||||
b, err := rc.EvalBool(table.in)
|
b, err := EvalBool(rc.ExprEval, table.in)
|
||||||
if table.wantErr {
|
if table.wantErr {
|
||||||
assertObject.Error(err)
|
assertObject.Error(err)
|
||||||
}
|
}
|
||||||
|
@ -178,7 +179,7 @@ func updateTestIfWorkflow(t *testing.T, tables []struct {
|
||||||
for _, k := range keys {
|
for _, k := range keys {
|
||||||
envs += fmt.Sprintf(" %s: %s\n", k, rc.Env[k])
|
envs += fmt.Sprintf(" %s: %s\n", k, rc.Env[k])
|
||||||
}
|
}
|
||||||
|
// editorconfig-checker-disable
|
||||||
workflow := fmt.Sprintf(`
|
workflow := fmt.Sprintf(`
|
||||||
name: "Test what expressions result in true and false on GitHub"
|
name: "Test what expressions result in true and false on GitHub"
|
||||||
on: push
|
on: push
|
||||||
|
@ -191,6 +192,7 @@ jobs:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
`, envs)
|
`, envs)
|
||||||
|
// editorconfig-checker-enable
|
||||||
|
|
||||||
for i, table := range tables {
|
for i, table := range tables {
|
||||||
if table.wantErr || strings.HasPrefix(table.in, "github.actor") {
|
if table.wantErr || strings.HasPrefix(table.in, "github.actor") {
|
||||||
|
@ -344,3 +346,138 @@ func TestGetGitHubContext(t *testing.T) {
|
||||||
assert.Equal(t, ghc.EventPath, ActPath+"/workflow/event.json")
|
assert.Equal(t, ghc.EventPath, ActPath+"/workflow/event.json")
|
||||||
assert.Equal(t, ghc.Token, rc.Config.Secrets["GITHUB_TOKEN"])
|
assert.Equal(t, ghc.Token, rc.Config.Secrets["GITHUB_TOKEN"])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func createIfTestRunContext(jobs map[string]*model.Job) *RunContext {
|
||||||
|
rc := &RunContext{
|
||||||
|
Config: &Config{
|
||||||
|
Workdir: ".",
|
||||||
|
Platforms: map[string]string{
|
||||||
|
"ubuntu-latest": "ubuntu-latest",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Env: map[string]string{},
|
||||||
|
Run: &model.Run{
|
||||||
|
JobID: "job1",
|
||||||
|
Workflow: &model.Workflow{
|
||||||
|
Name: "test-workflow",
|
||||||
|
Jobs: jobs,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
rc.ExprEval = rc.NewExpressionEvaluator()
|
||||||
|
|
||||||
|
return rc
|
||||||
|
}
|
||||||
|
|
||||||
|
func createJob(t *testing.T, input string, result string) *model.Job {
|
||||||
|
var job *model.Job
|
||||||
|
err := yaml.Unmarshal([]byte(input), &job)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
job.Result = result
|
||||||
|
|
||||||
|
return job
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestRunContextIsEnabled(t *testing.T) {
|
||||||
|
log.SetLevel(log.DebugLevel)
|
||||||
|
assertObject := assert.New(t)
|
||||||
|
|
||||||
|
// success()
|
||||||
|
rc := createIfTestRunContext(map[string]*model.Job{
|
||||||
|
"job1": createJob(t, `runs-on: ubuntu-latest
|
||||||
|
if: success()`, ""),
|
||||||
|
})
|
||||||
|
assertObject.True(rc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
rc = createIfTestRunContext(map[string]*model.Job{
|
||||||
|
"job1": createJob(t, `runs-on: ubuntu-latest`, "failure"),
|
||||||
|
"job2": createJob(t, `runs-on: ubuntu-latest
|
||||||
|
needs: [job1]
|
||||||
|
if: success()`, ""),
|
||||||
|
})
|
||||||
|
rc.Run.JobID = "job2"
|
||||||
|
assertObject.False(rc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
rc = createIfTestRunContext(map[string]*model.Job{
|
||||||
|
"job1": createJob(t, `runs-on: ubuntu-latest`, "success"),
|
||||||
|
"job2": createJob(t, `runs-on: ubuntu-latest
|
||||||
|
needs: [job1]
|
||||||
|
if: success()`, ""),
|
||||||
|
})
|
||||||
|
rc.Run.JobID = "job2"
|
||||||
|
assertObject.True(rc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
rc = createIfTestRunContext(map[string]*model.Job{
|
||||||
|
"job1": createJob(t, `runs-on: ubuntu-latest`, "failure"),
|
||||||
|
"job2": createJob(t, `runs-on: ubuntu-latest
|
||||||
|
if: success()`, ""),
|
||||||
|
})
|
||||||
|
rc.Run.JobID = "job2"
|
||||||
|
assertObject.True(rc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
// failure()
|
||||||
|
rc = createIfTestRunContext(map[string]*model.Job{
|
||||||
|
"job1": createJob(t, `runs-on: ubuntu-latest
|
||||||
|
if: failure()`, ""),
|
||||||
|
})
|
||||||
|
assertObject.False(rc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
rc = createIfTestRunContext(map[string]*model.Job{
|
||||||
|
"job1": createJob(t, `runs-on: ubuntu-latest`, "failure"),
|
||||||
|
"job2": createJob(t, `runs-on: ubuntu-latest
|
||||||
|
needs: [job1]
|
||||||
|
if: failure()`, ""),
|
||||||
|
})
|
||||||
|
rc.Run.JobID = "job2"
|
||||||
|
assertObject.True(rc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
rc = createIfTestRunContext(map[string]*model.Job{
|
||||||
|
"job1": createJob(t, `runs-on: ubuntu-latest`, "success"),
|
||||||
|
"job2": createJob(t, `runs-on: ubuntu-latest
|
||||||
|
needs: [job1]
|
||||||
|
if: failure()`, ""),
|
||||||
|
})
|
||||||
|
rc.Run.JobID = "job2"
|
||||||
|
assertObject.False(rc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
rc = createIfTestRunContext(map[string]*model.Job{
|
||||||
|
"job1": createJob(t, `runs-on: ubuntu-latest`, "failure"),
|
||||||
|
"job2": createJob(t, `runs-on: ubuntu-latest
|
||||||
|
if: failure()`, ""),
|
||||||
|
})
|
||||||
|
rc.Run.JobID = "job2"
|
||||||
|
assertObject.False(rc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
// always()
|
||||||
|
rc = createIfTestRunContext(map[string]*model.Job{
|
||||||
|
"job1": createJob(t, `runs-on: ubuntu-latest
|
||||||
|
if: always()`, ""),
|
||||||
|
})
|
||||||
|
assertObject.True(rc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
rc = createIfTestRunContext(map[string]*model.Job{
|
||||||
|
"job1": createJob(t, `runs-on: ubuntu-latest`, "failure"),
|
||||||
|
"job2": createJob(t, `runs-on: ubuntu-latest
|
||||||
|
needs: [job1]
|
||||||
|
if: always()`, ""),
|
||||||
|
})
|
||||||
|
rc.Run.JobID = "job2"
|
||||||
|
assertObject.True(rc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
rc = createIfTestRunContext(map[string]*model.Job{
|
||||||
|
"job1": createJob(t, `runs-on: ubuntu-latest`, "success"),
|
||||||
|
"job2": createJob(t, `runs-on: ubuntu-latest
|
||||||
|
needs: [job1]
|
||||||
|
if: always()`, ""),
|
||||||
|
})
|
||||||
|
rc.Run.JobID = "job2"
|
||||||
|
assertObject.True(rc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
rc = createIfTestRunContext(map[string]*model.Job{
|
||||||
|
"job1": createJob(t, `runs-on: ubuntu-latest`, "success"),
|
||||||
|
"job2": createJob(t, `runs-on: ubuntu-latest
|
||||||
|
if: always()`, ""),
|
||||||
|
})
|
||||||
|
rc.Run.JobID = "job2"
|
||||||
|
assertObject.True(rc.isEnabled(context.Background()))
|
||||||
|
}
|
||||||
|
|
|
@ -154,7 +154,7 @@ func (runner *runnerImpl) NewPlanExecutor(plan *model.Plan) common.Executor {
|
||||||
}
|
}
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
})(WithJobLogger(ctx, jobName, rc.Config.Secrets, rc.Config.InsecureSecrets))
|
})(common.WithJobErrorContainer(WithJobLogger(ctx, jobName, rc.Config.Secrets, rc.Config.InsecureSecrets)))
|
||||||
})
|
})
|
||||||
b++
|
b++
|
||||||
if b == maxParallel {
|
if b == maxParallel {
|
||||||
|
@ -166,7 +166,20 @@ func (runner *runnerImpl) NewPlanExecutor(plan *model.Plan) common.Executor {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return common.NewPipelineExecutor(pipeline...)
|
return common.NewPipelineExecutor(pipeline...).Then(handleFailure(plan))
|
||||||
|
}
|
||||||
|
|
||||||
|
func handleFailure(plan *model.Plan) common.Executor {
|
||||||
|
return func(ctx context.Context) error {
|
||||||
|
for _, stage := range plan.Stages {
|
||||||
|
for _, run := range stage.Runs {
|
||||||
|
if run.Job().Result == "failure" {
|
||||||
|
return fmt.Errorf("Job '%s' failed", run.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (runner *runnerImpl) newRunContext(run *model.Run, matrix map[string]interface{}) *RunContext {
|
func (runner *runnerImpl) newRunContext(run *model.Run, matrix map[string]interface{}) *RunContext {
|
||||||
|
|
|
@ -123,6 +123,8 @@ func TestRunEvent(t *testing.T) {
|
||||||
{"testdata", "outputs", "push", "", platforms, ""},
|
{"testdata", "outputs", "push", "", platforms, ""},
|
||||||
{"testdata", "steps-context/conclusion", "push", "", platforms, ""},
|
{"testdata", "steps-context/conclusion", "push", "", platforms, ""},
|
||||||
{"testdata", "steps-context/outcome", "push", "", platforms, ""},
|
{"testdata", "steps-context/outcome", "push", "", platforms, ""},
|
||||||
|
{"testdata", "job-status-check", "push", "job 'fail' failed", platforms, ""},
|
||||||
|
{"testdata", "if-expressions", "push", "Job 'mytest' failed", platforms, ""},
|
||||||
{"../model/testdata", "strategy", "push", "", platforms, ""}, // TODO: move all testdata into pkg so we can validate it with planner and runner
|
{"../model/testdata", "strategy", "push", "", platforms, ""}, // TODO: move all testdata into pkg so we can validate it with planner and runner
|
||||||
// {"testdata", "issue-228", "push", "", platforms, ""}, // TODO [igni]: Remove this once everything passes
|
// {"testdata", "issue-228", "push", "", platforms, ""}, // TODO [igni]: Remove this once everything passes
|
||||||
|
|
||||||
|
|
|
@ -145,6 +145,21 @@ func (sc *StepContext) interpolateEnv(exprEval ExpressionEvaluator) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (sc *StepContext) isEnabled(ctx context.Context) (bool, error) {
|
||||||
|
runStep, err := EvalBool(sc.NewExpressionEvaluator(), sc.Step.If.Value)
|
||||||
|
if err != nil {
|
||||||
|
common.Logger(ctx).Errorf(" \u274C Error in if: expression - %s", sc.Step)
|
||||||
|
exprEval, err := sc.setupEnv(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
sc.RunContext.ExprEval = exprEval
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return runStep, nil
|
||||||
|
}
|
||||||
|
|
||||||
func (sc *StepContext) setupEnv(ctx context.Context) (ExpressionEvaluator, error) {
|
func (sc *StepContext) setupEnv(ctx context.Context) (ExpressionEvaluator, error) {
|
||||||
rc := sc.RunContext
|
rc := sc.RunContext
|
||||||
sc.Env = sc.mergeEnv()
|
sc.Env = sc.mergeEnv()
|
||||||
|
|
|
@ -4,7 +4,12 @@ import (
|
||||||
"context"
|
"context"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
"github.com/stretchr/testify/assert"
|
||||||
|
"gopkg.in/yaml.v3"
|
||||||
|
|
||||||
"github.com/nektos/act/pkg/common"
|
"github.com/nektos/act/pkg/common"
|
||||||
|
"github.com/nektos/act/pkg/model"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestStepContextExecutor(t *testing.T) {
|
func TestStepContextExecutor(t *testing.T) {
|
||||||
|
@ -27,3 +32,85 @@ func TestStepContextExecutor(t *testing.T) {
|
||||||
runTestJobFile(ctx, t, table)
|
runTestJobFile(ctx, t, table)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func createIfTestStepContext(t *testing.T, input string) *StepContext {
|
||||||
|
var step *model.Step
|
||||||
|
err := yaml.Unmarshal([]byte(input), &step)
|
||||||
|
assert.NoError(t, err)
|
||||||
|
|
||||||
|
return &StepContext{
|
||||||
|
RunContext: &RunContext{
|
||||||
|
Config: &Config{
|
||||||
|
Workdir: ".",
|
||||||
|
Platforms: map[string]string{
|
||||||
|
"ubuntu-latest": "ubuntu-latest",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
StepResults: map[string]*stepResult{},
|
||||||
|
Env: map[string]string{},
|
||||||
|
Run: &model.Run{
|
||||||
|
JobID: "job1",
|
||||||
|
Workflow: &model.Workflow{
|
||||||
|
Name: "workflow1",
|
||||||
|
Jobs: map[string]*model.Job{
|
||||||
|
"job1": createJob(t, `runs-on: ubuntu-latest`, ""),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
Step: step,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestStepContextIsEnabled(t *testing.T) {
|
||||||
|
log.SetLevel(log.DebugLevel)
|
||||||
|
assertObject := assert.New(t)
|
||||||
|
|
||||||
|
// success()
|
||||||
|
sc := createIfTestStepContext(t, "if: success()")
|
||||||
|
assertObject.True(sc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
sc = createIfTestStepContext(t, "if: success()")
|
||||||
|
sc.RunContext.StepResults["a"] = &stepResult{
|
||||||
|
Conclusion: stepStatusSuccess,
|
||||||
|
}
|
||||||
|
assertObject.True(sc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
sc = createIfTestStepContext(t, "if: success()")
|
||||||
|
sc.RunContext.StepResults["a"] = &stepResult{
|
||||||
|
Conclusion: stepStatusFailure,
|
||||||
|
}
|
||||||
|
assertObject.False(sc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
// failure()
|
||||||
|
sc = createIfTestStepContext(t, "if: failure()")
|
||||||
|
assertObject.False(sc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
sc = createIfTestStepContext(t, "if: failure()")
|
||||||
|
sc.RunContext.StepResults["a"] = &stepResult{
|
||||||
|
Conclusion: stepStatusSuccess,
|
||||||
|
}
|
||||||
|
assertObject.False(sc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
sc = createIfTestStepContext(t, "if: failure()")
|
||||||
|
sc.RunContext.StepResults["a"] = &stepResult{
|
||||||
|
Conclusion: stepStatusFailure,
|
||||||
|
}
|
||||||
|
assertObject.True(sc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
// always()
|
||||||
|
sc = createIfTestStepContext(t, "if: always()")
|
||||||
|
assertObject.True(sc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
sc = createIfTestStepContext(t, "if: always()")
|
||||||
|
sc.RunContext.StepResults["a"] = &stepResult{
|
||||||
|
Conclusion: stepStatusSuccess,
|
||||||
|
}
|
||||||
|
assertObject.True(sc.isEnabled(context.Background()))
|
||||||
|
|
||||||
|
sc = createIfTestStepContext(t, "if: always()")
|
||||||
|
sc.RunContext.StepResults["a"] = &stepResult{
|
||||||
|
Conclusion: stepStatusFailure,
|
||||||
|
}
|
||||||
|
assertObject.True(sc.isEnabled(context.Background()))
|
||||||
|
}
|
||||||
|
|
29
pkg/runner/testdata/if-expressions/push.yml
vendored
Normal file
29
pkg/runner/testdata/if-expressions/push.yml
vendored
Normal file
|
@ -0,0 +1,29 @@
|
||||||
|
on: push
|
||||||
|
jobs:
|
||||||
|
mytest:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
||||||
|
# - run: exit 1
|
||||||
|
- uses: ./
|
||||||
|
if: failure()
|
||||||
|
- run: echo Success
|
||||||
|
shell: bash
|
||||||
|
- run: echo Success
|
||||||
|
if: success()
|
||||||
|
shell: bash
|
||||||
|
- run: exit 1
|
||||||
|
shell: bash
|
||||||
|
- run: echo "Shouldn't run"
|
||||||
|
if: success()
|
||||||
|
shell: bash
|
||||||
|
- run: echo "Shouldn't run2"
|
||||||
|
shell: bash
|
||||||
|
- run: echo expected to run
|
||||||
|
if: failure()
|
||||||
|
shell: bash
|
||||||
|
next:
|
||||||
|
needs: mytest
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v2
|
28
pkg/runner/testdata/job-status-check/push.yml
vendored
Normal file
28
pkg/runner/testdata/job-status-check/push.yml
vendored
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
on: push
|
||||||
|
jobs:
|
||||||
|
fail:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- run: exit 1
|
||||||
|
suc1:
|
||||||
|
if: success() || failure()
|
||||||
|
needs:
|
||||||
|
- fail
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- run: exit 0
|
||||||
|
suc2:
|
||||||
|
if: success() || failure()
|
||||||
|
needs:
|
||||||
|
- fail
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- run: exit 0
|
||||||
|
next:
|
||||||
|
needs:
|
||||||
|
- suc1
|
||||||
|
- suc2
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- run: echo should never reach here
|
||||||
|
- run: exit 1
|
Loading…
Reference in a new issue