gitea源码

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208
  1. // Copyright 2024 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package project
  4. import (
  5. "context"
  6. "errors"
  7. "code.gitea.io/gitea/models/db"
  8. issues_model "code.gitea.io/gitea/models/issues"
  9. project_model "code.gitea.io/gitea/models/project"
  10. user_model "code.gitea.io/gitea/models/user"
  11. "code.gitea.io/gitea/modules/optional"
  12. )
  13. // MoveIssuesOnProjectColumn moves or keeps issues in a column and sorts them inside that column
  14. func MoveIssuesOnProjectColumn(ctx context.Context, doer *user_model.User, column *project_model.Column, sortedIssueIDs map[int64]int64) error {
  15. return db.WithTx(ctx, func(ctx context.Context) error {
  16. issueIDs := make([]int64, 0, len(sortedIssueIDs))
  17. for _, issueID := range sortedIssueIDs {
  18. issueIDs = append(issueIDs, issueID)
  19. }
  20. count, err := db.GetEngine(ctx).
  21. Where("project_id=?", column.ProjectID).
  22. In("issue_id", issueIDs).
  23. Count(new(project_model.ProjectIssue))
  24. if err != nil {
  25. return err
  26. }
  27. if int(count) != len(sortedIssueIDs) {
  28. return errors.New("all issues have to be added to a project first")
  29. }
  30. issues, err := issues_model.GetIssuesByIDs(ctx, issueIDs)
  31. if err != nil {
  32. return err
  33. }
  34. if _, err := issues.LoadRepositories(ctx); err != nil {
  35. return err
  36. }
  37. project, err := project_model.GetProjectByID(ctx, column.ProjectID)
  38. if err != nil {
  39. return err
  40. }
  41. issuesMap := make(map[int64]*issues_model.Issue, len(issues))
  42. for _, issue := range issues {
  43. issuesMap[issue.ID] = issue
  44. }
  45. for sorting, issueID := range sortedIssueIDs {
  46. curIssue := issuesMap[issueID]
  47. if curIssue == nil {
  48. continue
  49. }
  50. projectColumnID, err := curIssue.ProjectColumnID(ctx)
  51. if err != nil {
  52. return err
  53. }
  54. if projectColumnID != column.ID {
  55. // add timeline to issue
  56. if _, err := issues_model.CreateComment(ctx, &issues_model.CreateCommentOptions{
  57. Type: issues_model.CommentTypeProjectColumn,
  58. Doer: doer,
  59. Repo: curIssue.Repo,
  60. Issue: curIssue,
  61. ProjectID: column.ProjectID,
  62. ProjectTitle: project.Title,
  63. ProjectColumnID: column.ID,
  64. ProjectColumnTitle: column.Title,
  65. }); err != nil {
  66. return err
  67. }
  68. }
  69. _, err = db.Exec(ctx, "UPDATE `project_issue` SET project_board_id=?, sorting=? WHERE issue_id=?", column.ID, sorting, issueID)
  70. if err != nil {
  71. return err
  72. }
  73. }
  74. return nil
  75. })
  76. }
  77. // LoadIssuesFromProject load issues assigned to each project column inside the given project
  78. func LoadIssuesFromProject(ctx context.Context, project *project_model.Project, opts *issues_model.IssuesOptions) (map[int64]issues_model.IssueList, error) {
  79. issueList, err := issues_model.Issues(ctx, opts.Copy(func(o *issues_model.IssuesOptions) {
  80. o.ProjectID = project.ID
  81. o.SortType = "project-column-sorting"
  82. }))
  83. if err != nil {
  84. return nil, err
  85. }
  86. if err := issueList.LoadComments(ctx); err != nil {
  87. return nil, err
  88. }
  89. defaultColumn, err := project.MustDefaultColumn(ctx)
  90. if err != nil {
  91. return nil, err
  92. }
  93. issueColumnMap, err := issues_model.LoadProjectIssueColumnMap(ctx, project.ID, defaultColumn.ID)
  94. if err != nil {
  95. return nil, err
  96. }
  97. results := make(map[int64]issues_model.IssueList)
  98. for _, issue := range issueList {
  99. projectColumnID, ok := issueColumnMap[issue.ID]
  100. if !ok {
  101. continue
  102. }
  103. if _, ok := results[projectColumnID]; !ok {
  104. results[projectColumnID] = make(issues_model.IssueList, 0)
  105. }
  106. results[projectColumnID] = append(results[projectColumnID], issue)
  107. }
  108. return results, nil
  109. }
  110. // NumClosedIssues return counter of closed issues assigned to a project
  111. func loadNumClosedIssues(ctx context.Context, p *project_model.Project) error {
  112. cnt, err := db.GetEngine(ctx).Table("project_issue").
  113. Join("INNER", "issue", "project_issue.issue_id=issue.id").
  114. Where("project_issue.project_id=? AND issue.is_closed=?", p.ID, true).
  115. Cols("issue_id").
  116. Count()
  117. if err != nil {
  118. return err
  119. }
  120. p.NumClosedIssues = cnt
  121. return nil
  122. }
  123. // NumOpenIssues return counter of open issues assigned to a project
  124. func loadNumOpenIssues(ctx context.Context, p *project_model.Project) error {
  125. cnt, err := db.GetEngine(ctx).Table("project_issue").
  126. Join("INNER", "issue", "project_issue.issue_id=issue.id").
  127. Where("project_issue.project_id=? AND issue.is_closed=?", p.ID, false).
  128. Cols("issue_id").
  129. Count()
  130. if err != nil {
  131. return err
  132. }
  133. p.NumOpenIssues = cnt
  134. return nil
  135. }
  136. func LoadIssueNumbersForProjects(ctx context.Context, projects []*project_model.Project, doer *user_model.User) error {
  137. for _, project := range projects {
  138. if err := LoadIssueNumbersForProject(ctx, project, doer); err != nil {
  139. return err
  140. }
  141. }
  142. return nil
  143. }
  144. func LoadIssueNumbersForProject(ctx context.Context, project *project_model.Project, doer *user_model.User) error {
  145. // for repository project, just get the numbers
  146. if project.OwnerID == 0 {
  147. if err := loadNumClosedIssues(ctx, project); err != nil {
  148. return err
  149. }
  150. if err := loadNumOpenIssues(ctx, project); err != nil {
  151. return err
  152. }
  153. project.NumIssues = project.NumClosedIssues + project.NumOpenIssues
  154. return nil
  155. }
  156. if err := project.LoadOwner(ctx); err != nil {
  157. return err
  158. }
  159. // for user or org projects, we need to check access permissions
  160. opts := issues_model.IssuesOptions{
  161. ProjectID: project.ID,
  162. Doer: doer,
  163. AllPublic: doer == nil,
  164. Owner: project.Owner,
  165. }
  166. var err error
  167. project.NumOpenIssues, err = issues_model.CountIssues(ctx, opts.Copy(func(o *issues_model.IssuesOptions) {
  168. o.IsClosed = optional.Some(false)
  169. }))
  170. if err != nil {
  171. return err
  172. }
  173. project.NumClosedIssues, err = issues_model.CountIssues(ctx, opts.Copy(func(o *issues_model.IssuesOptions) {
  174. o.IsClosed = optional.Some(true)
  175. }))
  176. if err != nil {
  177. return err
  178. }
  179. project.NumIssues = project.NumClosedIssues + project.NumOpenIssues
  180. return nil
  181. }