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.

packages.go 5.9KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226
  1. // Copyright 2022 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 packages
  5. import (
  6. "fmt"
  7. "net/http"
  8. "time"
  9. "code.gitea.io/gitea/models/db"
  10. packages_model "code.gitea.io/gitea/models/packages"
  11. user_model "code.gitea.io/gitea/models/user"
  12. "code.gitea.io/gitea/modules/base"
  13. "code.gitea.io/gitea/modules/context"
  14. "code.gitea.io/gitea/modules/util"
  15. "code.gitea.io/gitea/modules/web"
  16. "code.gitea.io/gitea/services/forms"
  17. container_service "code.gitea.io/gitea/services/packages/container"
  18. )
  19. func SetPackagesContext(ctx *context.Context, owner *user_model.User) {
  20. pcrs, err := packages_model.GetCleanupRulesByOwner(ctx, owner.ID)
  21. if err != nil {
  22. ctx.ServerError("GetCleanupRulesByOwner", err)
  23. return
  24. }
  25. ctx.Data["CleanupRules"] = pcrs
  26. }
  27. func SetRuleAddContext(ctx *context.Context) {
  28. setRuleEditContext(ctx, nil)
  29. }
  30. func SetRuleEditContext(ctx *context.Context, owner *user_model.User) {
  31. pcr := getCleanupRuleByContext(ctx, owner)
  32. if pcr == nil {
  33. return
  34. }
  35. setRuleEditContext(ctx, pcr)
  36. }
  37. func setRuleEditContext(ctx *context.Context, pcr *packages_model.PackageCleanupRule) {
  38. ctx.Data["IsEditRule"] = pcr != nil
  39. if pcr == nil {
  40. pcr = &packages_model.PackageCleanupRule{}
  41. }
  42. ctx.Data["CleanupRule"] = pcr
  43. ctx.Data["AvailableTypes"] = packages_model.TypeList
  44. }
  45. func PerformRuleAddPost(ctx *context.Context, owner *user_model.User, redirectURL string, template base.TplName) {
  46. performRuleEditPost(ctx, owner, nil, redirectURL, template)
  47. }
  48. func PerformRuleEditPost(ctx *context.Context, owner *user_model.User, redirectURL string, template base.TplName) {
  49. pcr := getCleanupRuleByContext(ctx, owner)
  50. if pcr == nil {
  51. return
  52. }
  53. form := web.GetForm(ctx).(*forms.PackageCleanupRuleForm)
  54. if form.Action == "remove" {
  55. if err := packages_model.DeleteCleanupRuleByID(ctx, pcr.ID); err != nil {
  56. ctx.ServerError("DeleteCleanupRuleByID", err)
  57. return
  58. }
  59. ctx.Flash.Success(ctx.Tr("packages.owner.settings.cleanuprules.success.delete"))
  60. ctx.Redirect(redirectURL)
  61. } else {
  62. performRuleEditPost(ctx, owner, pcr, redirectURL, template)
  63. }
  64. }
  65. func performRuleEditPost(ctx *context.Context, owner *user_model.User, pcr *packages_model.PackageCleanupRule, redirectURL string, template base.TplName) {
  66. isEditRule := pcr != nil
  67. if pcr == nil {
  68. pcr = &packages_model.PackageCleanupRule{}
  69. }
  70. form := web.GetForm(ctx).(*forms.PackageCleanupRuleForm)
  71. pcr.Enabled = form.Enabled
  72. pcr.OwnerID = owner.ID
  73. pcr.KeepCount = form.KeepCount
  74. pcr.KeepPattern = form.KeepPattern
  75. pcr.RemoveDays = form.RemoveDays
  76. pcr.RemovePattern = form.RemovePattern
  77. pcr.MatchFullName = form.MatchFullName
  78. ctx.Data["IsEditRule"] = isEditRule
  79. ctx.Data["CleanupRule"] = pcr
  80. ctx.Data["AvailableTypes"] = packages_model.TypeList
  81. if ctx.HasError() {
  82. ctx.HTML(http.StatusOK, template)
  83. return
  84. }
  85. if isEditRule {
  86. if err := packages_model.UpdateCleanupRule(ctx, pcr); err != nil {
  87. ctx.ServerError("UpdateCleanupRule", err)
  88. return
  89. }
  90. } else {
  91. pcr.Type = packages_model.Type(form.Type)
  92. if has, err := packages_model.HasOwnerCleanupRuleForPackageType(ctx, owner.ID, pcr.Type); err != nil {
  93. ctx.ServerError("HasOwnerCleanupRuleForPackageType", err)
  94. return
  95. } else if has {
  96. ctx.Data["Err_Type"] = true
  97. ctx.HTML(http.StatusOK, template)
  98. return
  99. }
  100. var err error
  101. if pcr, err = packages_model.InsertCleanupRule(ctx, pcr); err != nil {
  102. ctx.ServerError("InsertCleanupRule", err)
  103. return
  104. }
  105. }
  106. ctx.Flash.Success(ctx.Tr("packages.owner.settings.cleanuprules.success.update"))
  107. ctx.Redirect(fmt.Sprintf("%s/rules/%d", redirectURL, pcr.ID))
  108. }
  109. func SetRulePreviewContext(ctx *context.Context, owner *user_model.User) {
  110. pcr := getCleanupRuleByContext(ctx, owner)
  111. if pcr == nil {
  112. return
  113. }
  114. if err := pcr.CompiledPattern(); err != nil {
  115. ctx.ServerError("CompiledPattern", err)
  116. return
  117. }
  118. olderThan := time.Now().AddDate(0, 0, -pcr.RemoveDays)
  119. packages, err := packages_model.GetPackagesByType(ctx, pcr.OwnerID, pcr.Type)
  120. if err != nil {
  121. ctx.ServerError("GetPackagesByType", err)
  122. return
  123. }
  124. versionsToRemove := make([]*packages_model.PackageDescriptor, 0, 10)
  125. for _, p := range packages {
  126. pvs, _, err := packages_model.SearchVersions(ctx, &packages_model.PackageSearchOptions{
  127. PackageID: p.ID,
  128. IsInternal: util.OptionalBoolFalse,
  129. Sort: packages_model.SortCreatedDesc,
  130. Paginator: db.NewAbsoluteListOptions(pcr.KeepCount, 200),
  131. })
  132. if err != nil {
  133. ctx.ServerError("SearchVersions", err)
  134. return
  135. }
  136. for _, pv := range pvs {
  137. if skip, err := container_service.ShouldBeSkipped(ctx, pcr, p, pv); err != nil {
  138. ctx.ServerError("ShouldBeSkipped", err)
  139. return
  140. } else if skip {
  141. continue
  142. }
  143. toMatch := pv.LowerVersion
  144. if pcr.MatchFullName {
  145. toMatch = p.LowerName + "/" + pv.LowerVersion
  146. }
  147. if pcr.KeepPatternMatcher != nil && pcr.KeepPatternMatcher.MatchString(toMatch) {
  148. continue
  149. }
  150. if pv.CreatedUnix.AsLocalTime().After(olderThan) {
  151. continue
  152. }
  153. if pcr.RemovePatternMatcher != nil && !pcr.RemovePatternMatcher.MatchString(toMatch) {
  154. continue
  155. }
  156. pd, err := packages_model.GetPackageDescriptor(ctx, pv)
  157. if err != nil {
  158. ctx.ServerError("GetPackageDescriptor", err)
  159. return
  160. }
  161. versionsToRemove = append(versionsToRemove, pd)
  162. }
  163. }
  164. ctx.Data["CleanupRule"] = pcr
  165. ctx.Data["VersionsToRemove"] = versionsToRemove
  166. }
  167. func getCleanupRuleByContext(ctx *context.Context, owner *user_model.User) *packages_model.PackageCleanupRule {
  168. id := ctx.FormInt64("id")
  169. if id == 0 {
  170. id = ctx.ParamsInt64("id")
  171. }
  172. pcr, err := packages_model.GetCleanupRuleByID(ctx, id)
  173. if err != nil {
  174. if err == packages_model.ErrPackageCleanupRuleNotExist {
  175. ctx.NotFound("", err)
  176. } else {
  177. ctx.ServerError("GetCleanupRuleByID", err)
  178. }
  179. return nil
  180. }
  181. if pcr != nil && pcr.OwnerID == owner.ID {
  182. return pcr
  183. }
  184. ctx.NotFound("", fmt.Errorf("PackageCleanupRule[%v] not associated to owner %v", id, owner))
  185. return nil
  186. }