gitea源码

packages.go 6.7KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252
  1. // Copyright 2022 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package packages
  4. import (
  5. "fmt"
  6. "net/http"
  7. "time"
  8. packages_model "code.gitea.io/gitea/models/packages"
  9. user_model "code.gitea.io/gitea/models/user"
  10. "code.gitea.io/gitea/modules/log"
  11. "code.gitea.io/gitea/modules/optional"
  12. "code.gitea.io/gitea/modules/templates"
  13. "code.gitea.io/gitea/modules/web"
  14. "code.gitea.io/gitea/services/context"
  15. "code.gitea.io/gitea/services/forms"
  16. cargo_service "code.gitea.io/gitea/services/packages/cargo"
  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 templates.TplName) {
  46. performRuleEditPost(ctx, owner, nil, redirectURL, template)
  47. }
  48. func PerformRuleEditPost(ctx *context.Context, owner *user_model.User, redirectURL string, template templates.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 templates.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: optional.Some(false),
  129. Sort: packages_model.SortCreatedDesc,
  130. })
  131. if err != nil {
  132. ctx.ServerError("SearchVersions", err)
  133. return
  134. }
  135. if pcr.KeepCount > 0 {
  136. if pcr.KeepCount < len(pvs) {
  137. pvs = pvs[pcr.KeepCount:]
  138. } else {
  139. pvs = nil
  140. }
  141. }
  142. for _, pv := range pvs {
  143. if skip, err := container_service.ShouldBeSkipped(ctx, pcr, p, pv); err != nil {
  144. ctx.ServerError("ShouldBeSkipped", err)
  145. return
  146. } else if skip {
  147. continue
  148. }
  149. toMatch := pv.LowerVersion
  150. if pcr.MatchFullName {
  151. toMatch = p.LowerName + "/" + pv.LowerVersion
  152. }
  153. if pcr.KeepPatternMatcher != nil && pcr.KeepPatternMatcher.MatchString(toMatch) {
  154. continue
  155. }
  156. if pv.CreatedUnix.AsLocalTime().After(olderThan) {
  157. continue
  158. }
  159. if pcr.RemovePatternMatcher != nil && !pcr.RemovePatternMatcher.MatchString(toMatch) {
  160. continue
  161. }
  162. pd, err := packages_model.GetPackageDescriptor(ctx, pv)
  163. if err != nil {
  164. ctx.ServerError("GetPackageDescriptor", err)
  165. return
  166. }
  167. versionsToRemove = append(versionsToRemove, pd)
  168. }
  169. }
  170. ctx.Data["CleanupRule"] = pcr
  171. ctx.Data["VersionsToRemove"] = versionsToRemove
  172. }
  173. func getCleanupRuleByContext(ctx *context.Context, owner *user_model.User) *packages_model.PackageCleanupRule {
  174. id := ctx.FormInt64("id")
  175. if id == 0 {
  176. id = ctx.PathParamInt64("id")
  177. }
  178. pcr, err := packages_model.GetCleanupRuleByID(ctx, id)
  179. if err != nil {
  180. if err == packages_model.ErrPackageCleanupRuleNotExist {
  181. ctx.NotFound(err)
  182. } else {
  183. ctx.ServerError("GetCleanupRuleByID", err)
  184. }
  185. return nil
  186. }
  187. if pcr != nil && pcr.OwnerID == owner.ID {
  188. return pcr
  189. }
  190. ctx.NotFound(fmt.Errorf("PackageCleanupRule[%v] not associated to owner %v", id, owner))
  191. return nil
  192. }
  193. func InitializeCargoIndex(ctx *context.Context, owner *user_model.User) {
  194. err := cargo_service.InitializeIndexRepository(ctx, owner, owner)
  195. if err != nil {
  196. log.Error("InitializeIndexRepository failed: %v", err)
  197. ctx.Flash.Error(ctx.Tr("packages.owner.settings.cargo.initialize.error", err))
  198. } else {
  199. ctx.Flash.Success(ctx.Tr("packages.owner.settings.cargo.initialize.success"))
  200. }
  201. }
  202. func RebuildCargoIndex(ctx *context.Context, owner *user_model.User) {
  203. err := cargo_service.RebuildIndex(ctx, owner, owner)
  204. if err != nil {
  205. log.Error("RebuildIndex failed: %v", err)
  206. ctx.Flash.Error(ctx.Tr("packages.owner.settings.cargo.rebuild.error", err))
  207. } else {
  208. ctx.Flash.Success(ctx.Tr("packages.owner.settings.cargo.rebuild.success"))
  209. }
  210. }