| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | package runner | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"context" | 
					
						
							|  |  |  | 	"fmt" | 
					
						
							| 
									
										
										
										
											2025-08-15 04:54:13 +00:00
										 |  |  | 	"maps" | 
					
						
							| 
									
										
										
										
											2023-08-08 16:14:46 +02:00
										 |  |  | 	"runtime" | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 	"strings" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	"github.com/kballard/go-shellquote" | 
					
						
							| 
									
										
										
										
											2023-02-16 23:34:51 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-09-05 07:29:38 +00:00
										 |  |  | 	"code.forgejo.org/forgejo/runner/v11/act/common" | 
					
						
							|  |  |  | 	"code.forgejo.org/forgejo/runner/v11/act/container" | 
					
						
							|  |  |  | 	"code.forgejo.org/forgejo/runner/v11/act/lookpath" | 
					
						
							|  |  |  | 	"code.forgejo.org/forgejo/runner/v11/act/model" | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type stepRun struct { | 
					
						
							| 
									
										
										
										
											2023-03-28 14:24:03 +02:00
										 |  |  | 	Step             *model.Step | 
					
						
							|  |  |  | 	RunContext       *RunContext | 
					
						
							|  |  |  | 	cmd              []string | 
					
						
							| 
									
										
										
										
											2023-08-08 17:44:25 +02:00
										 |  |  | 	cmdline          string | 
					
						
							| 
									
										
										
										
											2023-03-28 14:24:03 +02:00
										 |  |  | 	env              map[string]string | 
					
						
							|  |  |  | 	WorkingDirectory string | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 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{} | 
					
						
							| 
									
										
										
										
											2022-05-24 15:36:06 +02:00
										 |  |  | 	return runStepExecutor(sr, stepStageMain, common.NewPipelineExecutor( | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 		sr.setupShellCommandExecutor(), | 
					
						
							|  |  |  | 		func(ctx context.Context) error { | 
					
						
							| 
									
										
										
										
											2023-02-04 14:35:13 +01:00
										 |  |  | 			sr.getRunContext().ApplyExtraPath(ctx, &sr.env) | 
					
						
							| 
									
										
										
										
											2023-08-08 17:44:25 +02:00
										 |  |  | 			if he, ok := sr.getRunContext().JobContainer.(*container.HostEnvironment); ok && he != nil { | 
					
						
							|  |  |  | 				return he.ExecWithCmdLine(sr.cmd, sr.cmdline, sr.env, "", sr.WorkingDirectory)(ctx) | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2023-03-28 14:24:03 +02:00
										 |  |  | 			return sr.getRunContext().JobContainer.Exec(sr.cmd, sr.env, "", sr.WorkingDirectory)(ctx) | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 		}, | 
					
						
							|  |  |  | 	)) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (sr *stepRun) post() common.Executor { | 
					
						
							|  |  |  | 	return func(ctx context.Context) error { | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (sr *stepRun) getRunContext() *RunContext { | 
					
						
							|  |  |  | 	return sr.RunContext | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-06 23:58:16 +02:00
										 |  |  | func (sr *stepRun) getGithubContext(ctx context.Context) *model.GithubContext { | 
					
						
							|  |  |  | 	return sr.getRunContext().getGithubContext(ctx) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | func (sr *stepRun) getStepModel() *model.Step { | 
					
						
							|  |  |  | 	return sr.Step | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (sr *stepRun) getEnv() *map[string]string { | 
					
						
							|  |  |  | 	return &sr.env | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-07-10 17:12:12 -07:00
										 |  |  | func (sr *stepRun) getIfExpression(_ context.Context, _ stepStage) string { | 
					
						
							| 
									
										
										
										
											2022-05-24 15:36:06 +02:00
										 |  |  | 	return sr.Step.If.Value | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | func (sr *stepRun) setupShellCommandExecutor() common.Executor { | 
					
						
							|  |  |  | 	return func(ctx context.Context) error { | 
					
						
							| 
									
										
										
										
											2022-05-11 21:06:05 +02:00
										 |  |  | 		scriptName, script, err := sr.setupShellCommand(ctx) | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 		if err != nil { | 
					
						
							|  |  |  | 			return err | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 22:29:45 +01:00
										 |  |  | 		rc := sr.getRunContext() | 
					
						
							|  |  |  | 		return rc.JobContainer.Copy(rc.JobContainer.GetActPath(), &container.FileEntry{ | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 			Name: scriptName, | 
					
						
							| 
									
										
										
										
											2023-02-16 23:34:51 +08:00
										 |  |  | 			Mode: 0o755, | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 			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) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | func shellCommand(shell string) string { | 
					
						
							|  |  |  | 	// Reference: https://github.com/actions/runner/blob/8109c962f09d9acc473d92c595ff43afceddb347/src/Runner.Worker/Handlers/ScriptHandlerHelpers.cs#L9-L17 | 
					
						
							|  |  |  | 	switch shell { | 
					
						
							|  |  |  | 	case "", "bash": | 
					
						
							|  |  |  | 		return "bash --noprofile --norc -e -o pipefail {0}" | 
					
						
							|  |  |  | 	case "pwsh": | 
					
						
							|  |  |  | 		return "pwsh -command . '{0}'" | 
					
						
							|  |  |  | 	case "python": | 
					
						
							|  |  |  | 		return "python {0}" | 
					
						
							|  |  |  | 	case "sh": | 
					
						
							|  |  |  | 		return "sh -e {0}" | 
					
						
							|  |  |  | 	case "cmd": | 
					
						
							|  |  |  | 		return "cmd /D /E:ON /V:OFF /S /C \"CALL \"{0}\"\"" | 
					
						
							|  |  |  | 	case "powershell": | 
					
						
							|  |  |  | 		return "powershell -command . '{0}'" | 
					
						
							|  |  |  | 	case "node": | 
					
						
							|  |  |  | 		return "node {0}" | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return shell | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | // 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 | 
					
						
							| 
									
										
										
										
											2022-05-11 21:06:05 +02:00
										 |  |  | func (sr *stepRun) setupShellCommand(ctx context.Context) (name, script string, err error) { | 
					
						
							| 
									
										
										
										
											2022-06-17 17:55:21 +02:00
										 |  |  | 	logger := common.Logger(ctx) | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | 	shell := sr.interpretShell(ctx) | 
					
						
							| 
									
										
										
										
											2022-06-17 17:55:21 +02:00
										 |  |  | 	sr.setupWorkingDirectory(ctx) | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	step := sr.Step | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 17:55:21 +02:00
										 |  |  | 	script = sr.RunContext.NewStepExpressionEvaluator(ctx, sr).Interpolate(ctx, step.Run) | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | 	shellCommand := shellCommand(shell) | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 	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 := "" | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | 	switch shell { | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 	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" | 
					
						
							| 
									
										
										
										
											2025-04-18 18:14:21 +03:00
										 |  |  | 	case "node": | 
					
						
							|  |  |  | 		name += ".js" | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	script = fmt.Sprintf("%s\n%s\n%s", runPrepend, script, runAppend) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-05-11 21:06:05 +02:00
										 |  |  | 	if !strings.Contains(script, "::add-mask::") && !sr.RunContext.Config.InsecureSecrets { | 
					
						
							| 
									
										
										
										
											2022-06-17 17:55:21 +02:00
										 |  |  | 		logger.Debugf("Wrote command \n%s\n to '%s'", script, name) | 
					
						
							| 
									
										
										
										
											2022-05-11 21:06:05 +02:00
										 |  |  | 	} else { | 
					
						
							| 
									
										
										
										
											2022-06-17 17:55:21 +02:00
										 |  |  | 		logger.Debugf("Wrote add-mask command to '%s'", name) | 
					
						
							| 
									
										
										
										
											2022-05-11 21:06:05 +02:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-16 22:29:45 +01:00
										 |  |  | 	rc := sr.getRunContext() | 
					
						
							|  |  |  | 	scriptPath := fmt.Sprintf("%s/%s", rc.JobContainer.GetActPath(), name) | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | 	sr.cmdline = strings.Replace(shellCommand, `{0}`, scriptPath, 1) | 
					
						
							| 
									
										
										
										
											2023-08-08 17:44:25 +02:00
										 |  |  | 	sr.cmd, err = shellquote.Split(sr.cmdline) | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return name, script, err | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-08-08 16:14:46 +02:00
										 |  |  | type localEnv struct { | 
					
						
							|  |  |  | 	env map[string]string | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (l *localEnv) Getenv(name string) string { | 
					
						
							|  |  |  | 	if runtime.GOOS == "windows" { | 
					
						
							|  |  |  | 		for k, v := range l.env { | 
					
						
							|  |  |  | 			if strings.EqualFold(name, k) { | 
					
						
							|  |  |  | 				return v | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		return "" | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return l.env[name] | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | func (sr *stepRun) interpretShell(ctx context.Context) string { | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 	rc := sr.RunContext | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | 	shell := sr.Step.RawShell | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | 	if shell == "" { | 
					
						
							|  |  |  | 		shell = rc.Run.Job().Defaults.Run.Shell | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | 	shell = rc.NewExpressionEvaluator(ctx).Interpolate(ctx, shell) | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | 	if shell == "" { | 
					
						
							|  |  |  | 		shell = rc.Run.Workflow.Defaults.Run.Shell | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | 	if shell == "" { | 
					
						
							| 
									
										
										
										
											2023-08-08 16:14:46 +02:00
										 |  |  | 		if _, ok := rc.JobContainer.(*container.HostEnvironment); ok { | 
					
						
							|  |  |  | 			shellWithFallback := []string{"bash", "sh"} | 
					
						
							|  |  |  | 			// Don't use bash on windows by default, if not using a docker container | 
					
						
							|  |  |  | 			if runtime.GOOS == "windows" { | 
					
						
							|  |  |  | 				shellWithFallback = []string{"pwsh", "powershell"} | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | 			shell = shellWithFallback[0] | 
					
						
							| 
									
										
										
										
											2023-08-08 16:14:46 +02:00
										 |  |  | 			lenv := &localEnv{env: map[string]string{}} | 
					
						
							| 
									
										
										
										
											2025-08-15 04:54:13 +00:00
										 |  |  | 			maps.Copy(lenv.env, sr.env) | 
					
						
							| 
									
										
										
										
											2023-08-08 16:14:46 +02:00
										 |  |  | 			sr.getRunContext().ApplyExtraPath(ctx, &lenv.env) | 
					
						
							|  |  |  | 			_, err := lookpath.LookPath2(shellWithFallback[0], lenv) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | 				shell = shellWithFallback[1] | 
					
						
							| 
									
										
										
										
											2023-08-08 16:14:46 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2025-07-12 07:53:43 +02:00
										 |  |  | 		} else { | 
					
						
							| 
									
										
										
										
											2025-07-28 12:26:41 +00:00
										 |  |  | 			shellFallback := ` | 
					
						
							| 
									
										
										
										
											2025-07-12 07:53:43 +02:00
										 |  |  | if command -v bash >/dev/null; then | 
					
						
							|  |  |  | 	echo -n bash | 
					
						
							|  |  |  | else | 
					
						
							|  |  |  | 	echo -n sh | 
					
						
							|  |  |  | fi | 
					
						
							|  |  |  | ` | 
					
						
							| 
									
										
										
										
											2025-07-28 12:26:41 +00:00
										 |  |  | 			stdout, _, err := rc.sh(ctx, shellFallback) | 
					
						
							| 
									
										
										
										
											2025-07-12 07:53:43 +02:00
										 |  |  | 			if err != nil { | 
					
						
							| 
									
										
										
										
											2025-07-28 12:26:41 +00:00
										 |  |  | 				common.Logger(ctx).Error("fail to run %q: %v", shellFallback, err) | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | 			} else { | 
					
						
							|  |  |  | 				shell = stdout | 
					
						
							| 
									
										
										
										
											2025-07-12 07:53:43 +02:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2025-08-15 21:10:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return shell | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-06-17 17:55:21 +02:00
										 |  |  | func (sr *stepRun) setupWorkingDirectory(ctx context.Context) { | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 	rc := sr.RunContext | 
					
						
							|  |  |  | 	step := sr.Step | 
					
						
							| 
									
										
										
										
											2025-07-28 12:26:41 +00:00
										 |  |  | 	var workingdirectory string | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if step.WorkingDirectory == "" { | 
					
						
							| 
									
										
										
										
											2023-03-28 14:24:03 +02:00
										 |  |  | 		workingdirectory = rc.Run.Job().Defaults.Run.WorkingDirectory | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		workingdirectory = step.WorkingDirectory | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// jobs can receive context values, so we interpolate | 
					
						
							| 
									
										
										
										
											2023-03-28 14:24:03 +02:00
										 |  |  | 	workingdirectory = rc.NewExpressionEvaluator(ctx).Interpolate(ctx, workingdirectory) | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// but top level keys in workflow file like `defaults` or `env` can't | 
					
						
							| 
									
										
										
										
											2023-03-28 14:24:03 +02:00
										 |  |  | 	if workingdirectory == "" { | 
					
						
							|  |  |  | 		workingdirectory = rc.Run.Workflow.Defaults.Run.WorkingDirectory | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2023-03-28 14:24:03 +02:00
										 |  |  | 	sr.WorkingDirectory = workingdirectory | 
					
						
							| 
									
										
										
										
											2022-03-22 22:13:00 +01:00
										 |  |  | } |