mirror of
https://codeberg.org/forgejo/forgejo.git
synced 2025-01-15 07:26:31 +03:00
Merge pull request 'Feat: Add support for pacman -F
in Arch package' (#6180) from dragon/forgejo:clear-arch-pkg into forgejo
Reviewed-on: https://codeberg.org/forgejo/forgejo/pulls/6180 Reviewed-by: Gusted <gusted@noreply.codeberg.org>
This commit is contained in:
commit
4bc0abac3c
5 changed files with 96 additions and 29 deletions
|
@ -4,6 +4,7 @@
|
||||||
package arch
|
package arch
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"archive/tar"
|
||||||
"bufio"
|
"bufio"
|
||||||
"bytes"
|
"bytes"
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
|
@ -25,7 +26,9 @@ import (
|
||||||
// https://man.archlinux.org/man/PKGBUILD.5
|
// https://man.archlinux.org/man/PKGBUILD.5
|
||||||
|
|
||||||
const (
|
const (
|
||||||
PropertyDescription = "arch.description"
|
PropertyDescription = "arch.description"
|
||||||
|
PropertyFiles = "arch.files"
|
||||||
|
|
||||||
PropertyArch = "arch.architecture"
|
PropertyArch = "arch.architecture"
|
||||||
PropertyDistribution = "arch.distribution"
|
PropertyDistribution = "arch.distribution"
|
||||||
|
|
||||||
|
@ -85,6 +88,8 @@ type FileMetadata struct {
|
||||||
Packager string `json:"packager"`
|
Packager string `json:"packager"`
|
||||||
Arch string `json:"arch"`
|
Arch string `json:"arch"`
|
||||||
PgpSigned string `json:"pgp"`
|
PgpSigned string `json:"pgp"`
|
||||||
|
|
||||||
|
Files []string `json:"files,omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// ParsePackage Function that receives arch package archive data and returns it's metadata.
|
// ParsePackage Function that receives arch package archive data and returns it's metadata.
|
||||||
|
@ -127,6 +132,8 @@ func ParsePackage(r *packages.HashedBuffer) (*Package, error) {
|
||||||
var pkg *Package
|
var pkg *Package
|
||||||
var mTree bool
|
var mTree bool
|
||||||
|
|
||||||
|
files := make([]string, 0)
|
||||||
|
|
||||||
for {
|
for {
|
||||||
f, err := tarball.Read()
|
f, err := tarball.Read()
|
||||||
if err == io.EOF {
|
if err == io.EOF {
|
||||||
|
@ -135,6 +142,11 @@ func ParsePackage(r *packages.HashedBuffer) (*Package, error) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
// ref:https://gitlab.archlinux.org/pacman/pacman/-/blob/91546004903eea5d5267d59898a6029ba1d64031/lib/libalpm/add.c#L529-L533
|
||||||
|
if !strings.HasPrefix(f.Name(), ".") {
|
||||||
|
files = append(files, (f.Header.(*tar.Header)).Name)
|
||||||
|
}
|
||||||
|
|
||||||
switch f.Name() {
|
switch f.Name() {
|
||||||
case ".PKGINFO":
|
case ".PKGINFO":
|
||||||
pkg, err = ParsePackageInfo(tarballType, f)
|
pkg, err = ParsePackageInfo(tarballType, f)
|
||||||
|
@ -155,7 +167,7 @@ func ParsePackage(r *packages.HashedBuffer) (*Package, error) {
|
||||||
if !mTree {
|
if !mTree {
|
||||||
return nil, util.NewInvalidArgumentErrorf(".MTREE file not found")
|
return nil, util.NewInvalidArgumentErrorf(".MTREE file not found")
|
||||||
}
|
}
|
||||||
|
pkg.FileMetadata.Files = files
|
||||||
pkg.FileMetadata.CompressedSize = r.Size()
|
pkg.FileMetadata.CompressedSize = r.Size()
|
||||||
pkg.FileMetadata.MD5 = hex.EncodeToString(md5)
|
pkg.FileMetadata.MD5 = hex.EncodeToString(md5)
|
||||||
pkg.FileMetadata.SHA256 = hex.EncodeToString(sha256)
|
pkg.FileMetadata.SHA256 = hex.EncodeToString(sha256)
|
||||||
|
@ -339,3 +351,12 @@ func (p *Package) Desc() string {
|
||||||
}
|
}
|
||||||
return buf.String()
|
return buf.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (p *Package) Files() string {
|
||||||
|
var buf bytes.Buffer
|
||||||
|
buf.WriteString("%FILES%\n")
|
||||||
|
for _, item := range p.FileMetadata.Files {
|
||||||
|
_, _ = fmt.Fprintf(&buf, "%s\n", item)
|
||||||
|
}
|
||||||
|
return buf.String()
|
||||||
|
}
|
||||||
|
|
|
@ -344,8 +344,8 @@ func TestValidatePackageSpec(t *testing.T) {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestDescString(t *testing.T) {
|
func TestDescAndFileString(t *testing.T) {
|
||||||
const pkgdesc = `%FILENAME%
|
const pkgDesc = `%FILENAME%
|
||||||
zstd-1.5.5-1-x86_64.pkg.tar.zst
|
zstd-1.5.5-1-x86_64.pkg.tar.zst
|
||||||
|
|
||||||
%NAME%
|
%NAME%
|
||||||
|
@ -415,6 +415,12 @@ ninja
|
||||||
dummy5
|
dummy5
|
||||||
dummy6
|
dummy6
|
||||||
|
|
||||||
|
`
|
||||||
|
|
||||||
|
const pkgFiles = `%FILES%
|
||||||
|
usr/
|
||||||
|
usr/bin/
|
||||||
|
usr/bin/zstd
|
||||||
`
|
`
|
||||||
|
|
||||||
md := &Package{
|
md := &Package{
|
||||||
|
@ -441,7 +447,9 @@ dummy6
|
||||||
BuildDate: 1681646714,
|
BuildDate: 1681646714,
|
||||||
Packager: "Jelle van der Waa <jelle@archlinux.org>",
|
Packager: "Jelle van der Waa <jelle@archlinux.org>",
|
||||||
Arch: "x86_64",
|
Arch: "x86_64",
|
||||||
|
Files: []string{"usr/", "usr/bin/", "usr/bin/zstd"},
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
require.Equal(t, pkgdesc, md.Desc())
|
require.Equal(t, pkgDesc, md.Desc())
|
||||||
|
require.Equal(t, pkgFiles, md.Files())
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,7 +26,7 @@ import (
|
||||||
|
|
||||||
var (
|
var (
|
||||||
archPkgOrSig = regexp.MustCompile(`^.*\.pkg\.tar\.\w+(\.sig)*$`)
|
archPkgOrSig = regexp.MustCompile(`^.*\.pkg\.tar\.\w+(\.sig)*$`)
|
||||||
archDBOrSig = regexp.MustCompile(`^.*.db(\.tar\.gz)*(\.sig)*$`)
|
archDBOrSig = regexp.MustCompile(`^.*.(db|files)(\.tar\.gz)*(\.sig)*$`)
|
||||||
|
|
||||||
locker = sync.NewExclusivePool()
|
locker = sync.NewExclusivePool()
|
||||||
)
|
)
|
||||||
|
@ -115,6 +115,7 @@ func PushPackage(ctx *context.Context) {
|
||||||
|
|
||||||
properties := map[string]string{
|
properties := map[string]string{
|
||||||
arch_module.PropertyDescription: p.Desc(),
|
arch_module.PropertyDescription: p.Desc(),
|
||||||
|
arch_module.PropertyFiles: p.Files(),
|
||||||
arch_module.PropertyArch: p.FileMetadata.Arch,
|
arch_module.PropertyArch: p.FileMetadata.Arch,
|
||||||
arch_module.PropertyDistribution: group,
|
arch_module.PropertyDistribution: group,
|
||||||
}
|
}
|
||||||
|
|
|
@ -225,22 +225,44 @@ func createDB(ctx context.Context, ownerID int64, group, arch string) (*packages
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if len(pps) >= 1 {
|
if len(pps) == 0 {
|
||||||
meta := []byte(pps[0].Value)
|
continue
|
||||||
|
}
|
||||||
|
pkgDesc := []byte(pps[0].Value)
|
||||||
|
header := &tar.Header{
|
||||||
|
Name: pkg.Name + "-" + ver.Version + "/desc",
|
||||||
|
Size: int64(len(pkgDesc)),
|
||||||
|
Mode: int64(os.ModePerm),
|
||||||
|
}
|
||||||
|
if err = tw.WriteHeader(header); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if _, err := tw.Write(pkgDesc); err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
pfs, err := packages_model.GetPropertiesByName(
|
||||||
|
ctx, packages_model.PropertyTypeFile, pf.ID, arch_module.PropertyFiles,
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if len(pfs) >= 1 {
|
||||||
|
pkgFiles := []byte(pfs[0].Value)
|
||||||
header := &tar.Header{
|
header := &tar.Header{
|
||||||
Name: pkg.Name + "-" + ver.Version + "/desc",
|
Name: pkg.Name + "-" + ver.Version + "/files",
|
||||||
Size: int64(len(meta)),
|
Size: int64(len(pkgFiles)),
|
||||||
Mode: int64(os.ModePerm),
|
Mode: int64(os.ModePerm),
|
||||||
}
|
}
|
||||||
if err = tw.WriteHeader(header); err != nil {
|
if err = tw.WriteHeader(header); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if _, err := tw.Write(meta); err != nil {
|
if _, err := tw.Write(pkgFiles); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
count++
|
|
||||||
break
|
|
||||||
}
|
}
|
||||||
|
count++
|
||||||
|
break
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if count == 0 {
|
if count == 0 {
|
||||||
|
|
|
@ -223,8 +223,14 @@ HMhNSS1IzUsBcpJAPFAwwUXSM0u4BjoaR8EoGAWjgGQAAILFeyQADAAA
|
||||||
|
|
||||||
t.Run(fmt.Sprintf("RepositoryDB[%s]", group), func(t *testing.T) {
|
t.Run(fmt.Sprintf("RepositoryDB[%s]", group), func(t *testing.T) {
|
||||||
defer tests.PrintCurrentTest(t)()
|
defer tests.PrintCurrentTest(t)()
|
||||||
req := NewRequest(t, "GET", rootURL+"/repository.key")
|
req := NewRequest(t, "GET", groupURL+"/x86_64/base.db.tar.gz")
|
||||||
respPub := MakeRequest(t, req, http.StatusOK)
|
MakeRequest(t, req, http.StatusOK)
|
||||||
|
|
||||||
|
req = NewRequest(t, "GET", groupURL+"/x86_64/base.files")
|
||||||
|
MakeRequest(t, req, http.StatusOK)
|
||||||
|
|
||||||
|
req = NewRequest(t, "GET", groupURL+"/x86_64/base.files.tar.gz")
|
||||||
|
MakeRequest(t, req, http.StatusOK)
|
||||||
|
|
||||||
req = NewRequest(t, "GET", groupURL+"/x86_64/base.db")
|
req = NewRequest(t, "GET", groupURL+"/x86_64/base.db")
|
||||||
respPkg := MakeRequest(t, req, http.StatusOK)
|
respPkg := MakeRequest(t, req, http.StatusOK)
|
||||||
|
@ -232,23 +238,32 @@ HMhNSS1IzUsBcpJAPFAwwUXSM0u4BjoaR8EoGAWjgGQAAILFeyQADAAA
|
||||||
req = NewRequest(t, "GET", groupURL+"/x86_64/base.db.sig")
|
req = NewRequest(t, "GET", groupURL+"/x86_64/base.db.sig")
|
||||||
respSig := MakeRequest(t, req, http.StatusOK)
|
respSig := MakeRequest(t, req, http.StatusOK)
|
||||||
|
|
||||||
|
req = NewRequest(t, "GET", rootURL+"/repository.key")
|
||||||
|
respPub := MakeRequest(t, req, http.StatusOK)
|
||||||
|
|
||||||
if err := gpgVerify(respPub.Body.Bytes(), respSig.Body.Bytes(), respPkg.Body.Bytes()); err != nil {
|
if err := gpgVerify(respPub.Body.Bytes(), respSig.Body.Bytes(), respPkg.Body.Bytes()); err != nil {
|
||||||
t.Fatal(err)
|
t.Fatal(err)
|
||||||
}
|
}
|
||||||
files, err := listTarGzFiles(respPkg.Body.Bytes())
|
files, err := listTarGzFiles(respPkg.Body.Bytes())
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.Len(t, files, 1)
|
require.Len(t, files, 2)
|
||||||
for s, d := range files {
|
for s, d := range files {
|
||||||
name := getProperty(string(d.Data), "NAME")
|
if strings.HasSuffix(s, "/desc") {
|
||||||
ver := getProperty(string(d.Data), "VERSION")
|
name := getProperty(string(d.Data), "NAME")
|
||||||
require.Equal(t, name+"-"+ver+"/desc", s)
|
ver := getProperty(string(d.Data), "VERSION")
|
||||||
fn := getProperty(string(d.Data), "FILENAME")
|
require.Equal(t, name+"-"+ver+"/desc", s)
|
||||||
pgp := getProperty(string(d.Data), "PGPSIG")
|
fn := getProperty(string(d.Data), "FILENAME")
|
||||||
req = NewRequest(t, "GET", groupURL+"/x86_64/"+fn+".sig")
|
pgp := getProperty(string(d.Data), "PGPSIG")
|
||||||
respSig := MakeRequest(t, req, http.StatusOK)
|
req = NewRequest(t, "GET", groupURL+"/x86_64/"+fn+".sig")
|
||||||
decodeString, err := base64.StdEncoding.DecodeString(pgp)
|
respSig := MakeRequest(t, req, http.StatusOK)
|
||||||
require.NoError(t, err)
|
decodeString, err := base64.StdEncoding.DecodeString(pgp)
|
||||||
require.Equal(t, respSig.Body.Bytes(), decodeString)
|
require.NoError(t, err)
|
||||||
|
require.Equal(t, respSig.Body.Bytes(), decodeString)
|
||||||
|
} else if strings.HasSuffix(s, "/files") {
|
||||||
|
require.True(t, strings.HasPrefix(string(d.Data), "%FILES%"))
|
||||||
|
} else {
|
||||||
|
require.Failf(t, "unknown item", "fileName:%s", s)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -275,7 +290,7 @@ HMhNSS1IzUsBcpJAPFAwwUXSM0u4BjoaR8EoGAWjgGQAAILFeyQADAAA
|
||||||
respPkg := MakeRequest(t, req, http.StatusOK)
|
respPkg := MakeRequest(t, req, http.StatusOK)
|
||||||
files, err := listTarGzFiles(respPkg.Body.Bytes())
|
files, err := listTarGzFiles(respPkg.Body.Bytes())
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.Len(t, files, 1)
|
require.Len(t, files, 2)
|
||||||
|
|
||||||
req = NewRequestWithBody(t, "DELETE", groupURL+"/test2/1.0.0-1/any", nil).
|
req = NewRequestWithBody(t, "DELETE", groupURL+"/test2/1.0.0-1/any", nil).
|
||||||
AddBasicAuth(user.Name)
|
AddBasicAuth(user.Name)
|
||||||
|
@ -347,7 +362,7 @@ HMhNSS1IzUsBcpJAPFAwwUXSM0u4BjoaR8EoGAWjgGQAAILFeyQADAAA
|
||||||
|
|
||||||
files, err := listTarGzFiles(respPkg.Body.Bytes())
|
files, err := listTarGzFiles(respPkg.Body.Bytes())
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.Len(t, files, 1)
|
require.Len(t, files, 2)
|
||||||
|
|
||||||
req = NewRequestWithBody(t, "PUT", rootURL, bytes.NewReader(pkgs["otherXZ"])).
|
req = NewRequestWithBody(t, "PUT", rootURL, bytes.NewReader(pkgs["otherXZ"])).
|
||||||
AddBasicAuth(user.Name)
|
AddBasicAuth(user.Name)
|
||||||
|
@ -358,7 +373,7 @@ HMhNSS1IzUsBcpJAPFAwwUXSM0u4BjoaR8EoGAWjgGQAAILFeyQADAAA
|
||||||
|
|
||||||
files, err = listTarGzFiles(respPkg.Body.Bytes())
|
files, err = listTarGzFiles(respPkg.Body.Bytes())
|
||||||
require.NoError(t, err)
|
require.NoError(t, err)
|
||||||
require.Len(t, files, 2)
|
require.Len(t, files, 4)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue