| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133 |
- // Copyright 2022 The Gitea Authors. All rights reserved.
- // SPDX-License-Identifier: MIT
-
- package actions
-
- import (
- "context"
- "errors"
- "fmt"
-
- actions_model "code.gitea.io/gitea/models/actions"
- "code.gitea.io/gitea/models/db"
- secret_model "code.gitea.io/gitea/models/secret"
- notify_service "code.gitea.io/gitea/services/notify"
-
- runnerv1 "code.gitea.io/actions-proto-go/runner/v1"
- "google.golang.org/protobuf/types/known/structpb"
- )
-
- func PickTask(ctx context.Context, runner *actions_model.ActionRunner) (*runnerv1.Task, bool, error) {
- var (
- task *runnerv1.Task
- job *actions_model.ActionRunJob
- actionTask *actions_model.ActionTask
- )
-
- if runner.Ephemeral {
- var task actions_model.ActionTask
- has, err := db.GetEngine(ctx).Where("runner_id = ?", runner.ID).Get(&task)
- // Let the runner retry the request, do not allow to proceed
- if err != nil {
- return nil, false, err
- }
- if has {
- if task.Status == actions_model.StatusWaiting || task.Status == actions_model.StatusRunning || task.Status == actions_model.StatusBlocked {
- return nil, false, nil
- }
- // task has been finished, remove it
- _, err = db.DeleteByID[actions_model.ActionRunner](ctx, runner.ID)
- if err != nil {
- return nil, false, err
- }
- return nil, false, errors.New("runner has been removed")
- }
- }
-
- if err := db.WithTx(ctx, func(ctx context.Context) error {
- t, ok, err := actions_model.CreateTaskForRunner(ctx, runner)
- if err != nil {
- return fmt.Errorf("CreateTaskForRunner: %w", err)
- }
- if !ok {
- return nil
- }
-
- if err := t.LoadAttributes(ctx); err != nil {
- return fmt.Errorf("task LoadAttributes: %w", err)
- }
- job = t.Job
- actionTask = t
-
- secrets, err := secret_model.GetSecretsOfTask(ctx, t)
- if err != nil {
- return fmt.Errorf("GetSecretsOfTask: %w", err)
- }
-
- vars, err := actions_model.GetVariablesOfRun(ctx, t.Job.Run)
- if err != nil {
- return fmt.Errorf("GetVariablesOfRun: %w", err)
- }
-
- needs, err := findTaskNeeds(ctx, job)
- if err != nil {
- return fmt.Errorf("findTaskNeeds: %w", err)
- }
-
- taskContext, err := generateTaskContext(t)
- if err != nil {
- return fmt.Errorf("generateTaskContext: %w", err)
- }
-
- task = &runnerv1.Task{
- Id: t.ID,
- WorkflowPayload: t.Job.WorkflowPayload,
- Context: taskContext,
- Secrets: secrets,
- Vars: vars,
- Needs: needs,
- }
-
- return nil
- }); err != nil {
- return nil, false, err
- }
-
- if task == nil {
- return nil, false, nil
- }
-
- CreateCommitStatus(ctx, job)
- notify_service.WorkflowJobStatusUpdate(ctx, job.Run.Repo, job.Run.TriggerUser, job, actionTask)
-
- return task, true, nil
- }
-
- func generateTaskContext(t *actions_model.ActionTask) (*structpb.Struct, error) {
- giteaRuntimeToken, err := CreateAuthorizationToken(t.ID, t.Job.RunID, t.JobID)
- if err != nil {
- return nil, err
- }
-
- gitCtx := GenerateGiteaContext(t.Job.Run, t.Job)
- gitCtx["token"] = t.Token
- gitCtx["gitea_runtime_token"] = giteaRuntimeToken
-
- return structpb.NewStruct(gitCtx)
- }
-
- func findTaskNeeds(ctx context.Context, taskJob *actions_model.ActionRunJob) (map[string]*runnerv1.TaskNeed, error) {
- taskNeeds, err := FindTaskNeeds(ctx, taskJob)
- if err != nil {
- return nil, err
- }
- ret := make(map[string]*runnerv1.TaskNeed, len(taskNeeds))
- for jobID, taskNeed := range taskNeeds {
- ret[jobID] = &runnerv1.TaskNeed{
- Outputs: taskNeed.Outputs,
- Result: runnerv1.Result(taskNeed.Result),
- }
- }
- return ret, nil
- }
|