fix: rework setupShellCommand (#930)
* fix: rework `setupShellCommand` * move all logic to separate function so we can test that later * split `step.Shell` and `step.WorkingDirectory` setup into own funcs * general cleanup of function * use `ActPath` to not collide with checked out repository * use `shellquote.Split()` instead of `strings.Fields()` for better command split * replace single string concat with `fmt` Signed-off-by: hackercat <me@hackerc.at> * lint(editorconfig): ignore *_test.go due to mixed style Signed-off-by: hackercat <me@hackerc.at>
This commit is contained in:
parent
4e0ba618d3
commit
adabf2a202
9 changed files with 168 additions and 83 deletions
|
@ -12,7 +12,7 @@ indent_size = 4
|
||||||
indent_style = space
|
indent_style = space
|
||||||
indent_size = 2
|
indent_size = 2
|
||||||
|
|
||||||
[{Dockerfile,*.md}]
|
[{Dockerfile,*.md,*_test.go}]
|
||||||
indent_style = unset
|
indent_style = unset
|
||||||
indent_size = unset
|
indent_size = unset
|
||||||
|
|
||||||
|
|
|
@ -266,6 +266,7 @@ func TestStep_ShellCommand(t *testing.T) {
|
||||||
shell string
|
shell string
|
||||||
want string
|
want string
|
||||||
}{
|
}{
|
||||||
|
{"pwsh -v '. {0}'", "pwsh -v '. {0}'"},
|
||||||
{"pwsh", "pwsh -command . '{0}'"},
|
{"pwsh", "pwsh -command . '{0}'"},
|
||||||
{"powershell", "powershell -command . '{0}'"},
|
{"powershell", "powershell -command . '{0}'"},
|
||||||
}
|
}
|
||||||
|
|
|
@ -98,6 +98,8 @@ func TestRunEvent(t *testing.T) {
|
||||||
{"testdata", "runs-on", "push", "", platforms, ""},
|
{"testdata", "runs-on", "push", "", platforms, ""},
|
||||||
{"testdata", "checkout", "push", "", platforms, ""},
|
{"testdata", "checkout", "push", "", platforms, ""},
|
||||||
{"testdata", "shells/defaults", "push", "", platforms, ""},
|
{"testdata", "shells/defaults", "push", "", platforms, ""},
|
||||||
|
// TODO: figure out why it fails
|
||||||
|
// {"testdata", "shells/custom", "push", "", map[string]string{"ubuntu-latest": "ghcr.io/justingrote/act-pwsh:latest"}, ""}, // custom image with pwsh
|
||||||
{"testdata", "shells/pwsh", "push", "", map[string]string{"ubuntu-latest": "ghcr.io/justingrote/act-pwsh:latest"}, ""}, // custom image with pwsh
|
{"testdata", "shells/pwsh", "push", "", map[string]string{"ubuntu-latest": "ghcr.io/justingrote/act-pwsh:latest"}, ""}, // custom image with pwsh
|
||||||
{"testdata", "shells/bash", "push", "", platforms, ""},
|
{"testdata", "shells/bash", "push", "", platforms, ""},
|
||||||
{"testdata", "shells/python", "push", "", map[string]string{"ubuntu-latest": "node:12-buster"}, ""}, // slim doesn't have python
|
{"testdata", "shells/python", "push", "", map[string]string{"ubuntu-latest": "node:12-buster"}, ""}, // slim doesn't have python
|
||||||
|
|
|
@ -53,7 +53,7 @@ func (sc *StepContext) Executor() common.Executor {
|
||||||
switch step.Type() {
|
switch step.Type() {
|
||||||
case model.StepTypeRun:
|
case model.StepTypeRun:
|
||||||
return common.NewPipelineExecutor(
|
return common.NewPipelineExecutor(
|
||||||
sc.setupShellCommand(),
|
sc.setupShellCommandExecutor(),
|
||||||
sc.execJobContainer(),
|
sc.execJobContainer(),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -185,87 +185,106 @@ func (sc *StepContext) setupEnv(ctx context.Context) (ExpressionEvaluator, error
|
||||||
return evaluator, nil
|
return evaluator, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// nolint:gocyclo
|
func (sc *StepContext) setupWorkingDirectory() {
|
||||||
func (sc *StepContext) setupShellCommand() common.Executor {
|
|
||||||
rc := sc.RunContext
|
rc := sc.RunContext
|
||||||
step := sc.Step
|
step := sc.Step
|
||||||
return func(ctx context.Context) error {
|
|
||||||
var script strings.Builder
|
|
||||||
var err error
|
|
||||||
|
|
||||||
if step.WorkingDirectory == "" {
|
if step.WorkingDirectory == "" {
|
||||||
step.WorkingDirectory = rc.Run.Job().Defaults.Run.WorkingDirectory
|
step.WorkingDirectory = rc.Run.Job().Defaults.Run.WorkingDirectory
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// jobs can receive context values, so we interpolate
|
||||||
|
step.WorkingDirectory = rc.ExprEval.Interpolate(step.WorkingDirectory)
|
||||||
|
|
||||||
|
// but top level keys in workflow file like `defaults` or `env` can't
|
||||||
if step.WorkingDirectory == "" {
|
if step.WorkingDirectory == "" {
|
||||||
step.WorkingDirectory = rc.Run.Workflow.Defaults.Run.WorkingDirectory
|
step.WorkingDirectory = rc.Run.Workflow.Defaults.Run.WorkingDirectory
|
||||||
}
|
}
|
||||||
step.WorkingDirectory = rc.ExprEval.Interpolate(step.WorkingDirectory)
|
|
||||||
|
|
||||||
run := rc.ExprEval.Interpolate(step.Run)
|
|
||||||
step.Shell = rc.ExprEval.Interpolate(step.Shell)
|
|
||||||
|
|
||||||
if _, err = script.WriteString(run); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
scriptName := fmt.Sprintf("workflow/%s", step.ID)
|
|
||||||
|
|
||||||
// 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 := ""
|
|
||||||
scriptExt := ""
|
|
||||||
switch step.Shell {
|
|
||||||
case "bash", "sh":
|
|
||||||
scriptExt = ".sh"
|
|
||||||
case "pwsh", "powershell":
|
|
||||||
scriptExt = ".ps1"
|
|
||||||
runPrepend = "$ErrorActionPreference = 'stop'"
|
|
||||||
runAppend = "if ((Test-Path -LiteralPath variable:/LASTEXITCODE)) { exit $LASTEXITCODE }"
|
|
||||||
case "cmd":
|
|
||||||
scriptExt = ".cmd"
|
|
||||||
runPrepend = "@echo off"
|
|
||||||
case "python":
|
|
||||||
scriptExt = ".py"
|
|
||||||
}
|
}
|
||||||
|
|
||||||
scriptName += scriptExt
|
func (sc *StepContext) setupShell() {
|
||||||
run = runPrepend + "\n" + run + "\n" + runAppend
|
rc := sc.RunContext
|
||||||
|
step := sc.Step
|
||||||
log.Debugf("Wrote command '%s' to '%s'", run, scriptName)
|
|
||||||
scriptPath := fmt.Sprintf("%s/%s", rc.Config.ContainerWorkdir(), scriptName)
|
|
||||||
|
|
||||||
if step.Shell == "" {
|
if step.Shell == "" {
|
||||||
step.Shell = rc.Run.Job().Defaults.Run.Shell
|
step.Shell = rc.Run.Job().Defaults.Run.Shell
|
||||||
}
|
}
|
||||||
|
|
||||||
|
step.Shell = rc.ExprEval.Interpolate(step.Shell)
|
||||||
|
|
||||||
if step.Shell == "" {
|
if step.Shell == "" {
|
||||||
step.Shell = rc.Run.Workflow.Defaults.Run.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() != nil {
|
||||||
if rc.Run.Job().Container().Image != "" && step.Shell == "" {
|
if rc.Run.Job().Container().Image != "" && step.Shell == "" {
|
||||||
step.Shell = "sh"
|
step.Shell = "sh"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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 (sc *StepContext) setupShellCommand() (name, script string, err error) {
|
||||||
|
sc.setupShell()
|
||||||
|
sc.setupWorkingDirectory()
|
||||||
|
|
||||||
|
step := sc.Step
|
||||||
|
|
||||||
|
script = sc.RunContext.ExprEval.Interpolate(step.Run)
|
||||||
|
|
||||||
scCmd := step.ShellCommand()
|
scCmd := step.ShellCommand()
|
||||||
|
|
||||||
var finalCMD []string
|
name = fmt.Sprintf("workflow/%s", step.ID)
|
||||||
if step.Shell == "pwsh" || step.Shell == "powershell" {
|
|
||||||
finalCMD = strings.SplitN(scCmd, " ", 3)
|
// Reference: https://github.com/actions/runner/blob/8109c962f09d9acc473d92c595ff43afceddb347/src/Runner.Worker/Handlers/ScriptHandlerHelpers.cs#L47-L64
|
||||||
} else {
|
// Reference: https://github.com/actions/runner/blob/8109c962f09d9acc473d92c595ff43afceddb347/src/Runner.Worker/Handlers/ScriptHandlerHelpers.cs#L19-L27
|
||||||
finalCMD = strings.Fields(scCmd)
|
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"
|
||||||
}
|
}
|
||||||
|
|
||||||
for k, v := range finalCMD {
|
script = fmt.Sprintf("%s\n%s\n%s", runPrepend, script, runAppend)
|
||||||
if strings.Contains(v, `{0}`) {
|
|
||||||
finalCMD[k] = strings.Replace(v, `{0}`, scriptPath, 1)
|
log.Debugf("Wrote command \n%s\n to '%s'", script, name)
|
||||||
}
|
|
||||||
|
scriptPath := fmt.Sprintf("%s/%s", ActPath, name)
|
||||||
|
sc.Cmd, err = shellquote.Split(strings.Replace(scCmd, `{0}`, scriptPath, 1))
|
||||||
|
|
||||||
|
return name, script, err
|
||||||
}
|
}
|
||||||
|
|
||||||
sc.Cmd = finalCMD
|
func (sc *StepContext) setupShellCommandExecutor() common.Executor {
|
||||||
|
rc := sc.RunContext
|
||||||
|
return func(ctx context.Context) error {
|
||||||
|
scriptName, script, err := sc.setupShellCommand()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
return rc.JobContainer.Copy(rc.Config.ContainerWorkdir(), &container.FileEntry{
|
return rc.JobContainer.Copy(ActPath, &container.FileEntry{
|
||||||
Name: scriptName,
|
Name: scriptName,
|
||||||
Mode: 0755,
|
Mode: 0755,
|
||||||
Body: script.String(),
|
Body: script,
|
||||||
})(ctx)
|
})(ctx)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
18
pkg/runner/testdata/shells/bash/push.yml
vendored
18
pkg/runner/testdata/shells/bash/push.yml
vendored
|
@ -1,9 +1,11 @@
|
||||||
on: push
|
on: push
|
||||||
|
env:
|
||||||
|
MY_SHELL: bash
|
||||||
jobs:
|
jobs:
|
||||||
check:
|
check:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- shell: bash
|
- shell: ${{ env.MY_SHELL }}
|
||||||
run: |
|
run: |
|
||||||
if [[ -n "$BASH" ]]; then
|
if [[ -n "$BASH" ]]; then
|
||||||
echo "I'm $BASH!"
|
echo "I'm $BASH!"
|
||||||
|
@ -14,10 +16,22 @@ jobs:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
container: node:12-buster-slim
|
container: node:12-buster-slim
|
||||||
steps:
|
steps:
|
||||||
- shell: bash
|
- shell: ${{ env.MY_SHELL }}
|
||||||
run: |
|
run: |
|
||||||
if [[ -n "$BASH" ]]; then
|
if [[ -n "$BASH" ]]; then
|
||||||
echo "I'm $BASH!"
|
echo "I'm $BASH!"
|
||||||
else
|
else
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
check-job-default:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
defaults:
|
||||||
|
run:
|
||||||
|
shell: ${{ env.MY_SHELL }}
|
||||||
|
steps:
|
||||||
|
- run: |
|
||||||
|
if [[ -n "$BASH" ]]; then
|
||||||
|
echo "I'm $BASH!"
|
||||||
|
else
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
14
pkg/runner/testdata/shells/custom/push.yml
vendored
Normal file
14
pkg/runner/testdata/shells/custom/push.yml
vendored
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
on: push
|
||||||
|
jobs:
|
||||||
|
check:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
# prints version and exits, it's not valid (for github) if {0} is not included
|
||||||
|
- shell: pwsh -v '. {0}'
|
||||||
|
run: ''
|
||||||
|
check-container:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
container: ghcr.io/justingrote/act-pwsh:latest
|
||||||
|
steps:
|
||||||
|
- shell: pwsh -v '. {0}'
|
||||||
|
run: ''
|
14
pkg/runner/testdata/shells/pwsh/push.yml
vendored
14
pkg/runner/testdata/shells/pwsh/push.yml
vendored
|
@ -1,15 +1,25 @@
|
||||||
on: push
|
on: push
|
||||||
|
env:
|
||||||
|
MY_SHELL: pwsh
|
||||||
jobs:
|
jobs:
|
||||||
check:
|
check:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- shell: pwsh
|
- shell: ${{ env.MY_SHELL }}
|
||||||
run: |
|
run: |
|
||||||
$PSVersionTable
|
$PSVersionTable
|
||||||
check-container:
|
check-container:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
container: ghcr.io/justingrote/act-pwsh:latest
|
container: ghcr.io/justingrote/act-pwsh:latest
|
||||||
steps:
|
steps:
|
||||||
- shell: pwsh
|
- shell: ${{ env.MY_SHELL }}
|
||||||
run: |
|
run: |
|
||||||
$PSVersionTable
|
$PSVersionTable
|
||||||
|
check-job-default:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
defaults:
|
||||||
|
run:
|
||||||
|
shell: ${{ env.MY_SHELL }}
|
||||||
|
steps:
|
||||||
|
- run: |
|
||||||
|
$PSVersionTable
|
||||||
|
|
15
pkg/runner/testdata/shells/python/push.yml
vendored
15
pkg/runner/testdata/shells/python/push.yml
vendored
|
@ -1,9 +1,11 @@
|
||||||
on: push
|
on: push
|
||||||
|
env:
|
||||||
|
MY_SHELL: python
|
||||||
jobs:
|
jobs:
|
||||||
check:
|
check:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- shell: python
|
- shell: ${{ env.MY_SHELL }}
|
||||||
run: |
|
run: |
|
||||||
import platform
|
import platform
|
||||||
print(platform.python_version())
|
print(platform.python_version())
|
||||||
|
@ -11,7 +13,16 @@ jobs:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
container: node:12-buster
|
container: node:12-buster
|
||||||
steps:
|
steps:
|
||||||
- shell: python
|
- shell: ${{ env.MY_SHELL }}
|
||||||
run: |
|
run: |
|
||||||
import platform
|
import platform
|
||||||
print(platform.python_version())
|
print(platform.python_version())
|
||||||
|
check-job-default:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
defaults:
|
||||||
|
run:
|
||||||
|
shell: ${{ env.MY_SHELL }}
|
||||||
|
steps:
|
||||||
|
- run: |
|
||||||
|
import platform
|
||||||
|
print(platform.python_version())
|
||||||
|
|
18
pkg/runner/testdata/shells/sh/push.yml
vendored
18
pkg/runner/testdata/shells/sh/push.yml
vendored
|
@ -1,9 +1,11 @@
|
||||||
on: push
|
on: push
|
||||||
|
env:
|
||||||
|
MY_SHELL: sh
|
||||||
jobs:
|
jobs:
|
||||||
check:
|
check:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- shell: sh
|
- shell: ${{ env.MY_SHELL }}
|
||||||
run: |
|
run: |
|
||||||
if [ -z ${BASH+x} ]; then
|
if [ -z ${BASH+x} ]; then
|
||||||
echo "I'm sh!"
|
echo "I'm sh!"
|
||||||
|
@ -14,10 +16,22 @@ jobs:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
container: alpine:latest
|
container: alpine:latest
|
||||||
steps:
|
steps:
|
||||||
- shell: sh
|
- shell: ${{ env.MY_SHELL }}
|
||||||
run: |
|
run: |
|
||||||
if [ -z ${BASH+x} ]; then
|
if [ -z ${BASH+x} ]; then
|
||||||
echo "I'm sh!"
|
echo "I'm sh!"
|
||||||
else
|
else
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
check-job-default:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
defaults:
|
||||||
|
run:
|
||||||
|
shell: ${{ env.MY_SHELL }}
|
||||||
|
steps:
|
||||||
|
- run: |
|
||||||
|
if [ -z ${BASH+x} ]; then
|
||||||
|
echo "I'm sh!"
|
||||||
|
else
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
Loading…
Reference in a new issue