2024-09-19 20:37:53 +02:00
|
|
|
// Copyright 2024 The Gitea Authors. All rights reserved.
|
|
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
|
|
|
|
package actions
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
2024-09-20 14:10:50 +02:00
|
|
|
"os"
|
2024-09-19 20:37:53 +02:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
actions_model "code.gitea.io/gitea/models/actions"
|
|
|
|
"code.gitea.io/gitea/models/db"
|
|
|
|
"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"
|
|
|
|
"code.gitea.io/gitea/modules/actions"
|
|
|
|
"code.gitea.io/gitea/modules/git"
|
|
|
|
api "code.gitea.io/gitea/modules/structs"
|
|
|
|
"code.gitea.io/gitea/services/context"
|
|
|
|
"code.gitea.io/gitea/services/convert"
|
|
|
|
|
|
|
|
"github.com/nektos/act/pkg/jobparser"
|
|
|
|
"github.com/nektos/act/pkg/model"
|
|
|
|
)
|
|
|
|
|
2024-09-20 14:10:50 +02:00
|
|
|
func getActionWorkflowPath(commit *git.Commit) string {
|
|
|
|
_, err := commit.SubTree(".gitea/workflows")
|
|
|
|
if err == nil {
|
|
|
|
return ".gitea/workflows"
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, ok := err.(git.ErrNotExist); ok {
|
|
|
|
_, err = commit.SubTree(".github/workflows")
|
|
|
|
return ".github/workflows"
|
|
|
|
}
|
|
|
|
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func getActionWorkflowEntry(ctx *context.APIContext, commit *git.Commit, entry *git.TreeEntry) (*api.ActionWorkflow, error) {
|
2024-09-19 20:37:53 +02:00
|
|
|
cfgUnit := ctx.Repo.Repository.MustGetUnit(ctx, unit.TypeActions)
|
|
|
|
cfg := cfgUnit.ActionsConfig()
|
|
|
|
|
2024-09-20 14:10:50 +02:00
|
|
|
defaultBranch, _ := commit.GetBranchName()
|
|
|
|
|
2024-09-19 20:37:53 +02:00
|
|
|
URL := fmt.Sprintf("%s/actions/workflows/%s", ctx.Repo.Repository.APIURL(), entry.Name())
|
2024-09-20 14:10:50 +02:00
|
|
|
HTMLURL := fmt.Sprintf("%s/src/branch/%s/%s/%s", ctx.Repo.Repository.HTMLURL(ctx), defaultBranch, getActionWorkflowPath(commit), entry.Name())
|
2024-09-19 20:37:53 +02:00
|
|
|
badgeURL := fmt.Sprintf("%s/actions/workflows/%s/badge.svg?branch=%s", ctx.Repo.Repository.HTMLURL(ctx), entry.Name(), ctx.Repo.Repository.DefaultBranch)
|
|
|
|
|
|
|
|
// See https://docs.github.com/en/rest/actions/workflows?apiVersion=2022-11-28#get-a-workflow
|
|
|
|
// State types:
|
|
|
|
// - active
|
|
|
|
// - deleted
|
|
|
|
// - disabled_fork
|
|
|
|
// - disabled_inactivity
|
|
|
|
// - disabled_manually
|
|
|
|
state := "active"
|
|
|
|
if cfg.IsWorkflowDisabled(entry.Name()) {
|
|
|
|
state = "disabled_manually"
|
|
|
|
}
|
|
|
|
|
2024-09-20 14:10:50 +02:00
|
|
|
// Currently, the NodeID returns the hostname of the server since, as far as I know, Gitea does not have a parameter
|
|
|
|
// similar to an instance ID.
|
|
|
|
hostname, err := os.Hostname()
|
|
|
|
if err != nil {
|
|
|
|
hostname = "unknown"
|
|
|
|
}
|
|
|
|
|
|
|
|
// The CreatedAt and UpdatedAt fields currently reflect the timestamp of the latest commit, which can later be refined
|
|
|
|
// by retrieving the first and last commits for the file history. The first commit would indicate the creation date,
|
|
|
|
// while the last commit would represent the modification date. The DeletedAt could be determined by identifying
|
|
|
|
// the last commit where the file existed. However, this implementation has not been done here yet, as it would likely
|
|
|
|
// cause a significant performance degradation.
|
|
|
|
createdAt := commit.Author.When
|
|
|
|
updatedAt := commit.Author.When
|
2024-09-19 20:37:53 +02:00
|
|
|
|
|
|
|
return &api.ActionWorkflow{
|
2024-09-20 14:10:50 +02:00
|
|
|
ID: entry.Name(),
|
|
|
|
NodeID: hostname,
|
|
|
|
Name: entry.Name(),
|
|
|
|
Path: entry.Name(),
|
|
|
|
State: state,
|
|
|
|
CreatedAt: createdAt,
|
|
|
|
UpdatedAt: updatedAt,
|
|
|
|
URL: URL,
|
|
|
|
HTMLURL: HTMLURL,
|
|
|
|
BadgeURL: badgeURL,
|
2024-09-19 20:37:53 +02:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func disableOrEnableWorkflow(ctx *context.APIContext, workflowID string, isEnable bool) error {
|
|
|
|
cfgUnit := ctx.Repo.Repository.MustGetUnit(ctx, unit.TypeActions)
|
|
|
|
cfg := cfgUnit.ActionsConfig()
|
|
|
|
|
|
|
|
if isEnable {
|
|
|
|
cfg.EnableWorkflow(workflowID)
|
|
|
|
} else {
|
|
|
|
cfg.DisableWorkflow(workflowID)
|
|
|
|
}
|
|
|
|
|
|
|
|
return repo_model.UpdateRepoUnit(ctx, cfgUnit)
|
|
|
|
}
|
|
|
|
|
|
|
|
func ListActionWorkflows(ctx *context.APIContext) ([]*api.ActionWorkflow, error) {
|
|
|
|
defaultBranchCommit, err := ctx.Repo.GitRepo.GetBranchCommit(ctx.Repo.Repository.DefaultBranch)
|
|
|
|
if err != nil {
|
|
|
|
ctx.Error(http.StatusInternalServerError, "WorkflowDefaultBranchError", err.Error())
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
entries, err := actions.ListWorkflows(defaultBranchCommit)
|
|
|
|
if err != nil {
|
|
|
|
ctx.Error(http.StatusNotFound, "WorkflowListNotFound", err.Error())
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
workflows := make([]*api.ActionWorkflow, len(entries))
|
|
|
|
for i, entry := range entries {
|
2024-09-20 14:10:50 +02:00
|
|
|
workflows[i], err = getActionWorkflowEntry(ctx, defaultBranchCommit, entry)
|
2024-09-19 20:37:53 +02:00
|
|
|
if err != nil {
|
|
|
|
ctx.Error(http.StatusInternalServerError, "WorkflowGetError", err.Error())
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return workflows, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func GetActionWorkflow(ctx *context.APIContext, workflowID string) (*api.ActionWorkflow, error) {
|
|
|
|
entries, err := ListActionWorkflows(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
workflows := make([]*api.ActionWorkflow, len(entries))
|
|
|
|
for i, entry := range entries {
|
|
|
|
if entry.Name == workflowID {
|
|
|
|
workflows[i] = entry
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return workflows[len(workflows)-1], nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func DisableActionWorkflow(ctx *context.APIContext, workflowID string) error {
|
|
|
|
return disableOrEnableWorkflow(ctx, workflowID, false)
|
|
|
|
}
|
|
|
|
|
|
|
|
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)
|
|
|
|
cfg := cfgUnit.ActionsConfig()
|
|
|
|
if cfg.IsWorkflowDisabled(workflowID) {
|
|
|
|
ctx.Error(http.StatusInternalServerError, "WorkflowDisabled", ctx.Tr("actions.workflow.disabled"))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// get target commit of run from specified ref
|
|
|
|
refName := git.RefName(opt.Ref)
|
|
|
|
var runTargetCommit *git.Commit
|
|
|
|
var err error
|
|
|
|
if refName.IsTag() {
|
|
|
|
runTargetCommit, err = ctx.Repo.GitRepo.GetTagCommit(refName.TagName())
|
|
|
|
} else if refName.IsBranch() {
|
|
|
|
runTargetCommit, err = ctx.Repo.GitRepo.GetBranchCommit(refName.BranchName())
|
|
|
|
} else {
|
|
|
|
ctx.Error(http.StatusInternalServerError, "WorkflowRefNameError", ctx.Tr("form.git_ref_name_error", opt.Ref))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
ctx.Error(http.StatusNotFound, "WorkflowRefNotFound", ctx.Tr("form.target_ref_not_exist", opt.Ref))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// get workflow entry from default branch commit
|
|
|
|
defaultBranchCommit, err := ctx.Repo.GitRepo.GetBranchCommit(ctx.Repo.Repository.DefaultBranch)
|
|
|
|
if err != nil {
|
|
|
|
ctx.Error(http.StatusInternalServerError, "WorkflowDefaultBranchError", err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
entries, err := actions.ListWorkflows(defaultBranchCommit)
|
|
|
|
if err != nil {
|
|
|
|
ctx.Error(http.StatusNotFound, "WorkflowListNotFound", err.Error())
|
|
|
|
}
|
|
|
|
|
|
|
|
// find workflow from commit
|
|
|
|
var workflows []*jobparser.SingleWorkflow
|
|
|
|
for _, entry := range entries {
|
|
|
|
if entry.Name() == workflowID {
|
|
|
|
content, err := actions.GetContentFromEntry(entry)
|
|
|
|
if err != nil {
|
|
|
|
ctx.Error(http.StatusInternalServerError, "WorkflowGetContentError", err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
workflows, err = jobparser.Parse(content)
|
|
|
|
if err != nil {
|
|
|
|
ctx.Error(http.StatusInternalServerError, "WorkflowParseError", err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(workflows) == 0 {
|
|
|
|
ctx.Error(http.StatusNotFound, "WorkflowNotFound", ctx.Tr("actions.workflow.not_found", workflowID))
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
workflow := &model.Workflow{
|
|
|
|
RawOn: workflows[0].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{
|
|
|
|
Workflow: workflowID,
|
|
|
|
Ref: opt.Ref,
|
|
|
|
Repository: convert.ToRepo(ctx, ctx.Repo.Repository, access_model.Permission{AccessMode: perm.AccessModeNone}),
|
|
|
|
Inputs: inputs,
|
|
|
|
Sender: convert.ToUserWithAccessMode(ctx, ctx.Doer, perm.AccessModeNone),
|
|
|
|
}
|
|
|
|
var eventPayload []byte
|
|
|
|
if eventPayload, err = workflowDispatchPayload.JSONPayload(); err != nil {
|
|
|
|
ctx.Error(http.StatusInternalServerError, "WorkflowDispatchJSONParseError", err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
run := &actions_model.ActionRun{
|
|
|
|
Title: strings.SplitN(runTargetCommit.CommitMessage, "\n", 2)[0],
|
|
|
|
RepoID: ctx.Repo.Repository.ID,
|
|
|
|
OwnerID: ctx.Repo.Repository.Owner.ID,
|
|
|
|
WorkflowID: workflowID,
|
|
|
|
TriggerUserID: ctx.Doer.ID,
|
|
|
|
Ref: opt.Ref,
|
|
|
|
CommitSHA: runTargetCommit.ID.String(),
|
|
|
|
IsForkPullRequest: false,
|
|
|
|
Event: "workflow_dispatch",
|
|
|
|
TriggerEvent: "workflow_dispatch",
|
|
|
|
EventPayload: string(eventPayload),
|
|
|
|
Status: actions_model.StatusWaiting,
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := actions_model.InsertRun(ctx, run, workflows); err != nil {
|
|
|
|
ctx.Error(http.StatusInternalServerError, "WorkflowInsertRunError", err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
alljobs, err := db.Find[actions_model.ActionRunJob](ctx, actions_model.FindRunJobOptions{RunID: run.ID})
|
|
|
|
if err != nil {
|
|
|
|
ctx.Error(http.StatusInternalServerError, "WorkflowFindRunJobError", err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
CreateCommitStatus(ctx, alljobs...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func EnableActionWorkflow(ctx *context.APIContext, workflowID string) error {
|
|
|
|
return disableOrEnableWorkflow(ctx, workflowID, true)
|
|
|
|
}
|