2023-05-12 19:27:50 +02:00
// Copyright 2023 The Gitea Authors. All rights reserved.
// SPDX-License-Identifier: MIT
package integration
import (
"archive/tar"
"bufio"
"bytes"
"compress/gzip"
"encoding/base64"
"fmt"
"io"
"net/http"
"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"
alpine_module "code.gitea.io/gitea/modules/packages/alpine"
2024-02-25 16:10:55 +01:00
alpine_service "code.gitea.io/gitea/services/packages/alpine"
2023-05-12 19:27:50 +02:00
"code.gitea.io/gitea/tests"
"github.com/stretchr/testify/assert"
)
func TestPackageAlpine ( t * testing . T ) {
defer tests . PrepareTestEnv ( t ) ( )
user := unittest . AssertExistsAndLoadBean ( t , & user_model . User { ID : 2 } )
packageName := "gitea-test"
packageVersion := "1.4.1-r3"
base64AlpinePackageContent := ` H4sIAAAAAAACA9ML9nT30wsKdtTLzjNJzjYuckjPLElN1DUzMUxMNTa11CsqTtQrKE1ioAAYAIGZ
iQmYBgJ02hDENjQxMTAzMzQ1MTVjMDA0MTQ1ZlAwYKADKC0uSSxSUGAYoWDm4sZZtypv75 + q2fVT
POD1bKkFB22ms + g1z + H4dk7AhC3HwUSj9EbT0Rk3Dn55dHxy / K7Q + Nl / i + L7Z036ypcRvvpZuMiN
s7wbZL / klqRGGshv9Gi0qHTgTZfw3HytnJdx9c3NTRp / PHn + Z50uq2pjkilzjtpfd + uzQMw1M7cY
i9RXJasnT2M + vDXCesLK7MilJt8sGplj4xUlLMUun9SzY + phFpxWxRXa06AseV9WvzH3jtGGoL5A
vQkea + VKPj5R + Cb461tIk97qpa9nJYsJujTNl2B / J1P52H / D2rPr / j19uU8p7cMSq5tmXk51ReXl
F / Yddr9XsMpEwFKlXSPo3QSGwnCOG8y2uadjm6ui998WYXNYubjg78N3a7bnXjhrl5fB8voI ++ LI
1 FP5W44e2xf4Ou2wrtyic1Onz7MzMV5ksuno2V / LVG4eN / 15 X / n2 / 2 vJ2VV + T68aT327dOrhd6e6
q5Y0V82Y83tdqkFa8TW2BvGCZ0ds / iibHVpzKuPcuSULO63 / bNmfrnhjWqXzhMSXTb5Cv4vPaxSL
8 LFMdqmxbN7 + Y + Yi0ZyZhz4UxexLuHHFd1VFvk + kwvniq3P + f9rh52InWnL8Lpvedcecoh1GFSc5
xZ9VBGex2V269HZfwxSVCvP35wQfi2xKX + lYMXtF48n1R65O2PLWpm69RdESMa79dlrTGazsZacu
MbMLeSSScPORZde76 / MBV6SFJAAEAAAfiwgAAAAAAAID7VRLaxsxEN6zfoUgZ ++ OVq + 1 aUIhUDeY
pKa49FhmJdkW3ofRysXpr69220t9SCk0gZJ + IGaY56eBmbxY4 / m9Q + vCUOTr1fLu4d2H7O8CEpQQ
k0y4lAClypgQoBSTQqoMGBMgMnrOXgCnIWJIVLLXCcaoib5110CSij / V7D9eCZ5p5f9o / 5 VkF / tf
MqUzCi + 5 / 6 Hv41Nxv / Nffu4fwRVdus4FjM7S + pFiffKNpTxnkMMsALmin5PnHgMtS8rkgvGFBPpp
c0tLKDk5HnYdto5e052PDmfRDXE0fnUh2VgucjYLU5h1g0mm5RhGNymMrtEccOfIKTTJsY / xOCyK
YqqT + 74 gExWbmI2VlJ6LeQUcyPFH2lh / 9 SBuV / wjfXPohDnw8HZKviGD / zYmCZgrgsHsk36u1Bcl
SB / 8 zne / 0 jV92 / qYbKRF38X0niiemN2QxhvXDWOL + 7 tNGhGeYt + m22mwaR6pddGZNM8FSeRxj8PY
X7PaqdqAVlqWXHKnmQGmK43VlqNlILRilbBSMI2jV5Vbu5XGSVsDyGc7yd8B / gK2qgAIAAAfiwgA
AAAAAAID7dNNSgMxGAbg7MSCOxcu5wJOv0x + OlkU7K5QoYXqVsxMMihlKMwP1Fu48QQewCN4DfEQ
egUz4sYuFKEtFN9n870hWSSQN + 7 P7GrsrfNV3Y9dW5Z3bNMo0FJ + zmB9EhcJ41KS1lxJpRnxbsWi
FduBtm5sFa7C / ifOo7y5Lf2QeiHar6jTaDSbnF5Mp + fzOL / x + aJuy3g + HvGhs8JY4b3yOpMZOZEo
lRW + MEoTTw3ZwqU0INNjsAe2VPk / 9 b / L3 / s / kIKzqOtk + IbJGTtmr + bx7WoxOUoun98frk / un14O
Djfa / 2 q5bH4699v ++ uMAAAAAAAAAAAAAAAAAAAAAAHbgA / eXQh8AKAAA `
content , err := base64 . StdEncoding . DecodeString ( base64AlpinePackageContent )
assert . NoError ( t , err )
2024-02-25 16:10:55 +01:00
base64AlpinePackageNoArchContent := ` H4sIAAAAAAACA9ML9nT30wsKdtQrLU4t0jUzTUo1NDVP0ysqTtQrKE1ioAYwAAIzExMwDQTotCGI
bWhiampuYmRiaGrMYGBoZGZkxKBgwEAHUFpcklikoMAwQkHLB7eoE40P9n5jvx32t7Dy9rq7x19k
66 cJPV38t / h + vWe2jdXy + / PzPT0YTF5z39i4cPFptcLa1C1lD0z / XvrNp6In / 7 nP4PPCF2pZu8uV
z74QXLxpY1XWJuVFysqVf + PdizccFbD6ZL / QPGXd1Ri1fec2XBNuYfK / rFa6wF / h3dK / W12f8mxP
04i P3aCy + vPx7h9S + 5 M1LLkWr5M / 4 ezGt3bDW / FjBp / S9hiKP72s / XrJ0vWtO0zr5wa + D / X8XluW
d7BLP7XS3YUhd8WbPPF / NW3691ONJbXsRb69O7BIMZC96uTri + utC / fbie5J + n7zhCxD4Aep / qet
QnlCZyN8MhNdVNlNl7965R1nExrrGvfI / YQZFx8Dg + d9122hZsYd / 24 WL / L69OWrDAN / y //nS7im
XEive3v7QeTe433TPj / X71 + 9 yHiV6 + E9k ++ 3 TL8V0Xoq9panhNt23fLgau / pTOvmKx6bV / pS26 + Y
5 UP4viyuklYeu4 / BZl6rLINe1L / uWuUXcH5z7pa2b9 + / rp / v / 8 dFgc1PL3bO3 / iVcrI //J/LMU2X
Nzu1IaMmWXnGp7CmyQIR39d0Nai9 / + tdPbfjvmsNH88Tu7uVrvNuJE0wjxfePXGv / KHNXD + mnG0t
yTPu + Na0b5WR9O4t0yMd9T5k6ui7hOyU / jL / 4 dOn6neLwhdrZIZfcl1ectnGvUTurWDo1vY5Gw9k
PTQLVgcA61F + 7 gAEAAAfiwgAAAAAAAID7VVNa9wwEPXZv2Ig53hHlizbCzkVkobQJtDkB4wl2SvW
lhdbTpP ++ oyXQGEPLYU2paTvIs3X05PQSNnmjp4 + OrJumjfZ3c3V9efL2 + T3AhlaqePIOB0Rc50I
VRSlypUoZIJCKJQJPCVvgGWONLGU5H1CCDDRD + 4 CU57S6zT5j3eCP9Tyv9T / GsuT / scyLxPAt + z /
aRzjj / J + Fv9HcQZXLriJorPQPAM1i + 8 tyEzkGZ5PmJ7BMvvQQUt7tx4BPPJH4ccAIpN5Jjj + hSJc
ugZAghDbArco4eH + A + SYq / Sw7wINDi6g89HReRhpMrvVzTzsFZlaV2Hbutmw4zVhmXo2djEe5u1m
c6zNzDikR3mW1a61JepaC0SZHsjsqTsyPoR9GL + GdPbf1iSFtU5Xyu / c4 + Q7H04lMfvgI3vT3hsX
5 rX40 / U9b5CWOA78Mhrq + 2 ewLjrDp7VNWQbtaF6ZXVWZIhdV09RWOIvU6BqNboSxLSEpkrpQq80x
W1Nla6NavuqtrJQ0sv17D + 4 L2oD1lwAIAAAfiwgAAAAAAAID7dM / SgNBFAbw6cSAnYXlXsDNm50 /
u1METBeIkEBMK87uzKKEJbB / IN7CxhN4AI / gNcRD6BWciI0WSiBGxO / XvA9mile8L + 5 P7WrkrfN1
04 9 dV1XXbNso0FK + zeDzJC4SxqVSqUwkV4IR51KkLFqxHeia1tZhFfY / cR4V7VXlB9QL0b5HnUXD
6 fj4bDI5ncXFpS8WTVfFs9GQD5wVxgrvlde5zMmJRKm89KVRmnhmyJYuo5RMj8Ef8EOV36j / 6 / yx
/ 5 qnxKJ1J8MZJifskD2Zu + fzxfggmT + 83 F4c3dw / 7 u1vtf / 1 ctl + 9e+7 dwAAAAAAAAAAAAAAAAAA
AACAX / AKARNTyAAoAAA = `
noarchContent , err := base64 . StdEncoding . DecodeString ( base64AlpinePackageNoArchContent )
assert . NoError ( t , err )
branches := [ ] string { "v3.16" , "v3.17" }
2023-05-12 19:27:50 +02:00
repositories := [ ] string { "main" , "testing" }
rootURL := fmt . Sprintf ( "/api/packages/%s/alpine" , user . Name )
t . Run ( "RepositoryKey" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
req := NewRequest ( t , "GET" , rootURL + "/key" )
resp := MakeRequest ( t , req , http . StatusOK )
assert . Equal ( t , "application/x-pem-file" , resp . Header ( ) . Get ( "Content-Type" ) )
assert . Contains ( t , resp . Body . String ( ) , "-----BEGIN PUBLIC KEY-----" )
} )
for _ , branch := range branches {
for _ , repository := range repositories {
t . Run ( fmt . Sprintf ( "[Branch:%s,Repository:%s]" , branch , repository ) , func ( t * testing . T ) {
t . Run ( "Upload" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
uploadURL := fmt . Sprintf ( "%s/%s/%s" , rootURL , branch , repository )
req := NewRequestWithBody ( t , "PUT" , uploadURL , bytes . NewReader ( [ ] byte { } ) )
MakeRequest ( t , req , http . StatusUnauthorized )
2023-12-22 00:59:59 +01:00
req = NewRequestWithBody ( t , "PUT" , uploadURL , bytes . NewReader ( [ ] byte { } ) ) .
AddBasicAuth ( user . Name )
2023-05-12 19:27:50 +02:00
MakeRequest ( t , req , http . StatusBadRequest )
2023-12-22 00:59:59 +01:00
req = NewRequestWithBody ( t , "PUT" , uploadURL , bytes . NewReader ( content ) ) .
AddBasicAuth ( user . Name )
2023-05-12 19:27:50 +02:00
MakeRequest ( t , req , http . StatusCreated )
pvs , err := packages . GetVersionsByPackageType ( db . DefaultContext , user . ID , packages . TypeAlpine )
assert . NoError ( t , err )
assert . Len ( t , pvs , 1 )
pd , err := packages . GetPackageDescriptor ( db . DefaultContext , pvs [ 0 ] )
assert . NoError ( t , err )
assert . Nil ( t , pd . SemVer )
assert . IsType ( t , & alpine_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 )
assert . NoError ( t , err )
assert . NotEmpty ( t , pfs )
assert . Condition ( t , func ( ) bool {
seen := false
expectedFilename := fmt . Sprintf ( "%s-%s.apk" , packageName , packageVersion )
expectedCompositeKey := fmt . Sprintf ( "%s|%s|x86_64" , branch , repository )
for _ , pf := range pfs {
if pf . Name == expectedFilename && pf . CompositeKey == expectedCompositeKey {
if seen {
return false
}
seen = true
assert . True ( t , pf . IsLead )
pfps , err := packages . GetProperties ( db . DefaultContext , packages . PropertyTypeFile , pf . ID )
assert . NoError ( t , err )
for _ , pfp := range pfps {
switch pfp . Name {
case alpine_module . PropertyBranch :
assert . Equal ( t , branch , pfp . Value )
case alpine_module . PropertyRepository :
assert . Equal ( t , repository , pfp . Value )
case alpine_module . PropertyArchitecture :
assert . Equal ( t , "x86_64" , pfp . Value )
}
}
}
}
return seen
} )
} )
2024-02-25 16:10:55 +01:00
readIndexContent := func ( r io . Reader ) ( string , error ) {
br := bufio . NewReader ( r )
2023-05-12 19:27:50 +02:00
2024-02-25 16:10:55 +01:00
gzr , err := gzip . NewReader ( br )
if err != nil {
return "" , err
}
2023-05-12 19:27:50 +02:00
2024-02-25 16:10:55 +01:00
for {
gzr . Multistream ( false )
2023-05-12 19:27:50 +02:00
2024-02-25 16:10:55 +01:00
tr := tar . NewReader ( gzr )
2023-05-12 19:27:50 +02:00
for {
2024-02-25 16:10:55 +01:00
hd , err := tr . Next ( )
if err == io . EOF {
break
}
if err != nil {
return "" , err
}
2023-05-12 19:27:50 +02:00
2024-02-25 16:10:55 +01:00
if hd . Name == alpine_service . IndexFilename {
buf , err := io . ReadAll ( tr )
if err != nil {
return "" , err
2023-05-12 19:27:50 +02:00
}
2024-02-25 16:10:55 +01:00
return string ( buf ) , nil
2023-05-12 19:27:50 +02:00
}
2024-02-25 16:10:55 +01:00
}
2023-05-12 19:27:50 +02:00
2024-02-25 16:10:55 +01:00
err = gzr . Reset ( br )
if err == io . EOF {
break
}
if err != nil {
return "" , err
2023-05-12 19:27:50 +02:00
}
2024-02-25 16:10:55 +01:00
}
2023-05-12 19:27:50 +02:00
2024-02-25 16:10:55 +01:00
return "" , io . EOF
}
t . Run ( "Index" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/%s/%s/x86_64/APKINDEX.tar.gz" , rootURL , branch , repository ) )
resp := MakeRequest ( t , req , http . StatusOK )
content , err := readIndexContent ( resp . Body )
assert . NoError ( t , err )
assert . Contains ( t , content , "C:Q1/se1PjO94hYXbfpNR1/61hVORIc=\n" )
assert . Contains ( t , content , "P:" + packageName + "\n" )
assert . Contains ( t , content , "V:" + packageVersion + "\n" )
assert . Contains ( t , content , "A:x86_64\n" )
assert . NotContains ( t , content , "A:noarch\n" )
assert . Contains ( t , content , "T:Gitea Test Package\n" )
assert . Contains ( t , content , "U:https://gitea.io/\n" )
assert . Contains ( t , content , "L:MIT\n" )
assert . Contains ( t , content , "S:1353\n" )
assert . Contains ( t , content , "I:4096\n" )
assert . Contains ( t , content , "o:gitea-test\n" )
assert . Contains ( t , content , "m:KN4CK3R <kn4ck3r@gitea.io>\n" )
assert . Contains ( t , content , "t:1679498030\n" )
2023-05-12 19:27:50 +02:00
} )
t . Run ( "Download" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/%s/%s/x86_64/%s-%s.apk" , rootURL , branch , repository , packageName , packageVersion ) )
MakeRequest ( t , req , http . StatusOK )
} )
2024-02-25 16:10:55 +01:00
t . Run ( "NoArch" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
req := NewRequestWithBody ( t , "PUT" , fmt . Sprintf ( "%s/%s/%s" , rootURL , branch , repository ) , bytes . NewReader ( noarchContent ) ) .
AddBasicAuth ( user . Name )
MakeRequest ( t , req , http . StatusCreated )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/%s/%s/x86_64/APKINDEX.tar.gz" , rootURL , branch , repository ) )
resp := MakeRequest ( t , req , http . StatusOK )
content , err := readIndexContent ( resp . Body )
assert . NoError ( t , err )
assert . Contains ( t , content , "C:Q1/se1PjO94hYXbfpNR1/61hVORIc=\n" )
assert . Contains ( t , content , "A:x86_64\n" )
assert . Contains ( t , content , "C:Q1kbH5WoIPFccQYyATanaKXd2cJcc=\n" )
assert . NotContains ( t , content , "A:noarch\n" )
// noarch package should be available with every architecture requested
for _ , arch := range [ ] string { alpine_module . NoArch , "x86_64" , "my_arch" } {
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/%s/%s/%s/gitea-noarch-1.4-r0.apk" , rootURL , branch , repository , arch ) )
MakeRequest ( t , req , http . StatusOK )
}
req = NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/%s/%s/noarch/gitea-noarch-1.4-r0.apk" , rootURL , branch , repository ) ) .
AddBasicAuth ( user . Name )
MakeRequest ( t , req , http . StatusNoContent )
} )
2023-05-12 19:27:50 +02:00
} )
}
}
t . Run ( "Delete" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
for _ , branch := range branches {
for _ , repository := range repositories {
req := NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/%s/%s/x86_64/%s-%s.apk" , rootURL , branch , repository , packageName , packageVersion ) )
MakeRequest ( t , req , http . StatusUnauthorized )
2023-12-22 00:59:59 +01:00
req = NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/%s/%s/x86_64/%s-%s.apk" , rootURL , branch , repository , packageName , packageVersion ) ) .
AddBasicAuth ( user . Name )
2023-05-12 19:27:50 +02:00
MakeRequest ( t , req , http . StatusNoContent )
// Deleting the last file of an architecture should remove that index
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/%s/%s/x86_64/APKINDEX.tar.gz" , rootURL , branch , repository ) )
MakeRequest ( t , req , http . StatusNotFound )
}
}
} )
}