gitea/routers/api/packages/arch/arch.go

230 lines
5.9 KiB
Go
Raw Normal View History

2023-06-20 20:08:48 +00:00
// Copyright 2023 The Gitea Authors. All rights reserved.
// SPDX-License-Identifier: MIT
package arch
import (
"bytes"
2023-06-20 20:08:48 +00:00
"encoding/hex"
"io"
"net/http"
"strings"
2023-08-03 17:43:13 +00:00
pkg_model "code.gitea.io/gitea/models/packages"
2023-06-20 20:08:48 +00:00
"code.gitea.io/gitea/modules/context"
pkg_module "code.gitea.io/gitea/modules/packages"
2023-06-20 20:08:48 +00:00
arch_module "code.gitea.io/gitea/modules/packages/arch"
"code.gitea.io/gitea/routers/api/packages/helper"
2023-08-03 17:43:13 +00:00
pkg_service "code.gitea.io/gitea/services/packages"
arch_service "code.gitea.io/gitea/services/packages/arch"
2023-06-20 20:08:48 +00:00
)
// Push new package to arch package registry.
func Push(ctx *context.Context) {
var (
owner = ctx.Params("username")
2023-07-27 10:21:11 +00:00
filename = ctx.Params("filename")
distro = ctx.Params("distro")
sign = ctx.Params("sign")
)
2023-06-20 20:08:48 +00:00
upload, close, err := ctx.UploadStream()
2023-06-20 20:08:48 +00:00
if err != nil {
apiError(ctx, http.StatusInternalServerError, err)
return
}
if close {
defer upload.Close()
}
buf, err := pkg_module.CreateHashedBufferFromReader(upload)
if err != nil {
apiError(ctx, http.StatusInternalServerError, err)
return
}
defer buf.Close()
2023-06-20 20:08:48 +00:00
// Parse metadata related to package contained in arch package archive.
pkgmd, err := arch_module.EjectMetadata(filename, distro, buf)
if err != nil {
apiError(ctx, http.StatusBadRequest, err)
return
}
if _, err := buf.Seek(0, io.SeekStart); err != nil {
apiError(ctx, http.StatusInternalServerError, err)
return
}
// Metadata related to SQL database.
dbmd := &arch_module.Metadata{
URL: pkgmd.URL,
Description: pkgmd.Description,
Provides: pkgmd.Provides,
License: pkgmd.License,
Depends: pkgmd.Depends,
OptDepends: pkgmd.OptDepends,
MakeDepends: pkgmd.MakeDepends,
CheckDepends: pkgmd.CheckDepends,
Backup: pkgmd.Backup,
DistroArch: []string{distro + "-" + pkgmd.Arch[0]},
}
// Save file related to arch package.
pkgid, err := arch_service.SaveFile(ctx, &arch_service.SaveFileParams{
Creator: ctx.ContextUser,
Owner: ctx.Package.Owner,
Metadata: dbmd,
2023-08-03 17:43:13 +00:00
Buf: buf,
Filename: filename,
Distro: distro,
2023-08-03 17:43:13 +00:00
PkgName: pkgmd.Name,
PkgVer: pkgmd.Version,
})
if err != nil {
apiError(ctx, http.StatusInternalServerError, err)
return
}
r := io.NopCloser(bytes.NewReader([]byte(pkgmd.GetDbDesc())))
buf, err = pkg_module.CreateHashedBufferFromReader(r)
if err != nil {
apiError(ctx, http.StatusInternalServerError, err)
return
}
defer buf.Close()
// Save file related to arch package description for pacman database.
_, err = arch_service.SaveFile(ctx, &arch_service.SaveFileParams{
Creator: ctx.ContextUser,
Owner: ctx.Package.Owner,
Filename: pkgmd.Name + "-" + pkgmd.Version + "-" + pkgmd.Arch[0] + ".desc",
Buf: buf,
Metadata: dbmd,
Distro: distro,
2023-08-03 17:43:13 +00:00
PkgName: pkgmd.Name,
PkgVer: pkgmd.Version,
})
2023-06-20 20:08:48 +00:00
if err != nil {
apiError(ctx, http.StatusInternalServerError, err)
2023-06-20 20:08:48 +00:00
return
}
// Decoding package signature, if present saving with package as file.
sigdata, err := hex.DecodeString(sign)
if err == nil {
r := io.NopCloser(bytes.NewReader(sigdata))
buf, err := pkg_module.CreateHashedBufferFromReader(r)
if err != nil {
apiError(ctx, http.StatusInternalServerError, err)
return
}
defer buf.Close()
_, err = arch_service.SaveFile(ctx, &arch_service.SaveFileParams{
Creator: ctx.ContextUser,
Owner: ctx.Package.Owner,
Buf: buf,
Filename: filename + ".sig",
Metadata: dbmd,
Distro: distro,
2023-08-03 17:43:13 +00:00
PkgName: pkgmd.Name,
PkgVer: pkgmd.Version,
})
if err != nil {
apiError(ctx, http.StatusInternalServerError, err)
return
}
2023-06-20 20:08:48 +00:00
}
// Add new architectures and distribution info to package version metadata.
err = arch_service.UpdateMetadata(ctx, &arch_service.UpdateMetadataParameters{
2023-08-03 17:43:13 +00:00
User: ctx.Package.Owner,
Metadata: dbmd,
DbDesc: pkgmd,
})
if err != nil {
apiError(ctx, http.StatusInternalServerError, err)
return
}
// Automatically connect repository for provided package if name matched.
2023-08-03 17:43:13 +00:00
err = arch_service.RepositoryAutoconnect(ctx, owner, pkgmd.Name, pkgid)
2023-06-20 20:08:48 +00:00
if err != nil {
apiError(ctx, http.StatusInternalServerError, err)
return
}
ctx.Status(http.StatusOK)
}
// Get file from arch package registry.
func Get(ctx *context.Context) {
var (
file = ctx.Params("file")
owner = ctx.Params("username")
distro = ctx.Params("distro")
arch = ctx.Params("arch")
)
2023-06-20 20:08:48 +00:00
2023-08-03 17:43:13 +00:00
// Packages and signatures are loaded directly from object storage.
if strings.HasSuffix(file, "tar.zst") || strings.HasSuffix(file, "zst.sig") {
pkg, err := arch_service.GetFileObject(ctx, distro, file)
2023-06-20 20:08:48 +00:00
if err != nil {
apiError(ctx, http.StatusNotFound, err)
2023-06-20 20:08:48 +00:00
return
}
ctx.ServeContent(pkg, &context.ServeHeaderOptions{
2023-07-27 10:21:11 +00:00
Filename: file,
})
2023-06-20 20:08:48 +00:00
return
}
2023-08-03 17:43:13 +00:00
// Pacman databases is not stored in giteas storage and created 'on-request'
// for user/organization scope with accordance to requested architecture
// and distribution.
if strings.HasSuffix(file, ".db.tar.gz") || strings.HasSuffix(file, ".db") {
db, err := arch_service.CreatePacmanDb(ctx, owner, arch, distro)
if err != nil {
apiError(ctx, http.StatusInternalServerError, err)
return
}
ctx.ServeContent(bytes.NewReader(db), &context.ServeHeaderOptions{
2023-07-27 10:21:11 +00:00
Filename: file,
})
return
2023-06-20 20:08:48 +00:00
}
ctx.Status(http.StatusNotFound)
2023-06-20 20:08:48 +00:00
}
2023-06-25 14:45:16 +00:00
// Remove specific package version, related files and pacman database entry.
func Remove(ctx *context.Context) {
var (
2023-07-27 10:21:11 +00:00
pkg = ctx.Params("package")
ver = ctx.Params("version")
)
2023-08-03 17:43:13 +00:00
version, err := pkg_model.GetVersionByNameAndVersion(ctx, ctx.Package.Owner.ID, pkg_model.TypeArch, pkg, ver)
if err != nil {
apiError(ctx, http.StatusInternalServerError, err)
return
}
err = pkg_service.RemovePackageVersion(ctx.Package.Owner, version)
if err != nil {
apiError(ctx, http.StatusInternalServerError, err)
return
}
ctx.Status(http.StatusOK)
}
2023-06-20 20:08:48 +00:00
func apiError(ctx *context.Context, status int, obj interface{}) {
helper.LogAndProcessError(ctx, status, obj, func(message string) {
ctx.PlainText(status, message)
})
}