679cac1677
* test: add test case for #1319 * fix: setup of composite inputs This change fixes the composite action setup handling of inputs. All inputs are taken from the env now. The env is composed of the 'level above'. For example: - step env -> taken from run context - action env -> taken from step env - composite env -> taken from action env Before this change the env setup for steps, actions and composite run contexts was harder to understand as all parts looked into one of these: parent run context, step, action, composite run context. Now the 'data flow' is from higher levels to lower levels which should make it more clean. Fixes #1319 * test: add simple remote composite action test Since we don't have a remote composite test at all before this, we need at least the simplest case. This does not check every feature, but ensures basic availability of remote composite actions. * refactor: move ActionRef and ActionRepository Moving ActionRef and ActionRepository from RunContext into the step, allows us to remove the - more or less - ugly copy operations from the RunContext. This is more clean, as each step does hold the data required anyway and the RunContext shouldn't know about the action details. * refactor: remove unused properties
178 lines
5.1 KiB
Go
178 lines
5.1 KiB
Go
package runner
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"strings"
|
|
|
|
"github.com/kballard/go-shellquote"
|
|
"github.com/nektos/act/pkg/common"
|
|
"github.com/nektos/act/pkg/container"
|
|
"github.com/nektos/act/pkg/model"
|
|
)
|
|
|
|
type stepRun struct {
|
|
Step *model.Step
|
|
RunContext *RunContext
|
|
cmd []string
|
|
env map[string]string
|
|
}
|
|
|
|
func (sr *stepRun) pre() common.Executor {
|
|
return func(ctx context.Context) error {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func (sr *stepRun) main() common.Executor {
|
|
sr.env = map[string]string{}
|
|
|
|
return runStepExecutor(sr, stepStageMain, common.NewPipelineExecutor(
|
|
sr.setupShellCommandExecutor(),
|
|
func(ctx context.Context) error {
|
|
return sr.getRunContext().JobContainer.Exec(sr.cmd, sr.env, "", sr.Step.WorkingDirectory)(ctx)
|
|
},
|
|
))
|
|
}
|
|
|
|
func (sr *stepRun) post() common.Executor {
|
|
return func(ctx context.Context) error {
|
|
return nil
|
|
}
|
|
}
|
|
|
|
func (sr *stepRun) getRunContext() *RunContext {
|
|
return sr.RunContext
|
|
}
|
|
|
|
func (sr *stepRun) getGithubContext(ctx context.Context) *model.GithubContext {
|
|
return sr.getRunContext().getGithubContext(ctx)
|
|
}
|
|
|
|
func (sr *stepRun) getStepModel() *model.Step {
|
|
return sr.Step
|
|
}
|
|
|
|
func (sr *stepRun) getEnv() *map[string]string {
|
|
return &sr.env
|
|
}
|
|
|
|
func (sr *stepRun) getIfExpression(context context.Context, stage stepStage) string {
|
|
return sr.Step.If.Value
|
|
}
|
|
|
|
func (sr *stepRun) setupShellCommandExecutor() common.Executor {
|
|
return func(ctx context.Context) error {
|
|
scriptName, script, err := sr.setupShellCommand(ctx)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return sr.RunContext.JobContainer.Copy(ActPath, &container.FileEntry{
|
|
Name: scriptName,
|
|
Mode: 0755,
|
|
Body: script,
|
|
})(ctx)
|
|
}
|
|
}
|
|
|
|
func getScriptName(rc *RunContext, step *model.Step) string {
|
|
scriptName := step.ID
|
|
for rcs := rc; rcs.Parent != nil; rcs = rcs.Parent {
|
|
scriptName = fmt.Sprintf("%s-composite-%s", rcs.Parent.CurrentStep, scriptName)
|
|
}
|
|
return fmt.Sprintf("workflow/%s", scriptName)
|
|
}
|
|
|
|
// TODO: Currently we just ignore top level keys, BUT we should return proper error on them
|
|
// BUTx2 I leave this for when we rewrite act to use actionlint for workflow validation
|
|
// so we return proper errors before any execution or spawning containers
|
|
// it will error anyway with:
|
|
// OCI runtime exec failed: exec failed: container_linux.go:380: starting container process caused: exec: "${{": executable file not found in $PATH: unknown
|
|
func (sr *stepRun) setupShellCommand(ctx context.Context) (name, script string, err error) {
|
|
logger := common.Logger(ctx)
|
|
sr.setupShell(ctx)
|
|
sr.setupWorkingDirectory(ctx)
|
|
|
|
step := sr.Step
|
|
|
|
script = sr.RunContext.NewStepExpressionEvaluator(ctx, sr).Interpolate(ctx, step.Run)
|
|
|
|
scCmd := step.ShellCommand()
|
|
|
|
name = getScriptName(sr.RunContext, step)
|
|
|
|
// Reference: https://github.com/actions/runner/blob/8109c962f09d9acc473d92c595ff43afceddb347/src/Runner.Worker/Handlers/ScriptHandlerHelpers.cs#L47-L64
|
|
// Reference: https://github.com/actions/runner/blob/8109c962f09d9acc473d92c595ff43afceddb347/src/Runner.Worker/Handlers/ScriptHandlerHelpers.cs#L19-L27
|
|
runPrepend := ""
|
|
runAppend := ""
|
|
switch step.Shell {
|
|
case "bash", "sh":
|
|
name += ".sh"
|
|
case "pwsh", "powershell":
|
|
name += ".ps1"
|
|
runPrepend = "$ErrorActionPreference = 'stop'"
|
|
runAppend = "if ((Test-Path -LiteralPath variable:/LASTEXITCODE)) { exit $LASTEXITCODE }"
|
|
case "cmd":
|
|
name += ".cmd"
|
|
runPrepend = "@echo off"
|
|
case "python":
|
|
name += ".py"
|
|
}
|
|
|
|
script = fmt.Sprintf("%s\n%s\n%s", runPrepend, script, runAppend)
|
|
|
|
if !strings.Contains(script, "::add-mask::") && !sr.RunContext.Config.InsecureSecrets {
|
|
logger.Debugf("Wrote command \n%s\n to '%s'", script, name)
|
|
} else {
|
|
logger.Debugf("Wrote add-mask command to '%s'", name)
|
|
}
|
|
|
|
scriptPath := fmt.Sprintf("%s/%s", ActPath, name)
|
|
sr.cmd, err = shellquote.Split(strings.Replace(scCmd, `{0}`, scriptPath, 1))
|
|
|
|
return name, script, err
|
|
}
|
|
|
|
func (sr *stepRun) setupShell(ctx context.Context) {
|
|
rc := sr.RunContext
|
|
step := sr.Step
|
|
|
|
if step.Shell == "" {
|
|
step.Shell = rc.Run.Job().Defaults.Run.Shell
|
|
}
|
|
|
|
step.Shell = rc.NewExpressionEvaluator(ctx).Interpolate(ctx, step.Shell)
|
|
|
|
if step.Shell == "" {
|
|
step.Shell = rc.Run.Workflow.Defaults.Run.Shell
|
|
}
|
|
|
|
// current GitHub Runner behaviour is that default is `sh`,
|
|
// but if it's not container it validates with `which` command
|
|
// if `bash` is available, and provides `bash` if it is
|
|
// for now I'm going to leave below logic, will address it in different PR
|
|
// https://github.com/actions/runner/blob/9a829995e02d2db64efb939dc2f283002595d4d9/src/Runner.Worker/Handlers/ScriptHandler.cs#L87-L91
|
|
if rc.Run.Job().Container() != nil {
|
|
if rc.Run.Job().Container().Image != "" && step.Shell == "" {
|
|
step.Shell = "sh"
|
|
}
|
|
}
|
|
}
|
|
|
|
func (sr *stepRun) setupWorkingDirectory(ctx context.Context) {
|
|
rc := sr.RunContext
|
|
step := sr.Step
|
|
|
|
if step.WorkingDirectory == "" {
|
|
step.WorkingDirectory = rc.Run.Job().Defaults.Run.WorkingDirectory
|
|
}
|
|
|
|
// jobs can receive context values, so we interpolate
|
|
step.WorkingDirectory = rc.NewExpressionEvaluator(ctx).Interpolate(ctx, step.WorkingDirectory)
|
|
|
|
// but top level keys in workflow file like `defaults` or `env` can't
|
|
if step.WorkingDirectory == "" {
|
|
step.WorkingDirectory = rc.Run.Workflow.Defaults.Run.WorkingDirectory
|
|
}
|
|
}
|