gitea/tests/integration/api_packages_arch_test.go

466 lines
11 KiB
Go
Raw Normal View History

// Copyright 2023 The Gitea Authors. All rights reserved.
// SPDX-License-Identifier: MIT
package integration
import (
"bufio"
"bytes"
"crypto/md5"
"encoding/hex"
"errors"
"fmt"
"io"
"net/http"
"os"
"path"
"testing"
"testing/fstest"
"time"
"code.gitea.io/gitea/models/unittest"
user_model "code.gitea.io/gitea/models/user"
"code.gitea.io/gitea/modules/packages/arch"
"code.gitea.io/gitea/tests"
2023-07-10 17:50:44 +00:00
"github.com/mholt/archiver/v3"
"github.com/minio/sha256-simd"
"github.com/stretchr/testify/assert"
)
func TestPackageArch(t *testing.T) {
assert.NoError(t, unittest.PrepareTestDatabase())
var (
firstV1x86_64 = BuildArchPackage(t, "first", "1-1", "x86_64")
firstV1i686 = BuildArchPackage(t, "first", "1-1", "i686")
secondV1any = BuildArchPackage(t, "second", "1-1", "any")
firstV2x86_64 = BuildArchPackage(t, "first", "2-1", "x86_64")
secondV2any = BuildArchPackage(t, "second", "2-1", "any")
firstSign = []byte{1, 2, 3, 4}
secondSign = []byte{4, 3, 2, 1}
V1x86_64database = BuildArchDatabase([]arch.Package{
{
Name: "first",
Version: "1-1",
VersionMetadata: arch.VersionMetadata{
Base: "first",
},
FileMetadata: arch.FileMetadata{
CompressedSize: firstV1x86_64.size,
MD5: firstV1x86_64.md5,
SHA256: firstV1x86_64.sha256,
Arch: "x86_64",
},
},
{
Name: "second",
Version: "1-1",
VersionMetadata: arch.VersionMetadata{
Base: "second",
},
FileMetadata: arch.FileMetadata{
CompressedSize: secondV1any.size,
MD5: secondV1any.md5,
SHA256: secondV1any.sha256,
Arch: "any",
},
},
})
V1i686database = BuildArchDatabase([]arch.Package{
{
Name: "first",
Version: "1-1",
VersionMetadata: arch.VersionMetadata{
Base: "first",
},
FileMetadata: arch.FileMetadata{
CompressedSize: firstV1i686.size,
MD5: firstV1i686.md5,
SHA256: firstV1i686.sha256,
Arch: "i686",
},
},
{
Name: "second",
Version: "1-1",
VersionMetadata: arch.VersionMetadata{
Base: "second",
},
FileMetadata: arch.FileMetadata{
CompressedSize: secondV1any.size,
MD5: secondV1any.md5,
SHA256: secondV1any.sha256,
Arch: "any",
},
},
})
V2x86_64database = BuildArchDatabase([]arch.Package{
{
Name: "first",
Version: "2-1",
VersionMetadata: arch.VersionMetadata{
Base: "first",
},
FileMetadata: arch.FileMetadata{
CompressedSize: firstV2x86_64.size,
MD5: firstV2x86_64.md5,
SHA256: firstV2x86_64.sha256,
Arch: "i686",
},
},
{
Name: "second",
Version: "2-1",
VersionMetadata: arch.VersionMetadata{
Base: "second",
},
FileMetadata: arch.FileMetadata{
CompressedSize: secondV2any.size,
MD5: secondV2any.md5,
SHA256: secondV2any.sha256,
Arch: "any",
},
},
})
V2i686database = BuildArchDatabase([]arch.Package{
{
Name: "first",
Version: "1-1",
VersionMetadata: arch.VersionMetadata{
Base: "first",
},
FileMetadata: arch.FileMetadata{
CompressedSize: firstV1i686.size,
MD5: firstV1i686.md5,
SHA256: firstV1i686.sha256,
Arch: "i686",
},
},
{
Name: "second",
Version: "2-1",
VersionMetadata: arch.VersionMetadata{
Base: "second",
},
FileMetadata: arch.FileMetadata{
CompressedSize: secondV2any.size,
MD5: secondV2any.md5,
SHA256: secondV2any.sha256,
Arch: "any",
},
},
})
user = unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
rootURL = fmt.Sprintf("/api/packages/%s/arch", user.Name)
)
t.Run("Version_1", func(t *testing.T) {
t.Run("Push_first_x86_64", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequestWithBody(t, "PUT",
path.Join(
rootURL, "push", "first-1-1-x86_64.pkg.tar.zst",
"archlinux", hex.EncodeToString(firstSign),
),
bytes.NewReader(firstV1x86_64.data),
)
req = AddBasicAuthHeader(req, user.Name)
MakeRequest(t, req, http.StatusOK)
})
t.Run("Push_first_i686", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequestWithBody(t, "PUT",
path.Join(
rootURL, "push", "first-1-1-i686.pkg.tar.zst",
"archlinux", hex.EncodeToString(secondSign),
),
bytes.NewReader(firstV1i686.data),
)
req = AddBasicAuthHeader(req, user.Name)
MakeRequest(t, req, http.StatusOK)
})
t.Run("Push_second_any", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequestWithBody(t, "PUT",
path.Join(rootURL, "push", "second-1-1-any.pkg.tar.zst", "archlinux"),
bytes.NewReader(secondV1any.data),
)
req = AddBasicAuthHeader(req, user.Name)
MakeRequest(t, req, http.StatusOK)
})
t.Run("Get_first_x86_64_package", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequest(t, "GET",
rootURL+"/archlinux/x86_64/first-1-1-x86_64.pkg.tar.zst",
)
resp := MakeRequest(t, req, http.StatusOK)
assert.Equal(t, firstV1x86_64.data, resp.Body.Bytes())
})
t.Run("Get_first_i686_package", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequest(t, "GET",
rootURL+"/archlinux/x86_64/first-1-1-i686.pkg.tar.zst",
)
resp := MakeRequest(t, req, http.StatusOK)
assert.Equal(t, firstV1i686.data, resp.Body.Bytes())
})
t.Run("Get_first_x86_64_package_signature", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequest(t, "GET",
rootURL+"/archlinux/x86_64/first-1-1-x86_64.pkg.tar.zst.sig",
)
resp := MakeRequest(t, req, http.StatusOK)
assert.Equal(t, firstSign, resp.Body.Bytes())
})
t.Run("Get_first_i686_package_signature", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequest(t, "GET",
rootURL+"/archlinux/x86_64/first-1-1-i686.pkg.tar.zst.sig",
)
resp := MakeRequest(t, req, http.StatusOK)
assert.Equal(t, secondSign, resp.Body.Bytes())
})
t.Run("Get_any_package_from_x86_64_group", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequest(t, "GET",
rootURL+"/archlinux/x86_64/second-1-1-any.pkg.tar.zst",
)
resp := MakeRequest(t, req, http.StatusOK)
assert.Equal(t, secondV1any.data, resp.Body.Bytes())
})
t.Run("Get_any_package_from_i686_group", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequest(t, "GET",
rootURL+"/archlinux/i686/second-1-1-any.pkg.tar.zst",
)
resp := MakeRequest(t, req, http.StatusOK)
assert.Equal(t, secondV1any.data, resp.Body.Bytes())
})
t.Run("Get_x86_64_pacman_database", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequest(t, "GET",
rootURL+"/archlinux/x86_64/user.db.tar.gz",
)
resp := MakeRequest(t, req, http.StatusOK)
assert.Equal(t, V1x86_64database, resp.Body.Bytes())
})
t.Run("Get_i686_pacman_database", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
2023-07-06 20:44:09 +00:00
req := NewRequest(t, "GET",
rootURL+"/archlinux/i686/user.db.tar.gz",
)
resp := MakeRequest(t, req, http.StatusOK)
assert.Equal(t, V1i686database, resp.Body.Bytes())
})
})
t.Run("Version_2", func(t *testing.T) {
t.Run("Push_first_x86_64", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequestWithBody(t, "PUT",
path.Join(rootURL, "push", "first-2-1-x86_64.pkg.tar.zst", "archlinux"),
bytes.NewReader(firstV2x86_64.data),
)
req = AddBasicAuthHeader(req, user.Name)
MakeRequest(t, req, http.StatusOK)
})
2023-07-06 20:44:09 +00:00
t.Run("Push_second_any", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequestWithBody(t, "PUT",
path.Join(rootURL, "push", "first-2-1-x86_64.pkg.tar.zst", "archlinux"),
bytes.NewReader(firstV1x86_64.data),
)
req = AddBasicAuthHeader(req, user.Name)
MakeRequest(t, req, http.StatusOK)
})
t.Run("Get_x86_64_pacman_database", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequest(t, "GET",
rootURL+"/archlinux/x86_64/user2.db.tar.gz",
)
resp := MakeRequest(t, req, http.StatusOK)
assert.Equal(t, V2x86_64database, resp.Body.Bytes())
})
t.Run("Get_i686_pacman_database", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequest(t, "GET",
rootURL+"/archlinux/i686/user2.db.tar.gz",
)
resp := MakeRequest(t, req, http.StatusOK)
assert.Equal(t, V2i686database, resp.Body.Bytes())
})
})
}
type testArchPackage struct {
data []byte
md5 string
sha256 string
size int64
}
2023-07-06 20:44:09 +00:00
func BuildArchPackage(t *testing.T, name, ver, arch string) testArchPackage {
fs := fstest.MapFS{
"pkginfo": &fstest.MapFile{
Data: []byte(fmt.Sprintf(
"pkgname = %s\npkgbase = %s\npkgver = %s\narch = %s\n",
name, name, ver, arch,
)),
Mode: os.ModePerm,
ModTime: time.Now(),
},
"mtree": &fstest.MapFile{
Data: []byte("test"),
Mode: os.ModePerm,
ModTime: time.Now(),
},
}
pinf, err := fs.Stat("pkginfo")
assert.NoError(t, err)
pfile, err := fs.Open("pkginfo")
assert.NoError(t, err)
parcname, err := archiver.NameInArchive(pinf, ".PKGINFO", ".PKGINFO")
assert.NoError(t, err)
minf, err := fs.Stat("mtree")
assert.NoError(t, err)
mfile, err := fs.Open("mtree")
assert.NoError(t, err)
marcname, err := archiver.NameInArchive(minf, ".MTREE", ".MTREE")
assert.NoError(t, err)
var buf bytes.Buffer
archive := archiver.NewTarZstd()
archive.Create(&buf)
err = archive.Write(archiver.File{
FileInfo: archiver.FileInfo{
FileInfo: pinf,
CustomName: parcname,
},
ReadCloser: pfile,
})
assert.NoError(t, errors.Join(pfile.Close(), err))
err = archive.Write(archiver.File{
FileInfo: archiver.FileInfo{
FileInfo: minf,
CustomName: marcname,
},
ReadCloser: mfile,
})
assert.NoError(t, errors.Join(mfile.Close(), archive.Close(), err))
md5, sha256, size := archPkgParams(buf.Bytes())
return testArchPackage{
data: buf.Bytes(),
md5: hex.EncodeToString(md5),
sha256: hex.EncodeToString(sha256),
size: size,
}
}
func archPkgParams(b []byte) ([]byte, []byte, int64) {
md5 := md5.New()
sha256 := sha256.New()
c := counter{bytes.NewReader(b), 0}
br := bufio.NewReader(io.TeeReader(&c, io.MultiWriter(md5, sha256)))
io.ReadAll(br)
return md5.Sum(nil), sha256.Sum(nil), int64(c.n)
}
type counter struct {
io.Reader
n int
}
func (w *counter) Read(p []byte) (int, error) {
n, err := w.Reader.Read(p)
w.n += n
return n, err
}
func BuildArchDatabase(pkgs []arch.Package) []byte {
entries := map[string][]byte{}
for _, p := range pkgs {
entries[fmt.Sprintf("%s-%s/desc", p.Name, p.Version)] = []byte(p.Desc())
}
b, err := arch.CreatePacmanDb(entries)
if err != nil {
panic(err)
}
return b.Bytes()
}