2014-02-12 21:49:46 +04:00
// Copyright 2014 The Gogs Authors. All rights reserved.
2018-06-20 08:06:01 +03:00
// Copyright 2018 The Gitea Authors. All rights reserved.
2014-02-12 21:49:46 +04:00
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
2021-09-19 14:49:59 +03:00
package db
2014-02-13 19:23:23 +04:00
2014-02-19 02:48:02 +04:00
import (
2019-12-15 12:51:28 +03:00
"context"
2014-10-19 09:35:24 +04:00
"database/sql"
2016-08-12 00:38:26 +03:00
"errors"
2014-02-19 02:48:02 +04:00
"fmt"
2021-09-19 14:49:59 +03:00
"io"
2020-09-07 00:52:01 +03:00
"reflect"
"strings"
2014-02-19 02:48:02 +04:00
2017-10-27 09:10:54 +03:00
"code.gitea.io/gitea/modules/setting"
2016-11-26 03:20:18 +03:00
// Needed for the MySQL driver
2014-02-19 02:48:02 +04:00
_ "github.com/go-sql-driver/mysql"
2019-10-17 12:26:49 +03:00
"xorm.io/xorm"
2020-03-22 18:12:55 +03:00
"xorm.io/xorm/names"
"xorm.io/xorm/schemas"
2016-11-26 03:20:18 +03:00
// Needed for the Postgresql driver
2014-03-17 22:03:58 +04:00
_ "github.com/lib/pq"
2014-02-19 02:48:02 +04:00
2020-03-27 17:12:39 +03:00
// Needed for the MSSQL driver
2016-12-24 04:37:35 +03:00
_ "github.com/denisenkom/go-mssqldb"
2014-02-19 02:48:02 +04:00
)
2014-02-13 19:23:23 +04:00
2021-09-19 14:49:59 +03:00
var (
x * xorm . Engine
tables [ ] interface { }
initFuncs [ ] func ( ) error
// HasEngine specifies if we have a xorm.Engine
HasEngine bool
)
2014-10-19 09:35:24 +04:00
// Engine represents a xorm engine or session.
type Engine interface {
2017-02-14 06:46:46 +03:00
Table ( tableNameOrBean interface { } ) * xorm . Session
2017-08-22 14:39:52 +03:00
Count ( ... interface { } ) ( int64 , error )
2017-02-04 19:00:07 +03:00
Decr ( column string , arg ... interface { } ) * xorm . Session
2021-08-13 02:11:42 +03:00
Delete ( ... interface { } ) ( int64 , error )
2018-12-12 04:01:41 +03:00
Exec ( ... interface { } ) ( sql . Result , error )
2015-02-13 08:58:46 +03:00
Find ( interface { } , ... interface { } ) error
2015-02-11 07:44:16 +03:00
Get ( interface { } ) ( bool , error )
2017-10-05 07:43:04 +03:00
ID ( interface { } ) * xorm . Session
2016-08-16 04:40:32 +03:00
In ( string , ... interface { } ) * xorm . Session
2017-02-04 19:00:07 +03:00
Incr ( column string , arg ... interface { } ) * xorm . Session
2014-10-19 09:35:24 +04:00
Insert ( ... interface { } ) ( int64 , error )
2015-02-13 08:58:46 +03:00
InsertOne ( interface { } ) ( int64 , error )
2016-07-26 12:26:48 +03:00
Iterate ( interface { } , xorm . IterFunc ) error
2017-02-11 15:01:33 +03:00
Join ( joinOperator string , tablename interface { } , condition string , args ... interface { } ) * xorm . Session
2016-11-10 10:20:48 +03:00
SQL ( interface { } , ... interface { } ) * xorm . Session
2016-09-23 02:38:12 +03:00
Where ( interface { } , ... interface { } ) * xorm . Session
2019-06-12 22:41:28 +03:00
Asc ( colNames ... string ) * xorm . Session
2020-08-17 06:07:38 +03:00
Desc ( colNames ... string ) * xorm . Session
2020-01-24 22:00:29 +03:00
Limit ( limit int , start ... int ) * xorm . Session
SumInt ( bean interface { } , columnName string ) ( res int64 , err error )
2021-09-19 14:49:59 +03:00
Sync2 ( ... interface { } ) error
Select ( string ) * xorm . Session
NotIn ( string , ... interface { } ) * xorm . Session
OrderBy ( string ) * xorm . Session
Exist ( ... interface { } ) ( bool , error )
Distinct ( ... string ) * xorm . Session
Query ( ... interface { } ) ( [ ] map [ string ] [ ] byte , error )
Cols ( ... string ) * xorm . Session
2014-10-19 09:35:24 +04:00
}
2021-09-19 14:49:59 +03:00
// TableInfo returns table's information via an object
func TableInfo ( v interface { } ) ( * schemas . Table , error ) {
return x . TableInfo ( v )
}
2020-02-15 13:51:25 +03:00
2021-09-19 14:49:59 +03:00
// DumpTables dump tables information
func DumpTables ( tables [ ] * schemas . Table , w io . Writer , tp ... schemas . DBType ) error {
return x . DumpTables ( tables , w , tp ... )
}
2016-11-26 03:20:18 +03:00
2021-09-19 14:49:59 +03:00
// RegisterModel registers model, if initfunc provided, it will be invoked after data model sync
func RegisterModel ( bean interface { } , initFunc ... func ( ) error ) {
tables = append ( tables , bean )
if len ( initFuncs ) > 0 && initFunc [ 0 ] != nil {
initFuncs = append ( initFuncs , initFunc [ 0 ] )
}
}
2014-03-21 09:48:10 +04:00
2014-04-05 18:46:32 +04:00
func init ( ) {
2016-11-26 03:20:18 +03:00
gonicNames := [ ] string { "SSL" , "UID" }
2015-08-27 18:06:14 +03:00
for _ , name := range gonicNames {
2020-03-22 18:12:55 +03:00
names . LintGonicMapper [ name ] = true
2015-08-27 18:06:14 +03:00
}
2014-04-05 18:46:32 +04:00
}
2021-10-30 17:32:11 +03:00
// NewEngine returns a new xorm engine from the configuration
func NewEngine ( ) ( * xorm . Engine , error ) {
2019-08-24 12:24:45 +03:00
connStr , err := setting . DBConnStr ( )
if err != nil {
return nil , err
2014-03-30 18:47:08 +04:00
}
2017-02-20 11:11:13 +03:00
2021-01-02 05:07:43 +03:00
var engine * xorm . Engine
if setting . Database . UsePostgreSQL && len ( setting . Database . Schema ) > 0 {
// OK whilst we sort out our schema issues - create a schema aware postgres
registerPostgresSchemaDriver ( )
engine , err = xorm . NewEngine ( "postgresschema" , connStr )
} else {
engine , err = xorm . NewEngine ( setting . Database . Type , connStr )
}
2020-01-20 18:45:14 +03:00
if err != nil {
return nil , err
}
2020-03-27 11:44:05 +03:00
if setting . Database . Type == "mysql" {
engine . Dialect ( ) . SetParams ( map [ string ] string { "rowFormat" : "DYNAMIC" } )
2020-07-21 15:28:27 +03:00
} else if setting . Database . Type == "mssql" {
engine . Dialect ( ) . SetParams ( map [ string ] string { "DEFAULT_VARCHAR" : "nvarchar" } )
2020-03-27 11:44:05 +03:00
}
2020-01-20 18:45:14 +03:00
engine . SetSchema ( setting . Database . Schema )
return engine , nil
2014-09-04 19:19:26 +04:00
}
2021-11-12 17:36:47 +03:00
//SyncAllTables sync the schemas of all tables, is required by unit test code
func SyncAllTables ( ) error {
2021-06-14 05:22:55 +03:00
return x . StoreEngine ( "InnoDB" ) . Sync2 ( tables ... )
}
2021-10-30 17:32:11 +03:00
// InitEngine sets the xorm.Engine
2021-11-07 06:11:27 +03:00
func InitEngine ( ctx context . Context ) ( err error ) {
2021-10-30 17:32:11 +03:00
x , err = NewEngine ( )
2014-02-19 02:48:02 +04:00
if err != nil {
2017-01-29 23:13:57 +03:00
return fmt . Errorf ( "Failed to connect to database: %v" , err )
2014-02-19 02:48:02 +04:00
}
2020-03-22 18:12:55 +03:00
x . SetMapper ( names . GonicMapper { } )
2014-12-07 04:22:48 +03:00
// WARNING: for serv command, MUST remove the output to os.stdout,
2014-03-21 00:04:56 +04:00
// so use log file to instead print to stdout.
2019-08-24 12:24:45 +03:00
x . SetLogger ( NewXORMLogger ( setting . Database . LogSQL ) )
x . ShowSQL ( setting . Database . LogSQL )
2019-10-22 00:20:47 +03:00
x . SetMaxOpenConns ( setting . Database . MaxOpenConns )
x . SetMaxIdleConns ( setting . Database . MaxIdleConns )
x . SetConnMaxLifetime ( setting . Database . ConnMaxLifetime )
2021-11-07 06:11:27 +03:00
DefaultContext = & Context {
Context : ctx ,
e : x ,
}
x . SetDefaultContext ( ctx )
2014-03-30 01:50:51 +04:00
return nil
2014-02-19 02:48:02 +04:00
}
2021-11-16 11:53:21 +03:00
// SetEngine is used by unit test code
func SetEngine ( eng * xorm . Engine ) {
x = eng
DefaultContext = & Context {
Context : context . Background ( ) ,
e : x ,
}
}
2021-10-30 17:32:11 +03:00
// InitEngineWithMigration initializes a new xorm.Engine
2020-05-29 16:24:15 +03:00
// This function must never call .Sync2() if the provided migration function fails.
// When called from the "doctor" command, the migration function is a version check
// that prevents the doctor from fixing anything in the database if the migration level
// is different from the expected value.
2021-10-30 17:32:11 +03:00
func InitEngineWithMigration ( ctx context . Context , migrateFunc func ( * xorm . Engine ) error ) ( err error ) {
2021-11-07 06:11:27 +03:00
if err = InitEngine ( ctx ) ; err != nil {
2014-03-30 01:50:51 +04:00
return err
2014-04-05 18:46:32 +04:00
}
2015-01-22 15:49:52 +03:00
2016-11-24 17:30:36 +03:00
if err = x . Ping ( ) ; err != nil {
return err
}
2021-11-07 06:11:27 +03:00
// We have to run migrateFunc here in case the user is re-running installation on a previously created DB.
// If we do not then table schemas will be changed and there will be conflicts when the migrations run properly.
//
// Installation should only be being re-run if users want to recover an old database.
// However, we should think carefully about should we support re-install on an installed instance,
// as there may be other problems due to secret reinitialization.
2017-07-02 16:50:57 +03:00
if err = migrateFunc ( x ) ; err != nil {
2015-02-12 05:58:37 +03:00
return fmt . Errorf ( "migrate: %v" , err )
2015-01-22 15:49:52 +03:00
}
2021-11-12 17:36:47 +03:00
if err = SyncAllTables ( ) ; err != nil {
2016-11-28 10:25:16 +03:00
return fmt . Errorf ( "sync database struct error: %v" , err )
2014-02-19 13:50:53 +04:00
}
2015-01-23 10:54:16 +03:00
2021-09-19 14:49:59 +03:00
for _ , initFunc := range initFuncs {
if err := initFunc ( ) ; err != nil {
return fmt . Errorf ( "initFunc failed: %v" , err )
2021-08-17 21:30:42 +03:00
}
}
2014-03-30 01:50:51 +04:00
return nil
2014-02-19 02:48:02 +04:00
}
2014-03-21 00:04:56 +04:00
2020-09-07 00:52:01 +03:00
// NamesToBean return a list of beans or an error
func NamesToBean ( names ... string ) ( [ ] interface { } , error ) {
beans := [ ] interface { } { }
if len ( names ) == 0 {
beans = append ( beans , tables ... )
return beans , nil
}
// Need to map provided names to beans...
beanMap := make ( map [ string ] interface { } )
for _ , bean := range tables {
beanMap [ strings . ToLower ( reflect . Indirect ( reflect . ValueOf ( bean ) ) . Type ( ) . Name ( ) ) ] = bean
beanMap [ strings . ToLower ( x . TableName ( bean ) ) ] = bean
beanMap [ strings . ToLower ( x . TableName ( bean , true ) ) ] = bean
}
gotBean := make ( map [ interface { } ] bool )
for _ , name := range names {
bean , ok := beanMap [ strings . ToLower ( strings . TrimSpace ( name ) ) ]
if ! ok {
return nil , fmt . Errorf ( "No table found that matches: %s" , name )
}
if ! gotBean [ bean ] {
beans = append ( beans , bean )
gotBean [ bean ] = true
}
}
return beans , nil
}
2016-11-26 03:20:18 +03:00
// Ping tests if database is alive
2014-08-07 01:21:24 +04:00
func Ping ( ) error {
2018-04-04 12:34:27 +03:00
if x != nil {
return x . Ping ( )
}
return errors . New ( "database not configured" )
2014-08-07 01:21:24 +04:00
}
2017-01-03 11:20:28 +03:00
// DumpDatabase dumps all data from database according the special database SQL syntax to file system.
2021-03-14 21:52:12 +03:00
func DumpDatabase ( filePath , dbType string ) error {
2020-03-22 18:12:55 +03:00
var tbs [ ] * schemas . Table
2017-01-03 11:20:28 +03:00
for _ , t := range tables {
2020-03-22 18:12:55 +03:00
t , err := x . TableInfo ( t )
if err != nil {
return err
}
tbs = append ( tbs , t )
2017-01-03 11:20:28 +03:00
}
2020-09-08 01:27:17 +03:00
type Version struct {
ID int64 ` xorm:"pk autoincr" `
Version int64
}
2021-05-07 02:17:43 +03:00
t , err := x . TableInfo ( & Version { } )
2020-09-08 01:27:17 +03:00
if err != nil {
return err
}
tbs = append ( tbs , t )
2017-01-03 11:20:28 +03:00
if len ( dbType ) > 0 {
2020-03-22 18:12:55 +03:00
return x . DumpTablesToFile ( tbs , filePath , schemas . DBType ( dbType ) )
2017-01-03 11:20:28 +03:00
}
return x . DumpTablesToFile ( tbs , filePath )
2014-05-05 08:55:17 +04:00
}
2019-07-06 22:24:50 +03:00
// MaxBatchInsertSize returns the table's max batch insert size
func MaxBatchInsertSize ( bean interface { } ) int {
2020-03-22 18:12:55 +03:00
t , err := x . TableInfo ( bean )
if err != nil {
return 50
}
2019-07-06 22:24:50 +03:00
return 999 / len ( t . ColumnsSeq ( ) )
}
2019-09-06 05:20:09 +03:00
// Count returns records number according struct's fields as database query conditions
func Count ( bean interface { } ) ( int64 , error ) {
return x . Count ( bean )
}
2019-12-08 22:15:35 +03:00
// IsTableNotEmpty returns true if table has at least one record
func IsTableNotEmpty ( tableName string ) ( bool , error ) {
return x . Table ( tableName ) . Exist ( )
}
// DeleteAllRecords will delete all the records of this table
func DeleteAllRecords ( tableName string ) error {
_ , err := x . Exec ( fmt . Sprintf ( "DELETE FROM %s" , tableName ) )
return err
}
// GetMaxID will return max id of the table
func GetMaxID ( beanOrTableName interface { } ) ( maxID int64 , err error ) {
_ , err = x . Select ( "MAX(id)" ) . Table ( beanOrTableName ) . Get ( & maxID )
return
}
// FindByMaxID filled results as the condition from database
func FindByMaxID ( maxID int64 , limit int , results interface { } ) error {
return x . Where ( "id <= ?" , maxID ) .
OrderBy ( "id DESC" ) .
Limit ( limit ) .
Find ( results )
}