gitea源码

api_packages_alpine_test.go 11KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294
  1. // Copyright 2023 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package integration
  4. import (
  5. "archive/tar"
  6. "bufio"
  7. "bytes"
  8. "compress/gzip"
  9. "encoding/base64"
  10. "fmt"
  11. "io"
  12. "net/http"
  13. "testing"
  14. "code.gitea.io/gitea/models/packages"
  15. "code.gitea.io/gitea/models/unittest"
  16. user_model "code.gitea.io/gitea/models/user"
  17. alpine_module "code.gitea.io/gitea/modules/packages/alpine"
  18. alpine_service "code.gitea.io/gitea/services/packages/alpine"
  19. "code.gitea.io/gitea/tests"
  20. "github.com/stretchr/testify/assert"
  21. )
  22. func TestPackageAlpine(t *testing.T) {
  23. defer tests.PrepareTestEnv(t)()
  24. user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
  25. packageName := "gitea-test"
  26. packageVersion := "1.4.1-r3"
  27. base64AlpinePackageContent := `H4sIAAAAAAACA9ML9nT30wsKdtTLzjNJzjYuckjPLElN1DUzMUxMNTa11CsqTtQrKE1ioAAYAIGZ
  28. iQmYBgJ02hDENjQxMTAzMzQ1MTVjMDA0MTQ1ZlAwYKADKC0uSSxSUGAYoWDm4sZZtypv75+q2fVT
  29. POD1bKkFB22ms+g1z+H4dk7AhC3HwUSj9EbT0Rk3Dn55dHxy/K7Q+Nl/i+L7Z036ypcRvvpZuMiN
  30. s7wbZL/klqRGGshv9Gi0qHTgTZfw3HytnJdx9c3NTRp/PHn+Z50uq2pjkilzjtpfd+uzQMw1M7cY
  31. i9RXJasnT2M+vDXCesLK7MilJt8sGplj4xUlLMUun9SzY+phFpxWxRXa06AseV9WvzH3jtGGoL5A
  32. vQkea+VKPj5R+Cb461tIk97qpa9nJYsJujTNl2B/J1P52H/D2rPr/j19uU8p7cMSq5tmXk51ReXl
  33. F/Yddr9XsMpEwFKlXSPo3QSGwnCOG8y2uadjm6ui998WYXNYubjg78N3a7bnXjhrl5fB8voI++LI
  34. 1FP5W44e2xf4Ou2wrtyic1Onz7MzMV5ksuno2V/LVG4eN/15X/n2/2vJ2VV+T68aT327dOrhd6e6
  35. q5Y0V82Y83tdqkFa8TW2BvGCZ0ds/iibHVpzKuPcuSULO63/bNmfrnhjWqXzhMSXTb5Cv4vPaxSL
  36. 8LFMdqmxbN7+Y+Yi0ZyZhz4UxexLuHHFd1VFvk+kwvniq3P+f9rh52InWnL8Lpvedcecoh1GFSc5
  37. xZ9VBGex2V269HZfwxSVCvP35wQfi2xKX+lYMXtF48n1R65O2PLWpm69RdESMa79dlrTGazsZacu
  38. MbMLeSSScPORZde76/MBV6SFJAAEAAAfiwgAAAAAAAID7VRLaxsxEN6zfoUgZ++OVq+1aUIhUDeY
  39. pKa49FhmJdkW3ofRysXpr69220t9SCk0gZJ+IGaY56eBmbxY4/m9Q+vCUOTr1fLu4d2H7O8CEpQQ
  40. k0y4lAClypgQoBSTQqoMGBMgMnrOXgCnIWJIVLLXCcaoib5110CSij/V7D9eCZ5p5f9o/5VkF/tf
  41. MqUzCi+5/6Hv41Nxv/Nffu4fwRVdus4FjM7S+pFiffKNpTxnkMMsALmin5PnHgMtS8rkgvGFBPpp
  42. c0tLKDk5HnYdto5e052PDmfRDXE0fnUh2VgucjYLU5h1g0mm5RhGNymMrtEccOfIKTTJsY/xOCyK
  43. YqqT+74gExWbmI2VlJ6LeQUcyPFH2lh/9SBuV/wjfXPohDnw8HZKviGD/zYmCZgrgsHsk36u1Bcl
  44. SB/8zne/0jV92/qYbKRF38X0niiemN2QxhvXDWOL+7tNGhGeYt+m22mwaR6pddGZNM8FSeRxj8PY
  45. X7PaqdqAVlqWXHKnmQGmK43VlqNlILRilbBSMI2jV5Vbu5XGSVsDyGc7yd8B/gK2qgAIAAAfiwgA
  46. AAAAAAID7dNNSgMxGAbg7MSCOxcu5wJOv0x+OlkU7K5QoYXqVsxMMihlKMwP1Fu48QQewCN4DfEQ
  47. egUz4sYuFKEtFN9n870hWSSQN+7P7GrsrfNV3Y9dW5Z3bNMo0FJ+zmB9EhcJ41KS1lxJpRnxbsWi
  48. FduBtm5sFa7C/ifOo7y5Lf2QeiHar6jTaDSbnF5Mp+fzOL/x+aJuy3g+HvGhs8JY4b3yOpMZOZEo
  49. lRW+MEoTTw3ZwqU0INNjsAe2VPk/9b/L3/s/kIKzqOtk+IbJGTtmr+bx7WoxOUoun98frk/un14O
  50. Djfa/2q5bH4699v++uMAAAAAAAAAAAAAAAAAAAAAAHbgA/eXQh8AKAAA`
  51. content, err := base64.StdEncoding.DecodeString(base64AlpinePackageContent)
  52. assert.NoError(t, err)
  53. base64AlpinePackageNoArchContent := `H4sIAAAAAAACA9ML9nT30wsKdtQrLU4t0jUzTUo1NDVP0ysqTtQrKE1ioAYwAAIzExMwDQTotCGI
  54. bWhiampuYmRiaGrMYGBoZGZkxKBgwEAHUFpcklikoMAwQkHLB7eoE40P9n5jvx32t7Dy9rq7x19k
  55. 66cJPV38t/h+vWe2jdXy+/PzPT0YTF5z39i4cPFptcLa1C1lD0z/XvrNp6In/7nP4PPCF2pZu8uV
  56. z74QXLxpY1XWJuVFysqVf+PdizccFbD6ZL/QPGXd1Ri1fec2XBNuYfK/rFa6wF/h3dK/W12f8mxP
  57. 04iP3aCy+vPx7h9S+5M1LLkWr5M/4ezGt3bDW/FjBp/S9hiKP72s/XrJ0vWtO0zr5wa+D/X8XluW
  58. d7BLP7XS3YUhd8WbPPF/NW3691ONJbXsRb69O7BIMZC96uTri+utC/fbie5J+n7zhCxD4Aep/qet
  59. QnlCZyN8MhNdVNlNl7965R1nExrrGvfI/YQZFx8Dg+d9122hZsYd/24WL/L69OWrDAN/y//nS7im
  60. XEive3v7QeTe433TPj/X71+9yHiV6+E9k++3TL8V0Xoq9panhNt23fLgau/pTOvmKx6bV/pS26+Y
  61. 5UP4viyuklYeu4/BZl6rLINe1L/uWuUXcH5z7pa2b9+/rp/v/8dFgc1PL3bO3/iVcrI//J/LMU2X
  62. Nzu1IaMmWXnGp7CmyQIR39d0Nai9/+tdPbfjvmsNH88Tu7uVrvNuJE0wjxfePXGv/KHNXD+mnG0t
  63. yTPu+Na0b5WR9O4t0yMd9T5k6ui7hOyU/jL/4dOn6neLwhdrZIZfcl1ectnGvUTurWDo1vY5Gw9k
  64. PTQLVgcA61F+7gAEAAAfiwgAAAAAAAID7VVNa9wwEPXZv2Ig53hHlizbCzkVkobQJtDkB4wl2SvW
  65. lhdbTpP++oyXQGEPLYU2paTvIs3X05PQSNnmjp4+OrJumjfZ3c3V9efL2+T3AhlaqePIOB0Rc50I
  66. VRSlypUoZIJCKJQJPCVvgGWONLGU5H1CCDDRD+4CU57S6zT5j3eCP9Tyv9T/GsuT/scyLxPAt+z/
  67. aRzjj/J+Fv9HcQZXLriJorPQPAM1i+8tyEzkGZ5PmJ7BMvvQQUt7tx4BPPJH4ccAIpN5Jjj+hSJc
  68. ugZAghDbArco4eH+A+SYq/Sw7wINDi6g89HReRhpMrvVzTzsFZlaV2Hbutmw4zVhmXo2djEe5u1m
  69. c6zNzDikR3mW1a61JepaC0SZHsjsqTsyPoR9GL+GdPbf1iSFtU5Xyu/c4+Q7H04lMfvgI3vT3hsX
  70. 5rX40/U9b5CWOA78Mhrq+2ewLjrDp7VNWQbtaF6ZXVWZIhdV09RWOIvU6BqNboSxLSEpkrpQq80x
  71. W1Nla6NavuqtrJQ0sv17D+4L2oD1lwAIAAAfiwgAAAAAAAID7dM/SgNBFAbw6cSAnYXlXsDNm50/
  72. u1METBeIkEBMK87uzKKEJbB/IN7CxhN4AI/gNcRD6BWciI0WSiBGxO/XvA9mile8L+5P7WrkrfN1
  73. 049dV1XXbNso0FK+zeDzJC4SxqVSqUwkV4IR51KkLFqxHeia1tZhFfY/cR4V7VXlB9QL0b5HnUXD
  74. 6fj4bDI5ncXFpS8WTVfFs9GQD5wVxgrvlde5zMmJRKm89KVRmnhmyJYuo5RMj8Ef8EOV36j/6/yx
  75. /5qnxKJ1J8MZJifskD2Zu+fzxfggmT+83F4c3dw/7u1vtf/1ctl+9e+7dwAAAAAAAAAAAAAAAAAA
  76. AACAX/AKARNTyAAoAAA=`
  77. noarchContent, err := base64.StdEncoding.DecodeString(base64AlpinePackageNoArchContent)
  78. assert.NoError(t, err)
  79. branches := []string{"v3.16", "v3.17"}
  80. repositories := []string{"main", "testing"}
  81. rootURL := fmt.Sprintf("/api/packages/%s/alpine", user.Name)
  82. t.Run("RepositoryKey", func(t *testing.T) {
  83. defer tests.PrintCurrentTest(t)()
  84. req := NewRequest(t, "GET", rootURL+"/key")
  85. resp := MakeRequest(t, req, http.StatusOK)
  86. assert.Equal(t, "application/x-pem-file", resp.Header().Get("Content-Type"))
  87. assert.Contains(t, resp.Body.String(), "-----BEGIN PUBLIC KEY-----")
  88. })
  89. for _, branch := range branches {
  90. for _, repository := range repositories {
  91. t.Run(fmt.Sprintf("[Branch:%s,Repository:%s]", branch, repository), func(t *testing.T) {
  92. t.Run("Upload", func(t *testing.T) {
  93. defer tests.PrintCurrentTest(t)()
  94. uploadURL := fmt.Sprintf("%s/%s/%s", rootURL, branch, repository)
  95. req := NewRequestWithBody(t, "PUT", uploadURL, bytes.NewReader([]byte{}))
  96. MakeRequest(t, req, http.StatusUnauthorized)
  97. req = NewRequestWithBody(t, "PUT", uploadURL, bytes.NewReader([]byte{})).
  98. AddBasicAuth(user.Name)
  99. MakeRequest(t, req, http.StatusBadRequest)
  100. req = NewRequestWithBody(t, "PUT", uploadURL, bytes.NewReader(content)).
  101. AddBasicAuth(user.Name)
  102. MakeRequest(t, req, http.StatusCreated)
  103. pvs, err := packages.GetVersionsByPackageType(t.Context(), user.ID, packages.TypeAlpine)
  104. assert.NoError(t, err)
  105. assert.Len(t, pvs, 1)
  106. pd, err := packages.GetPackageDescriptor(t.Context(), pvs[0])
  107. assert.NoError(t, err)
  108. assert.Nil(t, pd.SemVer)
  109. assert.IsType(t, &alpine_module.VersionMetadata{}, pd.Metadata)
  110. assert.Equal(t, packageName, pd.Package.Name)
  111. assert.Equal(t, packageVersion, pd.Version.Version)
  112. pfs, err := packages.GetFilesByVersionID(t.Context(), pvs[0].ID)
  113. assert.NoError(t, err)
  114. assert.NotEmpty(t, pfs)
  115. assert.Condition(t, func() bool {
  116. seen := false
  117. expectedFilename := fmt.Sprintf("%s-%s.apk", packageName, packageVersion)
  118. expectedCompositeKey := fmt.Sprintf("%s|%s|x86_64", branch, repository)
  119. for _, pf := range pfs {
  120. if pf.Name == expectedFilename && pf.CompositeKey == expectedCompositeKey {
  121. if seen {
  122. return false
  123. }
  124. seen = true
  125. assert.True(t, pf.IsLead)
  126. pfps, err := packages.GetProperties(t.Context(), packages.PropertyTypeFile, pf.ID)
  127. assert.NoError(t, err)
  128. for _, pfp := range pfps {
  129. switch pfp.Name {
  130. case alpine_module.PropertyBranch:
  131. assert.Equal(t, branch, pfp.Value)
  132. case alpine_module.PropertyRepository:
  133. assert.Equal(t, repository, pfp.Value)
  134. case alpine_module.PropertyArchitecture:
  135. assert.Equal(t, "x86_64", pfp.Value)
  136. }
  137. }
  138. }
  139. }
  140. return seen
  141. })
  142. })
  143. readIndexContent := func(r io.Reader) (string, error) {
  144. br := bufio.NewReader(r)
  145. gzr, err := gzip.NewReader(br)
  146. if err != nil {
  147. return "", err
  148. }
  149. for {
  150. gzr.Multistream(false)
  151. tr := tar.NewReader(gzr)
  152. for {
  153. hd, err := tr.Next()
  154. if err == io.EOF {
  155. break
  156. }
  157. if err != nil {
  158. return "", err
  159. }
  160. if hd.Name == alpine_service.IndexFilename {
  161. buf, err := io.ReadAll(tr)
  162. if err != nil {
  163. return "", err
  164. }
  165. return string(buf), nil
  166. }
  167. }
  168. err = gzr.Reset(br)
  169. if err == io.EOF {
  170. break
  171. }
  172. if err != nil {
  173. return "", err
  174. }
  175. }
  176. return "", io.EOF
  177. }
  178. t.Run("Index", func(t *testing.T) {
  179. defer tests.PrintCurrentTest(t)()
  180. req := NewRequest(t, "GET", fmt.Sprintf("%s/%s/%s/x86_64/APKINDEX.tar.gz", rootURL, branch, repository))
  181. resp := MakeRequest(t, req, http.StatusOK)
  182. content, err := readIndexContent(resp.Body)
  183. assert.NoError(t, err)
  184. assert.Contains(t, content, "C:Q1/se1PjO94hYXbfpNR1/61hVORIc=\n")
  185. assert.Contains(t, content, "P:"+packageName+"\n")
  186. assert.Contains(t, content, "V:"+packageVersion+"\n")
  187. assert.Contains(t, content, "A:x86_64\n")
  188. assert.NotContains(t, content, "A:noarch\n")
  189. assert.Contains(t, content, "T:Gitea Test Package\n")
  190. assert.Contains(t, content, "U:https://gitea.io/\n")
  191. assert.Contains(t, content, "L:MIT\n")
  192. assert.Contains(t, content, "S:1353\n")
  193. assert.Contains(t, content, "I:4096\n")
  194. assert.Contains(t, content, "o:gitea-test\n")
  195. assert.Contains(t, content, "m:KN4CK3R <kn4ck3r@gitea.io>\n")
  196. assert.Contains(t, content, "t:1679498030\n")
  197. })
  198. t.Run("Download", func(t *testing.T) {
  199. defer tests.PrintCurrentTest(t)()
  200. req := NewRequest(t, "GET", fmt.Sprintf("%s/%s/%s/x86_64/%s-%s.apk", rootURL, branch, repository, packageName, packageVersion))
  201. MakeRequest(t, req, http.StatusOK)
  202. })
  203. t.Run("NoArch", func(t *testing.T) {
  204. defer tests.PrintCurrentTest(t)()
  205. req := NewRequestWithBody(t, "PUT", fmt.Sprintf("%s/%s/%s", rootURL, branch, repository), bytes.NewReader(noarchContent)).
  206. AddBasicAuth(user.Name)
  207. MakeRequest(t, req, http.StatusCreated)
  208. req = NewRequest(t, "GET", fmt.Sprintf("%s/%s/%s/x86_64/APKINDEX.tar.gz", rootURL, branch, repository))
  209. resp := MakeRequest(t, req, http.StatusOK)
  210. content, err := readIndexContent(resp.Body)
  211. assert.NoError(t, err)
  212. assert.Contains(t, content, "C:Q1/se1PjO94hYXbfpNR1/61hVORIc=\n")
  213. assert.Contains(t, content, "A:x86_64\n")
  214. assert.Contains(t, content, "C:Q1kbH5WoIPFccQYyATanaKXd2cJcc=\n")
  215. assert.NotContains(t, content, "A:noarch\n")
  216. // noarch package should be available with every architecture requested
  217. for _, arch := range []string{alpine_module.NoArch, "x86_64", "my_arch"} {
  218. req := NewRequest(t, "GET", fmt.Sprintf("%s/%s/%s/%s/gitea-noarch-1.4-r0.apk", rootURL, branch, repository, arch))
  219. MakeRequest(t, req, http.StatusOK)
  220. }
  221. req = NewRequest(t, "DELETE", fmt.Sprintf("%s/%s/%s/noarch/gitea-noarch-1.4-r0.apk", rootURL, branch, repository)).
  222. AddBasicAuth(user.Name)
  223. MakeRequest(t, req, http.StatusNoContent)
  224. })
  225. })
  226. }
  227. }
  228. t.Run("Delete", func(t *testing.T) {
  229. defer tests.PrintCurrentTest(t)()
  230. for _, branch := range branches {
  231. for _, repository := range repositories {
  232. req := NewRequest(t, "DELETE", fmt.Sprintf("%s/%s/%s/x86_64/%s-%s.apk", rootURL, branch, repository, packageName, packageVersion))
  233. MakeRequest(t, req, http.StatusUnauthorized)
  234. req = NewRequest(t, "DELETE", fmt.Sprintf("%s/%s/%s/x86_64/%s-%s.apk", rootURL, branch, repository, packageName, packageVersion)).
  235. AddBasicAuth(user.Name)
  236. MakeRequest(t, req, http.StatusNoContent)
  237. // Deleting the last file of an architecture should remove that index
  238. req = NewRequest(t, "GET", fmt.Sprintf("%s/%s/%s/x86_64/APKINDEX.tar.gz", rootURL, branch, repository))
  239. MakeRequest(t, req, http.StatusNotFound)
  240. }
  241. }
  242. })
  243. }