62abf4fe11
Partially fixes https://gitea.com/gitea/act_runner/issues/91 If the repository is private, we need to provide the token to the caller workflows to access the called reusable workflows from the same repository. Reviewed-on: https://gitea.com/gitea/act/pulls/38 Reviewed-by: Jason Song <i@wolfogre.com> Co-authored-by: Zettat123 <zettat123@gmail.com> Co-committed-by: Zettat123 <zettat123@gmail.com>
180 lines
5.3 KiB
Go
180 lines
5.3 KiB
Go
package runner
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"io/fs"
|
|
"os"
|
|
"path"
|
|
"regexp"
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/nektos/act/pkg/common"
|
|
"github.com/nektos/act/pkg/common/git"
|
|
"github.com/nektos/act/pkg/model"
|
|
)
|
|
|
|
func newLocalReusableWorkflowExecutor(rc *RunContext) common.Executor {
|
|
// ./.gitea/workflows/wf.yml -> .gitea/workflows/wf.yml
|
|
trimmedUses := strings.TrimPrefix(rc.Run.Job().Uses, "./")
|
|
// uses string format is {owner}/{repo}/.{git_platform}/workflows/{filename}@{ref}
|
|
uses := fmt.Sprintf("%s/%s@%s", rc.Config.PresetGitHubContext.Repository, trimmedUses, rc.Config.PresetGitHubContext.Sha)
|
|
|
|
remoteReusableWorkflow := newRemoteReusableWorkflowWithPlat(uses)
|
|
if remoteReusableWorkflow == nil {
|
|
return common.NewErrorExecutor(fmt.Errorf("expected format {owner}/{repo}/.{git_platform}/workflows/{filename}@{ref}. Actual '%s' Input string was not in a correct format", uses))
|
|
}
|
|
remoteReusableWorkflow.URL = rc.Config.GitHubInstance
|
|
|
|
workflowDir := fmt.Sprintf("%s/%s", rc.ActionCacheDir(), safeFilename(uses))
|
|
|
|
// If the repository is private, we need a token to clone it
|
|
token := rc.Config.GetToken()
|
|
|
|
return common.NewPipelineExecutor(
|
|
newMutexExecutor(cloneIfRequired(rc, *remoteReusableWorkflow, workflowDir, token)),
|
|
newReusableWorkflowExecutor(rc, workflowDir, remoteReusableWorkflow.FilePath()),
|
|
)
|
|
}
|
|
|
|
func newRemoteReusableWorkflowExecutor(rc *RunContext) common.Executor {
|
|
uses := rc.Run.Job().Uses
|
|
|
|
remoteReusableWorkflow := newRemoteReusableWorkflowWithPlat(uses)
|
|
if remoteReusableWorkflow == nil {
|
|
return common.NewErrorExecutor(fmt.Errorf("expected format {owner}/{repo}/.{git_platform}/workflows/{filename}@{ref}. Actual '%s' Input string was not in a correct format", uses))
|
|
}
|
|
remoteReusableWorkflow.URL = rc.Config.GitHubInstance
|
|
|
|
workflowDir := fmt.Sprintf("%s/%s", rc.ActionCacheDir(), safeFilename(uses))
|
|
|
|
// FIXME: if the reusable workflow is from a private repository, we need to provide a token to access the repository.
|
|
token := ""
|
|
|
|
return common.NewPipelineExecutor(
|
|
newMutexExecutor(cloneIfRequired(rc, *remoteReusableWorkflow, workflowDir, token)),
|
|
newReusableWorkflowExecutor(rc, workflowDir, remoteReusableWorkflow.FilePath()),
|
|
)
|
|
}
|
|
|
|
var (
|
|
executorLock sync.Mutex
|
|
)
|
|
|
|
func newMutexExecutor(executor common.Executor) common.Executor {
|
|
return func(ctx context.Context) error {
|
|
executorLock.Lock()
|
|
defer executorLock.Unlock()
|
|
|
|
return executor(ctx)
|
|
}
|
|
}
|
|
|
|
func cloneIfRequired(rc *RunContext, remoteReusableWorkflow remoteReusableWorkflow, targetDirectory, token string) common.Executor {
|
|
return common.NewConditionalExecutor(
|
|
func(ctx context.Context) bool {
|
|
_, err := os.Stat(targetDirectory)
|
|
notExists := errors.Is(err, fs.ErrNotExist)
|
|
return notExists
|
|
},
|
|
git.NewGitCloneExecutor(git.NewGitCloneExecutorInput{
|
|
URL: remoteReusableWorkflow.CloneURL(),
|
|
Ref: remoteReusableWorkflow.Ref,
|
|
Dir: targetDirectory,
|
|
Token: token,
|
|
}),
|
|
nil,
|
|
)
|
|
}
|
|
|
|
func newReusableWorkflowExecutor(rc *RunContext, directory string, workflow string) common.Executor {
|
|
return func(ctx context.Context) error {
|
|
planner, err := model.NewWorkflowPlanner(path.Join(directory, workflow), true)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
plan, err := planner.PlanEvent("workflow_call")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
runner, err := NewReusableWorkflowRunner(rc)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return runner.NewPlanExecutor(plan)(ctx)
|
|
}
|
|
}
|
|
|
|
func NewReusableWorkflowRunner(rc *RunContext) (Runner, error) {
|
|
runner := &runnerImpl{
|
|
config: rc.Config,
|
|
eventJSON: rc.EventJSON,
|
|
caller: &caller{
|
|
runContext: rc,
|
|
},
|
|
}
|
|
|
|
return runner.configure()
|
|
}
|
|
|
|
type remoteReusableWorkflow struct {
|
|
URL string
|
|
Org string
|
|
Repo string
|
|
Filename string
|
|
Ref string
|
|
|
|
GitPlatform string
|
|
}
|
|
|
|
func (r *remoteReusableWorkflow) CloneURL() string {
|
|
// In Gitea, r.URL always has the protocol prefix, we don't need to add extra prefix in this case.
|
|
if strings.HasPrefix(r.URL, "http://") || strings.HasPrefix(r.URL, "https://") {
|
|
return fmt.Sprintf("%s/%s/%s", r.URL, r.Org, r.Repo)
|
|
}
|
|
return fmt.Sprintf("https://%s/%s/%s", r.URL, r.Org, r.Repo)
|
|
}
|
|
|
|
func (r *remoteReusableWorkflow) FilePath() string {
|
|
return fmt.Sprintf("./.%s/workflows/%s", r.GitPlatform, r.Filename)
|
|
}
|
|
|
|
func newRemoteReusableWorkflowWithPlat(uses string) *remoteReusableWorkflow {
|
|
// GitHub docs:
|
|
// https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions#jobsjob_iduses
|
|
r := regexp.MustCompile(`^([^/]+)/([^/]+)/\.([^/]+)/workflows/([^@]+)@(.*)$`)
|
|
matches := r.FindStringSubmatch(uses)
|
|
if len(matches) != 6 {
|
|
return nil
|
|
}
|
|
return &remoteReusableWorkflow{
|
|
Org: matches[1],
|
|
Repo: matches[2],
|
|
GitPlatform: matches[3],
|
|
Filename: matches[4],
|
|
Ref: matches[5],
|
|
}
|
|
}
|
|
|
|
// deprecated: use newRemoteReusableWorkflowWithPlat
|
|
func newRemoteReusableWorkflow(uses string) *remoteReusableWorkflow {
|
|
// GitHub docs:
|
|
// https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions#jobsjob_iduses
|
|
r := regexp.MustCompile(`^([^/]+)/([^/]+)/.github/workflows/([^@]+)@(.*)$`)
|
|
matches := r.FindStringSubmatch(uses)
|
|
if len(matches) != 5 {
|
|
return nil
|
|
}
|
|
return &remoteReusableWorkflow{
|
|
Org: matches[1],
|
|
Repo: matches[2],
|
|
Filename: matches[3],
|
|
Ref: matches[4],
|
|
URL: "github.com",
|
|
}
|
|
}
|