gitea源码

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. // Copyright 2022 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package pub
  4. import (
  5. "errors"
  6. "fmt"
  7. "io"
  8. "net/http"
  9. "net/url"
  10. "sort"
  11. "strings"
  12. "time"
  13. packages_model "code.gitea.io/gitea/models/packages"
  14. "code.gitea.io/gitea/modules/json"
  15. packages_module "code.gitea.io/gitea/modules/packages"
  16. pub_module "code.gitea.io/gitea/modules/packages/pub"
  17. "code.gitea.io/gitea/modules/setting"
  18. "code.gitea.io/gitea/modules/util"
  19. "code.gitea.io/gitea/routers/api/packages/helper"
  20. "code.gitea.io/gitea/services/context"
  21. packages_service "code.gitea.io/gitea/services/packages"
  22. )
  23. func jsonResponse(ctx *context.Context, status int, obj any) {
  24. resp := ctx.Resp
  25. resp.Header().Set("Content-Type", "application/vnd.pub.v2+json")
  26. resp.WriteHeader(status)
  27. _ = json.NewEncoder(resp).Encode(obj)
  28. }
  29. func apiError(ctx *context.Context, status int, obj any) {
  30. type Error struct {
  31. Code string `json:"code"`
  32. Message string `json:"message"`
  33. }
  34. type ErrorWrapper struct {
  35. Error Error `json:"error"`
  36. }
  37. message := helper.ProcessErrorForUser(ctx, status, obj)
  38. jsonResponse(ctx, status, ErrorWrapper{
  39. Error: Error{
  40. Code: http.StatusText(status),
  41. Message: message,
  42. },
  43. })
  44. }
  45. type packageVersions struct {
  46. Name string `json:"name"`
  47. Latest *versionMetadata `json:"latest"`
  48. Versions []*versionMetadata `json:"versions"`
  49. }
  50. type versionMetadata struct {
  51. Version string `json:"version"`
  52. ArchiveURL string `json:"archive_url"`
  53. Published time.Time `json:"published"`
  54. Pubspec any `json:"pubspec,omitempty"`
  55. }
  56. func packageDescriptorToMetadata(baseURL string, pd *packages_model.PackageDescriptor) *versionMetadata {
  57. return &versionMetadata{
  58. Version: pd.Version.Version,
  59. ArchiveURL: fmt.Sprintf("%s/files/%s.tar.gz", baseURL, url.PathEscape(pd.Version.Version)),
  60. Published: pd.Version.CreatedUnix.AsLocalTime(),
  61. Pubspec: pd.Metadata.(*pub_module.Metadata).Pubspec,
  62. }
  63. }
  64. func baseURL(ctx *context.Context) string {
  65. return setting.AppURL + "api/packages/" + ctx.Package.Owner.Name + "/pub/api/packages"
  66. }
  67. // https://github.com/dart-lang/pub/blob/master/doc/repository-spec-v2.md#list-all-versions-of-a-package
  68. func EnumeratePackageVersions(ctx *context.Context) {
  69. packageName := ctx.PathParam("id")
  70. pvs, err := packages_model.GetVersionsByPackageName(ctx, ctx.Package.Owner.ID, packages_model.TypePub, packageName)
  71. if err != nil {
  72. apiError(ctx, http.StatusInternalServerError, err)
  73. return
  74. }
  75. if len(pvs) == 0 {
  76. apiError(ctx, http.StatusNotFound, err)
  77. return
  78. }
  79. pds, err := packages_model.GetPackageDescriptors(ctx, pvs)
  80. if err != nil {
  81. apiError(ctx, http.StatusInternalServerError, err)
  82. return
  83. }
  84. sort.Slice(pds, func(i, j int) bool {
  85. return pds[i].SemVer.LessThan(pds[j].SemVer)
  86. })
  87. baseURL := fmt.Sprintf("%s/%s", baseURL(ctx), url.PathEscape(pds[0].Package.Name))
  88. versions := make([]*versionMetadata, 0, len(pds))
  89. for _, pd := range pds {
  90. versions = append(versions, packageDescriptorToMetadata(baseURL, pd))
  91. }
  92. jsonResponse(ctx, http.StatusOK, &packageVersions{
  93. Name: pds[0].Package.Name,
  94. Latest: packageDescriptorToMetadata(baseURL, pds[0]),
  95. Versions: versions,
  96. })
  97. }
  98. // https://github.com/dart-lang/pub/blob/master/doc/repository-spec-v2.md#deprecated-inspect-a-specific-version-of-a-package
  99. func PackageVersionMetadata(ctx *context.Context) {
  100. packageName := ctx.PathParam("id")
  101. packageVersion := ctx.PathParam("version")
  102. pv, err := packages_model.GetVersionByNameAndVersion(ctx, ctx.Package.Owner.ID, packages_model.TypePub, packageName, packageVersion)
  103. if err != nil {
  104. if errors.Is(err, packages_model.ErrPackageNotExist) {
  105. apiError(ctx, http.StatusNotFound, err)
  106. return
  107. }
  108. apiError(ctx, http.StatusInternalServerError, err)
  109. return
  110. }
  111. pd, err := packages_model.GetPackageDescriptor(ctx, pv)
  112. if err != nil {
  113. apiError(ctx, http.StatusInternalServerError, err)
  114. return
  115. }
  116. jsonResponse(ctx, http.StatusOK, packageDescriptorToMetadata(
  117. fmt.Sprintf("%s/%s", baseURL(ctx), url.PathEscape(pd.Package.Name)),
  118. pd,
  119. ))
  120. }
  121. // https://github.com/dart-lang/pub/blob/master/doc/repository-spec-v2.md#publishing-packages
  122. func RequestUpload(ctx *context.Context) {
  123. type UploadRequest struct {
  124. URL string `json:"url"`
  125. Fields map[string]string `json:"fields"`
  126. }
  127. jsonResponse(ctx, http.StatusOK, UploadRequest{
  128. URL: baseURL(ctx) + "/versions/new/upload",
  129. Fields: make(map[string]string),
  130. })
  131. }
  132. // https://github.com/dart-lang/pub/blob/master/doc/repository-spec-v2.md#publishing-packages
  133. func UploadPackageFile(ctx *context.Context) {
  134. file, _, err := ctx.Req.FormFile("file")
  135. if err != nil {
  136. apiError(ctx, http.StatusBadRequest, err)
  137. return
  138. }
  139. defer file.Close()
  140. buf, err := packages_module.CreateHashedBufferFromReader(file)
  141. if err != nil {
  142. apiError(ctx, http.StatusInternalServerError, err)
  143. return
  144. }
  145. defer buf.Close()
  146. pck, err := pub_module.ParsePackage(buf)
  147. if err != nil {
  148. if errors.Is(err, util.ErrInvalidArgument) {
  149. apiError(ctx, http.StatusBadRequest, err)
  150. } else {
  151. apiError(ctx, http.StatusInternalServerError, err)
  152. }
  153. return
  154. }
  155. if _, err := buf.Seek(0, io.SeekStart); err != nil {
  156. apiError(ctx, http.StatusInternalServerError, err)
  157. return
  158. }
  159. _, _, err = packages_service.CreatePackageAndAddFile(
  160. ctx,
  161. &packages_service.PackageCreationInfo{
  162. PackageInfo: packages_service.PackageInfo{
  163. Owner: ctx.Package.Owner,
  164. PackageType: packages_model.TypePub,
  165. Name: pck.Name,
  166. Version: pck.Version,
  167. },
  168. SemverCompatible: true,
  169. Creator: ctx.Doer,
  170. Metadata: pck.Metadata,
  171. },
  172. &packages_service.PackageFileCreationInfo{
  173. PackageFileInfo: packages_service.PackageFileInfo{
  174. Filename: strings.ToLower(pck.Version + ".tar.gz"),
  175. },
  176. Creator: ctx.Doer,
  177. Data: buf,
  178. IsLead: true,
  179. },
  180. )
  181. if err != nil {
  182. switch err {
  183. case packages_model.ErrDuplicatePackageVersion:
  184. apiError(ctx, http.StatusConflict, err)
  185. case packages_service.ErrQuotaTotalCount, packages_service.ErrQuotaTypeSize, packages_service.ErrQuotaTotalSize:
  186. apiError(ctx, http.StatusForbidden, err)
  187. default:
  188. apiError(ctx, http.StatusInternalServerError, err)
  189. }
  190. return
  191. }
  192. ctx.Resp.Header().Set("Location", fmt.Sprintf("%s/versions/new/finalize/%s/%s", baseURL(ctx), url.PathEscape(pck.Name), url.PathEscape(pck.Version)))
  193. ctx.Status(http.StatusNoContent)
  194. }
  195. // https://github.com/dart-lang/pub/blob/master/doc/repository-spec-v2.md#publishing-packages
  196. func FinalizePackage(ctx *context.Context) {
  197. packageName := ctx.PathParam("id")
  198. packageVersion := ctx.PathParam("version")
  199. _, err := packages_model.GetVersionByNameAndVersion(ctx, ctx.Package.Owner.ID, packages_model.TypePub, packageName, packageVersion)
  200. if err != nil {
  201. if errors.Is(err, packages_model.ErrPackageNotExist) {
  202. apiError(ctx, http.StatusNotFound, err)
  203. return
  204. }
  205. apiError(ctx, http.StatusInternalServerError, err)
  206. return
  207. }
  208. type Success struct {
  209. Message string `json:"message"`
  210. }
  211. type SuccessWrapper struct {
  212. Success Success `json:"success"`
  213. }
  214. jsonResponse(ctx, http.StatusOK, SuccessWrapper{Success{}})
  215. }
  216. // https://github.com/dart-lang/pub/blob/master/doc/repository-spec-v2.md#deprecated-download-a-specific-version-of-a-package
  217. func DownloadPackageFile(ctx *context.Context) {
  218. packageName := ctx.PathParam("id")
  219. packageVersion := strings.TrimSuffix(ctx.PathParam("version"), ".tar.gz")
  220. pv, err := packages_model.GetVersionByNameAndVersion(ctx, ctx.Package.Owner.ID, packages_model.TypePub, packageName, packageVersion)
  221. if err != nil {
  222. if errors.Is(err, packages_model.ErrPackageNotExist) {
  223. apiError(ctx, http.StatusNotFound, err)
  224. return
  225. }
  226. apiError(ctx, http.StatusInternalServerError, err)
  227. return
  228. }
  229. pd, err := packages_model.GetPackageDescriptor(ctx, pv)
  230. if err != nil {
  231. apiError(ctx, http.StatusInternalServerError, err)
  232. return
  233. }
  234. pf := pd.Files[0].File
  235. s, u, _, err := packages_service.OpenFileForDownload(ctx, pf, ctx.Req.Method)
  236. if err != nil {
  237. apiError(ctx, http.StatusInternalServerError, err)
  238. return
  239. }
  240. helper.ServePackageFile(ctx, s, u, pf)
  241. }