mirror of https://github.com/go-gitea/gitea.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
173 lines
5.2 KiB
173 lines
5.2 KiB
// Copyright 2024 The Gitea Authors. All rights reserved. |
|
// SPDX-License-Identifier: MIT |
|
|
|
package actions |
|
|
|
import ( |
|
"fmt" |
|
"strings" |
|
|
|
actions_model "code.gitea.io/gitea/models/actions" |
|
"code.gitea.io/gitea/models/perm" |
|
access_model "code.gitea.io/gitea/models/perm/access" |
|
repo_model "code.gitea.io/gitea/models/repo" |
|
"code.gitea.io/gitea/models/unit" |
|
user_model "code.gitea.io/gitea/models/user" |
|
"code.gitea.io/gitea/modules/actions" |
|
"code.gitea.io/gitea/modules/git" |
|
"code.gitea.io/gitea/modules/reqctx" |
|
api "code.gitea.io/gitea/modules/structs" |
|
"code.gitea.io/gitea/modules/util" |
|
"code.gitea.io/gitea/services/context" |
|
"code.gitea.io/gitea/services/convert" |
|
|
|
"github.com/nektos/act/pkg/jobparser" |
|
"github.com/nektos/act/pkg/model" |
|
"gopkg.in/yaml.v3" |
|
) |
|
|
|
func EnableOrDisableWorkflow(ctx *context.APIContext, workflowID string, isEnable bool) error { |
|
workflow, err := convert.GetActionWorkflow(ctx, ctx.Repo.GitRepo, ctx.Repo.Repository, workflowID) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
cfgUnit := ctx.Repo.Repository.MustGetUnit(ctx, unit.TypeActions) |
|
cfg := cfgUnit.ActionsConfig() |
|
|
|
if isEnable { |
|
cfg.EnableWorkflow(workflow.ID) |
|
} else { |
|
cfg.DisableWorkflow(workflow.ID) |
|
} |
|
|
|
return repo_model.UpdateRepoUnit(ctx, cfgUnit) |
|
} |
|
|
|
func DispatchActionWorkflow(ctx reqctx.RequestContext, doer *user_model.User, repo *repo_model.Repository, gitRepo *git.Repository, workflowID, ref string, processInputs func(model *model.WorkflowDispatch, inputs map[string]any) error) error { |
|
if workflowID == "" { |
|
return util.ErrorWrapTranslatable( |
|
util.NewNotExistErrorf("workflowID is empty"), |
|
"actions.workflow.not_found", workflowID, |
|
) |
|
} |
|
|
|
if ref == "" { |
|
return util.ErrorWrapTranslatable( |
|
util.NewNotExistErrorf("ref is empty"), |
|
"form.target_ref_not_exist", ref, |
|
) |
|
} |
|
|
|
// can not rerun job when workflow is disabled |
|
cfgUnit := repo.MustGetUnit(ctx, unit.TypeActions) |
|
cfg := cfgUnit.ActionsConfig() |
|
if cfg.IsWorkflowDisabled(workflowID) { |
|
return util.ErrorWrapTranslatable( |
|
util.NewPermissionDeniedErrorf("workflow is disabled"), |
|
"actions.workflow.disabled", |
|
) |
|
} |
|
|
|
// get target commit of run from specified ref |
|
refName := git.RefName(ref) |
|
var runTargetCommit *git.Commit |
|
var err error |
|
if refName.IsTag() { |
|
runTargetCommit, err = gitRepo.GetTagCommit(refName.TagName()) |
|
} else if refName.IsBranch() { |
|
runTargetCommit, err = gitRepo.GetBranchCommit(refName.BranchName()) |
|
} else { |
|
refName = git.RefNameFromBranch(ref) |
|
runTargetCommit, err = gitRepo.GetBranchCommit(ref) |
|
} |
|
if err != nil { |
|
return util.ErrorWrapTranslatable( |
|
util.NewNotExistErrorf("ref %q doesn't exist", ref), |
|
"form.target_ref_not_exist", ref, |
|
) |
|
} |
|
|
|
// get workflow entry from runTargetCommit |
|
_, entries, err := actions.ListWorkflows(runTargetCommit) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
// find workflow from commit |
|
var entry *git.TreeEntry |
|
|
|
run := &actions_model.ActionRun{ |
|
Title: strings.SplitN(runTargetCommit.CommitMessage, "\n", 2)[0], |
|
RepoID: repo.ID, |
|
Repo: repo, |
|
OwnerID: repo.OwnerID, |
|
WorkflowID: workflowID, |
|
TriggerUserID: doer.ID, |
|
TriggerUser: doer, |
|
Ref: string(refName), |
|
CommitSHA: runTargetCommit.ID.String(), |
|
IsForkPullRequest: false, |
|
Event: "workflow_dispatch", |
|
TriggerEvent: "workflow_dispatch", |
|
Status: actions_model.StatusWaiting, |
|
} |
|
|
|
for _, e := range entries { |
|
if e.Name() != workflowID { |
|
continue |
|
} |
|
entry = e |
|
break |
|
} |
|
|
|
if entry == nil { |
|
return util.ErrorWrapTranslatable( |
|
util.NewNotExistErrorf("workflow %q doesn't exist", workflowID), |
|
"actions.workflow.not_found", workflowID, |
|
) |
|
} |
|
|
|
content, err := actions.GetContentFromEntry(entry) |
|
if err != nil { |
|
return err |
|
} |
|
|
|
singleWorkflow := &jobparser.SingleWorkflow{} |
|
if err := yaml.Unmarshal(content, singleWorkflow); err != nil { |
|
return fmt.Errorf("failed to unmarshal workflow content: %w", err) |
|
} |
|
// get inputs from post |
|
workflow := &model.Workflow{ |
|
RawOn: singleWorkflow.RawOn, |
|
} |
|
inputsWithDefaults := make(map[string]any) |
|
if workflowDispatch := workflow.WorkflowDispatchConfig(); workflowDispatch != nil { |
|
if err = processInputs(workflowDispatch, inputsWithDefaults); err != nil { |
|
return err |
|
} |
|
} |
|
|
|
// ctx.Req.PostForm -> WorkflowDispatchPayload.Inputs -> ActionRun.EventPayload -> runner: ghc.Event |
|
// https://docs.github.com/en/actions/learn-github-actions/contexts#github-context |
|
// https://docs.github.com/en/webhooks/webhook-events-and-payloads#workflow_dispatch |
|
workflowDispatchPayload := &api.WorkflowDispatchPayload{ |
|
Workflow: workflowID, |
|
Ref: ref, |
|
Repository: convert.ToRepo(ctx, repo, access_model.Permission{AccessMode: perm.AccessModeNone}), |
|
Inputs: inputsWithDefaults, |
|
Sender: convert.ToUserWithAccessMode(ctx, doer, perm.AccessModeNone), |
|
} |
|
|
|
var eventPayload []byte |
|
if eventPayload, err = workflowDispatchPayload.JSONPayload(); err != nil { |
|
return fmt.Errorf("JSONPayload: %w", err) |
|
} |
|
run.EventPayload = string(eventPayload) |
|
|
|
// Insert the action run and its associated jobs into the database |
|
if err := PrepareRunAndInsert(ctx, content, run, inputsWithDefaults); err != nil { |
|
return fmt.Errorf("PrepareRun: %w", err) |
|
} |
|
return nil |
|
}
|
|
|