123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402 |
- // Copyright 2023 The Gitea Authors. All rights reserved.
- // SPDX-License-Identifier: MIT
-
- package chef
-
- import (
- "errors"
- "fmt"
- "io"
- "net/http"
- "net/url"
- "sort"
- "strings"
- "time"
-
- "code.gitea.io/gitea/models/db"
- packages_model "code.gitea.io/gitea/models/packages"
- "code.gitea.io/gitea/modules/context"
- packages_module "code.gitea.io/gitea/modules/packages"
- chef_module "code.gitea.io/gitea/modules/packages/chef"
- "code.gitea.io/gitea/modules/setting"
- "code.gitea.io/gitea/modules/util"
- "code.gitea.io/gitea/routers/api/packages/helper"
- packages_service "code.gitea.io/gitea/services/packages"
- )
-
- func apiError(ctx *context.Context, status int, obj any) {
- type Error struct {
- ErrorMessages []string `json:"error_messages"`
- }
-
- helper.LogAndProcessError(ctx, status, obj, func(message string) {
- ctx.JSON(status, Error{
- ErrorMessages: []string{message},
- })
- })
- }
-
- func PackagesUniverse(ctx *context.Context) {
- pvs, _, err := packages_model.SearchVersions(ctx, &packages_model.PackageSearchOptions{
- OwnerID: ctx.Package.Owner.ID,
- Type: packages_model.TypeChef,
- IsInternal: util.OptionalBoolFalse,
- })
- if err != nil {
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
-
- pds, err := packages_model.GetPackageDescriptors(ctx, pvs)
- if err != nil {
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
-
- type VersionInfo struct {
- LocationType string `json:"location_type"`
- LocationPath string `json:"location_path"`
- DownloadURL string `json:"download_url"`
- Dependencies map[string]string `json:"dependencies"`
- }
-
- baseURL := setting.AppURL + "api/packages/" + ctx.Package.Owner.Name + "/chef/api/v1"
-
- universe := make(map[string]map[string]*VersionInfo)
- for _, pd := range pds {
- if _, ok := universe[pd.Package.Name]; !ok {
- universe[pd.Package.Name] = make(map[string]*VersionInfo)
- }
- universe[pd.Package.Name][pd.Version.Version] = &VersionInfo{
- LocationType: "opscode",
- LocationPath: baseURL,
- DownloadURL: fmt.Sprintf("%s/cookbooks/%s/versions/%s/download", baseURL, url.PathEscape(pd.Package.Name), pd.Version.Version),
- Dependencies: pd.Metadata.(*chef_module.Metadata).Dependencies,
- }
- }
-
- ctx.JSON(http.StatusOK, universe)
- }
-
- // https://github.com/chef/chef/blob/main/knife/lib/chef/knife/supermarket_list.rb
- // https://github.com/chef/chef/blob/main/knife/lib/chef/knife/supermarket_search.rb
- func EnumeratePackages(ctx *context.Context) {
- opts := &packages_model.PackageSearchOptions{
- OwnerID: ctx.Package.Owner.ID,
- Type: packages_model.TypeChef,
- Name: packages_model.SearchValue{Value: ctx.FormTrim("q")},
- IsInternal: util.OptionalBoolFalse,
- Paginator: db.NewAbsoluteListOptions(
- ctx.FormInt("start"),
- ctx.FormInt("items"),
- ),
- }
-
- switch strings.ToLower(ctx.FormTrim("order")) {
- case "recently_updated", "recently_added":
- opts.Sort = packages_model.SortCreatedDesc
- default:
- opts.Sort = packages_model.SortNameAsc
- }
-
- pvs, total, err := packages_model.SearchLatestVersions(ctx, opts)
- if err != nil {
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
-
- pds, err := packages_model.GetPackageDescriptors(ctx, pvs)
- if err != nil {
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
-
- type Item struct {
- CookbookName string `json:"cookbook_name"`
- CookbookMaintainer string `json:"cookbook_maintainer"`
- CookbookDescription string `json:"cookbook_description"`
- Cookbook string `json:"cookbook"`
- }
-
- type Result struct {
- Start int `json:"start"`
- Total int `json:"total"`
- Items []*Item `json:"items"`
- }
-
- baseURL := setting.AppURL + "api/packages/" + ctx.Package.Owner.Name + "/chef/api/v1/cookbooks/"
-
- items := make([]*Item, 0, len(pds))
- for _, pd := range pds {
- metadata := pd.Metadata.(*chef_module.Metadata)
-
- items = append(items, &Item{
- CookbookName: pd.Package.Name,
- CookbookMaintainer: metadata.Author,
- CookbookDescription: metadata.Description,
- Cookbook: baseURL + url.PathEscape(pd.Package.Name),
- })
- }
-
- skip, _ := opts.Paginator.GetSkipTake()
-
- ctx.JSON(http.StatusOK, &Result{
- Start: skip,
- Total: int(total),
- Items: items,
- })
- }
-
- // https://github.com/chef/chef/blob/main/knife/lib/chef/knife/supermarket_show.rb
- func PackageMetadata(ctx *context.Context) {
- packageName := ctx.Params("name")
-
- pvs, err := packages_model.GetVersionsByPackageName(ctx, ctx.Package.Owner.ID, packages_model.TypeChef, packageName)
- if err != nil {
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
- if len(pvs) == 0 {
- apiError(ctx, http.StatusNotFound, nil)
- return
- }
-
- pds, err := packages_model.GetPackageDescriptors(ctx, pvs)
- if err != nil {
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
-
- sort.Slice(pds, func(i, j int) bool {
- return pds[i].SemVer.LessThan(pds[j].SemVer)
- })
-
- type Result struct {
- Name string `json:"name"`
- Maintainer string `json:"maintainer"`
- Description string `json:"description"`
- Category string `json:"category"`
- LatestVersion string `json:"latest_version"`
- SourceURL string `json:"source_url"`
- CreatedAt time.Time `json:"created_at"`
- UpdatedAt time.Time `json:"updated_at"`
- Deprecated bool `json:"deprecated"`
- Versions []string `json:"versions"`
- }
-
- baseURL := fmt.Sprintf("%sapi/packages/%s/chef/api/v1/cookbooks/%s/versions/", setting.AppURL, ctx.Package.Owner.Name, url.PathEscape(packageName))
-
- versions := make([]string, 0, len(pds))
- for _, pd := range pds {
- versions = append(versions, baseURL+pd.Version.Version)
- }
-
- latest := pds[len(pds)-1]
-
- metadata := latest.Metadata.(*chef_module.Metadata)
-
- ctx.JSON(http.StatusOK, &Result{
- Name: latest.Package.Name,
- Maintainer: metadata.Author,
- Description: metadata.Description,
- LatestVersion: baseURL + latest.Version.Version,
- SourceURL: metadata.RepositoryURL,
- CreatedAt: latest.Version.CreatedUnix.AsLocalTime(),
- UpdatedAt: latest.Version.CreatedUnix.AsLocalTime(),
- Deprecated: false,
- Versions: versions,
- })
- }
-
- // https://github.com/chef/chef/blob/main/knife/lib/chef/knife/supermarket_show.rb
- func PackageVersionMetadata(ctx *context.Context) {
- packageName := ctx.Params("name")
- packageVersion := strings.ReplaceAll(ctx.Params("version"), "_", ".") // Chef calls this endpoint with "_" instead of "."?!
-
- pv, err := packages_model.GetVersionByNameAndVersion(ctx, ctx.Package.Owner.ID, packages_model.TypeChef, packageName, packageVersion)
- if err != nil {
- if err == packages_model.ErrPackageNotExist {
- apiError(ctx, http.StatusNotFound, err)
- return
- }
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
-
- pd, err := packages_model.GetPackageDescriptor(ctx, pv)
- if err != nil {
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
-
- type Result struct {
- Version string `json:"version"`
- TarballFileSize int64 `json:"tarball_file_size"`
- PublishedAt time.Time `json:"published_at"`
- Cookbook string `json:"cookbook"`
- File string `json:"file"`
- License string `json:"license"`
- Dependencies map[string]string `json:"dependencies"`
- }
-
- baseURL := fmt.Sprintf("%sapi/packages/%s/chef/api/v1/cookbooks/%s", setting.AppURL, ctx.Package.Owner.Name, url.PathEscape(pd.Package.Name))
-
- metadata := pd.Metadata.(*chef_module.Metadata)
-
- ctx.JSON(http.StatusOK, &Result{
- Version: pd.Version.Version,
- TarballFileSize: pd.Files[0].Blob.Size,
- PublishedAt: pd.Version.CreatedUnix.AsLocalTime(),
- Cookbook: baseURL,
- File: fmt.Sprintf("%s/versions/%s/download", baseURL, pd.Version.Version),
- License: metadata.License,
- Dependencies: metadata.Dependencies,
- })
- }
-
- // https://github.com/chef/chef/blob/main/knife/lib/chef/knife/supermarket_share.rb
- func UploadPackage(ctx *context.Context) {
- file, _, err := ctx.Req.FormFile("tarball")
- if err != nil {
- apiError(ctx, http.StatusBadRequest, err)
- return
- }
- defer file.Close()
-
- buf, err := packages_module.CreateHashedBufferFromReader(file)
- if err != nil {
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
- defer buf.Close()
-
- pck, err := chef_module.ParsePackage(buf)
- if err != nil {
- if errors.Is(err, util.ErrInvalidArgument) {
- apiError(ctx, http.StatusBadRequest, err)
- } else {
- apiError(ctx, http.StatusInternalServerError, err)
- }
- return
- }
-
- if _, err := buf.Seek(0, io.SeekStart); err != nil {
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
-
- _, _, err = packages_service.CreatePackageAndAddFile(
- ctx,
- &packages_service.PackageCreationInfo{
- PackageInfo: packages_service.PackageInfo{
- Owner: ctx.Package.Owner,
- PackageType: packages_model.TypeChef,
- Name: pck.Name,
- Version: pck.Version,
- },
- Creator: ctx.Doer,
- SemverCompatible: true,
- Metadata: pck.Metadata,
- },
- &packages_service.PackageFileCreationInfo{
- PackageFileInfo: packages_service.PackageFileInfo{
- Filename: strings.ToLower(pck.Version + ".tar.gz"),
- },
- Creator: ctx.Doer,
- Data: buf,
- IsLead: true,
- },
- )
- if err != nil {
- switch err {
- case packages_model.ErrDuplicatePackageVersion:
- apiError(ctx, http.StatusBadRequest, err)
- case packages_service.ErrQuotaTotalCount, packages_service.ErrQuotaTypeSize, packages_service.ErrQuotaTotalSize:
- apiError(ctx, http.StatusForbidden, err)
- default:
- apiError(ctx, http.StatusInternalServerError, err)
- }
- return
- }
-
- ctx.JSON(http.StatusCreated, make(map[any]any))
- }
-
- // https://github.com/chef/chef/blob/main/knife/lib/chef/knife/supermarket_download.rb
- func DownloadPackage(ctx *context.Context) {
- pv, err := packages_model.GetVersionByNameAndVersion(ctx, ctx.Package.Owner.ID, packages_model.TypeChef, ctx.Params("name"), ctx.Params("version"))
- if err != nil {
- if err == packages_model.ErrPackageNotExist {
- apiError(ctx, http.StatusNotFound, err)
- return
- }
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
-
- pd, err := packages_model.GetPackageDescriptor(ctx, pv)
- if err != nil {
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
-
- pf := pd.Files[0].File
-
- s, u, _, err := packages_service.GetPackageFileStream(ctx, pf)
- if err != nil {
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
-
- helper.ServePackageFile(ctx, s, u, pf)
- }
-
- // https://github.com/chef/chef/blob/main/knife/lib/chef/knife/supermarket_unshare.rb
- func DeletePackageVersion(ctx *context.Context) {
- packageName := ctx.Params("name")
- packageVersion := ctx.Params("version")
-
- err := packages_service.RemovePackageVersionByNameAndVersion(
- ctx,
- ctx.Doer,
- &packages_service.PackageInfo{
- Owner: ctx.Package.Owner,
- PackageType: packages_model.TypeChef,
- Name: packageName,
- Version: packageVersion,
- },
- )
- if err != nil {
- if err == packages_model.ErrPackageNotExist {
- apiError(ctx, http.StatusNotFound, err)
- } else {
- apiError(ctx, http.StatusInternalServerError, err)
- }
- return
- }
-
- ctx.Status(http.StatusOK)
- }
-
- // https://github.com/chef/chef/blob/main/knife/lib/chef/knife/supermarket_unshare.rb
- func DeletePackage(ctx *context.Context) {
- pvs, err := packages_model.GetVersionsByPackageName(ctx, ctx.Package.Owner.ID, packages_model.TypeChef, ctx.Params("name"))
- if err != nil {
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
-
- if len(pvs) == 0 {
- apiError(ctx, http.StatusNotFound, err)
- return
- }
-
- for _, pv := range pvs {
- if err := packages_service.RemovePackageVersion(ctx, ctx.Doer, pv); err != nil {
- apiError(ctx, http.StatusInternalServerError, err)
- return
- }
- }
-
- ctx.Status(http.StatusOK)
- }
|