2021-11-12 22:36:47 +08:00
|
|
|
// Copyright 2021 The Gitea Authors. All rights reserved.
|
2022-11-27 13:20:29 -05:00
|
|
|
// SPDX-License-Identifier: MIT
|
2017-04-25 03:24:51 -04:00
|
|
|
|
2021-11-12 22:36:47 +08:00
|
|
|
package unittest
|
2017-04-25 03:24:51 -04:00
|
|
|
|
|
|
|
import (
|
2018-10-21 22:09:17 +08:00
|
|
|
"fmt"
|
|
|
|
"time"
|
|
|
|
|
2021-11-12 22:36:47 +08:00
|
|
|
"code.gitea.io/gitea/models/db"
|
2023-02-20 05:20:30 +00:00
|
|
|
"code.gitea.io/gitea/modules/auth/password/hash"
|
|
|
|
"code.gitea.io/gitea/modules/setting"
|
2021-11-12 22:36:47 +08:00
|
|
|
|
2020-06-17 15:07:58 -04:00
|
|
|
"github.com/go-testfixtures/testfixtures/v3"
|
2021-03-24 18:27:22 +00:00
|
|
|
"xorm.io/xorm"
|
2020-03-22 23:12:55 +08:00
|
|
|
"xorm.io/xorm/schemas"
|
2017-04-25 03:24:51 -04:00
|
|
|
)
|
|
|
|
|
2023-04-19 21:40:42 +08:00
|
|
|
var fixturesLoader *testfixtures.Loader
|
2017-04-25 03:24:51 -04:00
|
|
|
|
2021-12-01 15:50:01 +08:00
|
|
|
// GetXORMEngine gets the XORM engine
|
|
|
|
func GetXORMEngine(engine ...*xorm.Engine) (x *xorm.Engine) {
|
2021-03-24 18:27:22 +00:00
|
|
|
if len(engine) == 1 {
|
2021-11-12 22:36:47 +08:00
|
|
|
return engine[0]
|
2021-03-24 18:27:22 +00:00
|
|
|
}
|
2024-10-28 06:48:07 +08:00
|
|
|
return db.GetEngine(db.DefaultContext).(*xorm.Engine)
|
2021-11-12 22:36:47 +08:00
|
|
|
}
|
2021-03-24 18:27:22 +00:00
|
|
|
|
2021-11-12 22:36:47 +08:00
|
|
|
// InitFixtures initialize test fixtures for a test database
|
|
|
|
func InitFixtures(opts FixturesOptions, engine ...*xorm.Engine) (err error) {
|
2021-12-01 15:50:01 +08:00
|
|
|
e := GetXORMEngine(engine...)
|
2023-04-19 21:40:42 +08:00
|
|
|
var fixtureOptionFiles func(*testfixtures.Loader) error
|
2021-09-24 19:32:56 +08:00
|
|
|
if opts.Dir != "" {
|
2023-04-19 21:40:42 +08:00
|
|
|
fixtureOptionFiles = testfixtures.Directory(opts.Dir)
|
2021-09-24 19:32:56 +08:00
|
|
|
} else {
|
2023-04-19 21:40:42 +08:00
|
|
|
fixtureOptionFiles = testfixtures.Files(opts.Files...)
|
2021-09-24 19:32:56 +08:00
|
|
|
}
|
2024-12-29 09:05:56 +08:00
|
|
|
var dialect string
|
2021-03-24 18:27:22 +00:00
|
|
|
switch e.Dialect().URI().DBType {
|
2020-06-17 15:07:58 -04:00
|
|
|
case schemas.POSTGRES:
|
|
|
|
dialect = "postgres"
|
|
|
|
case schemas.MYSQL:
|
|
|
|
dialect = "mysql"
|
|
|
|
case schemas.MSSQL:
|
|
|
|
dialect = "mssql"
|
|
|
|
case schemas.SQLITE:
|
|
|
|
dialect = "sqlite3"
|
|
|
|
default:
|
2024-12-29 09:05:56 +08:00
|
|
|
return fmt.Errorf("unsupported RDBMS for integration tests: %q", e.Dialect().URI().DBType)
|
2020-06-17 15:07:58 -04:00
|
|
|
}
|
|
|
|
loaderOptions := []func(loader *testfixtures.Loader) error{
|
2021-03-24 18:27:22 +00:00
|
|
|
testfixtures.Database(e.DB().DB),
|
2020-06-17 15:07:58 -04:00
|
|
|
testfixtures.Dialect(dialect),
|
|
|
|
testfixtures.DangerousSkipTestDatabaseCheck(),
|
2023-04-19 21:40:42 +08:00
|
|
|
fixtureOptionFiles,
|
2020-06-17 15:07:58 -04:00
|
|
|
}
|
|
|
|
|
2021-03-24 18:27:22 +00:00
|
|
|
if e.Dialect().URI().DBType == schemas.POSTGRES {
|
2020-06-17 15:07:58 -04:00
|
|
|
loaderOptions = append(loaderOptions, testfixtures.SkipResetSequences())
|
|
|
|
}
|
|
|
|
|
2023-04-19 21:40:42 +08:00
|
|
|
fixturesLoader, err = testfixtures.New(loaderOptions...)
|
2020-06-17 15:07:58 -04:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-02-20 05:20:30 +00:00
|
|
|
// register the dummy hash algorithm function used in the test fixtures
|
|
|
|
_ = hash.Register("dummy", hash.NewDummyHasher)
|
|
|
|
setting.PasswordHashAlgo, _ = hash.SetDefaultPasswordHashAlgorithm("dummy")
|
2017-04-25 03:24:51 -04:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// LoadFixtures load fixtures for a test database
|
2021-03-24 18:27:22 +00:00
|
|
|
func LoadFixtures(engine ...*xorm.Engine) error {
|
2021-12-01 15:50:01 +08:00
|
|
|
e := GetXORMEngine(engine...)
|
2018-10-21 04:48:33 +08:00
|
|
|
var err error
|
2023-04-19 21:40:42 +08:00
|
|
|
// (doubt) database transaction conflicts could occur and result in ROLLBACK? just try for a few times.
|
|
|
|
for i := 0; i < 5; i++ {
|
|
|
|
if err = fixturesLoader.Load(); err == nil {
|
2018-10-21 04:48:33 +08:00
|
|
|
break
|
|
|
|
}
|
2018-10-21 22:09:17 +08:00
|
|
|
time.Sleep(200 * time.Millisecond)
|
|
|
|
}
|
|
|
|
if err != nil {
|
2024-12-29 09:05:56 +08:00
|
|
|
return fmt.Errorf("LoadFixtures failed after retries: %w", err)
|
2018-10-21 04:48:33 +08:00
|
|
|
}
|
2019-12-10 07:49:05 +00:00
|
|
|
// Now if we're running postgres we need to tell it to update the sequences
|
2021-03-24 18:27:22 +00:00
|
|
|
if e.Dialect().URI().DBType == schemas.POSTGRES {
|
|
|
|
results, err := e.QueryString(`SELECT 'SELECT SETVAL(' ||
|
2019-12-10 07:49:05 +00:00
|
|
|
quote_literal(quote_ident(PGT.schemaname) || '.' || quote_ident(S.relname)) ||
|
|
|
|
', COALESCE(MAX(' ||quote_ident(C.attname)|| '), 1) ) FROM ' ||
|
|
|
|
quote_ident(PGT.schemaname)|| '.'||quote_ident(T.relname)|| ';'
|
|
|
|
FROM pg_class AS S,
|
|
|
|
pg_depend AS D,
|
|
|
|
pg_class AS T,
|
|
|
|
pg_attribute AS C,
|
|
|
|
pg_tables AS PGT
|
|
|
|
WHERE S.relkind = 'S'
|
|
|
|
AND S.oid = D.objid
|
|
|
|
AND D.refobjid = T.oid
|
|
|
|
AND D.refobjid = C.attrelid
|
|
|
|
AND D.refobjsubid = C.attnum
|
|
|
|
AND T.relname = PGT.tablename
|
|
|
|
ORDER BY S.relname;`)
|
|
|
|
if err != nil {
|
2024-12-29 09:05:56 +08:00
|
|
|
return fmt.Errorf("failed to generate sequence update: %w", err)
|
2019-12-10 07:49:05 +00:00
|
|
|
}
|
|
|
|
for _, r := range results {
|
|
|
|
for _, value := range r {
|
2021-03-24 18:27:22 +00:00
|
|
|
_, err = e.Exec(value)
|
2019-12-10 07:49:05 +00:00
|
|
|
if err != nil {
|
2024-12-29 09:05:56 +08:00
|
|
|
return fmt.Errorf("failed to update sequence: %s, error: %w", value, err)
|
2019-12-10 07:49:05 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-02-20 05:20:30 +00:00
|
|
|
_ = hash.Register("dummy", hash.NewDummyHasher)
|
|
|
|
setting.PasswordHashAlgo, _ = hash.SetDefaultPasswordHashAlgorithm("dummy")
|
2024-12-29 09:05:56 +08:00
|
|
|
return nil
|
2017-04-25 03:24:51 -04:00
|
|
|
}
|