2022-03-30 11:42:47 +03:00
// Copyright 2022 The Gitea Authors. All rights reserved.
2022-11-27 21:20:29 +03:00
// SPDX-License-Identifier: MIT
2022-03-30 11:42:47 +03:00
2022-09-02 22:18:23 +03:00
package integration
2022-03-30 11:42:47 +03:00
import (
"fmt"
"net/http"
stdurl "net/url"
"strings"
"testing"
"time"
2024-09-05 10:05:42 +03:00
auth_model "code.gitea.io/gitea/models/auth"
2022-03-30 11:42:47 +03:00
"code.gitea.io/gitea/models/db"
"code.gitea.io/gitea/models/packages"
conan_model "code.gitea.io/gitea/models/packages/conan"
"code.gitea.io/gitea/models/unittest"
user_model "code.gitea.io/gitea/models/user"
conan_module "code.gitea.io/gitea/modules/packages/conan"
"code.gitea.io/gitea/modules/setting"
conan_router "code.gitea.io/gitea/routers/api/packages/conan"
2024-09-05 10:05:42 +03:00
package_service "code.gitea.io/gitea/services/packages"
2022-09-02 22:18:23 +03:00
"code.gitea.io/gitea/tests"
2022-03-30 11:42:47 +03:00
"github.com/stretchr/testify/assert"
)
const (
conanfileName = "conanfile.py"
conaninfoName = "conaninfo.txt"
conanLicense = "MIT"
conanAuthor = "Gitea <info@gitea.io>"
conanHomepage = "https://gitea.io/"
conanURL = "https://gitea.com/"
conanDescription = "Description of ConanPackage"
conanTopic = "gitea"
conanPackageReference = "dummyreference"
contentConaninfo = ` [ settings ]
arch = x84_64
[ requires ]
fmt / 7.1 .3
[ options ]
shared = False
[ full_settings ]
arch = x84_64
[ full_requires ]
fmt / 7.1 .3
[ full_options ]
shared = False
[ recipe_hash ]
74714915 a51073acb548ca1ce29afbac
[ env ]
CC = gcc - 10 `
)
func buildConanfileContent ( name , version string ) string {
return ` from conans import ConanFile , CMake , tools
class ConanPackageConan ( ConanFile ) :
name = "` + name + `"
version = "` + version + `"
license = "` + conanLicense + `"
author = "` + conanAuthor + `"
homepage = "` + conanHomepage + `"
url = "` + conanURL + `"
description = "` + conanDescription + `"
topics = ( "` + conanTopic + `" )
settings = "os" , "compiler" , "build_type" , "arch"
options = { "shared" : [ True , False ] , "fPIC" : [ True , False ] }
default_options = { "shared" : False , "fPIC" : True }
generators = "cmake" `
}
func uploadConanPackageV1 ( t * testing . T , baseURL , token , name , version , user , channel string ) {
contentConanfile := buildConanfileContent ( name , version )
recipeURL := fmt . Sprintf ( "%s/v1/conans/%s/%s/%s/%s" , baseURL , name , version , user , channel )
2023-12-22 02:59:59 +03:00
req := NewRequest ( t , "GET" , recipeURL ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusNotFound )
2023-12-22 02:59:59 +03:00
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/digest" , recipeURL ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusNotFound )
2023-12-22 02:59:59 +03:00
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/download_urls" , recipeURL ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusNotFound )
req = NewRequest ( t , "POST" , fmt . Sprintf ( "%s/upload_urls" , recipeURL ) )
MakeRequest ( t , req , http . StatusUnauthorized )
req = NewRequestWithJSON ( t , "POST" , fmt . Sprintf ( "%s/upload_urls" , recipeURL ) , map [ string ] int64 {
conanfileName : int64 ( len ( contentConanfile ) ) ,
"removed.txt" : 0 ,
2023-12-22 02:59:59 +03:00
} ) . AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
resp := MakeRequest ( t , req , http . StatusOK )
uploadURLs := make ( map [ string ] string )
DecodeJSON ( t , resp , & uploadURLs )
assert . Contains ( t , uploadURLs , conanfileName )
assert . NotContains ( t , uploadURLs , "removed.txt" )
uploadURL := uploadURLs [ conanfileName ]
assert . NotEmpty ( t , uploadURL )
2023-12-22 02:59:59 +03:00
req = NewRequestWithBody ( t , "PUT" , uploadURL , strings . NewReader ( contentConanfile ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusCreated )
packageURL := fmt . Sprintf ( "%s/packages/%s" , recipeURL , conanPackageReference )
2023-12-22 02:59:59 +03:00
req = NewRequest ( t , "GET" , packageURL ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusNotFound )
2023-12-22 02:59:59 +03:00
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/digest" , packageURL ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusNotFound )
2023-12-22 02:59:59 +03:00
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/download_urls" , packageURL ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusNotFound )
req = NewRequest ( t , "POST" , fmt . Sprintf ( "%s/upload_urls" , packageURL ) )
MakeRequest ( t , req , http . StatusUnauthorized )
req = NewRequestWithJSON ( t , "POST" , fmt . Sprintf ( "%s/upload_urls" , packageURL ) , map [ string ] int64 {
conaninfoName : int64 ( len ( contentConaninfo ) ) ,
"removed.txt" : 0 ,
2023-12-22 02:59:59 +03:00
} ) . AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
resp = MakeRequest ( t , req , http . StatusOK )
uploadURLs = make ( map [ string ] string )
DecodeJSON ( t , resp , & uploadURLs )
assert . Contains ( t , uploadURLs , conaninfoName )
assert . NotContains ( t , uploadURLs , "removed.txt" )
uploadURL = uploadURLs [ conaninfoName ]
assert . NotEmpty ( t , uploadURL )
2023-12-22 02:59:59 +03:00
req = NewRequestWithBody ( t , "PUT" , uploadURL , strings . NewReader ( contentConaninfo ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusCreated )
}
func uploadConanPackageV2 ( t * testing . T , baseURL , token , name , version , user , channel , recipeRevision , packageRevision string ) {
contentConanfile := buildConanfileContent ( name , version )
recipeURL := fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s/revisions/%s" , baseURL , name , version , user , channel , recipeRevision )
2023-12-22 02:59:59 +03:00
req := NewRequestWithBody ( t , "PUT" , fmt . Sprintf ( "%s/files/%s" , recipeURL , conanfileName ) , strings . NewReader ( contentConanfile ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusCreated )
2023-12-22 02:59:59 +03:00
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/files" , recipeURL ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
resp := MakeRequest ( t , req , http . StatusOK )
var list * struct {
2023-07-04 21:36:08 +03:00
Files map [ string ] any ` json:"files" `
2022-03-30 11:42:47 +03:00
}
DecodeJSON ( t , resp , & list )
assert . Len ( t , list . Files , 1 )
assert . Contains ( t , list . Files , conanfileName )
packageURL := fmt . Sprintf ( "%s/packages/%s/revisions/%s" , recipeURL , conanPackageReference , packageRevision )
2023-12-22 02:59:59 +03:00
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/files" , packageURL ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusNotFound )
2023-12-22 02:59:59 +03:00
req = NewRequestWithBody ( t , "PUT" , fmt . Sprintf ( "%s/files/%s" , packageURL , conaninfoName ) , strings . NewReader ( contentConaninfo ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusCreated )
2023-12-22 02:59:59 +03:00
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/files" , packageURL ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
resp = MakeRequest ( t , req , http . StatusOK )
list = nil
DecodeJSON ( t , resp , & list )
assert . Len ( t , list . Files , 1 )
assert . Contains ( t , list . Files , conaninfoName )
}
func TestPackageConan ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrepareTestEnv ( t ) ( )
2023-02-23 17:11:56 +03:00
2022-08-16 05:22:25 +03:00
user := unittest . AssertExistsAndLoadBean ( t , & user_model . User { ID : 2 } )
2022-03-30 11:42:47 +03:00
name := "ConanPackage"
version1 := "1.2"
version2 := "1.3"
user1 := "dummy"
user2 := "gitea"
channel1 := "test"
channel2 := "final"
revision1 := "rev1"
revision2 := "rev2"
url := fmt . Sprintf ( "%sapi/packages/%s/conan" , setting . AppURL , user . Name )
t . Run ( "v1" , func ( t * testing . T ) {
t . Run ( "Ping" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/v1/ping" , url ) )
resp := MakeRequest ( t , req , http . StatusOK )
assert . Equal ( t , "revisions" , resp . Header ( ) . Get ( "X-Conan-Server-Capabilities" ) )
} )
token := ""
2024-09-05 10:05:42 +03:00
t . Run ( "UserName/Password Authenticate" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
2023-12-22 02:59:59 +03:00
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/v1/users/authenticate" , url ) ) .
AddBasicAuth ( user . Name )
2022-03-30 11:42:47 +03:00
resp := MakeRequest ( t , req , http . StatusOK )
2023-12-22 02:59:59 +03:00
token = resp . Body . String ( )
assert . NotEmpty ( t , token )
2024-09-05 10:05:42 +03:00
pkgMeta , err := package_service . ParseAuthorizationToken ( token )
assert . NoError ( t , err )
assert . Equal ( t , user . ID , pkgMeta . UserID )
assert . Equal ( t , auth_model . AccessTokenScopeAll , pkgMeta . Scope )
} )
badToken := ""
t . Run ( "Token Scope Authentication" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
session := loginUser ( t , user . Name )
badToken = getTokenForLoggedInUser ( t , session , auth_model . AccessTokenScopeReadNotification )
testCase := func ( t * testing . T , scope auth_model . AccessTokenScope , expectedAuthStatusCode , expectedStatusCode int ) {
t . Helper ( )
token := getTokenForLoggedInUser ( t , session , scope )
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/v1/users/authenticate" , url ) ) .
AddTokenAuth ( token )
resp := MakeRequest ( t , req , expectedAuthStatusCode )
if expectedAuthStatusCode != http . StatusOK {
return
}
body := resp . Body . String ( )
assert . NotEmpty ( t , body )
pkgMeta , err := package_service . ParseAuthorizationToken ( body )
assert . NoError ( t , err )
assert . Equal ( t , user . ID , pkgMeta . UserID )
assert . Equal ( t , scope , pkgMeta . Scope )
recipeURL := fmt . Sprintf ( "%s/v1/conans/%s/%s/%s/%s" , url , "TestScope" , version1 , "testing" , channel1 )
req = NewRequestWithJSON ( t , "POST" , fmt . Sprintf ( "%s/upload_urls" , recipeURL ) , map [ string ] int64 {
conanfileName : 64 ,
"removed.txt" : 0 ,
} ) . AddTokenAuth ( token )
MakeRequest ( t , req , expectedStatusCode )
}
t . Run ( "No Package permission" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
testCase ( t , auth_model . AccessTokenScopeReadNotification , http . StatusUnauthorized , http . StatusForbidden )
} )
t . Run ( "Package Read permission" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
testCase ( t , auth_model . AccessTokenScopeReadPackage , http . StatusOK , http . StatusUnauthorized )
} )
t . Run ( "Package Write permission" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
testCase ( t , auth_model . AccessTokenScopeWritePackage , http . StatusOK , http . StatusOK )
} )
t . Run ( "All permission" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
testCase ( t , auth_model . AccessTokenScopeAll , http . StatusOK , http . StatusOK )
} )
2022-03-30 11:42:47 +03:00
} )
t . Run ( "CheckCredentials" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
2023-12-22 02:59:59 +03:00
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/v1/users/check_credentials" , url ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusOK )
} )
t . Run ( "Upload" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
uploadConanPackageV1 ( t , url , token , name , version1 , user1 , channel1 )
t . Run ( "Validate" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
pvs , err := packages . GetVersionsByPackageType ( db . DefaultContext , user . ID , packages . TypeConan )
assert . NoError ( t , err )
assert . Len ( t , pvs , 1 )
pd , err := packages . GetPackageDescriptor ( db . DefaultContext , pvs [ 0 ] )
assert . NoError ( t , err )
2022-10-07 07:22:05 +03:00
assert . Nil ( t , pd . SemVer )
2022-03-30 11:42:47 +03:00
assert . Equal ( t , name , pd . Package . Name )
assert . Equal ( t , version1 , pd . Version . Version )
assert . IsType ( t , & conan_module . Metadata { } , pd . Metadata )
metadata := pd . Metadata . ( * conan_module . Metadata )
assert . Equal ( t , conanLicense , metadata . License )
assert . Equal ( t , conanAuthor , metadata . Author )
assert . Equal ( t , conanHomepage , metadata . ProjectURL )
assert . Equal ( t , conanURL , metadata . RepositoryURL )
assert . Equal ( t , conanDescription , metadata . Description )
assert . Equal ( t , [ ] string { conanTopic } , metadata . Keywords )
pfs , err := packages . GetFilesByVersionID ( db . DefaultContext , pvs [ 0 ] . ID )
assert . NoError ( t , err )
assert . Len ( t , pfs , 2 )
for _ , pf := range pfs {
pb , err := packages . GetBlobByID ( db . DefaultContext , pf . BlobID )
assert . NoError ( t , err )
if pf . Name == conanfileName {
assert . True ( t , pf . IsLead )
assert . Equal ( t , int64 ( len ( buildConanfileContent ( name , version1 ) ) ) , pb . Size )
} else if pf . Name == conaninfoName {
assert . False ( t , pf . IsLead )
assert . Equal ( t , int64 ( len ( contentConaninfo ) ) , pb . Size )
} else {
2023-10-11 14:02:24 +03:00
assert . FailNow ( t , "unknown file: %s" , pf . Name )
2022-03-30 11:42:47 +03:00
}
}
} )
} )
t . Run ( "Download" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
recipeURL := fmt . Sprintf ( "%s/v1/conans/%s/%s/%s/%s" , url , name , version1 , user1 , channel1 )
req := NewRequest ( t , "GET" , recipeURL )
resp := MakeRequest ( t , req , http . StatusOK )
fileHashes := make ( map [ string ] string )
DecodeJSON ( t , resp , & fileHashes )
assert . Len ( t , fileHashes , 1 )
assert . Contains ( t , fileHashes , conanfileName )
assert . Equal ( t , "7abc52241c22090782c54731371847a8" , fileHashes [ conanfileName ] )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/digest" , recipeURL ) )
resp = MakeRequest ( t , req , http . StatusOK )
downloadURLs := make ( map [ string ] string )
DecodeJSON ( t , resp , & downloadURLs )
assert . Contains ( t , downloadURLs , conanfileName )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/download_urls" , recipeURL ) )
resp = MakeRequest ( t , req , http . StatusOK )
DecodeJSON ( t , resp , & downloadURLs )
assert . Contains ( t , downloadURLs , conanfileName )
req = NewRequest ( t , "GET" , downloadURLs [ conanfileName ] )
resp = MakeRequest ( t , req , http . StatusOK )
assert . Equal ( t , buildConanfileContent ( name , version1 ) , resp . Body . String ( ) )
packageURL := fmt . Sprintf ( "%s/packages/%s" , recipeURL , conanPackageReference )
req = NewRequest ( t , "GET" , packageURL )
resp = MakeRequest ( t , req , http . StatusOK )
fileHashes = make ( map [ string ] string )
DecodeJSON ( t , resp , & fileHashes )
assert . Len ( t , fileHashes , 1 )
assert . Contains ( t , fileHashes , conaninfoName )
assert . Equal ( t , "7628bfcc5b17f1470c468621a78df394" , fileHashes [ conaninfoName ] )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/digest" , packageURL ) )
resp = MakeRequest ( t , req , http . StatusOK )
downloadURLs = make ( map [ string ] string )
DecodeJSON ( t , resp , & downloadURLs )
assert . Contains ( t , downloadURLs , conaninfoName )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/download_urls" , packageURL ) )
resp = MakeRequest ( t , req , http . StatusOK )
DecodeJSON ( t , resp , & downloadURLs )
assert . Contains ( t , downloadURLs , conaninfoName )
req = NewRequest ( t , "GET" , downloadURLs [ conaninfoName ] )
resp = MakeRequest ( t , req , http . StatusOK )
assert . Equal ( t , contentConaninfo , resp . Body . String ( ) )
} )
t . Run ( "Search" , func ( t * testing . T ) {
uploadConanPackageV1 ( t , url , token , name , version2 , user1 , channel1 )
uploadConanPackageV1 ( t , url , token , name , version1 , user1 , channel2 )
uploadConanPackageV1 ( t , url , token , name , version1 , user2 , channel1 )
uploadConanPackageV1 ( t , url , token , name , version1 , user2 , channel2 )
t . Run ( "Recipe" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
cases := [ ] struct {
Query string
Expected [ ] string
} {
{ "ConanPackage" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.3@dummy/test" , "ConanPackage/1.2@dummy/final" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/1.2" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.2@dummy/final" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/1.1" , [ ] string { } } ,
{ "Conan*" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.3@dummy/test" , "ConanPackage/1.2@dummy/final" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.3@dummy/test" , "ConanPackage/1.2@dummy/final" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/*" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.3@dummy/test" , "ConanPackage/1.2@dummy/final" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/1*" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.3@dummy/test" , "ConanPackage/1.2@dummy/final" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/*2" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.2@dummy/final" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/1*2" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.2@dummy/final" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/1.2@" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.2@dummy/final" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/1.2@du*" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.2@dummy/final" } } ,
{ "ConanPackage/1.2@du*/" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.2@dummy/final" } } ,
{ "ConanPackage/1.2@du*/*test" , [ ] string { "ConanPackage/1.2@dummy/test" } } ,
{ "ConanPackage/1.2@du*/*st" , [ ] string { "ConanPackage/1.2@dummy/test" } } ,
{ "ConanPackage/1.2@gitea/*" , [ ] string { "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "*/*@dummy" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.3@dummy/test" , "ConanPackage/1.2@dummy/final" } } ,
{ "*/*@*/final" , [ ] string { "ConanPackage/1.2@dummy/final" , "ConanPackage/1.2@gitea/final" } } ,
}
for i , c := range cases {
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/v1/conans/search?q=%s" , url , stdurl . QueryEscape ( c . Query ) ) )
resp := MakeRequest ( t , req , http . StatusOK )
var result * conan_router . SearchResult
DecodeJSON ( t , resp , & result )
assert . ElementsMatch ( t , c . Expected , result . Results , "case %d: unexpected result" , i )
}
} )
t . Run ( "Package" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/v1/conans/%s/%s/%s/%s/search" , url , name , version1 , user1 , channel2 ) )
resp := MakeRequest ( t , req , http . StatusOK )
var result map [ string ] * conan_module . Conaninfo
DecodeJSON ( t , resp , & result )
assert . Contains ( t , result , conanPackageReference )
info := result [ conanPackageReference ]
assert . NotEmpty ( t , info . Settings )
} )
} )
t . Run ( "Delete" , func ( t * testing . T ) {
t . Run ( "Package" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
cases := [ ] struct {
Channel string
References [ ] string
} {
{ channel1 , [ ] string { conanPackageReference } } ,
{ channel2 , [ ] string { } } ,
}
for i , c := range cases {
rref , _ := conan_module . NewRecipeReference ( name , version1 , user1 , c . Channel , conan_module . DefaultRevision )
references , err := conan_model . GetPackageReferences ( db . DefaultContext , user . ID , rref )
assert . NoError ( t , err )
assert . NotEmpty ( t , references )
req := NewRequestWithJSON ( t , "POST" , fmt . Sprintf ( "%s/v1/conans/%s/%s/%s/%s/packages/delete" , url , name , version1 , user1 , c . Channel ) , map [ string ] [ ] string {
"package_ids" : c . References ,
2024-09-05 10:05:42 +03:00
} ) . AddTokenAuth ( badToken )
MakeRequest ( t , req , http . StatusUnauthorized )
req = NewRequestWithJSON ( t , "POST" , fmt . Sprintf ( "%s/v1/conans/%s/%s/%s/%s/packages/delete" , url , name , version1 , user1 , c . Channel ) , map [ string ] [ ] string {
"package_ids" : c . References ,
2023-12-22 02:59:59 +03:00
} ) . AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusOK )
references , err = conan_model . GetPackageReferences ( db . DefaultContext , user . ID , rref )
assert . NoError ( t , err )
assert . Empty ( t , references , "case %d: should be empty" , i )
}
} )
t . Run ( "Recipe" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
cases := [ ] struct {
Channel string
} {
{ channel1 } ,
{ channel2 } ,
}
for i , c := range cases {
rref , _ := conan_module . NewRecipeReference ( name , version1 , user1 , c . Channel , conan_module . DefaultRevision )
revisions , err := conan_model . GetRecipeRevisions ( db . DefaultContext , user . ID , rref )
assert . NoError ( t , err )
assert . NotEmpty ( t , revisions )
2023-12-22 02:59:59 +03:00
req := NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/v1/conans/%s/%s/%s/%s" , url , name , version1 , user1 , c . Channel ) ) .
2024-09-05 10:05:42 +03:00
AddTokenAuth ( badToken )
MakeRequest ( t , req , http . StatusUnauthorized )
req = NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/v1/conans/%s/%s/%s/%s" , url , name , version1 , user1 , c . Channel ) ) .
2023-12-22 02:59:59 +03:00
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusOK )
revisions , err = conan_model . GetRecipeRevisions ( db . DefaultContext , user . ID , rref )
assert . NoError ( t , err )
assert . Empty ( t , revisions , "case %d: should be empty" , i )
}
} )
} )
} )
t . Run ( "v2" , func ( t * testing . T ) {
t . Run ( "Ping" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/v2/ping" , url ) )
resp := MakeRequest ( t , req , http . StatusOK )
assert . Equal ( t , "revisions" , resp . Header ( ) . Get ( "X-Conan-Server-Capabilities" ) )
} )
token := ""
2024-09-05 10:05:42 +03:00
t . Run ( "UserName/Password Authenticate" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
2023-12-22 02:59:59 +03:00
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/v2/users/authenticate" , url ) ) .
AddBasicAuth ( user . Name )
2022-03-30 11:42:47 +03:00
resp := MakeRequest ( t , req , http . StatusOK )
body := resp . Body . String ( )
assert . NotEmpty ( t , body )
2024-09-05 10:05:42 +03:00
pkgMeta , err := package_service . ParseAuthorizationToken ( body )
assert . NoError ( t , err )
assert . Equal ( t , user . ID , pkgMeta . UserID )
assert . Equal ( t , auth_model . AccessTokenScopeAll , pkgMeta . Scope )
2022-03-30 11:42:47 +03:00
token = fmt . Sprintf ( "Bearer %s" , body )
} )
2024-09-05 10:05:42 +03:00
badToken := ""
t . Run ( "Token Scope Authentication" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
session := loginUser ( t , user . Name )
badToken = getTokenForLoggedInUser ( t , session , auth_model . AccessTokenScopeReadNotification )
testCase := func ( t * testing . T , scope auth_model . AccessTokenScope , expectedAuthStatusCode , expectedStatusCode int ) {
t . Helper ( )
token := getTokenForLoggedInUser ( t , session , scope )
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/v2/users/authenticate" , url ) ) .
AddTokenAuth ( token )
resp := MakeRequest ( t , req , expectedAuthStatusCode )
if expectedAuthStatusCode != http . StatusOK {
return
}
body := resp . Body . String ( )
assert . NotEmpty ( t , body )
pkgMeta , err := package_service . ParseAuthorizationToken ( body )
assert . NoError ( t , err )
assert . Equal ( t , user . ID , pkgMeta . UserID )
assert . Equal ( t , scope , pkgMeta . Scope )
recipeURL := fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s/revisions/%s" , url , "TestScope" , version1 , "testing" , channel1 , revision1 )
req = NewRequestWithBody ( t , "PUT" , fmt . Sprintf ( "%s/files/%s" , recipeURL , conanfileName ) , strings . NewReader ( "Demo Conan file" ) ) .
AddTokenAuth ( token )
MakeRequest ( t , req , expectedStatusCode )
}
t . Run ( "No Package permission" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
testCase ( t , auth_model . AccessTokenScopeReadNotification , http . StatusUnauthorized , http . StatusUnauthorized )
} )
t . Run ( "Package Read permission" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
testCase ( t , auth_model . AccessTokenScopeReadPackage , http . StatusOK , http . StatusUnauthorized )
} )
t . Run ( "Package Write permission" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
testCase ( t , auth_model . AccessTokenScopeWritePackage , http . StatusOK , http . StatusCreated )
} )
t . Run ( "All permission" , func ( t * testing . T ) {
defer tests . PrintCurrentTest ( t ) ( )
testCase ( t , auth_model . AccessTokenScopeAll , http . StatusOK , http . StatusCreated )
} )
} )
2022-03-30 11:42:47 +03:00
t . Run ( "CheckCredentials" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
2023-12-22 02:59:59 +03:00
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/v2/users/check_credentials" , url ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusOK )
} )
t . Run ( "Upload" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
uploadConanPackageV2 ( t , url , token , name , version1 , user1 , channel1 , revision1 , revision1 )
t . Run ( "Validate" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
pvs , err := packages . GetVersionsByPackageType ( db . DefaultContext , user . ID , packages . TypeConan )
assert . NoError ( t , err )
2024-09-05 10:05:42 +03:00
assert . Len ( t , pvs , 3 )
2022-03-30 11:42:47 +03:00
} )
} )
t . Run ( "Latest" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
recipeURL := fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s" , url , name , version1 , user1 , channel1 )
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/latest" , recipeURL ) )
resp := MakeRequest ( t , req , http . StatusOK )
obj := make ( map [ string ] string )
DecodeJSON ( t , resp , & obj )
assert . Contains ( t , obj , "revision" )
assert . Equal ( t , revision1 , obj [ "revision" ] )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/revisions/%s/packages/%s/latest" , recipeURL , revision1 , conanPackageReference ) )
resp = MakeRequest ( t , req , http . StatusOK )
obj = make ( map [ string ] string )
DecodeJSON ( t , resp , & obj )
assert . Contains ( t , obj , "revision" )
assert . Equal ( t , revision1 , obj [ "revision" ] )
} )
t . Run ( "ListRevisions" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
uploadConanPackageV2 ( t , url , token , name , version1 , user1 , channel1 , revision1 , revision2 )
uploadConanPackageV2 ( t , url , token , name , version1 , user1 , channel1 , revision2 , revision1 )
uploadConanPackageV2 ( t , url , token , name , version1 , user1 , channel1 , revision2 , revision2 )
recipeURL := fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s/revisions" , url , name , version1 , user1 , channel1 )
req := NewRequest ( t , "GET" , recipeURL )
resp := MakeRequest ( t , req , http . StatusOK )
type RevisionInfo struct {
Revision string ` json:"revision" `
Time time . Time ` json:"time" `
}
type RevisionList struct {
Revisions [ ] * RevisionInfo ` json:"revisions" `
}
var list * RevisionList
DecodeJSON ( t , resp , & list )
assert . Len ( t , list . Revisions , 2 )
revs := make ( [ ] string , 0 , len ( list . Revisions ) )
for _ , rev := range list . Revisions {
revs = append ( revs , rev . Revision )
}
assert . ElementsMatch ( t , [ ] string { revision1 , revision2 } , revs )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/%s/packages/%s/revisions" , recipeURL , revision1 , conanPackageReference ) )
resp = MakeRequest ( t , req , http . StatusOK )
DecodeJSON ( t , resp , & list )
assert . Len ( t , list . Revisions , 2 )
revs = make ( [ ] string , 0 , len ( list . Revisions ) )
for _ , rev := range list . Revisions {
revs = append ( revs , rev . Revision )
}
assert . ElementsMatch ( t , [ ] string { revision1 , revision2 } , revs )
} )
t . Run ( "Search" , func ( t * testing . T ) {
t . Run ( "Recipe" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
cases := [ ] struct {
Query string
Expected [ ] string
} {
{ "ConanPackage" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.3@dummy/test" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/1.2" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/1.1" , [ ] string { } } ,
{ "Conan*" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.3@dummy/test" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.3@dummy/test" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/*" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.3@dummy/test" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/1*" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.3@dummy/test" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/*2" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/1*2" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/1.2@" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "ConanPackage/1.2@du*" , [ ] string { "ConanPackage/1.2@dummy/test" } } ,
{ "ConanPackage/1.2@du*/" , [ ] string { "ConanPackage/1.2@dummy/test" } } ,
{ "ConanPackage/1.2@du*/*test" , [ ] string { "ConanPackage/1.2@dummy/test" } } ,
{ "ConanPackage/1.2@du*/*st" , [ ] string { "ConanPackage/1.2@dummy/test" } } ,
{ "ConanPackage/1.2@gitea/*" , [ ] string { "ConanPackage/1.2@gitea/test" , "ConanPackage/1.2@gitea/final" } } ,
{ "*/*@dummy" , [ ] string { "ConanPackage/1.2@dummy/test" , "ConanPackage/1.3@dummy/test" } } ,
{ "*/*@*/final" , [ ] string { "ConanPackage/1.2@gitea/final" } } ,
}
for i , c := range cases {
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/v2/conans/search?q=%s" , url , stdurl . QueryEscape ( c . Query ) ) )
resp := MakeRequest ( t , req , http . StatusOK )
var result * conan_router . SearchResult
DecodeJSON ( t , resp , & result )
assert . ElementsMatch ( t , c . Expected , result . Results , "case %d: unexpected result" , i )
}
} )
t . Run ( "Package" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
req := NewRequest ( t , "GET" , fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s/search" , url , name , version1 , user1 , channel1 ) )
resp := MakeRequest ( t , req , http . StatusOK )
var result map [ string ] * conan_module . Conaninfo
DecodeJSON ( t , resp , & result )
assert . Contains ( t , result , conanPackageReference )
info := result [ conanPackageReference ]
assert . NotEmpty ( t , info . Settings )
req = NewRequest ( t , "GET" , fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s/revisions/%s/search" , url , name , version1 , user1 , channel1 , revision1 ) )
resp = MakeRequest ( t , req , http . StatusOK )
result = make ( map [ string ] * conan_module . Conaninfo )
DecodeJSON ( t , resp , & result )
assert . Contains ( t , result , conanPackageReference )
info = result [ conanPackageReference ]
assert . NotEmpty ( t , info . Settings )
} )
} )
t . Run ( "Delete" , func ( t * testing . T ) {
t . Run ( "Package" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
rref , _ := conan_module . NewRecipeReference ( name , version1 , user1 , channel1 , revision1 )
pref , _ := conan_module . NewPackageReference ( rref , conanPackageReference , conan_module . DefaultRevision )
checkPackageRevisionCount := func ( count int ) {
revisions , err := conan_model . GetPackageRevisions ( db . DefaultContext , user . ID , pref )
assert . NoError ( t , err )
assert . Len ( t , revisions , count )
}
checkPackageReferenceCount := func ( count int ) {
references , err := conan_model . GetPackageReferences ( db . DefaultContext , user . ID , rref )
assert . NoError ( t , err )
assert . Len ( t , references , count )
}
checkPackageRevisionCount ( 2 )
2023-12-22 02:59:59 +03:00
req := NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s/revisions/%s/packages/%s/revisions/%s" , url , name , version1 , user1 , channel1 , revision1 , conanPackageReference , revision1 ) ) .
2024-09-05 10:05:42 +03:00
AddTokenAuth ( badToken )
MakeRequest ( t , req , http . StatusUnauthorized )
req = NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s/revisions/%s/packages/%s/revisions/%s" , url , name , version1 , user1 , channel1 , revision1 , conanPackageReference , revision1 ) ) .
2023-12-22 02:59:59 +03:00
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusOK )
checkPackageRevisionCount ( 1 )
2024-09-05 10:05:42 +03:00
req = NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s/revisions/%s/packages/%s" , url , name , version1 , user1 , channel1 , revision1 , conanPackageReference ) ) .
AddTokenAuth ( badToken )
MakeRequest ( t , req , http . StatusUnauthorized )
2023-12-22 02:59:59 +03:00
req = NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s/revisions/%s/packages/%s" , url , name , version1 , user1 , channel1 , revision1 , conanPackageReference ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusOK )
checkPackageRevisionCount ( 0 )
rref = rref . WithRevision ( revision2 )
checkPackageReferenceCount ( 1 )
2024-09-05 10:05:42 +03:00
req = NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s/revisions/%s/packages" , url , name , version1 , user1 , channel1 , revision2 ) ) .
AddTokenAuth ( badToken )
MakeRequest ( t , req , http . StatusUnauthorized )
2023-12-22 02:59:59 +03:00
req = NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s/revisions/%s/packages" , url , name , version1 , user1 , channel1 , revision2 ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusOK )
checkPackageReferenceCount ( 0 )
} )
t . Run ( "Recipe" , func ( t * testing . T ) {
2022-09-02 22:18:23 +03:00
defer tests . PrintCurrentTest ( t ) ( )
2022-03-30 11:42:47 +03:00
rref , _ := conan_module . NewRecipeReference ( name , version1 , user1 , channel1 , conan_module . DefaultRevision )
checkRecipeRevisionCount := func ( count int ) {
revisions , err := conan_model . GetRecipeRevisions ( db . DefaultContext , user . ID , rref )
assert . NoError ( t , err )
assert . Len ( t , revisions , count )
}
checkRecipeRevisionCount ( 2 )
2023-12-22 02:59:59 +03:00
req := NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s/revisions/%s" , url , name , version1 , user1 , channel1 , revision1 ) ) .
2024-09-05 10:05:42 +03:00
AddTokenAuth ( badToken )
MakeRequest ( t , req , http . StatusUnauthorized )
req = NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s/revisions/%s" , url , name , version1 , user1 , channel1 , revision1 ) ) .
2023-12-22 02:59:59 +03:00
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusOK )
checkRecipeRevisionCount ( 1 )
2024-09-05 10:05:42 +03:00
req = NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s" , url , name , version1 , user1 , channel1 ) ) .
AddTokenAuth ( badToken )
MakeRequest ( t , req , http . StatusUnauthorized )
2023-12-22 02:59:59 +03:00
req = NewRequest ( t , "DELETE" , fmt . Sprintf ( "%s/v2/conans/%s/%s/%s/%s" , url , name , version1 , user1 , channel1 ) ) .
AddTokenAuth ( token )
2022-03-30 11:42:47 +03:00
MakeRequest ( t , req , http . StatusOK )
checkRecipeRevisionCount ( 0 )
} )
} )
} )
}