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.

update.go 6.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. // Copyright 2014 The Gogs 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 models
  5. import (
  6. "container/list"
  7. "fmt"
  8. "os/exec"
  9. "strings"
  10. "code.gitea.io/git"
  11. "code.gitea.io/gitea/modules/log"
  12. )
  13. // env keys for git hooks need
  14. const (
  15. EnvRepoName = "GITEA_REPO_NAME"
  16. EnvRepoUsername = "GITEA_REPO_USER_NAME"
  17. EnvRepoIsWiki = "GITEA_REPO_IS_WIKI"
  18. EnvPusherName = "GITEA_PUSHER_NAME"
  19. EnvPusherID = "GITEA_PUSHER_ID"
  20. )
  21. // CommitToPushCommit transforms a git.Commit to PushCommit type.
  22. func CommitToPushCommit(commit *git.Commit) *PushCommit {
  23. return &PushCommit{
  24. Sha1: commit.ID.String(),
  25. Message: commit.Message(),
  26. AuthorEmail: commit.Author.Email,
  27. AuthorName: commit.Author.Name,
  28. CommitterEmail: commit.Committer.Email,
  29. CommitterName: commit.Committer.Name,
  30. Timestamp: commit.Author.When,
  31. }
  32. }
  33. // ListToPushCommits transforms a list.List to PushCommits type.
  34. func ListToPushCommits(l *list.List) *PushCommits {
  35. var commits []*PushCommit
  36. var actEmail string
  37. for e := l.Front(); e != nil; e = e.Next() {
  38. commit := e.Value.(*git.Commit)
  39. if actEmail == "" {
  40. actEmail = commit.Committer.Email
  41. }
  42. commits = append(commits, CommitToPushCommit(commit))
  43. }
  44. return &PushCommits{l.Len(), commits, "", nil}
  45. }
  46. // PushUpdateOptions defines the push update options
  47. type PushUpdateOptions struct {
  48. PusherID int64
  49. PusherName string
  50. RepoUserName string
  51. RepoName string
  52. RefFullName string
  53. OldCommitID string
  54. NewCommitID string
  55. }
  56. // PushUpdate must be called for any push actions in order to
  57. // generates necessary push action history feeds.
  58. func PushUpdate(branch string, opt PushUpdateOptions) error {
  59. repo, err := pushUpdate(opt)
  60. if err != nil {
  61. return err
  62. }
  63. pusher, err := GetUserByID(opt.PusherID)
  64. if err != nil {
  65. return err
  66. }
  67. log.Trace("TriggerTask '%s/%s' by %s", repo.Name, branch, pusher.Name)
  68. go AddTestPullRequestTask(pusher, repo.ID, branch, true)
  69. return nil
  70. }
  71. func pushUpdateDeleteTag(repo *Repository, gitRepo *git.Repository, tagName string) error {
  72. rel, err := GetRelease(repo.ID, tagName)
  73. if err != nil {
  74. if IsErrReleaseNotExist(err) {
  75. return nil
  76. }
  77. return fmt.Errorf("GetRelease: %v", err)
  78. }
  79. if rel.IsTag {
  80. if _, err = x.ID(rel.ID).Delete(new(Release)); err != nil {
  81. return fmt.Errorf("Delete: %v", err)
  82. }
  83. } else {
  84. rel.IsDraft = true
  85. rel.NumCommits = 0
  86. rel.Sha1 = ""
  87. if _, err = x.ID(rel.ID).AllCols().Update(rel); err != nil {
  88. return fmt.Errorf("Update: %v", err)
  89. }
  90. }
  91. return nil
  92. }
  93. func pushUpdateAddTag(repo *Repository, gitRepo *git.Repository, tagName string) error {
  94. rel, err := GetRelease(repo.ID, tagName)
  95. if err != nil && !IsErrReleaseNotExist(err) {
  96. return fmt.Errorf("GetRelease: %v", err)
  97. }
  98. tag, err := gitRepo.GetTag(tagName)
  99. if err != nil {
  100. return fmt.Errorf("GetTag: %v", err)
  101. }
  102. commit, err := tag.Commit()
  103. if err != nil {
  104. return fmt.Errorf("Commit: %v", err)
  105. }
  106. tagCreatedUnix := commit.Author.When.Unix()
  107. author, err := GetUserByEmail(commit.Author.Email)
  108. if err != nil && !IsErrUserNotExist(err) {
  109. return fmt.Errorf("GetUserByEmail: %v", err)
  110. }
  111. commitsCount, err := commit.CommitsCount()
  112. if err != nil {
  113. return fmt.Errorf("CommitsCount: %v", err)
  114. }
  115. if rel == nil {
  116. rel = &Release{
  117. RepoID: repo.ID,
  118. Title: "",
  119. TagName: tagName,
  120. LowerTagName: strings.ToLower(tagName),
  121. Target: "",
  122. Sha1: commit.ID.String(),
  123. NumCommits: commitsCount,
  124. Note: "",
  125. IsDraft: false,
  126. IsPrerelease: false,
  127. IsTag: true,
  128. CreatedUnix: tagCreatedUnix,
  129. }
  130. if author != nil {
  131. rel.PublisherID = author.ID
  132. }
  133. if _, err = x.InsertOne(rel); err != nil {
  134. return fmt.Errorf("InsertOne: %v", err)
  135. }
  136. } else {
  137. rel.Sha1 = commit.ID.String()
  138. rel.CreatedUnix = tagCreatedUnix
  139. rel.NumCommits = commitsCount
  140. rel.IsDraft = false
  141. if rel.IsTag && author != nil {
  142. rel.PublisherID = author.ID
  143. }
  144. if _, err = x.ID(rel.ID).AllCols().Update(rel); err != nil {
  145. return fmt.Errorf("Update: %v", err)
  146. }
  147. }
  148. return nil
  149. }
  150. func pushUpdate(opts PushUpdateOptions) (repo *Repository, err error) {
  151. isNewRef := opts.OldCommitID == git.EmptySHA
  152. isDelRef := opts.NewCommitID == git.EmptySHA
  153. if isNewRef && isDelRef {
  154. return nil, fmt.Errorf("Old and new revisions are both %s", git.EmptySHA)
  155. }
  156. repoPath := RepoPath(opts.RepoUserName, opts.RepoName)
  157. gitUpdate := exec.Command("git", "update-server-info")
  158. gitUpdate.Dir = repoPath
  159. if err = gitUpdate.Run(); err != nil {
  160. return nil, fmt.Errorf("Failed to call 'git update-server-info': %v", err)
  161. }
  162. owner, err := GetUserByName(opts.RepoUserName)
  163. if err != nil {
  164. return nil, fmt.Errorf("GetUserByName: %v", err)
  165. }
  166. repo, err = GetRepositoryByName(owner.ID, opts.RepoName)
  167. if err != nil {
  168. return nil, fmt.Errorf("GetRepositoryByName: %v", err)
  169. }
  170. gitRepo, err := git.OpenRepository(repoPath)
  171. if err != nil {
  172. return nil, fmt.Errorf("OpenRepository: %v", err)
  173. }
  174. if err = repo.UpdateSize(); err != nil {
  175. log.Error(4, "Failed to update size for repository: %v", err)
  176. }
  177. var commits = &PushCommits{}
  178. if strings.HasPrefix(opts.RefFullName, git.TagPrefix) {
  179. // If is tag reference
  180. if isDelRef {
  181. err = pushUpdateDeleteTag(repo, gitRepo, opts.RefFullName[len(git.TagPrefix):])
  182. if err != nil {
  183. return nil, fmt.Errorf("pushUpdateDeleteTag: %v", err)
  184. }
  185. } else {
  186. err = pushUpdateAddTag(repo, gitRepo, opts.RefFullName[len(git.TagPrefix):])
  187. if err != nil {
  188. return nil, fmt.Errorf("pushUpdateAddTag: %v", err)
  189. }
  190. }
  191. } else if !isDelRef {
  192. // If is branch reference
  193. newCommit, err := gitRepo.GetCommit(opts.NewCommitID)
  194. if err != nil {
  195. return nil, fmt.Errorf("gitRepo.GetCommit: %v", err)
  196. }
  197. // Push new branch.
  198. var l *list.List
  199. if isNewRef {
  200. l, err = newCommit.CommitsBeforeLimit(10)
  201. if err != nil {
  202. return nil, fmt.Errorf("newCommit.CommitsBeforeLimit: %v", err)
  203. }
  204. } else {
  205. l, err = newCommit.CommitsBeforeUntil(opts.OldCommitID)
  206. if err != nil {
  207. return nil, fmt.Errorf("newCommit.CommitsBeforeUntil: %v", err)
  208. }
  209. }
  210. commits = ListToPushCommits(l)
  211. }
  212. if err := CommitRepoAction(CommitRepoActionOptions{
  213. PusherName: opts.PusherName,
  214. RepoOwnerID: owner.ID,
  215. RepoName: repo.Name,
  216. RefFullName: opts.RefFullName,
  217. OldCommitID: opts.OldCommitID,
  218. NewCommitID: opts.NewCommitID,
  219. Commits: commits,
  220. }); err != nil {
  221. return nil, fmt.Errorf("CommitRepoAction: %v", err)
  222. }
  223. return repo, nil
  224. }