2022-09-02 22:18:23 +03:00
// Copyright 2017 The Gitea Authors. All rights reserved.
2022-11-27 21:20:29 +03:00
// SPDX-License-Identifier: MIT
2022-09-02 22:18:23 +03:00
package tests
import (
"context"
"database/sql"
"fmt"
"os"
"path"
"path/filepath"
"testing"
2023-02-23 17:11:56 +03:00
"code.gitea.io/gitea/models/db"
packages_model "code.gitea.io/gitea/models/packages"
2022-09-02 22:18:23 +03:00
"code.gitea.io/gitea/models/unittest"
"code.gitea.io/gitea/modules/base"
"code.gitea.io/gitea/modules/git"
"code.gitea.io/gitea/modules/graceful"
"code.gitea.io/gitea/modules/log"
"code.gitea.io/gitea/modules/queue"
repo_module "code.gitea.io/gitea/modules/repository"
"code.gitea.io/gitea/modules/setting"
"code.gitea.io/gitea/modules/storage"
"code.gitea.io/gitea/modules/util"
"code.gitea.io/gitea/routers"
"github.com/stretchr/testify/assert"
)
2023-04-19 16:40:42 +03:00
func exitf ( format string , args ... interface { } ) {
fmt . Printf ( format + "\n" , args ... )
os . Exit ( 1 )
}
2022-09-02 22:18:23 +03:00
func InitTest ( requireGitea bool ) {
giteaRoot := base . SetupGiteaRoot ( )
if giteaRoot == "" {
2023-04-19 16:40:42 +03:00
exitf ( "Environment variable $GITEA_ROOT not set" )
2022-09-02 22:18:23 +03:00
}
2023-04-19 16:40:42 +03:00
setting . AppWorkPath = giteaRoot
2022-09-02 22:18:23 +03:00
if requireGitea {
giteaBinary := "gitea"
2023-04-19 16:40:42 +03:00
if setting . IsWindows {
2022-09-02 22:18:23 +03:00
giteaBinary += ".exe"
}
setting . AppPath = path . Join ( giteaRoot , giteaBinary )
if _ , err := os . Stat ( setting . AppPath ) ; err != nil {
2023-04-19 16:40:42 +03:00
exitf ( "Could not find gitea binary at %s" , setting . AppPath )
2022-09-02 22:18:23 +03:00
}
}
giteaConf := os . Getenv ( "GITEA_CONF" )
if giteaConf == "" {
2023-04-19 16:40:42 +03:00
// By default, use sqlite.ini for testing, then IDE like GoLand can start the test process with debugger.
// It's easier for developers to debug bugs step by step with a debugger.
// Notice: when doing "ssh push", Gitea executes sub processes, debugger won't work for the sub processes.
giteaConf = "tests/sqlite.ini"
_ = os . Setenv ( "GITEA_CONF" , giteaConf )
fmt . Printf ( "Environment variable $GITEA_CONF not set, use default: %s\n" , giteaConf )
if ! setting . EnableSQLite3 {
exitf ( ` Need to enable SQLite3 for sqlite.ini testing, please set: -tags "sqlite,sqlite_unlock_notify" ` )
}
}
setting . IsInTesting = true
if ! path . IsAbs ( giteaConf ) {
2022-09-02 22:18:23 +03:00
setting . CustomConf = path . Join ( giteaRoot , giteaConf )
} else {
setting . CustomConf = giteaConf
}
setting . SetCustomPathAndConf ( "" , "" , "" )
2023-02-19 19:12:01 +03:00
setting . InitProviderAndLoadCommonSettingsForTest ( )
2022-09-02 22:18:23 +03:00
setting . Repository . DefaultBranch = "master" // many test code still assume that default branch is called "master"
_ = util . RemoveAll ( repo_module . LocalCopyPath ( ) )
if err := git . InitFull ( context . Background ( ) ) ; err != nil {
log . Fatal ( "git.InitOnceWithSync: %v" , err )
}
2023-02-19 19:12:01 +03:00
setting . LoadDBSetting ( )
2022-09-02 22:18:23 +03:00
if err := storage . Init ( ) ; err != nil {
2023-04-19 16:40:42 +03:00
exitf ( "Init storage failed: %v" , err )
2022-09-02 22:18:23 +03:00
}
switch {
2023-03-07 13:51:06 +03:00
case setting . Database . Type . IsMySQL ( ) :
2022-09-02 22:18:23 +03:00
connType := "tcp"
if len ( setting . Database . Host ) > 0 && setting . Database . Host [ 0 ] == '/' { // looks like a unix socket
connType = "unix"
}
db , err := sql . Open ( "mysql" , fmt . Sprintf ( "%s:%s@%s(%s)/" ,
setting . Database . User , setting . Database . Passwd , connType , setting . Database . Host ) )
defer db . Close ( )
if err != nil {
log . Fatal ( "sql.Open: %v" , err )
}
if _ , err = db . Exec ( fmt . Sprintf ( "CREATE DATABASE IF NOT EXISTS %s" , setting . Database . Name ) ) ; err != nil {
log . Fatal ( "db.Exec: %v" , err )
}
2023-03-07 13:51:06 +03:00
case setting . Database . Type . IsPostgreSQL ( ) :
2022-09-02 22:18:23 +03:00
var db * sql . DB
var err error
if setting . Database . Host [ 0 ] == '/' {
db , err = sql . Open ( "postgres" , fmt . Sprintf ( "postgres://%s:%s@/%s?sslmode=%s&host=%s" ,
setting . Database . User , setting . Database . Passwd , setting . Database . Name , setting . Database . SSLMode , setting . Database . Host ) )
} else {
db , err = sql . Open ( "postgres" , fmt . Sprintf ( "postgres://%s:%s@%s/%s?sslmode=%s" ,
setting . Database . User , setting . Database . Passwd , setting . Database . Host , setting . Database . Name , setting . Database . SSLMode ) )
}
defer db . Close ( )
if err != nil {
log . Fatal ( "sql.Open: %v" , err )
}
dbrows , err := db . Query ( fmt . Sprintf ( "SELECT 1 FROM pg_database WHERE datname = '%s'" , setting . Database . Name ) )
if err != nil {
log . Fatal ( "db.Query: %v" , err )
}
defer dbrows . Close ( )
if ! dbrows . Next ( ) {
if _ , err = db . Exec ( fmt . Sprintf ( "CREATE DATABASE %s" , setting . Database . Name ) ) ; err != nil {
log . Fatal ( "db.Exec: CREATE DATABASE: %v" , err )
}
}
// Check if we need to setup a specific schema
if len ( setting . Database . Schema ) == 0 {
break
}
db . Close ( )
if setting . Database . Host [ 0 ] == '/' {
db , err = sql . Open ( "postgres" , fmt . Sprintf ( "postgres://%s:%s@/%s?sslmode=%s&host=%s" ,
setting . Database . User , setting . Database . Passwd , setting . Database . Name , setting . Database . SSLMode , setting . Database . Host ) )
} else {
db , err = sql . Open ( "postgres" , fmt . Sprintf ( "postgres://%s:%s@%s/%s?sslmode=%s" ,
setting . Database . User , setting . Database . Passwd , setting . Database . Host , setting . Database . Name , setting . Database . SSLMode ) )
}
// This is a different db object; requires a different Close()
defer db . Close ( )
if err != nil {
log . Fatal ( "sql.Open: %v" , err )
}
schrows , err := db . Query ( fmt . Sprintf ( "SELECT 1 FROM information_schema.schemata WHERE schema_name = '%s'" , setting . Database . Schema ) )
if err != nil {
log . Fatal ( "db.Query: %v" , err )
}
defer schrows . Close ( )
if ! schrows . Next ( ) {
// Create and setup a DB schema
if _ , err = db . Exec ( fmt . Sprintf ( "CREATE SCHEMA %s" , setting . Database . Schema ) ) ; err != nil {
log . Fatal ( "db.Exec: CREATE SCHEMA: %v" , err )
}
}
2023-03-07 13:51:06 +03:00
case setting . Database . Type . IsMSSQL ( ) :
2022-09-02 22:18:23 +03:00
host , port := setting . ParseMSSQLHostPort ( setting . Database . Host )
db , err := sql . Open ( "mssql" , fmt . Sprintf ( "server=%s; port=%s; database=%s; user id=%s; password=%s;" ,
host , port , "master" , setting . Database . User , setting . Database . Passwd ) )
if err != nil {
log . Fatal ( "sql.Open: %v" , err )
}
if _ , err := db . Exec ( fmt . Sprintf ( "If(db_id(N'%s') IS NULL) BEGIN CREATE DATABASE %s; END;" , setting . Database . Name , setting . Database . Name ) ) ; err != nil {
log . Fatal ( "db.Exec: %v" , err )
}
defer db . Close ( )
}
routers . GlobalInitInstalled ( graceful . GetManager ( ) . HammerContext ( ) )
}
func PrepareTestEnv ( t testing . TB , skip ... int ) func ( ) {
t . Helper ( )
ourSkip := 2
if len ( skip ) > 0 {
ourSkip += skip [ 0 ]
}
deferFn := PrintCurrentTest ( t , ourSkip )
2022-12-23 02:41:56 +03:00
// load database fixtures
2022-09-02 22:18:23 +03:00
assert . NoError ( t , unittest . LoadFixtures ( ) )
2022-12-23 02:41:56 +03:00
// load git repo fixtures
2022-09-02 22:18:23 +03:00
assert . NoError ( t , util . RemoveAll ( setting . RepoRootPath ) )
assert . NoError ( t , unittest . CopyDir ( path . Join ( filepath . Dir ( setting . AppPath ) , "tests/gitea-repositories-meta" ) , setting . RepoRootPath ) )
ownerDirs , err := os . ReadDir ( setting . RepoRootPath )
if err != nil {
assert . NoError ( t , err , "unable to read the new repo root: %v\n" , err )
}
for _ , ownerDir := range ownerDirs {
if ! ownerDir . Type ( ) . IsDir ( ) {
continue
}
repoDirs , err := os . ReadDir ( filepath . Join ( setting . RepoRootPath , ownerDir . Name ( ) ) )
if err != nil {
assert . NoError ( t , err , "unable to read the new repo root: %v\n" , err )
}
for _ , repoDir := range repoDirs {
_ = os . MkdirAll ( filepath . Join ( setting . RepoRootPath , ownerDir . Name ( ) , repoDir . Name ( ) , "objects" , "pack" ) , 0 o755 )
_ = os . MkdirAll ( filepath . Join ( setting . RepoRootPath , ownerDir . Name ( ) , repoDir . Name ( ) , "objects" , "info" ) , 0 o755 )
_ = os . MkdirAll ( filepath . Join ( setting . RepoRootPath , ownerDir . Name ( ) , repoDir . Name ( ) , "refs" , "heads" ) , 0 o755 )
_ = os . MkdirAll ( filepath . Join ( setting . RepoRootPath , ownerDir . Name ( ) , repoDir . Name ( ) , "refs" , "tag" ) , 0 o755 )
}
}
2022-12-23 02:41:56 +03:00
// load LFS object fixtures
// (LFS storage can be on any of several backends, including remote servers, so we init it with the storage API)
lfsFixtures , err := storage . NewStorage ( "" , storage . LocalStorageConfig { Path : path . Join ( filepath . Dir ( setting . AppPath ) , "tests/gitea-lfs-meta" ) } )
assert . NoError ( t , err )
assert . NoError ( t , storage . Clean ( storage . LFS ) )
2023-03-13 13:23:51 +03:00
assert . NoError ( t , lfsFixtures . IterateObjects ( "" , func ( path string , _ storage . Object ) error {
2022-12-23 02:41:56 +03:00
_ , err := storage . Copy ( storage . LFS , path , lfsFixtures , path )
return err
} ) )
2023-02-23 17:11:56 +03:00
// clear all package data
assert . NoError ( t , db . TruncateBeans ( db . DefaultContext ,
& packages_model . Package { } ,
& packages_model . PackageVersion { } ,
& packages_model . PackageFile { } ,
& packages_model . PackageBlob { } ,
& packages_model . PackageProperty { } ,
& packages_model . PackageBlobUpload { } ,
& packages_model . PackageCleanupRule { } ,
) )
assert . NoError ( t , storage . Clean ( storage . Packages ) )
2022-09-02 22:18:23 +03:00
return deferFn
}
2023-04-19 16:40:42 +03:00
// ResetFixtures flushes queues, reloads fixtures and resets test repositories within a single test.
2022-09-02 22:18:23 +03:00
// Most tests should call defer tests.PrepareTestEnv(t)() (or have onGiteaRun do that for them) but sometimes
// within a single test this is required
func ResetFixtures ( t * testing . T ) {
assert . NoError ( t , queue . GetManager ( ) . FlushAll ( context . Background ( ) , - 1 ) )
2022-12-23 02:41:56 +03:00
// load database fixtures
2022-09-02 22:18:23 +03:00
assert . NoError ( t , unittest . LoadFixtures ( ) )
2022-12-23 02:41:56 +03:00
// load git repo fixtures
2022-09-02 22:18:23 +03:00
assert . NoError ( t , util . RemoveAll ( setting . RepoRootPath ) )
assert . NoError ( t , unittest . CopyDir ( path . Join ( filepath . Dir ( setting . AppPath ) , "tests/gitea-repositories-meta" ) , setting . RepoRootPath ) )
ownerDirs , err := os . ReadDir ( setting . RepoRootPath )
if err != nil {
assert . NoError ( t , err , "unable to read the new repo root: %v\n" , err )
}
for _ , ownerDir := range ownerDirs {
if ! ownerDir . Type ( ) . IsDir ( ) {
continue
}
repoDirs , err := os . ReadDir ( filepath . Join ( setting . RepoRootPath , ownerDir . Name ( ) ) )
if err != nil {
assert . NoError ( t , err , "unable to read the new repo root: %v\n" , err )
}
for _ , repoDir := range repoDirs {
_ = os . MkdirAll ( filepath . Join ( setting . RepoRootPath , ownerDir . Name ( ) , repoDir . Name ( ) , "objects" , "pack" ) , 0 o755 )
_ = os . MkdirAll ( filepath . Join ( setting . RepoRootPath , ownerDir . Name ( ) , repoDir . Name ( ) , "objects" , "info" ) , 0 o755 )
_ = os . MkdirAll ( filepath . Join ( setting . RepoRootPath , ownerDir . Name ( ) , repoDir . Name ( ) , "refs" , "heads" ) , 0 o755 )
_ = os . MkdirAll ( filepath . Join ( setting . RepoRootPath , ownerDir . Name ( ) , repoDir . Name ( ) , "refs" , "tag" ) , 0 o755 )
}
}
2022-12-23 02:41:56 +03:00
// load LFS object fixtures
// (LFS storage can be on any of several backends, including remote servers, so we init it with the storage API)
lfsFixtures , err := storage . NewStorage ( "" , storage . LocalStorageConfig { Path : path . Join ( filepath . Dir ( setting . AppPath ) , "tests/gitea-lfs-meta" ) } )
assert . NoError ( t , err )
assert . NoError ( t , storage . Clean ( storage . LFS ) )
2023-03-13 13:23:51 +03:00
assert . NoError ( t , lfsFixtures . IterateObjects ( "" , func ( path string , _ storage . Object ) error {
2022-12-23 02:41:56 +03:00
_ , err := storage . Copy ( storage . LFS , path , lfsFixtures , path )
return err
} ) )
2022-09-02 22:18:23 +03:00
}