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.

pub.go 8.0KB

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