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.

action.go 5.9KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. // Copyright 2019 The Gitea Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package repofiles
  5. import (
  6. "encoding/json"
  7. "fmt"
  8. "strings"
  9. "code.gitea.io/gitea/models"
  10. "code.gitea.io/gitea/modules/git"
  11. "code.gitea.io/gitea/modules/log"
  12. "code.gitea.io/gitea/modules/setting"
  13. api "code.gitea.io/gitea/modules/structs"
  14. )
  15. // CommitRepoActionOptions represent options of a new commit action.
  16. type CommitRepoActionOptions struct {
  17. PusherName string
  18. RepoOwnerID int64
  19. RepoName string
  20. RefFullName string
  21. OldCommitID string
  22. NewCommitID string
  23. Commits *models.PushCommits
  24. }
  25. // CommitRepoAction adds new commit action to the repository, and prepare
  26. // corresponding webhooks.
  27. func CommitRepoAction(opts CommitRepoActionOptions) error {
  28. pusher, err := models.GetUserByName(opts.PusherName)
  29. if err != nil {
  30. return fmt.Errorf("GetUserByName [%s]: %v", opts.PusherName, err)
  31. }
  32. repo, err := models.GetRepositoryByName(opts.RepoOwnerID, opts.RepoName)
  33. if err != nil {
  34. return fmt.Errorf("GetRepositoryByName [owner_id: %d, name: %s]: %v", opts.RepoOwnerID, opts.RepoName, err)
  35. }
  36. refName := git.RefEndName(opts.RefFullName)
  37. // Change default branch and empty status only if pushed ref is non-empty branch.
  38. if repo.IsEmpty && opts.NewCommitID != git.EmptySHA && strings.HasPrefix(opts.RefFullName, git.BranchPrefix) {
  39. repo.DefaultBranch = refName
  40. repo.IsEmpty = false
  41. if refName != "master" {
  42. gitRepo, err := git.OpenRepository(repo.RepoPath())
  43. if err != nil {
  44. return err
  45. }
  46. if err := gitRepo.SetDefaultBranch(repo.DefaultBranch); err != nil {
  47. if !git.IsErrUnsupportedVersion(err) {
  48. return err
  49. }
  50. }
  51. }
  52. }
  53. // Change repository empty status and update last updated time.
  54. if err = models.UpdateRepository(repo, false); err != nil {
  55. return fmt.Errorf("UpdateRepository: %v", err)
  56. }
  57. isNewBranch := false
  58. opType := models.ActionCommitRepo
  59. // Check it's tag push or branch.
  60. if strings.HasPrefix(opts.RefFullName, git.TagPrefix) {
  61. opType = models.ActionPushTag
  62. if opts.NewCommitID == git.EmptySHA {
  63. opType = models.ActionDeleteTag
  64. }
  65. opts.Commits = &models.PushCommits{}
  66. } else if opts.NewCommitID == git.EmptySHA {
  67. opType = models.ActionDeleteBranch
  68. opts.Commits = &models.PushCommits{}
  69. } else {
  70. // if not the first commit, set the compare URL.
  71. if opts.OldCommitID == git.EmptySHA {
  72. isNewBranch = true
  73. } else {
  74. opts.Commits.CompareURL = repo.ComposeCompareURL(opts.OldCommitID, opts.NewCommitID)
  75. }
  76. if err = models.UpdateIssuesCommit(pusher, repo, opts.Commits.Commits, refName); err != nil {
  77. log.Error("updateIssuesCommit: %v", err)
  78. }
  79. }
  80. if len(opts.Commits.Commits) > setting.UI.FeedMaxCommitNum {
  81. opts.Commits.Commits = opts.Commits.Commits[:setting.UI.FeedMaxCommitNum]
  82. }
  83. data, err := json.Marshal(opts.Commits)
  84. if err != nil {
  85. return fmt.Errorf("Marshal: %v", err)
  86. }
  87. if err = models.NotifyWatchers(&models.Action{
  88. ActUserID: pusher.ID,
  89. ActUser: pusher,
  90. OpType: opType,
  91. Content: string(data),
  92. RepoID: repo.ID,
  93. Repo: repo,
  94. RefName: refName,
  95. IsPrivate: repo.IsPrivate,
  96. }); err != nil {
  97. return fmt.Errorf("NotifyWatchers: %v", err)
  98. }
  99. defer func() {
  100. go models.HookQueue.Add(repo.ID)
  101. }()
  102. apiPusher := pusher.APIFormat()
  103. apiRepo := repo.APIFormat(models.AccessModeNone)
  104. var shaSum string
  105. var isHookEventPush = false
  106. switch opType {
  107. case models.ActionCommitRepo: // Push
  108. isHookEventPush = true
  109. if isNewBranch {
  110. gitRepo, err := git.OpenRepository(repo.RepoPath())
  111. if err != nil {
  112. log.Error("OpenRepository[%s]: %v", repo.RepoPath(), err)
  113. }
  114. shaSum, err = gitRepo.GetBranchCommitID(refName)
  115. if err != nil {
  116. log.Error("GetBranchCommitID[%s]: %v", opts.RefFullName, err)
  117. }
  118. if err = models.PrepareWebhooks(repo, models.HookEventCreate, &api.CreatePayload{
  119. Ref: refName,
  120. Sha: shaSum,
  121. RefType: "branch",
  122. Repo: apiRepo,
  123. Sender: apiPusher,
  124. }); err != nil {
  125. return fmt.Errorf("PrepareWebhooks: %v", err)
  126. }
  127. }
  128. case models.ActionDeleteBranch: // Delete Branch
  129. isHookEventPush = true
  130. if err = models.PrepareWebhooks(repo, models.HookEventDelete, &api.DeletePayload{
  131. Ref: refName,
  132. RefType: "branch",
  133. PusherType: api.PusherTypeUser,
  134. Repo: apiRepo,
  135. Sender: apiPusher,
  136. }); err != nil {
  137. return fmt.Errorf("PrepareWebhooks.(delete branch): %v", err)
  138. }
  139. case models.ActionPushTag: // Create
  140. isHookEventPush = true
  141. gitRepo, err := git.OpenRepository(repo.RepoPath())
  142. if err != nil {
  143. log.Error("OpenRepository[%s]: %v", repo.RepoPath(), err)
  144. }
  145. shaSum, err = gitRepo.GetTagCommitID(refName)
  146. if err != nil {
  147. log.Error("GetTagCommitID[%s]: %v", opts.RefFullName, err)
  148. }
  149. if err = models.PrepareWebhooks(repo, models.HookEventCreate, &api.CreatePayload{
  150. Ref: refName,
  151. Sha: shaSum,
  152. RefType: "tag",
  153. Repo: apiRepo,
  154. Sender: apiPusher,
  155. }); err != nil {
  156. return fmt.Errorf("PrepareWebhooks: %v", err)
  157. }
  158. case models.ActionDeleteTag: // Delete Tag
  159. isHookEventPush = true
  160. if err = models.PrepareWebhooks(repo, models.HookEventDelete, &api.DeletePayload{
  161. Ref: refName,
  162. RefType: "tag",
  163. PusherType: api.PusherTypeUser,
  164. Repo: apiRepo,
  165. Sender: apiPusher,
  166. }); err != nil {
  167. return fmt.Errorf("PrepareWebhooks.(delete tag): %v", err)
  168. }
  169. }
  170. if isHookEventPush {
  171. commits, err := opts.Commits.ToAPIPayloadCommits(repo.RepoPath(), repo.HTMLURL())
  172. if err != nil {
  173. return err
  174. }
  175. if err = models.PrepareWebhooks(repo, models.HookEventPush, &api.PushPayload{
  176. Ref: opts.RefFullName,
  177. Before: opts.OldCommitID,
  178. After: opts.NewCommitID,
  179. CompareURL: setting.AppURL + opts.Commits.CompareURL,
  180. Commits: commits,
  181. Repo: apiRepo,
  182. Pusher: apiPusher,
  183. Sender: apiPusher,
  184. }); err != nil {
  185. return fmt.Errorf("PrepareWebhooks: %v", err)
  186. }
  187. }
  188. return nil
  189. }