2023-05-02 18:31:35 +02:00
// Copyright 2023 The Gitea Authors. All rights reserved.
// SPDX-License-Identifier: MIT
package integration
import (
"archive/tar"
"bytes"
"compress/gzip"
"fmt"
"io"
"net/http"
2024-10-28 06:48:07 +08:00
"strconv"
2023-05-02 18:31:35 +02:00
"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"
"code.gitea.io/gitea/modules/base"
debian_module "code.gitea.io/gitea/modules/packages/debian"
2024-10-28 06:48:07 +08:00
packages_cleanup_service "code.gitea.io/gitea/services/packages/cleanup"
2023-05-02 18:31:35 +02:00
"code.gitea.io/gitea/tests"
"github.com/blakesmith/ar"
"github.com/stretchr/testify/assert"
)
func TestPackageDebian ( t * testing . T ) {
defer tests . PrepareTestEnv ( t ) ( )
user := unittest . AssertExistsAndLoadBean ( t , & user_model . User { ID : 2 } )
packageName := "gitea"
packageVersion := "1.0.3"
2023-10-29 15:14:47 +01:00
packageVersion2 := "1.0.4"
2023-05-02 18:31:35 +02:00
packageDescription := "Package Description"
createArchive := func ( name , version , architecture string ) io . Reader {
var cbuf bytes . Buffer
zw := gzip . NewWriter ( & cbuf )
tw := tar . NewWriter ( zw )
tw . WriteHeader ( & tar . Header {
Name : "control" ,
Mode : 0 o600 ,
Size : 50 ,
} )
fmt . Fprintf ( tw , "Package: %s\nVersion: %s\nArchitecture: %s\nDescription: %s\n" , name , version , architecture , packageDescription )
tw . Close ( )
zw . Close ( )
var buf bytes . Buffer
aw := ar . NewWriter ( & buf )
aw . WriteGlobalHeader ( )
hdr := & ar . Header {
Name : "control.tar.gz" ,
Mode : 0 o600 ,
Size : int64 ( cbuf . Len ( ) ) ,
}
aw . WriteHeader ( hdr )
aw . Write ( cbuf . Bytes ( ) )
return & buf
}
distributions := [ ] string { "test" , "gitea" }
components := [ ] string { "main" , "stable" }
architectures := [ ] string { "all" , "amd64" }
rootURL := fmt . Sprintf ( "/api/packages/%s/debian" , user . Name )
t . Run ( "RepositoryKey" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
req := NewRequest ( t , "GET" , rootURL + "/repository.key" )
resp := MakeRequest ( t , req , http . StatusOK )
assert . Equal ( t , "application/pgp-keys" , resp . Header ( ) . Get ( "Content-Type" ) )
assert . Contains ( t , resp . Body . String ( ) , "-----BEGIN PGP PUBLIC KEY BLOCK-----" )
} )
for _ , distribution := range distributions {
t . Run ( fmt . Sprintf ( "[Distribution:%s]" , distribution ) , func ( t * testing . T ) {
for _ , component := range components {
for _ , architecture := range architectures {
t . Run ( fmt . Sprintf ( "[Component:%s,Architecture:%s]" , component , architecture ) , func ( t * testing . T ) {
2023-10-29 15:14:47 +01:00
uploadURL := fmt . Sprintf ( "%s/pool/%s/%s/upload" , rootURL , distribution , component )
2023-05-02 18:31:35 +02:00
t . Run ( "Upload" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
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-02 18:31:35 +02:00
MakeRequest ( t , req , http . StatusBadRequest )
2023-12-22 00:59:59 +01:00
req = NewRequestWithBody ( t , "PUT" , uploadURL , createArchive ( "" , "" , "" ) ) .
AddBasicAuth ( user . Name )
2023-05-02 18:31:35 +02:00
MakeRequest ( t , req , http . StatusBadRequest )
2023-12-22 00:59:59 +01:00
req = NewRequestWithBody ( t , "PUT" , uploadURL , createArchive ( packageName , packageVersion , architecture ) ) .
AddBasicAuth ( user . Name )
2023-05-02 18:31:35 +02:00
MakeRequest ( t , req , http . StatusCreated )
2023-10-29 15:14:47 +01:00
pv , err := packages . GetVersionByNameAndVersion ( db . DefaultContext , user . ID , packages . TypeDebian , packageName , packageVersion )
2023-05-02 18:31:35 +02:00
assert . NoError ( t , err )
2023-10-29 15:14:47 +01:00
pd , err := packages . GetPackageDescriptor ( db . DefaultContext , pv )
2023-05-02 18:31:35 +02:00
assert . NoError ( t , err )
assert . Nil ( t , pd . SemVer )
assert . IsType ( t , & debian_module . Metadata { } , pd . Metadata )
assert . Equal ( t , packageName , pd . Package . Name )
assert . Equal ( t , packageVersion , pd . Version . Version )
2023-10-29 15:14:47 +01:00
pfs , err := packages . GetFilesByVersionID ( db . DefaultContext , pv . ID )
2023-05-02 18:31:35 +02:00
assert . NoError ( t , err )
assert . NotEmpty ( t , pfs )
assert . Condition ( t , func ( ) bool {
seen := false
expectedFilename := fmt . Sprintf ( "%s_%s_%s.deb" , packageName , packageVersion , architecture )
expectedCompositeKey := fmt . Sprintf ( "%s|%s" , distribution , component )
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 debian_module . PropertyDistribution :
assert . Equal ( t , distribution , pfp . Value )
case debian_module . PropertyComponent :
assert . Equal ( t , component , pfp . Value )
case debian_module . PropertyArchitecture :
assert . Equal ( t , architecture , pfp . Value )
}
}
}
}
return seen
} )
2023-07-24 16:19:44 +02:00
2023-12-22 00:59:59 +01:00
req = NewRequestWithBody ( t , "PUT" , uploadURL , createArchive ( packageName , packageVersion , architecture ) ) .
AddBasicAuth ( user . Name )
2023-10-10 21:39:58 +08:00
MakeRequest ( t , req , http . StatusConflict )
2023-05-02 18:31:35 +02:00
} )
t . Run ( "Download" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/pool/%s/%s/%s_%s_%s.deb" , rootURL , distribution , component , packageName , packageVersion , architecture ) )
resp := MakeRequest ( t , req , http . StatusOK )
assert . Equal ( t , "application/vnd.debian.binary-package" , resp . Header ( ) . Get ( "Content-Type" ) )
} )
t . Run ( "Packages" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
2023-12-22 00:59:59 +01:00
req := NewRequestWithBody ( t , "PUT" , uploadURL , createArchive ( packageName , packageVersion2 , architecture ) ) .
AddBasicAuth ( user . Name )
2023-10-29 15:14:47 +01:00
MakeRequest ( t , req , http . StatusCreated )
2023-05-02 18:31:35 +02:00
url := fmt . Sprintf ( "%s/dists/%s/%s/binary-%s/Packages" , rootURL , distribution , component , architecture )
2023-10-29 15:14:47 +01:00
req = NewRequest ( t , "GET" , url )
2023-05-02 18:31:35 +02:00
resp := MakeRequest ( t , req , http . StatusOK )
body := resp . Body . String ( )
2023-10-29 15:14:47 +01:00
assert . Contains ( t , body , "Package: " + packageName + "\n" )
assert . Contains ( t , body , "Version: " + packageVersion + "\n" )
assert . Contains ( t , body , "Version: " + packageVersion2 + "\n" )
assert . Contains ( t , body , "Architecture: " + architecture + "\n" )
assert . Contains ( t , body , fmt . Sprintf ( "Filename: pool/%s/%s/%s_%s_%s.deb\n" , distribution , component , packageName , packageVersion , architecture ) )
assert . Contains ( t , body , fmt . Sprintf ( "Filename: pool/%s/%s/%s_%s_%s.deb\n" , distribution , component , packageName , packageVersion2 , architecture ) )
2023-05-02 18:31:35 +02:00
req = NewRequest ( t , "GET" , url + ".gz" )
MakeRequest ( t , req , http . StatusOK )
req = NewRequest ( t , "GET" , url + ".xz" )
MakeRequest ( t , req , http . StatusOK )
url = fmt . Sprintf ( "%s/dists/%s/%s/%s/by-hash/SHA256/%s" , rootURL , distribution , component , architecture , base . EncodeSha256 ( body ) )
req = NewRequest ( t , "GET" , url )
resp = MakeRequest ( t , req , http . StatusOK )
assert . Equal ( t , body , resp . Body . String ( ) )
} )
} )
}
}
t . Run ( "Release" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/dists/%s/Release" , rootURL , distribution ) )
resp := MakeRequest ( t , req , http . StatusOK )
body := resp . Body . String ( )
2023-10-29 15:14:47 +01:00
assert . Contains ( t , body , "Components: " + strings . Join ( components , " " ) + "\n" )
assert . Contains ( t , body , "Architectures: " + strings . Join ( architectures , " " ) + "\n" )
2023-05-02 18:31:35 +02:00
for _ , component := range components {
for _ , architecture := range architectures {
2023-10-29 15:14:47 +01:00
assert . Contains ( t , body , fmt . Sprintf ( "%s/binary-%s/Packages\n" , component , architecture ) )
assert . Contains ( t , body , fmt . Sprintf ( "%s/binary-%s/Packages.gz\n" , component , architecture ) )
assert . Contains ( t , body , fmt . Sprintf ( "%s/binary-%s/Packages.xz\n" , component , architecture ) )
2023-05-02 18:31:35 +02:00
}
}
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/dists/%s/by-hash/SHA256/%s" , rootURL , distribution , base . EncodeSha256 ( body ) ) )
resp = MakeRequest ( t , req , http . StatusOK )
assert . Equal ( t , body , resp . Body . String ( ) )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/dists/%s/Release.gpg" , rootURL , distribution ) )
resp = MakeRequest ( t , req , http . StatusOK )
assert . Contains ( t , resp . Body . String ( ) , "-----BEGIN PGP SIGNATURE-----" )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/dists/%s/InRelease" , rootURL , distribution ) )
resp = MakeRequest ( t , req , http . StatusOK )
assert . Contains ( t , resp . Body . String ( ) , "-----BEGIN PGP SIGNED MESSAGE-----" )
} )
} )
}
t . Run ( "Delete" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
distribution := distributions [ 0 ]
architecture := architectures [ 0 ]
for _ , component := range components {
req := NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/pool/%s/%s/%s/%s/%s" , rootURL , distribution , component , packageName , packageVersion , architecture ) )
MakeRequest ( t , req , http . StatusUnauthorized )
2023-12-22 00:59:59 +01:00
req = NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/pool/%s/%s/%s/%s/%s" , rootURL , distribution , component , packageName , packageVersion , architecture ) ) .
AddBasicAuth ( user . Name )
2023-05-02 18:31:35 +02:00
MakeRequest ( t , req , http . StatusNoContent )
2023-12-22 00:59:59 +01:00
req = NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/pool/%s/%s/%s/%s/%s" , rootURL , distribution , component , packageName , packageVersion2 , architecture ) ) .
AddBasicAuth ( user . Name )
2023-10-29 15:14:47 +01:00
MakeRequest ( t , req , http . StatusNoContent )
2023-05-02 18:31:35 +02:00
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/dists/%s/%s/binary-%s/Packages" , rootURL , distribution , component , architecture ) )
MakeRequest ( t , req , http . StatusNotFound )
}
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/dists/%s/Release" , rootURL , distribution ) )
resp := MakeRequest ( t , req , http . StatusOK )
body := resp . Body . String ( )
2023-10-29 15:14:47 +01:00
assert . Contains ( t , body , "Components: " + strings . Join ( components , " " ) + "\n" )
assert . Contains ( t , body , "Architectures: " + architectures [ 1 ] + "\n" )
2023-05-02 18:31:35 +02:00
} )
2024-10-28 06:48:07 +08:00
t . Run ( "Cleanup" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
rule := & packages . PackageCleanupRule {
Enabled : true ,
RemovePattern : ` .* ` ,
MatchFullName : true ,
OwnerID : user . ID ,
Type : packages . TypeDebian ,
}
_ , err := packages . InsertCleanupRule ( db . DefaultContext , rule )
assert . NoError ( t , err )
// When there were a lot of packages (> 50 or 100) and the code used "Iterate" to get all packages, it ever caused bugs,
// because "Iterate" keeps a dangling SQL session but the callback function still uses the same session to execute statements.
// The "Iterate" problem has been checked by TestContextSafety now, so here we only need to check the cleanup logic with a small number
packagesCount := 2
for i := 0 ; i < packagesCount ; i ++ {
uploadURL := fmt . Sprintf ( "%s/pool/%s/%s/upload" , rootURL , "test" , "main" )
req := NewRequestWithBody ( t , "PUT" , uploadURL , createArchive ( packageName , "1.0." + strconv . Itoa ( i ) , "all" ) ) . AddBasicAuth ( user . Name )
MakeRequest ( t , req , http . StatusCreated )
}
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/dists/%s/Release" , rootURL , "test" ) )
MakeRequest ( t , req , http . StatusOK )
err = packages_cleanup_service . CleanupTask ( db . DefaultContext , 0 )
assert . NoError ( t , err )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/dists/%s/Release" , rootURL , "test" ) )
MakeRequest ( t , req , http . StatusNotFound )
} )
2023-05-02 18:31:35 +02:00
}