gitea源码

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. // Copyright 2022 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package actions
  4. import (
  5. "context"
  6. "errors"
  7. "fmt"
  8. actions_model "code.gitea.io/gitea/models/actions"
  9. "code.gitea.io/gitea/models/db"
  10. secret_model "code.gitea.io/gitea/models/secret"
  11. notify_service "code.gitea.io/gitea/services/notify"
  12. runnerv1 "code.gitea.io/actions-proto-go/runner/v1"
  13. "google.golang.org/protobuf/types/known/structpb"
  14. )
  15. func PickTask(ctx context.Context, runner *actions_model.ActionRunner) (*runnerv1.Task, bool, error) {
  16. var (
  17. task *runnerv1.Task
  18. job *actions_model.ActionRunJob
  19. actionTask *actions_model.ActionTask
  20. )
  21. if runner.Ephemeral {
  22. var task actions_model.ActionTask
  23. has, err := db.GetEngine(ctx).Where("runner_id = ?", runner.ID).Get(&task)
  24. // Let the runner retry the request, do not allow to proceed
  25. if err != nil {
  26. return nil, false, err
  27. }
  28. if has {
  29. if task.Status == actions_model.StatusWaiting || task.Status == actions_model.StatusRunning || task.Status == actions_model.StatusBlocked {
  30. return nil, false, nil
  31. }
  32. // task has been finished, remove it
  33. _, err = db.DeleteByID[actions_model.ActionRunner](ctx, runner.ID)
  34. if err != nil {
  35. return nil, false, err
  36. }
  37. return nil, false, errors.New("runner has been removed")
  38. }
  39. }
  40. if err := db.WithTx(ctx, func(ctx context.Context) error {
  41. t, ok, err := actions_model.CreateTaskForRunner(ctx, runner)
  42. if err != nil {
  43. return fmt.Errorf("CreateTaskForRunner: %w", err)
  44. }
  45. if !ok {
  46. return nil
  47. }
  48. if err := t.LoadAttributes(ctx); err != nil {
  49. return fmt.Errorf("task LoadAttributes: %w", err)
  50. }
  51. job = t.Job
  52. actionTask = t
  53. secrets, err := secret_model.GetSecretsOfTask(ctx, t)
  54. if err != nil {
  55. return fmt.Errorf("GetSecretsOfTask: %w", err)
  56. }
  57. vars, err := actions_model.GetVariablesOfRun(ctx, t.Job.Run)
  58. if err != nil {
  59. return fmt.Errorf("GetVariablesOfRun: %w", err)
  60. }
  61. needs, err := findTaskNeeds(ctx, job)
  62. if err != nil {
  63. return fmt.Errorf("findTaskNeeds: %w", err)
  64. }
  65. taskContext, err := generateTaskContext(t)
  66. if err != nil {
  67. return fmt.Errorf("generateTaskContext: %w", err)
  68. }
  69. task = &runnerv1.Task{
  70. Id: t.ID,
  71. WorkflowPayload: t.Job.WorkflowPayload,
  72. Context: taskContext,
  73. Secrets: secrets,
  74. Vars: vars,
  75. Needs: needs,
  76. }
  77. return nil
  78. }); err != nil {
  79. return nil, false, err
  80. }
  81. if task == nil {
  82. return nil, false, nil
  83. }
  84. CreateCommitStatus(ctx, job)
  85. notify_service.WorkflowJobStatusUpdate(ctx, job.Run.Repo, job.Run.TriggerUser, job, actionTask)
  86. return task, true, nil
  87. }
  88. func generateTaskContext(t *actions_model.ActionTask) (*structpb.Struct, error) {
  89. giteaRuntimeToken, err := CreateAuthorizationToken(t.ID, t.Job.RunID, t.JobID)
  90. if err != nil {
  91. return nil, err
  92. }
  93. gitCtx := GenerateGiteaContext(t.Job.Run, t.Job)
  94. gitCtx["token"] = t.Token
  95. gitCtx["gitea_runtime_token"] = giteaRuntimeToken
  96. return structpb.NewStruct(gitCtx)
  97. }
  98. func findTaskNeeds(ctx context.Context, taskJob *actions_model.ActionRunJob) (map[string]*runnerv1.TaskNeed, error) {
  99. taskNeeds, err := FindTaskNeeds(ctx, taskJob)
  100. if err != nil {
  101. return nil, err
  102. }
  103. ret := make(map[string]*runnerv1.TaskNeed, len(taskNeeds))
  104. for jobID, taskNeed := range taskNeeds {
  105. ret[jobID] = &runnerv1.TaskNeed{
  106. Outputs: taskNeed.Outputs,
  107. Result: runnerv1.Result(taskNeed.Result),
  108. }
  109. }
  110. return ret, nil
  111. }