1
1
mirror of https://github.com/go-gitea/gitea synced 2025-01-21 23:24:29 +00:00
Signed-off-by: Bence Santha <git@santha.eu>
This commit is contained in:
Bence Santha 2024-09-24 09:16:50 +02:00
parent 35bb7e7ed5
commit 27d1906283
2 changed files with 67 additions and 64 deletions

View File

@ -616,8 +616,6 @@ func (a ActionWorkflow) ListRepositoryWorkflows(ctx *context.APIContext) {
// "$ref": "#/responses/forbidden" // "$ref": "#/responses/forbidden"
// "404": // "404":
// "$ref": "#/responses/notFound" // "$ref": "#/responses/notFound"
// "409":
// "$ref": "#/responses/conflict"
// "422": // "422":
// "$ref": "#/responses/validationError" // "$ref": "#/responses/validationError"
// "500": // "500":
@ -630,6 +628,7 @@ func (a ActionWorkflow) ListRepositoryWorkflows(ctx *context.APIContext) {
if len(workflows) == 0 { if len(workflows) == 0 {
ctx.JSON(http.StatusNotFound, nil) ctx.JSON(http.StatusNotFound, nil)
return
} }
ctx.SetTotalCountHeader(int64(len(workflows))) ctx.SetTotalCountHeader(int64(len(workflows)))
@ -667,8 +666,6 @@ func (a ActionWorkflow) GetWorkflow(ctx *context.APIContext) {
// "$ref": "#/responses/forbidden" // "$ref": "#/responses/forbidden"
// "404": // "404":
// "$ref": "#/responses/notFound" // "$ref": "#/responses/notFound"
// "409":
// "$ref": "#/responses/conflict"
// "422": // "422":
// "$ref": "#/responses/validationError" // "$ref": "#/responses/validationError"
// "500": // "500":
@ -687,6 +684,7 @@ func (a ActionWorkflow) GetWorkflow(ctx *context.APIContext) {
if workflow == nil { if workflow == nil {
ctx.JSON(http.StatusNotFound, nil) ctx.JSON(http.StatusNotFound, nil)
return
} }
ctx.JSON(http.StatusOK, workflow) ctx.JSON(http.StatusOK, workflow)
@ -723,8 +721,6 @@ func (a ActionWorkflow) DisableWorkflow(ctx *context.APIContext) {
// "$ref": "#/responses/forbidden" // "$ref": "#/responses/forbidden"
// "404": // "404":
// "$ref": "#/responses/notFound" // "$ref": "#/responses/notFound"
// "409":
// "$ref": "#/responses/conflict"
// "422": // "422":
// "$ref": "#/responses/validationError" // "$ref": "#/responses/validationError"
@ -737,6 +733,7 @@ func (a ActionWorkflow) DisableWorkflow(ctx *context.APIContext) {
err := actions_service.DisableActionWorkflow(ctx, workflowID) err := actions_service.DisableActionWorkflow(ctx, workflowID)
if err != nil { if err != nil {
ctx.Error(http.StatusInternalServerError, "DisableActionWorkflow", err) ctx.Error(http.StatusInternalServerError, "DisableActionWorkflow", err)
return
} }
ctx.Status(http.StatusNoContent) ctx.Status(http.StatusNoContent)
@ -777,8 +774,6 @@ func (a ActionWorkflow) DispatchWorkflow(ctx *context.APIContext) {
// "$ref": "#/responses/forbidden" // "$ref": "#/responses/forbidden"
// "404": // "404":
// "$ref": "#/responses/notFound" // "$ref": "#/responses/notFound"
// "409":
// "$ref": "#/responses/conflict"
// "422": // "422":
// "$ref": "#/responses/validationError" // "$ref": "#/responses/validationError"
@ -846,6 +841,7 @@ func (a ActionWorkflow) EnableWorkflow(ctx *context.APIContext) {
err := actions_service.EnableActionWorkflow(ctx, workflowID) err := actions_service.EnableActionWorkflow(ctx, workflowID)
if err != nil { if err != nil {
ctx.Error(http.StatusInternalServerError, "EnableActionWorkflow", err) ctx.Error(http.StatusInternalServerError, "EnableActionWorkflow", err)
return
} }
ctx.Status(http.StatusNoContent) ctx.Status(http.StatusNoContent)

View File

@ -27,20 +27,16 @@ import (
) )
func getActionWorkflowPath(commit *git.Commit) string { func getActionWorkflowPath(commit *git.Commit) string {
_, err := commit.SubTree(".gitea/workflows") paths := []string{".gitea/workflows", ".github/workflows"}
if err == nil { for _, path := range paths {
return ".gitea/workflows" if _, err := commit.SubTree(path); err == nil {
return path
}
} }
if _, ok := err.(git.ErrNotExist); ok {
_, err = commit.SubTree(".github/workflows")
return ".github/workflows"
}
return "" return ""
} }
func getActionWorkflowEntry(ctx *context.APIContext, commit *git.Commit, entry *git.TreeEntry) (*api.ActionWorkflow, error) { func getActionWorkflowEntry(ctx *context.APIContext, commit *git.Commit, entry *git.TreeEntry) *api.ActionWorkflow {
cfgUnit := ctx.Repo.Repository.MustGetUnit(ctx, unit.TypeActions) cfgUnit := ctx.Repo.Repository.MustGetUnit(ctx, unit.TypeActions)
cfg := cfgUnit.ActionsConfig() cfg := cfgUnit.ActionsConfig()
@ -88,17 +84,22 @@ func getActionWorkflowEntry(ctx *context.APIContext, commit *git.Commit, entry *
URL: URL, URL: URL,
HTMLURL: HTMLURL, HTMLURL: HTMLURL,
BadgeURL: badgeURL, BadgeURL: badgeURL,
}, nil }
} }
func disableOrEnableWorkflow(ctx *context.APIContext, workflowID string, isEnable bool) error { func disableOrEnableWorkflow(ctx *context.APIContext, workflowID string, isEnable bool) error {
workflow, err := GetActionWorkflow(ctx, workflowID)
if err != nil {
return err
}
cfgUnit := ctx.Repo.Repository.MustGetUnit(ctx, unit.TypeActions) cfgUnit := ctx.Repo.Repository.MustGetUnit(ctx, unit.TypeActions)
cfg := cfgUnit.ActionsConfig() cfg := cfgUnit.ActionsConfig()
if isEnable { if isEnable {
cfg.EnableWorkflow(workflowID) cfg.EnableWorkflow(workflow.ID)
} else { } else {
cfg.DisableWorkflow(workflowID) cfg.DisableWorkflow(workflow.ID)
} }
return repo_model.UpdateRepoUnit(ctx, cfgUnit) return repo_model.UpdateRepoUnit(ctx, cfgUnit)
@ -119,11 +120,7 @@ func ListActionWorkflows(ctx *context.APIContext) ([]*api.ActionWorkflow, error)
workflows := make([]*api.ActionWorkflow, len(entries)) workflows := make([]*api.ActionWorkflow, len(entries))
for i, entry := range entries { for i, entry := range entries {
workflows[i], err = getActionWorkflowEntry(ctx, defaultBranchCommit, entry) workflows[i] = getActionWorkflowEntry(ctx, defaultBranchCommit, entry)
if err != nil {
ctx.Error(http.StatusInternalServerError, "WorkflowGetError", err.Error())
return nil, err
}
} }
return workflows, nil return workflows, nil
@ -135,15 +132,13 @@ func GetActionWorkflow(ctx *context.APIContext, workflowID string) (*api.ActionW
return nil, err return nil, err
} }
workflows := make([]*api.ActionWorkflow, len(entries)) for _, entry := range entries {
for i, entry := range entries {
if entry.Name == workflowID { if entry.Name == workflowID {
workflows[i] = entry return entry, nil
break
} }
} }
return workflows[len(workflows)-1], nil return nil, fmt.Errorf("workflow not found")
} }
func DisableActionWorkflow(ctx *context.APIContext, workflowID string) error { func DisableActionWorkflow(ctx *context.APIContext, workflowID string) error {
@ -151,44 +146,46 @@ func DisableActionWorkflow(ctx *context.APIContext, workflowID string) error {
} }
func DispatchActionWorkflow(ctx *context.APIContext, workflowID string, opt *api.CreateActionWorkflowDispatch) { func DispatchActionWorkflow(ctx *context.APIContext, workflowID string, opt *api.CreateActionWorkflowDispatch) {
// can not run job when workflow is disabled
cfgUnit := ctx.Repo.Repository.MustGetUnit(ctx, unit.TypeActions) cfgUnit := ctx.Repo.Repository.MustGetUnit(ctx, unit.TypeActions)
cfg := cfgUnit.ActionsConfig() cfg := cfgUnit.ActionsConfig()
if cfg.IsWorkflowDisabled(workflowID) { if cfg.IsWorkflowDisabled(workflowID) {
ctx.Error(http.StatusInternalServerError, "WorkflowDisabled", ctx.Tr("actions.workflow.disabled")) ctx.Error(http.StatusInternalServerError, "WorkflowDisabled", ctx.Tr("actions.workflow.disabled"))
return return
} }
// get target commit of run from specified ref
refName := git.RefName(opt.Ref) refName := git.RefName(opt.Ref)
var runTargetCommit *git.Commit var runTargetCommit *git.Commit
var err error var err error
if refName.IsTag() {
switch {
case refName.IsTag():
runTargetCommit, err = ctx.Repo.GitRepo.GetTagCommit(refName.TagName()) runTargetCommit, err = ctx.Repo.GitRepo.GetTagCommit(refName.TagName())
} else if refName.IsBranch() { case refName.IsBranch():
runTargetCommit, err = ctx.Repo.GitRepo.GetBranchCommit(refName.BranchName()) runTargetCommit, err = ctx.Repo.GitRepo.GetBranchCommit(refName.BranchName())
} else { default:
ctx.Error(http.StatusInternalServerError, "WorkflowRefNameError", ctx.Tr("form.git_ref_name_error", opt.Ref)) ctx.Error(http.StatusInternalServerError, "WorkflowRefNameError", ctx.Tr("form.git_ref_name_error", opt.Ref))
return return
} }
if err != nil { if err != nil {
ctx.Error(http.StatusNotFound, "WorkflowRefNotFound", ctx.Tr("form.target_ref_not_exist", opt.Ref)) ctx.Error(http.StatusNotFound, "WorkflowRefNotFound", ctx.Tr("form.target_ref_not_exist", opt.Ref))
return return
} }
// get workflow entry from default branch commit
defaultBranchCommit, err := ctx.Repo.GitRepo.GetBranchCommit(ctx.Repo.Repository.DefaultBranch) defaultBranchCommit, err := ctx.Repo.GitRepo.GetBranchCommit(ctx.Repo.Repository.DefaultBranch)
if err != nil { if err != nil {
ctx.Error(http.StatusInternalServerError, "WorkflowDefaultBranchError", err.Error()) ctx.Error(http.StatusInternalServerError, "WorkflowDefaultBranchError", err.Error())
return return
} }
entries, err := actions.ListWorkflows(defaultBranchCommit) entries, err := actions.ListWorkflows(defaultBranchCommit)
if err != nil { if err != nil {
ctx.Error(http.StatusNotFound, "WorkflowListNotFound", err.Error()) ctx.Error(http.StatusNotFound, "WorkflowListNotFound", err.Error())
return
} }
// find workflow from commit var workflow *jobparser.SingleWorkflow
var workflows []*jobparser.SingleWorkflow
for _, entry := range entries { for _, entry := range entries {
if entry.Name() == workflowID { if entry.Name() == workflowID {
content, err := actions.GetContentFromEntry(entry) content, err := actions.GetContentFromEntry(entry)
@ -196,39 +193,25 @@ func DispatchActionWorkflow(ctx *context.APIContext, workflowID string, opt *api
ctx.Error(http.StatusInternalServerError, "WorkflowGetContentError", err.Error()) ctx.Error(http.StatusInternalServerError, "WorkflowGetContentError", err.Error())
return return
} }
workflows, err = jobparser.Parse(content) workflows, err := jobparser.Parse(content)
if err != nil { if err != nil || len(workflows) == 0 {
ctx.Error(http.StatusInternalServerError, "WorkflowParseError", err.Error()) ctx.Error(http.StatusInternalServerError, "WorkflowParseError", err.Error())
return return
} }
workflow = workflows[0]
break break
} }
} }
if len(workflows) == 0 { if workflow == nil {
ctx.Error(http.StatusNotFound, "WorkflowNotFound", ctx.Tr("actions.workflow.not_found", workflowID)) ctx.Error(http.StatusNotFound, "WorkflowNotFound", ctx.Tr("actions.workflow.not_found", workflowID))
return return
} }
workflow := &model.Workflow{ // Process workflow inputs
RawOn: workflows[0].RawOn, inputs := processWorkflowInputs(opt, &model.Workflow{
} RawOn: workflow.RawOn,
inputs := make(map[string]any) })
if workflowDispatch := workflow.WorkflowDispatchConfig(); workflowDispatch != nil {
for name, config := range workflowDispatch.Inputs {
value, exists := opt.Inputs[name]
if !exists {
continue
}
if config.Type == "boolean" {
inputs[name] = strconv.FormatBool(value == "on")
} else if value != "" {
inputs[name] = value
} else {
inputs[name] = config.Default
}
}
}
workflowDispatchPayload := &api.WorkflowDispatchPayload{ workflowDispatchPayload := &api.WorkflowDispatchPayload{
Workflow: workflowID, Workflow: workflowID,
@ -237,8 +220,9 @@ func DispatchActionWorkflow(ctx *context.APIContext, workflowID string, opt *api
Inputs: inputs, Inputs: inputs,
Sender: convert.ToUserWithAccessMode(ctx, ctx.Doer, perm.AccessModeNone), Sender: convert.ToUserWithAccessMode(ctx, ctx.Doer, perm.AccessModeNone),
} }
var eventPayload []byte
if eventPayload, err = workflowDispatchPayload.JSONPayload(); err != nil { eventPayload, err := workflowDispatchPayload.JSONPayload()
if err != nil {
ctx.Error(http.StatusInternalServerError, "WorkflowDispatchJSONParseError", err.Error()) ctx.Error(http.StatusInternalServerError, "WorkflowDispatchJSONParseError", err.Error())
return return
} }
@ -258,7 +242,7 @@ func DispatchActionWorkflow(ctx *context.APIContext, workflowID string, opt *api
Status: actions_model.StatusWaiting, Status: actions_model.StatusWaiting,
} }
if err := actions_model.InsertRun(ctx, run, workflows); err != nil { if err := actions_model.InsertRun(ctx, run, []*jobparser.SingleWorkflow{workflow}); err != nil {
ctx.Error(http.StatusInternalServerError, "WorkflowInsertRunError", err.Error()) ctx.Error(http.StatusInternalServerError, "WorkflowInsertRunError", err.Error())
return return
} }
@ -268,9 +252,32 @@ func DispatchActionWorkflow(ctx *context.APIContext, workflowID string, opt *api
ctx.Error(http.StatusInternalServerError, "WorkflowFindRunJobError", err.Error()) ctx.Error(http.StatusInternalServerError, "WorkflowFindRunJobError", err.Error())
return return
} }
CreateCommitStatus(ctx, alljobs...) CreateCommitStatus(ctx, alljobs...)
} }
func processWorkflowInputs(opt *api.CreateActionWorkflowDispatch, workflow *model.Workflow) map[string]any {
inputs := make(map[string]any)
if workflowDispatch := workflow.WorkflowDispatchConfig(); workflowDispatch != nil {
for name, config := range workflowDispatch.Inputs {
value, exists := opt.Inputs[name]
if !exists {
continue
}
if value == "" {
value = config.Default
}
switch config.Type {
case "boolean":
inputs[name] = strconv.FormatBool(value == "on")
default:
inputs[name] = value
}
}
}
return inputs
}
func EnableActionWorkflow(ctx *context.APIContext, workflowID string) error { func EnableActionWorkflow(ctx *context.APIContext, workflowID string) error {
return disableOrEnableWorkflow(ctx, workflowID, true) return disableOrEnableWorkflow(ctx, workflowID, true)
} }