You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

workflow_notifier.go 4.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  1. // Copyright 2024 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package projects
  4. import (
  5. "context"
  6. "strings"
  7. issues_model "code.gitea.io/gitea/models/issues"
  8. project_model "code.gitea.io/gitea/models/project"
  9. repo_model "code.gitea.io/gitea/models/repo"
  10. user_model "code.gitea.io/gitea/models/user"
  11. "code.gitea.io/gitea/modules/git"
  12. "code.gitea.io/gitea/modules/gitrepo"
  13. "code.gitea.io/gitea/modules/log"
  14. project_module "code.gitea.io/gitea/modules/projects"
  15. notify_service "code.gitea.io/gitea/services/notify"
  16. )
  17. func init() {
  18. notify_service.RegisterNotifier(&workflowNotifier{})
  19. }
  20. type workflowNotifier struct {
  21. notify_service.NullNotifier
  22. }
  23. var _ notify_service.Notifier = &workflowNotifier{}
  24. // NewNotifier create a new workflowNotifier notifier
  25. func NewNotifier() notify_service.Notifier {
  26. return &workflowNotifier{}
  27. }
  28. func findRepoProjectsWorkflows(ctx context.Context, repo *repo_model.Repository) ([]*project_module.Workflow, error) {
  29. gitRepo, err := gitrepo.OpenRepository(ctx, repo)
  30. if err != nil {
  31. log.Error("IssueChangeStatus: OpenRepository: %v", err)
  32. return nil, err
  33. }
  34. defer gitRepo.Close()
  35. // Get the commit object for the ref
  36. commit, err := gitRepo.GetCommit(repo.DefaultBranch)
  37. if err != nil {
  38. log.Error("gitRepo.GetCommit: %w", err)
  39. return nil, err
  40. }
  41. tree, err := commit.SubTree(".gitea/projects")
  42. if _, ok := err.(git.ErrNotExist); ok {
  43. return nil, nil
  44. }
  45. if err != nil {
  46. log.Error("commit.SubTree: %w", err)
  47. return nil, err
  48. }
  49. entries, err := tree.ListEntriesRecursiveFast()
  50. if err != nil {
  51. log.Error("tree.ListEntriesRecursiveFast: %w", err)
  52. return nil, err
  53. }
  54. ret := make(git.Entries, 0, len(entries))
  55. for _, entry := range entries {
  56. if strings.HasSuffix(entry.Name(), ".yml") || strings.HasSuffix(entry.Name(), ".yaml") {
  57. ret = append(ret, entry)
  58. }
  59. }
  60. if len(ret) == 0 {
  61. return nil, nil
  62. }
  63. wfs := make([]*project_module.Workflow, 0, len(ret))
  64. for _, entry := range ret {
  65. workflowContent, err := commit.GetFileContent(".gitea/projects/"+entry.Name(), 1024*1024)
  66. if err != nil {
  67. log.Error("gitRepo.GetCommit: %w", err)
  68. return nil, err
  69. }
  70. wf, err := project_module.ParseWorkflow(workflowContent)
  71. if err != nil {
  72. log.Error("IssueChangeStatus: OpenRepository: %v", err)
  73. return nil, err
  74. }
  75. projectName := strings.TrimSuffix(strings.TrimSuffix(entry.Name(), ".yml"), ".yaml")
  76. project, err := project_model.GetProjectByName(ctx, repo.ID, projectName)
  77. if err != nil {
  78. log.Error("IssueChangeStatus: GetProjectByName: %v", err)
  79. return nil, err
  80. }
  81. wf.ProjectID = project.ID
  82. wfs = append(wfs, wf)
  83. }
  84. return wfs, nil
  85. }
  86. func (m *workflowNotifier) NewIssue(ctx context.Context, issue *issues_model.Issue, mentions []*user_model.User) {
  87. if err := issue.LoadRepo(ctx); err != nil {
  88. log.Error("NewIssue: LoadRepo: %v", err)
  89. return
  90. }
  91. wfs, err := findRepoProjectsWorkflows(ctx, issue.Repo)
  92. if err != nil {
  93. log.Error("NewIssue: findRepoProjectsWorkflows: %v", err)
  94. return
  95. }
  96. for _, wf := range wfs {
  97. if err := wf.FireAction(project_module.EventItemClosed, func(action project_module.Action) error {
  98. board, err := project_model.GetBoardByProjectIDAndBoardName(ctx, wf.ProjectID, action.SetValue)
  99. if err != nil {
  100. log.Error("NewIssue: GetBoardByProjectIDAndBoardName: %v", err)
  101. return err
  102. }
  103. return project_model.AddIssueToBoard(ctx, issue.ID, board)
  104. }); err != nil {
  105. log.Error("NewIssue: FireAction: %v", err)
  106. return
  107. }
  108. }
  109. }
  110. }
  111. func (m *workflowNotifier) IssueChangeStatus(ctx context.Context, doer *user_model.User, commitID string, issue *issues_model.Issue, actionComment *issues_model.Comment, isClosed bool) {
  112. if isClosed {
  113. if err := issue.LoadRepo(ctx); err != nil {
  114. log.Error("IssueChangeStatus: LoadRepo: %v", err)
  115. return
  116. }
  117. wfs, err := findRepoProjectsWorkflows(ctx, issue.Repo)
  118. if err != nil {
  119. log.Error("IssueChangeStatus: findRepoProjectsWorkflows: %v", err)
  120. return
  121. }
  122. for _, wf := range wfs {
  123. if err := wf.FireAction(project_module.EventItemClosed, func(action project_module.Action) error {
  124. board, err := project_model.GetBoardByProjectIDAndBoardName(ctx, wf.ProjectID, action.SetValue)
  125. if err != nil {
  126. log.Error("IssueChangeStatus: GetBoardByProjectIDAndBoardName: %v", err)
  127. return err
  128. }
  129. return project_model.MoveIssueToAnotherBoard(ctx, issue.ID, board)
  130. }); err != nil {
  131. log.Error("IssueChangeStatus: FireAction: %v", err)
  132. return
  133. }
  134. }
  135. }
  136. }