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.

api_packages_test.go 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374
  1. // Copyright 2021 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package integration
  4. import (
  5. "bytes"
  6. "fmt"
  7. "net/http"
  8. "testing"
  9. "time"
  10. auth_model "code.gitea.io/gitea/models/auth"
  11. "code.gitea.io/gitea/models/db"
  12. packages_model "code.gitea.io/gitea/models/packages"
  13. container_model "code.gitea.io/gitea/models/packages/container"
  14. "code.gitea.io/gitea/models/unittest"
  15. user_model "code.gitea.io/gitea/models/user"
  16. "code.gitea.io/gitea/modules/setting"
  17. api "code.gitea.io/gitea/modules/structs"
  18. packages_service "code.gitea.io/gitea/services/packages"
  19. "code.gitea.io/gitea/tests"
  20. "github.com/stretchr/testify/assert"
  21. )
  22. func TestPackageAPI(t *testing.T) {
  23. defer tests.PrepareTestEnv(t)()
  24. user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 4})
  25. session := loginUser(t, user.Name)
  26. tokenReadPackage := getTokenForLoggedInUser(t, session, auth_model.AccessTokenScopeReadPackage)
  27. tokenDeletePackage := getTokenForLoggedInUser(t, session, auth_model.AccessTokenScopeDeletePackage)
  28. packageName := "test-package"
  29. packageVersion := "1.0.3"
  30. filename := "file.bin"
  31. url := fmt.Sprintf("/api/packages/%s/generic/%s/%s/%s", user.Name, packageName, packageVersion, filename)
  32. req := NewRequestWithBody(t, "PUT", url, bytes.NewReader([]byte{}))
  33. AddBasicAuthHeader(req, user.Name)
  34. MakeRequest(t, req, http.StatusCreated)
  35. t.Run("ListPackages", func(t *testing.T) {
  36. defer tests.PrintCurrentTest(t)()
  37. req := NewRequest(t, "GET", fmt.Sprintf("/api/v1/packages/%s?token=%s", user.Name, tokenReadPackage))
  38. resp := MakeRequest(t, req, http.StatusOK)
  39. var apiPackages []*api.Package
  40. DecodeJSON(t, resp, &apiPackages)
  41. assert.Len(t, apiPackages, 1)
  42. assert.Equal(t, string(packages_model.TypeGeneric), apiPackages[0].Type)
  43. assert.Equal(t, packageName, apiPackages[0].Name)
  44. assert.Equal(t, packageVersion, apiPackages[0].Version)
  45. assert.NotNil(t, apiPackages[0].Creator)
  46. assert.Equal(t, user.Name, apiPackages[0].Creator.UserName)
  47. })
  48. t.Run("GetPackage", func(t *testing.T) {
  49. defer tests.PrintCurrentTest(t)()
  50. req := NewRequest(t, "GET", fmt.Sprintf("/api/v1/packages/%s/dummy/%s/%s?token=%s", user.Name, packageName, packageVersion, tokenReadPackage))
  51. MakeRequest(t, req, http.StatusNotFound)
  52. req = NewRequest(t, "GET", fmt.Sprintf("/api/v1/packages/%s/generic/%s/%s?token=%s", user.Name, packageName, packageVersion, tokenReadPackage))
  53. resp := MakeRequest(t, req, http.StatusOK)
  54. var p *api.Package
  55. DecodeJSON(t, resp, &p)
  56. assert.Equal(t, string(packages_model.TypeGeneric), p.Type)
  57. assert.Equal(t, packageName, p.Name)
  58. assert.Equal(t, packageVersion, p.Version)
  59. assert.NotNil(t, p.Creator)
  60. assert.Equal(t, user.Name, p.Creator.UserName)
  61. t.Run("RepositoryLink", func(t *testing.T) {
  62. defer tests.PrintCurrentTest(t)()
  63. p, err := packages_model.GetPackageByName(db.DefaultContext, user.ID, packages_model.TypeGeneric, packageName)
  64. assert.NoError(t, err)
  65. // no repository link
  66. req := NewRequest(t, "GET", fmt.Sprintf("/api/v1/packages/%s/generic/%s/%s?token=%s", user.Name, packageName, packageVersion, tokenReadPackage))
  67. resp := MakeRequest(t, req, http.StatusOK)
  68. var ap1 *api.Package
  69. DecodeJSON(t, resp, &ap1)
  70. assert.Nil(t, ap1.Repository)
  71. // link to public repository
  72. assert.NoError(t, packages_model.SetRepositoryLink(db.DefaultContext, p.ID, 1))
  73. req = NewRequest(t, "GET", fmt.Sprintf("/api/v1/packages/%s/generic/%s/%s?token=%s", user.Name, packageName, packageVersion, tokenReadPackage))
  74. resp = MakeRequest(t, req, http.StatusOK)
  75. var ap2 *api.Package
  76. DecodeJSON(t, resp, &ap2)
  77. assert.NotNil(t, ap2.Repository)
  78. assert.EqualValues(t, 1, ap2.Repository.ID)
  79. // link to private repository
  80. assert.NoError(t, packages_model.SetRepositoryLink(db.DefaultContext, p.ID, 2))
  81. req = NewRequest(t, "GET", fmt.Sprintf("/api/v1/packages/%s/generic/%s/%s?token=%s", user.Name, packageName, packageVersion, tokenReadPackage))
  82. resp = MakeRequest(t, req, http.StatusOK)
  83. var ap3 *api.Package
  84. DecodeJSON(t, resp, &ap3)
  85. assert.Nil(t, ap3.Repository)
  86. assert.NoError(t, packages_model.UnlinkRepositoryFromAllPackages(db.DefaultContext, 2))
  87. })
  88. })
  89. t.Run("ListPackageFiles", func(t *testing.T) {
  90. defer tests.PrintCurrentTest(t)()
  91. req := NewRequest(t, "GET", fmt.Sprintf("/api/v1/packages/%s/dummy/%s/%s/files?token=%s", user.Name, packageName, packageVersion, tokenReadPackage))
  92. MakeRequest(t, req, http.StatusNotFound)
  93. req = NewRequest(t, "GET", fmt.Sprintf("/api/v1/packages/%s/generic/%s/%s/files?token=%s", user.Name, packageName, packageVersion, tokenReadPackage))
  94. resp := MakeRequest(t, req, http.StatusOK)
  95. var files []*api.PackageFile
  96. DecodeJSON(t, resp, &files)
  97. assert.Len(t, files, 1)
  98. assert.Equal(t, int64(0), files[0].Size)
  99. assert.Equal(t, filename, files[0].Name)
  100. assert.Equal(t, "d41d8cd98f00b204e9800998ecf8427e", files[0].HashMD5)
  101. assert.Equal(t, "da39a3ee5e6b4b0d3255bfef95601890afd80709", files[0].HashSHA1)
  102. assert.Equal(t, "e3b0c44298fc1c149afbf4c8996fb92427ae41e4649b934ca495991b7852b855", files[0].HashSHA256)
  103. assert.Equal(t, "cf83e1357eefb8bdf1542850d66d8007d620e4050b5715dc83f4a921d36ce9ce47d0d13c5d85f2b0ff8318d2877eec2f63b931bd47417a81a538327af927da3e", files[0].HashSHA512)
  104. })
  105. t.Run("DeletePackage", func(t *testing.T) {
  106. defer tests.PrintCurrentTest(t)()
  107. req := NewRequest(t, "DELETE", fmt.Sprintf("/api/v1/packages/%s/dummy/%s/%s?token=%s", user.Name, packageName, packageVersion, tokenDeletePackage))
  108. MakeRequest(t, req, http.StatusNotFound)
  109. req = NewRequest(t, "DELETE", fmt.Sprintf("/api/v1/packages/%s/generic/%s/%s?token=%s", user.Name, packageName, packageVersion, tokenDeletePackage))
  110. MakeRequest(t, req, http.StatusNoContent)
  111. })
  112. }
  113. func TestPackageAccess(t *testing.T) {
  114. defer tests.PrepareTestEnv(t)()
  115. admin := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 1})
  116. user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 5})
  117. inactive := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 9})
  118. uploadPackage := func(doer, owner *user_model.User, expectedStatus int) {
  119. url := fmt.Sprintf("/api/packages/%s/generic/test-package/1.0/file.bin", owner.Name)
  120. req := NewRequestWithBody(t, "PUT", url, bytes.NewReader([]byte{1}))
  121. AddBasicAuthHeader(req, doer.Name)
  122. MakeRequest(t, req, expectedStatus)
  123. }
  124. uploadPackage(user, inactive, http.StatusUnauthorized)
  125. uploadPackage(inactive, inactive, http.StatusUnauthorized)
  126. uploadPackage(inactive, user, http.StatusUnauthorized)
  127. uploadPackage(admin, inactive, http.StatusCreated)
  128. uploadPackage(admin, user, http.StatusCreated)
  129. }
  130. func TestPackageQuota(t *testing.T) {
  131. defer tests.PrepareTestEnv(t)()
  132. limitTotalOwnerCount, limitTotalOwnerSize, limitSizeGeneric := setting.Packages.LimitTotalOwnerCount, setting.Packages.LimitTotalOwnerSize, setting.Packages.LimitSizeGeneric
  133. admin := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 1})
  134. user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 10})
  135. uploadPackage := func(doer *user_model.User, version string, expectedStatus int) {
  136. url := fmt.Sprintf("/api/packages/%s/generic/test-package/%s/file.bin", user.Name, version)
  137. req := NewRequestWithBody(t, "PUT", url, bytes.NewReader([]byte{1}))
  138. AddBasicAuthHeader(req, doer.Name)
  139. MakeRequest(t, req, expectedStatus)
  140. }
  141. // Exceeded quota result in StatusForbidden for normal users but admins are always allowed to upload.
  142. setting.Packages.LimitTotalOwnerCount = 0
  143. uploadPackage(user, "1.0", http.StatusForbidden)
  144. uploadPackage(admin, "1.0", http.StatusCreated)
  145. setting.Packages.LimitTotalOwnerCount = limitTotalOwnerCount
  146. setting.Packages.LimitTotalOwnerSize = 0
  147. uploadPackage(user, "1.1", http.StatusForbidden)
  148. uploadPackage(admin, "1.1", http.StatusCreated)
  149. setting.Packages.LimitTotalOwnerSize = limitTotalOwnerSize
  150. setting.Packages.LimitSizeGeneric = 0
  151. uploadPackage(user, "1.2", http.StatusForbidden)
  152. uploadPackage(admin, "1.2", http.StatusCreated)
  153. setting.Packages.LimitSizeGeneric = limitSizeGeneric
  154. }
  155. func TestPackageCleanup(t *testing.T) {
  156. defer tests.PrepareTestEnv(t)()
  157. duration, _ := time.ParseDuration("-1h")
  158. t.Run("Common", func(t *testing.T) {
  159. defer tests.PrintCurrentTest(t)()
  160. pbs, err := packages_model.FindExpiredUnreferencedBlobs(db.DefaultContext, duration)
  161. assert.NoError(t, err)
  162. assert.NotEmpty(t, pbs)
  163. _, err = packages_model.GetInternalVersionByNameAndVersion(db.DefaultContext, 2, packages_model.TypeContainer, "test", container_model.UploadVersion)
  164. assert.NoError(t, err)
  165. err = packages_service.Cleanup(db.DefaultContext, duration)
  166. assert.NoError(t, err)
  167. pbs, err = packages_model.FindExpiredUnreferencedBlobs(db.DefaultContext, duration)
  168. assert.NoError(t, err)
  169. assert.Empty(t, pbs)
  170. _, err = packages_model.GetInternalVersionByNameAndVersion(db.DefaultContext, 2, packages_model.TypeContainer, "test", container_model.UploadVersion)
  171. assert.ErrorIs(t, err, packages_model.ErrPackageNotExist)
  172. })
  173. t.Run("CleanupRules", func(t *testing.T) {
  174. defer tests.PrintCurrentTest(t)()
  175. user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
  176. type version struct {
  177. Version string
  178. ShouldExist bool
  179. Created int64
  180. }
  181. cases := []struct {
  182. Name string
  183. Versions []version
  184. Rule *packages_model.PackageCleanupRule
  185. }{
  186. {
  187. Name: "Disabled",
  188. Versions: []version{
  189. {Version: "keep", ShouldExist: true},
  190. },
  191. Rule: &packages_model.PackageCleanupRule{
  192. Enabled: false,
  193. },
  194. },
  195. {
  196. Name: "KeepCount",
  197. Versions: []version{
  198. {Version: "keep", ShouldExist: true},
  199. {Version: "v1.0", ShouldExist: true},
  200. {Version: "test-3", ShouldExist: false, Created: 1},
  201. {Version: "test-4", ShouldExist: false, Created: 1},
  202. },
  203. Rule: &packages_model.PackageCleanupRule{
  204. Enabled: true,
  205. KeepCount: 2,
  206. },
  207. },
  208. {
  209. Name: "KeepPattern",
  210. Versions: []version{
  211. {Version: "keep", ShouldExist: true},
  212. {Version: "v1.0", ShouldExist: false},
  213. },
  214. Rule: &packages_model.PackageCleanupRule{
  215. Enabled: true,
  216. KeepPattern: "k.+p",
  217. },
  218. },
  219. {
  220. Name: "RemoveDays",
  221. Versions: []version{
  222. {Version: "keep", ShouldExist: true},
  223. {Version: "v1.0", ShouldExist: false, Created: 1},
  224. },
  225. Rule: &packages_model.PackageCleanupRule{
  226. Enabled: true,
  227. RemoveDays: 60,
  228. },
  229. },
  230. {
  231. Name: "RemovePattern",
  232. Versions: []version{
  233. {Version: "test", ShouldExist: true},
  234. {Version: "test-3", ShouldExist: false},
  235. {Version: "test-4", ShouldExist: false},
  236. },
  237. Rule: &packages_model.PackageCleanupRule{
  238. Enabled: true,
  239. RemovePattern: `t[e]+st-\d+`,
  240. },
  241. },
  242. {
  243. Name: "MatchFullName",
  244. Versions: []version{
  245. {Version: "keep", ShouldExist: true},
  246. {Version: "test", ShouldExist: false},
  247. },
  248. Rule: &packages_model.PackageCleanupRule{
  249. Enabled: true,
  250. RemovePattern: `package/test|different/keep`,
  251. MatchFullName: true,
  252. },
  253. },
  254. {
  255. Name: "Mixed",
  256. Versions: []version{
  257. {Version: "keep", ShouldExist: true, Created: time.Now().Add(time.Duration(10000)).Unix()},
  258. {Version: "dummy", ShouldExist: true, Created: 1},
  259. {Version: "test-3", ShouldExist: true},
  260. {Version: "test-4", ShouldExist: false, Created: 1},
  261. },
  262. Rule: &packages_model.PackageCleanupRule{
  263. Enabled: true,
  264. KeepCount: 1,
  265. KeepPattern: `dummy`,
  266. RemoveDays: 7,
  267. RemovePattern: `t[e]+st-\d+`,
  268. },
  269. },
  270. }
  271. for _, c := range cases {
  272. t.Run(c.Name, func(t *testing.T) {
  273. defer tests.PrintCurrentTest(t)()
  274. for _, v := range c.Versions {
  275. url := fmt.Sprintf("/api/packages/%s/generic/package/%s/file.bin", user.Name, v.Version)
  276. req := NewRequestWithBody(t, "PUT", url, bytes.NewReader([]byte{1}))
  277. AddBasicAuthHeader(req, user.Name)
  278. MakeRequest(t, req, http.StatusCreated)
  279. if v.Created != 0 {
  280. pv, err := packages_model.GetVersionByNameAndVersion(db.DefaultContext, user.ID, packages_model.TypeGeneric, "package", v.Version)
  281. assert.NoError(t, err)
  282. _, err = db.GetEngine(db.DefaultContext).Exec("UPDATE package_version SET created_unix = ? WHERE id = ?", v.Created, pv.ID)
  283. assert.NoError(t, err)
  284. }
  285. }
  286. c.Rule.OwnerID = user.ID
  287. c.Rule.Type = packages_model.TypeGeneric
  288. pcr, err := packages_model.InsertCleanupRule(db.DefaultContext, c.Rule)
  289. assert.NoError(t, err)
  290. err = packages_service.Cleanup(db.DefaultContext, duration)
  291. assert.NoError(t, err)
  292. for _, v := range c.Versions {
  293. pv, err := packages_model.GetVersionByNameAndVersion(db.DefaultContext, user.ID, packages_model.TypeGeneric, "package", v.Version)
  294. if v.ShouldExist {
  295. assert.NoError(t, err)
  296. err = packages_service.DeletePackageVersionAndReferences(db.DefaultContext, pv)
  297. assert.NoError(t, err)
  298. } else {
  299. assert.ErrorIs(t, err, packages_model.ErrPackageNotExist)
  300. }
  301. }
  302. assert.NoError(t, packages_model.DeleteCleanupRuleByID(db.DefaultContext, pcr.ID))
  303. })
  304. }
  305. })
  306. }