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

update.go 8.5 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335
  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. "strings"
  9. "time"
  10. "code.gitea.io/gitea/modules/git"
  11. "code.gitea.io/gitea/modules/timeutil"
  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. EnvPusherEmail = "GITEA_PUSHER_EMAIL"
  20. EnvPusherID = "GITEA_PUSHER_ID"
  21. EnvKeyID = "GITEA_KEY_ID"
  22. EnvIsDeployKey = "GITEA_IS_DEPLOY_KEY"
  23. )
  24. // CommitToPushCommit transforms a git.Commit to PushCommit type.
  25. func CommitToPushCommit(commit *git.Commit) *PushCommit {
  26. return &PushCommit{
  27. Sha1: commit.ID.String(),
  28. Message: commit.Message(),
  29. AuthorEmail: commit.Author.Email,
  30. AuthorName: commit.Author.Name,
  31. CommitterEmail: commit.Committer.Email,
  32. CommitterName: commit.Committer.Name,
  33. Timestamp: commit.Author.When,
  34. }
  35. }
  36. // ListToPushCommits transforms a list.List to PushCommits type.
  37. func ListToPushCommits(l *list.List) *PushCommits {
  38. var commits []*PushCommit
  39. var actEmail string
  40. for e := l.Front(); e != nil; e = e.Next() {
  41. commit := e.Value.(*git.Commit)
  42. if actEmail == "" {
  43. actEmail = commit.Committer.Email
  44. }
  45. commits = append(commits, CommitToPushCommit(commit))
  46. }
  47. return &PushCommits{l.Len(), commits, "", make(map[string]string), make(map[string]*User)}
  48. }
  49. // PushUpdateAddDeleteTags updates a number of added and delete tags
  50. func PushUpdateAddDeleteTags(repo *Repository, gitRepo *git.Repository, addTags, delTags []string) error {
  51. sess := x.NewSession()
  52. defer sess.Close()
  53. if err := sess.Begin(); err != nil {
  54. return fmt.Errorf("Unable to begin sess in PushUpdateDeleteTags: %v", err)
  55. }
  56. if err := pushUpdateDeleteTags(sess, repo, delTags); err != nil {
  57. return err
  58. }
  59. if err := pushUpdateAddTags(sess, repo, gitRepo, addTags); err != nil {
  60. return err
  61. }
  62. return sess.Commit()
  63. }
  64. // PushUpdateDeleteTags updates a number of delete tags
  65. func PushUpdateDeleteTags(repo *Repository, tags []string) error {
  66. sess := x.NewSession()
  67. defer sess.Close()
  68. if err := sess.Begin(); err != nil {
  69. return fmt.Errorf("Unable to begin sess in PushUpdateDeleteTags: %v", err)
  70. }
  71. if err := pushUpdateDeleteTags(sess, repo, tags); err != nil {
  72. return err
  73. }
  74. return sess.Commit()
  75. }
  76. func pushUpdateDeleteTags(e Engine, repo *Repository, tags []string) error {
  77. if len(tags) == 0 {
  78. return nil
  79. }
  80. lowerTags := make([]string, 0, len(tags))
  81. for _, tag := range tags {
  82. lowerTags = append(lowerTags, strings.ToLower(tag))
  83. }
  84. if _, err := e.
  85. Where("repo_id = ? AND is_tag = ?", repo.ID, true).
  86. In("lower_tag_name", lowerTags).
  87. Delete(new(Release)); err != nil {
  88. return fmt.Errorf("Delete: %v", err)
  89. }
  90. if _, err := e.
  91. Where("repo_id = ? AND is_tag = ?", repo.ID, false).
  92. In("lower_tag_name", lowerTags).
  93. Cols("is_draft", "num_commits", "sha1").
  94. Update(&Release{
  95. IsDraft: true,
  96. }); err != nil {
  97. return fmt.Errorf("Update: %v", err)
  98. }
  99. return nil
  100. }
  101. // PushUpdateDeleteTag must be called for any push actions to delete tag
  102. func PushUpdateDeleteTag(repo *Repository, tagName string) error {
  103. rel, err := GetRelease(repo.ID, tagName)
  104. if err != nil {
  105. if IsErrReleaseNotExist(err) {
  106. return nil
  107. }
  108. return fmt.Errorf("GetRelease: %v", err)
  109. }
  110. if rel.IsTag {
  111. if _, err = x.ID(rel.ID).Delete(new(Release)); err != nil {
  112. return fmt.Errorf("Delete: %v", err)
  113. }
  114. } else {
  115. rel.IsDraft = true
  116. rel.NumCommits = 0
  117. rel.Sha1 = ""
  118. if _, err = x.ID(rel.ID).AllCols().Update(rel); err != nil {
  119. return fmt.Errorf("Update: %v", err)
  120. }
  121. }
  122. return nil
  123. }
  124. // PushUpdateAddTags updates a number of add tags
  125. func PushUpdateAddTags(repo *Repository, gitRepo *git.Repository, tags []string) error {
  126. sess := x.NewSession()
  127. defer sess.Close()
  128. if err := sess.Begin(); err != nil {
  129. return fmt.Errorf("Unable to begin sess in PushUpdateAddTags: %v", err)
  130. }
  131. if err := pushUpdateAddTags(sess, repo, gitRepo, tags); err != nil {
  132. return err
  133. }
  134. return sess.Commit()
  135. }
  136. func pushUpdateAddTags(e Engine, repo *Repository, gitRepo *git.Repository, tags []string) error {
  137. if len(tags) == 0 {
  138. return nil
  139. }
  140. lowerTags := make([]string, 0, len(tags))
  141. for _, tag := range tags {
  142. lowerTags = append(lowerTags, strings.ToLower(tag))
  143. }
  144. releases := make([]Release, 0, len(tags))
  145. if err := e.Where("repo_id = ?", repo.ID).
  146. In("lower_tag_name", lowerTags).Find(&releases); err != nil {
  147. return fmt.Errorf("GetRelease: %v", err)
  148. }
  149. relMap := make(map[string]*Release)
  150. for _, rel := range releases {
  151. relMap[rel.LowerTagName] = &rel
  152. }
  153. newReleases := make([]*Release, 0, len(lowerTags)-len(relMap))
  154. emailToUser := make(map[string]*User)
  155. for i, lowerTag := range lowerTags {
  156. tag, err := gitRepo.GetTag(tags[i])
  157. if err != nil {
  158. return fmt.Errorf("GetTag: %v", err)
  159. }
  160. commit, err := tag.Commit()
  161. if err != nil {
  162. return fmt.Errorf("Commit: %v", err)
  163. }
  164. sig := tag.Tagger
  165. if sig == nil {
  166. sig = commit.Author
  167. }
  168. if sig == nil {
  169. sig = commit.Committer
  170. }
  171. var author *User
  172. var createdAt = time.Unix(1, 0)
  173. if sig != nil {
  174. var ok bool
  175. author, ok = emailToUser[sig.Email]
  176. if !ok {
  177. author, err = GetUserByEmail(sig.Email)
  178. if err != nil && !IsErrUserNotExist(err) {
  179. return fmt.Errorf("GetUserByEmail: %v", err)
  180. }
  181. }
  182. createdAt = sig.When
  183. }
  184. commitsCount, err := commit.CommitsCount()
  185. if err != nil {
  186. return fmt.Errorf("CommitsCount: %v", err)
  187. }
  188. rel, has := relMap[lowerTag]
  189. if !has {
  190. rel = &Release{
  191. RepoID: repo.ID,
  192. Title: "",
  193. TagName: tags[i],
  194. LowerTagName: lowerTag,
  195. Target: "",
  196. Sha1: commit.ID.String(),
  197. NumCommits: commitsCount,
  198. Note: "",
  199. IsDraft: false,
  200. IsPrerelease: false,
  201. IsTag: true,
  202. CreatedUnix: timeutil.TimeStamp(createdAt.Unix()),
  203. }
  204. if author != nil {
  205. rel.PublisherID = author.ID
  206. }
  207. newReleases = append(newReleases, rel)
  208. } else {
  209. rel.Sha1 = commit.ID.String()
  210. rel.CreatedUnix = timeutil.TimeStamp(createdAt.Unix())
  211. rel.NumCommits = commitsCount
  212. rel.IsDraft = false
  213. if rel.IsTag && author != nil {
  214. rel.PublisherID = author.ID
  215. }
  216. if _, err = e.ID(rel.ID).AllCols().Update(rel); err != nil {
  217. return fmt.Errorf("Update: %v", err)
  218. }
  219. }
  220. }
  221. if len(newReleases) > 0 {
  222. if _, err := e.Insert(newReleases); err != nil {
  223. return fmt.Errorf("Insert: %v", err)
  224. }
  225. }
  226. return nil
  227. }
  228. // PushUpdateAddTag must be called for any push actions to add tag
  229. func PushUpdateAddTag(repo *Repository, gitRepo *git.Repository, tagName string) error {
  230. rel, err := GetRelease(repo.ID, tagName)
  231. if err != nil && !IsErrReleaseNotExist(err) {
  232. return fmt.Errorf("GetRelease: %v", err)
  233. }
  234. tag, err := gitRepo.GetTag(tagName)
  235. if err != nil {
  236. return fmt.Errorf("GetTag: %v", err)
  237. }
  238. commit, err := tag.Commit()
  239. if err != nil {
  240. return fmt.Errorf("Commit: %v", err)
  241. }
  242. sig := tag.Tagger
  243. if sig == nil {
  244. sig = commit.Author
  245. }
  246. if sig == nil {
  247. sig = commit.Committer
  248. }
  249. var author *User
  250. var createdAt = time.Unix(1, 0)
  251. if sig != nil {
  252. author, err = GetUserByEmail(sig.Email)
  253. if err != nil && !IsErrUserNotExist(err) {
  254. return fmt.Errorf("GetUserByEmail: %v", err)
  255. }
  256. createdAt = sig.When
  257. }
  258. commitsCount, err := commit.CommitsCount()
  259. if err != nil {
  260. return fmt.Errorf("CommitsCount: %v", err)
  261. }
  262. if rel == nil {
  263. rel = &Release{
  264. RepoID: repo.ID,
  265. Title: "",
  266. TagName: tagName,
  267. LowerTagName: strings.ToLower(tagName),
  268. Target: "",
  269. Sha1: commit.ID.String(),
  270. NumCommits: commitsCount,
  271. Note: "",
  272. IsDraft: false,
  273. IsPrerelease: false,
  274. IsTag: true,
  275. CreatedUnix: timeutil.TimeStamp(createdAt.Unix()),
  276. }
  277. if author != nil {
  278. rel.PublisherID = author.ID
  279. }
  280. if _, err = x.InsertOne(rel); err != nil {
  281. return fmt.Errorf("InsertOne: %v", err)
  282. }
  283. } else {
  284. rel.Sha1 = commit.ID.String()
  285. rel.CreatedUnix = timeutil.TimeStamp(createdAt.Unix())
  286. rel.NumCommits = commitsCount
  287. rel.IsDraft = false
  288. if rel.IsTag && author != nil {
  289. rel.PublisherID = author.ID
  290. }
  291. if _, err = x.ID(rel.ID).AllCols().Update(rel); err != nil {
  292. return fmt.Errorf("Update: %v", err)
  293. }
  294. }
  295. return nil
  296. }