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.

release.go 5.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  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 release
  5. import (
  6. "fmt"
  7. "strings"
  8. "code.gitea.io/gitea/models"
  9. "code.gitea.io/gitea/modules/git"
  10. "code.gitea.io/gitea/modules/log"
  11. "code.gitea.io/gitea/modules/notification"
  12. "code.gitea.io/gitea/modules/repository"
  13. "code.gitea.io/gitea/modules/storage"
  14. "code.gitea.io/gitea/modules/timeutil"
  15. )
  16. func createTag(gitRepo *git.Repository, rel *models.Release) error {
  17. // Only actual create when publish.
  18. if !rel.IsDraft {
  19. if !gitRepo.IsTagExist(rel.TagName) {
  20. commit, err := gitRepo.GetCommit(rel.Target)
  21. if err != nil {
  22. return fmt.Errorf("GetCommit: %v", err)
  23. }
  24. // Trim '--' prefix to prevent command line argument vulnerability.
  25. rel.TagName = strings.TrimPrefix(rel.TagName, "--")
  26. if err = gitRepo.CreateTag(rel.TagName, commit.ID.String()); err != nil {
  27. if strings.Contains(err.Error(), "is not a valid tag name") {
  28. return models.ErrInvalidTagName{
  29. TagName: rel.TagName,
  30. }
  31. }
  32. return err
  33. }
  34. rel.LowerTagName = strings.ToLower(rel.TagName)
  35. // Prepare Notify
  36. if err := rel.LoadAttributes(); err != nil {
  37. log.Error("LoadAttributes: %v", err)
  38. return err
  39. }
  40. notification.NotifyPushCommits(
  41. rel.Publisher, rel.Repo, git.TagPrefix+rel.TagName,
  42. git.EmptySHA, commit.ID.String(), repository.NewPushCommits())
  43. notification.NotifyCreateRef(rel.Publisher, rel.Repo, "tag", git.TagPrefix+rel.TagName)
  44. rel.CreatedUnix = timeutil.TimeStampNow()
  45. }
  46. commit, err := gitRepo.GetTagCommit(rel.TagName)
  47. if err != nil {
  48. return fmt.Errorf("GetTagCommit: %v", err)
  49. }
  50. rel.Sha1 = commit.ID.String()
  51. rel.NumCommits, err = commit.CommitsCount()
  52. if err != nil {
  53. return fmt.Errorf("CommitsCount: %v", err)
  54. }
  55. u, err := models.GetUserByEmail(commit.Author.Email)
  56. if err == nil {
  57. rel.PublisherID = u.ID
  58. }
  59. } else {
  60. rel.CreatedUnix = timeutil.TimeStampNow()
  61. }
  62. return nil
  63. }
  64. // CreateRelease creates a new release of repository.
  65. func CreateRelease(gitRepo *git.Repository, rel *models.Release, attachmentUUIDs []string) error {
  66. isExist, err := models.IsReleaseExist(rel.RepoID, rel.TagName)
  67. if err != nil {
  68. return err
  69. } else if isExist {
  70. return models.ErrReleaseAlreadyExist{
  71. TagName: rel.TagName,
  72. }
  73. }
  74. if err = createTag(gitRepo, rel); err != nil {
  75. return err
  76. }
  77. rel.LowerTagName = strings.ToLower(rel.TagName)
  78. if err = models.InsertRelease(rel); err != nil {
  79. return err
  80. }
  81. if err = models.AddReleaseAttachments(rel.ID, attachmentUUIDs); err != nil {
  82. return err
  83. }
  84. if !rel.IsDraft {
  85. notification.NotifyNewRelease(rel)
  86. }
  87. return nil
  88. }
  89. // UpdateReleaseOrCreatReleaseFromTag updates information of a release or create release from tag.
  90. func UpdateReleaseOrCreatReleaseFromTag(doer *models.User, gitRepo *git.Repository, rel *models.Release, attachmentUUIDs []string, isCreate bool) (err error) {
  91. if err = createTag(gitRepo, rel); err != nil {
  92. return err
  93. }
  94. rel.LowerTagName = strings.ToLower(rel.TagName)
  95. if err = models.UpdateRelease(models.DefaultDBContext(), rel); err != nil {
  96. return err
  97. }
  98. if err = models.AddReleaseAttachments(rel.ID, attachmentUUIDs); err != nil {
  99. log.Error("AddReleaseAttachments: %v", err)
  100. }
  101. if !isCreate {
  102. notification.NotifyUpdateRelease(doer, rel)
  103. return
  104. }
  105. if !rel.IsDraft {
  106. notification.NotifyNewRelease(rel)
  107. }
  108. return err
  109. }
  110. // DeleteReleaseByID deletes a release and corresponding Git tag by given ID.
  111. func DeleteReleaseByID(id int64, doer *models.User, delTag bool) error {
  112. rel, err := models.GetReleaseByID(id)
  113. if err != nil {
  114. return fmt.Errorf("GetReleaseByID: %v", err)
  115. }
  116. repo, err := models.GetRepositoryByID(rel.RepoID)
  117. if err != nil {
  118. return fmt.Errorf("GetRepositoryByID: %v", err)
  119. }
  120. if delTag {
  121. if stdout, err := git.NewCommand("tag", "-d", rel.TagName).
  122. SetDescription(fmt.Sprintf("DeleteReleaseByID (git tag -d): %d", rel.ID)).
  123. RunInDir(repo.RepoPath()); err != nil && !strings.Contains(err.Error(), "not found") {
  124. log.Error("DeleteReleaseByID (git tag -d): %d in %v Failed:\nStdout: %s\nError: %v", rel.ID, repo, stdout, err)
  125. return fmt.Errorf("git tag -d: %v", err)
  126. }
  127. if err := models.DeleteReleaseByID(id); err != nil {
  128. return fmt.Errorf("DeleteReleaseByID: %v", err)
  129. }
  130. } else {
  131. rel.IsTag = true
  132. rel.IsDraft = false
  133. rel.IsPrerelease = false
  134. rel.Title = ""
  135. rel.Note = ""
  136. if err = models.UpdateRelease(models.DefaultDBContext(), rel); err != nil {
  137. return fmt.Errorf("Update: %v", err)
  138. }
  139. }
  140. rel.Repo = repo
  141. if err = rel.LoadAttributes(); err != nil {
  142. return fmt.Errorf("LoadAttributes: %v", err)
  143. }
  144. if err := models.DeleteAttachmentsByRelease(rel.ID); err != nil {
  145. return fmt.Errorf("DeleteAttachments: %v", err)
  146. }
  147. for i := range rel.Attachments {
  148. attachment := rel.Attachments[i]
  149. if err := storage.Attachments.Delete(attachment.RelativePath()); err != nil {
  150. log.Error("Delete attachment %s of release %s failed: %v", attachment.UUID, rel.ID, err)
  151. }
  152. }
  153. notification.NotifyDeleteRelease(doer, rel)
  154. return nil
  155. }