0
Fork 0
mirror of https://codeberg.org/forgejo/forgejo.git synced 2025-02-22 22:06:21 -05:00
forgejo/tests/integration/api_packages_alt_test.go
Alex619829 7ae5376573 Alt Linux Apt-Rpm repository support for Forgejo packages. (#6351)
Co-authored-by: Aleksandr Gamzin alexgamz1119@gmail.com

Adds support for the Apt-Rpm registry of the Alt Lunux distribution.

Alt Linux uses RPM packages to store and distribute software to its users. But the logic of the Alt Linux package registry is different from the Red Hat package registry.
I have added support for the Alt Linux package registry.

## Checklist

The [contributor guide](https://forgejo.org/docs/next/contributor/) contains information that will be helpful to first time contributors. There also are a few [conditions for merging Pull Requests in Forgejo repositories](https://codeberg.org/forgejo/governance/src/branch/main/PullRequestsAgreement.md). You are also welcome to join the [Forgejo development chatroom](https://matrix.to/#/#forgejo-development:matrix.org).

### Tests

- I added test coverage for Go changes...
  - [ ] in their respective `*_test.go` for unit tests.
  - [x] in the `tests/integration` directory if it involves interactions with a live Forgejo server.
- I added test coverage for JavaScript changes...
  - [ ] in `web_src/js/*.test.js` if it can be unit tested.
  - [ ] in `tests/e2e/*.test.e2e.js` if it requires interactions with a live Forgejo server (see also the [developer guide for JavaScript testing](https://codeberg.org/forgejo/forgejo/src/branch/forgejo/tests/e2e/README.md#end-to-end-tests)).

### Documentation

- [x] I created a pull request [to the documentation](https://codeberg.org/forgejo/docs) to explain to Forgejo users how to use this change.
- [ ] I did not document these changes and I do not expect someone else to do it.

### Release notes

- [ ] I do not want this change to show in the release notes.
- [x] I want the title to show in the release notes with a link to this pull request.
- [ ] I want the content of the `release-notes/<pull request number>.md` to be be used for the release notes instead of the title.

Co-authored-by: Aleksandr Gamzin <gamzin@altlinux.org>
Reviewed-on: https://codeberg.org/forgejo/forgejo/pulls/6351
Reviewed-by: Earl Warren <earl-warren@noreply.codeberg.org>
Co-authored-by: Alex619829 <alex619829@noreply.codeberg.org>
Co-committed-by: Alex619829 <alex619829@noreply.codeberg.org>
2025-01-22 14:01:49 +00:00

658 lines
23 KiB
Go

// Copyright 2023 The Gitea Authors. All rights reserved.
// SPDX-License-Identifier: MIT
package integration
import (
"bytes"
"compress/gzip"
"crypto/sha256"
"encoding/base64"
"encoding/binary"
"fmt"
"io"
"net/http"
"strconv"
"strings"
"testing"
"code.gitea.io/gitea/models/db"
"code.gitea.io/gitea/models/packages"
"code.gitea.io/gitea/models/unittest"
user_model "code.gitea.io/gitea/models/user"
packages_module "code.gitea.io/gitea/modules/packages"
rpm_module "code.gitea.io/gitea/modules/packages/rpm"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/util"
"code.gitea.io/gitea/tests"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/require"
"github.com/ulikunitz/xz"
)
func TestPackageAlt(t *testing.T) {
defer tests.PrepareTestEnv(t)()
packageName := "gitea-test"
packageVersion := "1.0.2-1"
packageArchitecture := "x86_64"
user := unittest.AssertExistsAndLoadBean(t, &user_model.User{ID: 2})
base64RpmPackageContent := `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`
rpmPackageContent, err := base64.StdEncoding.DecodeString(base64RpmPackageContent)
require.NoError(t, err)
zr, err := gzip.NewReader(bytes.NewReader(rpmPackageContent))
require.NoError(t, err)
content, err := io.ReadAll(zr)
require.NoError(t, err)
rootURL := fmt.Sprintf("/api/packages/%s/alt", user.Name)
for _, group := range []string{"", "el9", "el9/stable"} {
t.Run(fmt.Sprintf("[Group:%s]", group), func(t *testing.T) {
var groupParts []string
uploadURL := rootURL
if group != "" {
groupParts = strings.Split(group, "/")
uploadURL = strings.Join(append([]string{rootURL}, groupParts...), "/")
} else {
groupParts = strings.Split("alt", "/")
}
groupURL := strings.Join(append([]string{rootURL}, groupParts...), "/")
t.Run("RepositoryConfig", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequest(t, "GET", groupURL+".repo")
resp := MakeRequest(t, req, http.StatusOK)
expected := fmt.Sprintf(`[gitea-%s]
name=%s
baseurl=%s
enabled=1`,
strings.Join(append([]string{user.LowerName}, groupParts...), "-"),
strings.Join(append([]string{user.Name, setting.AppName}, groupParts...), " - "),
util.URLJoin(setting.AppURL, groupURL),
)
assert.Equal(t, expected, resp.Body.String())
})
t.Run("Upload", func(t *testing.T) {
url := uploadURL + "/upload"
req := NewRequestWithBody(t, "PUT", url, bytes.NewReader(content))
MakeRequest(t, req, http.StatusUnauthorized)
req = NewRequestWithBody(t, "PUT", url, bytes.NewReader(content)).
AddBasicAuth(user.Name)
MakeRequest(t, req, http.StatusCreated)
pvs, err := packages.GetVersionsByPackageType(db.DefaultContext, user.ID, packages.TypeAlt)
require.NoError(t, err)
assert.Len(t, pvs, 1)
pd, err := packages.GetPackageDescriptor(db.DefaultContext, pvs[0])
require.NoError(t, err)
assert.Nil(t, pd.SemVer)
assert.IsType(t, &rpm_module.VersionMetadata{}, pd.Metadata)
assert.Equal(t, packageName, pd.Package.Name)
assert.Equal(t, packageVersion, pd.Version.Version)
pfs, err := packages.GetFilesByVersionID(db.DefaultContext, pvs[0].ID)
require.NoError(t, err)
assert.Len(t, pfs, 1)
assert.Equal(t, fmt.Sprintf("%s-%s.%s.rpm", packageName, packageVersion, packageArchitecture), pfs[0].Name)
assert.True(t, pfs[0].IsLead)
pb, err := packages.GetBlobByID(db.DefaultContext, pfs[0].BlobID)
require.NoError(t, err)
assert.Equal(t, int64(len(content)), pb.Size)
req = NewRequestWithBody(t, "PUT", url, bytes.NewReader(content)).
AddBasicAuth(user.Name)
MakeRequest(t, req, http.StatusConflict)
})
t.Run("Download", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequest(t, "GET", fmt.Sprintf("%s.repo/%s/RPMS.classic/%s-%s.%s.rpm", groupURL, packageArchitecture, packageName, packageVersion, packageArchitecture))
resp := MakeRequest(t, req, http.StatusOK)
assert.Equal(t, content, resp.Body.Bytes())
})
t.Run("Repository", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
url := fmt.Sprintf("%s.repo/%s/base", groupURL, packageArchitecture)
req := NewRequest(t, "HEAD", url+"/dummy.xml")
MakeRequest(t, req, http.StatusNotFound)
req = NewRequest(t, "GET", url+"/dummy.xml")
MakeRequest(t, req, http.StatusNotFound)
t.Run("release.classic", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req = NewRequest(t, "HEAD", url+"/release.classic")
MakeRequest(t, req, http.StatusOK)
req = NewRequest(t, "GET", url+"/release.classic")
resp := MakeRequest(t, req, http.StatusOK).Body.String()
type ReleaseClassic struct {
Archive string
Component string
Origin string
Label string
Architecture string
NotAutomatic bool
}
var result ReleaseClassic
lines := strings.Split(resp, "\n")
for _, line := range lines {
parts := strings.SplitN(line, ": ", 2)
if len(parts) < 2 {
continue
}
switch parts[0] {
case "Archive":
result.Archive = parts[1]
case "Component":
result.Component = parts[1]
case "Origin":
result.Origin = parts[1]
case "Label":
result.Label = parts[1]
case "Architecture":
result.Architecture = parts[1]
case "NotAutomatic":
notAuto, err := strconv.ParseBool(parts[1])
if err != nil {
require.NoError(t, err)
}
result.NotAutomatic = notAuto
}
}
assert.Equal(t, "classic", result.Component)
assert.Equal(t, "Alt Linux Team", result.Origin)
assert.Equal(t, "Forgejo", result.Label)
assert.Equal(t, "x86_64", result.Architecture)
assert.False(t, result.NotAutomatic)
assert.NotEmpty(t, result.Archive)
})
t.Run("release", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req = NewRequest(t, "HEAD", url+"/release")
MakeRequest(t, req, http.StatusOK)
req = NewRequest(t, "GET", url+"/release")
resp := MakeRequest(t, req, http.StatusOK).Body.String()
type Checksum struct {
Hash string
Size int
File string
}
type Release struct {
Origin string
Label string
Suite string
Architectures string
MD5Sum []Checksum
BLAKE2B []Checksum
}
var result Release
lines := strings.Split(resp, "\n")
var isMD5Sum, isBLAKE2b bool
for _, line := range lines {
line = strings.TrimSpace(line)
if line == "" {
continue
}
switch {
case strings.HasPrefix(line, "Origin:"):
result.Origin = strings.TrimSpace(strings.TrimPrefix(line, "Origin:"))
case strings.HasPrefix(line, "Label:"):
result.Label = strings.TrimSpace(strings.TrimPrefix(line, "Label:"))
case strings.HasPrefix(line, "Suite:"):
result.Suite = strings.TrimSpace(strings.TrimPrefix(line, "Suite:"))
case strings.HasPrefix(line, "Architectures:"):
result.Architectures = strings.TrimSpace(strings.TrimPrefix(line, "Architectures:"))
case line == "MD5Sum:":
isMD5Sum = true
isBLAKE2b = false
case line == "BLAKE2b:":
isBLAKE2b = true
isMD5Sum = false
case isMD5Sum || isBLAKE2b:
parts := strings.Fields(line)
if len(parts) >= 3 {
hash := parts[0]
size, err := strconv.Atoi(parts[1])
if err != nil {
continue
}
file := parts[2]
checksum := Checksum{
Hash: hash,
Size: size,
File: file,
}
if isMD5Sum {
result.MD5Sum = append(result.MD5Sum, checksum)
} else if isBLAKE2b {
result.BLAKE2B = append(result.BLAKE2B, checksum)
}
}
}
}
assert.Equal(t, "Forgejo", result.Origin)
assert.Equal(t, "Forgejo", result.Label)
assert.Equal(t, "Sisyphus", result.Suite)
assert.Equal(t, "x86_64", result.Architectures)
assert.Len(t, result.MD5Sum, 3)
assert.Equal(t, "bbf7ae6b2f540673ed1cfc0266b5f319", result.MD5Sum[0].Hash)
assert.Equal(t, 1003, result.MD5Sum[0].Size)
assert.Equal(t, "base/pkglist.classic", result.MD5Sum[0].File)
assert.Len(t, result.BLAKE2B, 3)
assert.Equal(t, "b527bf038895ce29107ec3a6d2eebd7c365e8ce5ab767276eeddd7c549a159025225cb0ecfdbf7b71da13db7e865e77bcb0e2dae4d21335df01a4a17e0056a70", result.BLAKE2B[0].Hash)
assert.Equal(t, 1003, result.BLAKE2B[0].Size)
assert.Equal(t, "base/pkglist.classic", result.BLAKE2B[0].File)
})
t.Run("pkglist.classic", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req = NewRequest(t, "GET", url+"/pkglist.classic")
resp := MakeRequest(t, req, http.StatusOK)
body := resp.Body
defer body.Reset()
type RpmHeader struct {
Magic [8]byte
Nindex uint32
Hsize uint32
}
type RpmHdrIndex struct {
Tag uint32
Type uint32
Offset uint32
Count uint32
}
type Metadata struct {
Name string
Version string
Release string
Summary []string
Description []string
BuildTime int
Size int
License string
Packager string
Group []string
URL string
Arch string
SourceRpm string
ProvideNames []string
RequireFlags []int
RequireNames []string
RequireVersions []string
ChangeLogTimes []int
ChangeLogNames []string
ChangeLogTexts []string
ProvideFlags []int
ProvideVersions []string
DirIndexes []int
BaseNames []string
DirNames []string
DistTag string
AptIndexLegacyFileName string
AptIndexLegacyFileSize int
MD5Sum string
BLAKE2B string
AptIndexLegacyDirectory string
}
var result Metadata
const rpmHeaderMagic = "\x8e\xad\xe8\x01\x00\x00\x00\x00"
var hdr RpmHeader
for {
if err := binary.Read(body, binary.BigEndian, &hdr); err != nil {
if err == io.EOF {
break
}
require.NoError(t, err)
}
if !bytes.Equal(hdr.Magic[:], []byte(rpmHeaderMagic)) {
require.NoError(t, err)
}
nindex := hdr.Nindex
index := make([]RpmHdrIndex, nindex)
if err := binary.Read(body, binary.BigEndian, &index); err != nil {
require.NoError(t, err)
}
data := make([]byte, hdr.Hsize)
if err := binary.Read(body, binary.BigEndian, &data); err != nil {
require.NoError(t, err)
}
var indexPtrs []*RpmHdrIndex
for i := range index {
indexPtrs = append(indexPtrs, &index[i])
}
for _, idx := range indexPtrs {
tag := binary.BigEndian.Uint32([]byte{byte(idx.Tag >> 24), byte(idx.Tag >> 16), byte(idx.Tag >> 8), byte(idx.Tag)})
typ := binary.BigEndian.Uint32([]byte{byte(idx.Type >> 24), byte(idx.Type >> 16), byte(idx.Type >> 8), byte(idx.Type)})
offset := binary.BigEndian.Uint32([]byte{byte(idx.Offset >> 24), byte(idx.Offset >> 16), byte(idx.Offset >> 8), byte(idx.Offset)})
count := binary.BigEndian.Uint32([]byte{byte(idx.Count >> 24), byte(idx.Count >> 16), byte(idx.Count >> 8), byte(idx.Count)})
if typ == 6 || typ == 8 || typ == 9 {
elem := data[offset:]
for j := uint32(0); j < count; j++ {
strEnd := bytes.IndexByte(elem, 0)
if strEnd == -1 {
require.NoError(t, err)
}
switch tag {
case 1000:
result.Name = string(elem[:strEnd])
case 1001:
result.Version = string(elem[:strEnd])
case 1002:
result.Release = string(elem[:strEnd])
case 1004:
var summaries []string
for i := uint32(0); i < count; i++ {
summaries = append(summaries, string(elem[:strEnd]))
}
result.Summary = summaries
case 1005:
var descriptions []string
for i := uint32(0); i < count; i++ {
descriptions = append(descriptions, string(elem[:strEnd]))
}
result.Description = descriptions
case 1014:
result.License = string(elem[:strEnd])
case 1015:
result.Packager = string(elem[:strEnd])
case 1016:
var groups []string
for i := uint32(0); i < count; i++ {
groups = append(groups, string(elem[:strEnd]))
}
result.Group = groups
case 1020:
result.URL = string(elem[:strEnd])
case 1022:
result.Arch = string(elem[:strEnd])
case 1044:
result.SourceRpm = string(elem[:strEnd])
case 1047:
var provideNames []string
for i := uint32(0); i < count; i++ {
provideNames = append(provideNames, string(elem[:strEnd]))
}
result.ProvideNames = provideNames
case 1049:
var requireNames []string
for i := uint32(0); i < count; i++ {
requireNames = append(requireNames, string(elem[:strEnd]))
}
result.RequireNames = requireNames
case 1050:
var requireVersions []string
for i := uint32(0); i < count; i++ {
requireVersions = append(requireVersions, string(elem[:strEnd]))
}
result.RequireVersions = requireVersions
case 1081:
var changeLogNames []string
for i := uint32(0); i < count; i++ {
changeLogNames = append(changeLogNames, string(elem[:strEnd]))
}
result.ChangeLogNames = changeLogNames
case 1082:
var changeLogTexts []string
for i := uint32(0); i < count; i++ {
changeLogTexts = append(changeLogTexts, string(elem[:strEnd]))
}
result.ChangeLogTexts = changeLogTexts
case 1113:
var provideVersions []string
for i := uint32(0); i < count; i++ {
provideVersions = append(provideVersions, string(elem[:strEnd]))
}
result.ProvideVersions = provideVersions
case 1117:
var baseNames []string
for i := uint32(0); i < count; i++ {
baseNames = append(baseNames, string(elem[:strEnd]))
}
result.BaseNames = baseNames
case 1118:
var dirNames []string
for i := uint32(0); i < count; i++ {
dirNames = append(dirNames, string(elem[:strEnd]))
}
result.DirNames = dirNames
case 1155:
result.DistTag = string(elem[:strEnd])
case 1000000:
result.AptIndexLegacyFileName = string(elem[:strEnd])
case 1000005:
result.MD5Sum = string(elem[:strEnd])
case 1000009:
result.BLAKE2B = string(elem[:strEnd])
case 1000010:
result.AptIndexLegacyDirectory = string(elem[:strEnd])
}
elem = elem[strEnd+1:]
}
} else if typ == 4 {
elem := data[offset:]
for j := uint32(0); j < count; j++ {
val := binary.BigEndian.Uint32(elem)
switch tag {
case 1006:
result.BuildTime = int(val)
case 1009:
result.Size = int(val)
case 1048:
var requireFlags []int
for i := uint32(0); i < count; i++ {
requireFlags = append(requireFlags, int(val))
}
result.RequireFlags = requireFlags
case 1080:
var changeLogTimes []int
for i := uint32(0); i < count; i++ {
changeLogTimes = append(changeLogTimes, int(val))
}
result.ChangeLogTimes = changeLogTimes
case 1112:
var provideFlags []int
for i := uint32(0); i < count; i++ {
provideFlags = append(provideFlags, int(val))
}
result.ProvideFlags = provideFlags
case 1116:
var dirIndexes []int
for i := uint32(0); i < count; i++ {
dirIndexes = append(dirIndexes, int(val))
}
result.DirIndexes = dirIndexes
case 1000001:
result.AptIndexLegacyFileSize = int(val)
}
elem = elem[4:]
}
} else {
require.NoError(t, err)
}
}
}
assert.Equal(t, "gitea-test", result.Name)
assert.Equal(t, "1.0.2", result.Version)
assert.Equal(t, "1", result.Release)
assert.Equal(t, []string{"RPM package summary"}, result.Summary)
assert.Equal(t, []string{"RPM package description"}, result.Description)
assert.Equal(t, 1678225964, result.BuildTime)
assert.Equal(t, 13, result.Size)
assert.Equal(t, "MIT", result.License)
assert.Equal(t, "KN4CK3R", result.Packager)
assert.Equal(t, []string{"System"}, result.Group)
assert.Equal(t, "https://gitea.io", result.URL)
assert.Equal(t, "x86_64", result.Arch)
assert.Equal(t, "gitea-test-1.0.2-1.src.rpm", result.SourceRpm)
assert.Equal(t, []string{"", ""}, result.ProvideNames)
assert.Equal(t, []int{16777226, 16777226, 16777226, 16777226, 16777226, 16777226, 16777226}, result.RequireFlags)
assert.Equal(t, []string{"", "", "", "", "", "", ""}, result.RequireNames)
assert.Equal(t, []string{"5.2-1", "5.2-1", "5.2-1", "5.2-1", "5.2-1", "5.2-1", "5.2-1"}, result.RequireVersions)
assert.Equal(t, []int{1678276800}, result.ChangeLogTimes)
assert.Equal(t, []string{"KN4CK3R <dummy@gitea.io>"}, result.ChangeLogNames)
assert.Equal(t, []string{"- Changelog message."}, result.ChangeLogTexts)
assert.Equal(t, []int{8, 8}, result.ProvideFlags)
assert.Equal(t, []string{"1.0.2-1", "1.0.2-1"}, result.ProvideVersions)
assert.Equal(t, []int(nil), result.DirIndexes)
assert.Equal(t, []string{"hello"}, result.BaseNames)
assert.Equal(t, []string{"/usr/local/bin/"}, result.DirNames)
assert.Equal(t, "", result.DistTag)
assert.Equal(t, "gitea-test-1.0.2-1.x86_64.rpm", result.AptIndexLegacyFileName)
assert.Equal(t, 7116, result.AptIndexLegacyFileSize)
assert.Equal(t, "9ea82dd62968719aea19c08cd2ced79a", result.MD5Sum)
assert.Equal(t, "8ba7f1f52a47b23997aa2de21b305cc71974d51f0c54fb53cb927156284dafdcc233d514a46c020e4a0666e218529e0284933c5873d24c2555830d7627140f7d", result.BLAKE2B)
assert.Equal(t, "RPMS.classic", result.AptIndexLegacyDirectory)
})
t.Run("pkglist.classic.xz", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequest(t, "GET", url+"/pkglist.classic.xz")
pkglistXZResp := MakeRequest(t, req, http.StatusOK)
pkglistXZ := pkglistXZResp.Body
defer pkglistXZ.Reset()
req2 := NewRequest(t, "GET", url+"/pkglist.classic")
pkglistResp := MakeRequest(t, req2, http.StatusOK)
pkglist := pkglistResp.Body
defer pkglist.Reset()
assert.Less(t, pkglistXZ.Len(), pkglist.Len())
xzReader, err := xz.NewReader(pkglistXZ)
require.NoError(t, err)
var unxzData bytes.Buffer
_, err = io.Copy(&unxzData, xzReader)
require.NoError(t, err)
assert.Equal(t, unxzData.Len(), pkglist.Len())
content, _ := packages_module.NewHashedBuffer()
defer content.Close()
h := sha256.New()
w := io.MultiWriter(content, h)
_, err = io.Copy(w, pkglist)
require.NoError(t, err)
hashMD5Classic, _, hashSHA256Classic, _, hashBlake2bClassic := content.Sums()
contentUnxz, _ := packages_module.NewHashedBuffer()
defer contentUnxz.Close()
_, err = io.Copy(io.MultiWriter(contentUnxz, sha256.New()), &unxzData)
require.NoError(t, err)
hashMD5Unxz, _, hashSHA256Unxz, _, hashBlake2bUnxz := contentUnxz.Sums()
assert.Equal(t, fmt.Sprintf("%x", hashSHA256Classic), fmt.Sprintf("%x", hashSHA256Unxz))
assert.Equal(t, fmt.Sprintf("%x", hashBlake2bClassic), fmt.Sprintf("%x", hashBlake2bUnxz))
assert.Equal(t, fmt.Sprintf("%x", hashMD5Classic), fmt.Sprintf("%x", hashMD5Unxz))
})
})
t.Run("Delete", func(t *testing.T) {
defer tests.PrintCurrentTest(t)()
req := NewRequest(t, "DELETE", fmt.Sprintf("%s.repo/%s/RPMS.classic/%s-%s.%s.rpm", groupURL, packageArchitecture, packageName, packageVersion, packageArchitecture))
MakeRequest(t, req, http.StatusUnauthorized)
req = NewRequest(t, "DELETE", fmt.Sprintf("%s.repo/%s/RPMS.classic/%s-%s.%s.rpm", groupURL, packageArchitecture, packageName, packageVersion, packageArchitecture)).
AddBasicAuth(user.Name)
MakeRequest(t, req, http.StatusNoContent)
pvs, err := packages.GetVersionsByPackageType(db.DefaultContext, user.ID, packages.TypeAlt)
require.NoError(t, err)
assert.Empty(t, pvs)
req = NewRequest(t, "DELETE", fmt.Sprintf("%s.repo/%s/RPMS.classic/%s-%s.%s.rpm", groupURL, packageArchitecture, packageName, packageVersion, packageArchitecture)).
AddBasicAuth(user.Name)
MakeRequest(t, req, http.StatusNotFound)
})
})
}
}