mirror of
https://gitea.com/gitea/act_runner.git
synced 2026-05-01 16:30:20 +08:00
## Consumer-facing breaking changes
- **Go module path**: `gitea.com/gitea/act_runner` → `gitea.com/gitea/runner`. Anything importing `act/...` or `internal/...` packages (notably Gitea itself) must update imports.
- **Binary name**: `act_runner` → `gitea-runner`. Wrapper scripts, systemd units, init scripts, and documentation referencing the binary by `act_runner` will break.
- **Docker image**: `gitea/act_runner` → `gitea/runner` (incl. `*-dind-rootless` variants). Users pulling `gitea/act_runner:nightly` etc. will get stale images. Note: the image name is `gitea/runner`, not `gitea/gitea-runner`.
- **Release artifact paths**: S3 directory `act_runner/{{.Version}}` → `gitea-runner/{{.Version}}`, and artifact filenames change with the new project name. Existing download URLs break.
- **Metrics namespace**: changed from `act_runner` to `gitea_runner` (e.g. `act_runner_jobs_total` → `gitea_runner_jobs_total`); existing monitors/dashboards must be updated.
- **ldflags version path**: `gitea.com/gitea/act_runner/internal/pkg/ver.version` → `gitea.com/gitea/runner/internal/pkg/ver.version`. Affects anyone building with custom ldflags.
- **Kubernetes example resource names**: `act-runner` / `act-runner-vol` → `runner` / `runner-vol`. Users who copied the manifests verbatim will see resource churn on apply.
- **s6 service name**: `scripts/s6/act_runner/` → `scripts/s6/gitea-runner/` (image-internal; only matters for downstream image overrides).
Unchanged: YAML config field names, env vars (`GITEA_*`), CLI flags/subcommands, registration file format.
---------
Co-authored-by: silverwind <me@silverwind.io>
Reviewed-on: https://gitea.com/gitea/runner/pulls/850
Reviewed-by: Zettat123 <39446+zettat123@noreply.gitea.com>
Reviewed-by: silverwind <2021+silverwind@noreply.gitea.com>
Reviewed-by: Nicolas <bircni@icloud.com>
Co-authored-by: Lunny Xiao <xiaolunwen@gmail.com>
Co-committed-by: Lunny Xiao <xiaolunwen@gmail.com>
201 lines
4.9 KiB
Go
201 lines
4.9 KiB
Go
// Copyright 2022 The Gitea Authors. All rights reserved.
|
|
// Copyright 2020 The nektos/act Authors. All rights reserved.
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
package runner
|
|
|
|
import (
|
|
"context"
|
|
"regexp"
|
|
"strings"
|
|
|
|
"gitea.com/gitea/runner/act/common"
|
|
)
|
|
|
|
var commandPatternGA *regexp.Regexp
|
|
|
|
var commandPatternADO *regexp.Regexp
|
|
|
|
func init() {
|
|
commandPatternGA = regexp.MustCompile("^::([^ ]+)( (.+))?::([^\r\n]*)[\r\n]+$")
|
|
commandPatternADO = regexp.MustCompile("^##\\[([^ ]+)( (.+))?]([^\r\n]*)[\r\n]+$")
|
|
}
|
|
|
|
func tryParseRawActionCommand(line string) (command string, kvPairs map[string]string, arg string, ok bool) {
|
|
if m := commandPatternGA.FindStringSubmatch(line); m != nil {
|
|
command = m[1]
|
|
kvPairs = parseKeyValuePairs(m[3], ",")
|
|
arg = m[4]
|
|
ok = true
|
|
} else if m := commandPatternADO.FindStringSubmatch(line); m != nil {
|
|
command = m[1]
|
|
kvPairs = parseKeyValuePairs(m[3], ";")
|
|
arg = m[4]
|
|
ok = true
|
|
}
|
|
return command, kvPairs, arg, ok
|
|
}
|
|
|
|
func (rc *RunContext) commandHandler(ctx context.Context) common.LineHandler {
|
|
logger := common.Logger(ctx)
|
|
resumeCommand := ""
|
|
return func(line string) bool {
|
|
command, kvPairs, arg, ok := tryParseRawActionCommand(line)
|
|
if !ok {
|
|
return true
|
|
}
|
|
|
|
if resumeCommand != "" && command != resumeCommand {
|
|
// There should not be any emojis in the log output for Gitea.
|
|
// The code in the switch statement is the same.
|
|
logger.Infof("%s", line)
|
|
return false
|
|
}
|
|
arg = unescapeCommandData(arg)
|
|
kvPairs = unescapeKvPairs(kvPairs)
|
|
switch command {
|
|
case "set-env":
|
|
rc.setEnv(ctx, kvPairs, arg)
|
|
case "set-output":
|
|
rc.setOutput(ctx, kvPairs, arg)
|
|
case "add-path":
|
|
rc.addPath(ctx, arg)
|
|
case "debug":
|
|
logger.Infof("%s", line)
|
|
case "warning":
|
|
logger.Infof("%s", line)
|
|
case "error":
|
|
logger.Infof("%s", line)
|
|
case "add-mask":
|
|
rc.AddMask(arg)
|
|
logger.Infof("%s", "***")
|
|
case "stop-commands":
|
|
resumeCommand = arg
|
|
logger.Infof("%s", line)
|
|
case resumeCommand:
|
|
resumeCommand = ""
|
|
logger.Infof("%s", line)
|
|
case "save-state":
|
|
logger.Infof("%s", line)
|
|
rc.saveState(ctx, kvPairs, arg)
|
|
case "add-matcher":
|
|
logger.Infof("%s", line)
|
|
default:
|
|
logger.Infof("%s", line)
|
|
}
|
|
|
|
// return true to let gitea's logger handle these special outputs also
|
|
return true
|
|
}
|
|
}
|
|
|
|
func (rc *RunContext) setEnv(ctx context.Context, kvPairs map[string]string, arg string) {
|
|
name := kvPairs["name"]
|
|
common.Logger(ctx).Infof("::set-env:: %s=%s", name, arg)
|
|
if rc.Env == nil {
|
|
rc.Env = make(map[string]string)
|
|
}
|
|
if rc.GlobalEnv == nil {
|
|
rc.GlobalEnv = map[string]string{}
|
|
}
|
|
newenv := map[string]string{
|
|
name: arg,
|
|
}
|
|
mergeIntoMap := mergeIntoMapCaseSensitive
|
|
if rc.JobContainer != nil && rc.JobContainer.IsEnvironmentCaseInsensitive() {
|
|
mergeIntoMap = mergeIntoMapCaseInsensitive
|
|
}
|
|
mergeIntoMap(rc.Env, newenv)
|
|
mergeIntoMap(rc.GlobalEnv, newenv)
|
|
}
|
|
|
|
func (rc *RunContext) setOutput(ctx context.Context, kvPairs map[string]string, arg string) {
|
|
logger := common.Logger(ctx)
|
|
stepID := rc.CurrentStep
|
|
outputName := kvPairs["name"]
|
|
if outputMapping, ok := rc.OutputMappings[MappableOutput{StepID: stepID, OutputName: outputName}]; ok {
|
|
stepID = outputMapping.StepID
|
|
outputName = outputMapping.OutputName
|
|
}
|
|
|
|
result, ok := rc.StepResults[stepID]
|
|
if !ok {
|
|
logger.Infof(" \U00002757 no outputs used step '%s'", stepID)
|
|
return
|
|
}
|
|
|
|
logger.Infof("::set-output:: %s=%s", outputName, arg)
|
|
result.Outputs[outputName] = arg
|
|
}
|
|
|
|
func (rc *RunContext) addPath(ctx context.Context, arg string) {
|
|
common.Logger(ctx).Infof("::add-path:: %s", arg)
|
|
extraPath := []string{arg}
|
|
for _, v := range rc.ExtraPath {
|
|
if v != arg {
|
|
extraPath = append(extraPath, v)
|
|
}
|
|
}
|
|
rc.ExtraPath = extraPath
|
|
}
|
|
|
|
func parseKeyValuePairs(kvPairs, separator string) map[string]string {
|
|
rtn := make(map[string]string)
|
|
kvPairList := strings.SplitSeq(kvPairs, separator)
|
|
for kvPair := range kvPairList {
|
|
kv := strings.Split(kvPair, "=")
|
|
if len(kv) == 2 {
|
|
rtn[kv[0]] = kv[1]
|
|
}
|
|
}
|
|
return rtn
|
|
}
|
|
|
|
func unescapeCommandData(arg string) string {
|
|
escapeMap := map[string]string{
|
|
"%25": "%",
|
|
"%0D": "\r",
|
|
"%0A": "\n",
|
|
}
|
|
for k, v := range escapeMap {
|
|
arg = strings.ReplaceAll(arg, k, v)
|
|
}
|
|
return arg
|
|
}
|
|
|
|
func unescapeCommandProperty(arg string) string {
|
|
escapeMap := map[string]string{
|
|
"%25": "%",
|
|
"%0D": "\r",
|
|
"%0A": "\n",
|
|
"%3A": ":",
|
|
"%2C": ",",
|
|
}
|
|
for k, v := range escapeMap {
|
|
arg = strings.ReplaceAll(arg, k, v)
|
|
}
|
|
return arg
|
|
}
|
|
|
|
func unescapeKvPairs(kvPairs map[string]string) map[string]string {
|
|
for k, v := range kvPairs {
|
|
kvPairs[k] = unescapeCommandProperty(v)
|
|
}
|
|
return kvPairs
|
|
}
|
|
|
|
func (rc *RunContext) saveState(_ context.Context, kvPairs map[string]string, arg string) {
|
|
stepID := rc.CurrentStep
|
|
if stepID != "" {
|
|
if rc.IntraActionState == nil {
|
|
rc.IntraActionState = map[string]map[string]string{}
|
|
}
|
|
state, ok := rc.IntraActionState[stepID]
|
|
if !ok {
|
|
state = map[string]string{}
|
|
rc.IntraActionState[stepID] = state
|
|
}
|
|
state[kvPairs["name"]] = arg
|
|
}
|
|
}
|