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

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