2014-02-12 12:49:46 -05:00
// Copyright 2014 The Gogs Authors. All rights reserved.
2018-06-20 01:06:01 -04:00
// Copyright 2018 The Gitea Authors. All rights reserved.
2014-02-12 12:49:46 -05:00
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package models
2014-02-13 23:23:23 +08:00
2014-02-18 17:48:02 -05:00
import (
2019-12-15 09:51:28 +00:00
"context"
2014-10-19 01:35:24 -04:00
"database/sql"
2016-08-11 14:38:26 -07:00
"errors"
2014-02-18 17:48:02 -05:00
"fmt"
2020-09-06 22:52:01 +01:00
"reflect"
"strings"
2014-02-18 17:48:02 -05:00
2017-10-26 23:10:54 -07:00
"code.gitea.io/gitea/modules/setting"
2016-11-26 01:20:18 +01:00
// Needed for the MySQL driver
2014-02-18 17:48:02 -05:00
_ "github.com/go-sql-driver/mysql"
2019-10-17 17:26:49 +08:00
"xorm.io/xorm"
2020-03-22 23:12:55 +08:00
"xorm.io/xorm/names"
"xorm.io/xorm/schemas"
2016-11-26 01:20:18 +01:00
// Needed for the Postgresql driver
2014-03-17 14:03:58 -04:00
_ "github.com/lib/pq"
2014-02-18 17:48:02 -05:00
2020-03-27 22:12:39 +08:00
// Needed for the MSSQL driver
2016-12-24 09:37:35 +08:00
_ "github.com/denisenkom/go-mssqldb"
2014-02-18 17:48:02 -05:00
)
2014-02-13 23:23:23 +08:00
2014-10-19 01:35:24 -04:00
// Engine represents a xorm engine or session.
type Engine interface {
2017-02-14 11:46:46 +08:00
Table ( tableNameOrBean interface { } ) * xorm . Session
2017-08-22 19:39:52 +08:00
Count ( ... interface { } ) ( int64 , error )
2017-02-04 11:00:07 -05:00
Decr ( column string , arg ... interface { } ) * xorm . Session
2014-10-19 01:35:24 -04:00
Delete ( interface { } ) ( int64 , error )
2018-12-12 02:01:41 +01:00
Exec ( ... interface { } ) ( sql . Result , error )
2015-02-13 00:58:46 -05:00
Find ( interface { } , ... interface { } ) error
2015-02-10 23:44:16 -05:00
Get ( interface { } ) ( bool , error )
2017-10-04 21:43:04 -07:00
ID ( interface { } ) * xorm . Session
2016-08-15 18:40:32 -07:00
In ( string , ... interface { } ) * xorm . Session
2017-02-04 11:00:07 -05:00
Incr ( column string , arg ... interface { } ) * xorm . Session
2014-10-19 01:35:24 -04:00
Insert ( ... interface { } ) ( int64 , error )
2015-02-13 00:58:46 -05:00
InsertOne ( interface { } ) ( int64 , error )
2016-07-26 17:26:48 +08:00
Iterate ( interface { } , xorm . IterFunc ) error
2017-02-11 20:01:33 +08:00
Join ( joinOperator string , tablename interface { } , condition string , args ... interface { } ) * xorm . Session
2016-11-10 15:20:48 +08:00
SQL ( interface { } , ... interface { } ) * xorm . Session
2016-09-23 07:38:12 +08:00
Where ( interface { } , ... interface { } ) * xorm . Session
2019-06-12 21:41:28 +02:00
Asc ( colNames ... string ) * xorm . Session
2020-08-17 04:07:38 +01:00
Desc ( colNames ... string ) * xorm . Session
2020-01-24 19:00:29 +00:00
Limit ( limit int , start ... int ) * xorm . Session
SumInt ( bean interface { } , columnName string ) ( res int64 , err error )
2014-10-19 01:35:24 -04:00
}
2020-02-15 07:51:25 -03:00
const (
// When queries are broken down in parts because of the number
// of parameters, attempt to break by this amount
maxQueryParameters = 300
)
2014-03-21 01:48:10 -04:00
var (
2019-08-24 17:24:45 +08:00
x * xorm . Engine
tables [ ] interface { }
2016-11-26 01:20:18 +01:00
// HasEngine specifies if we have a xorm.Engine
2014-03-30 10:47:08 -04:00
HasEngine bool
2014-03-21 01:48:10 -04:00
)
2014-04-05 22:46:32 +08:00
func init ( ) {
2014-11-12 06:48:50 -05:00
tables = append ( tables ,
2016-12-30 14:44:54 -02:00
new ( User ) ,
new ( PublicKey ) ,
new ( AccessToken ) ,
new ( Repository ) ,
new ( DeployKey ) ,
new ( Collaboration ) ,
new ( Access ) ,
new ( Upload ) ,
new ( Watch ) ,
new ( Star ) ,
new ( Follow ) ,
new ( Action ) ,
new ( Issue ) ,
new ( PullRequest ) ,
new ( Comment ) ,
new ( Attachment ) ,
new ( Label ) ,
new ( IssueLabel ) ,
new ( Milestone ) ,
new ( Mirror ) ,
new ( Release ) ,
new ( LoginSource ) ,
new ( Webhook ) ,
new ( HookTask ) ,
new ( Team ) ,
new ( OrgUser ) ,
new ( TeamUser ) ,
new ( TeamRepo ) ,
new ( Notice ) ,
new ( EmailAddress ) ,
new ( Notification ) ,
new ( IssueUser ) ,
new ( LFSMetaObject ) ,
2017-01-15 21:14:29 -05:00
new ( TwoFactor ) ,
2017-03-16 02:27:35 +01:00
new ( GPGKey ) ,
2019-04-14 18:43:56 +02:00
new ( GPGKeyImport ) ,
2017-02-04 23:53:46 +08:00
new ( RepoUnit ) ,
2017-02-05 09:35:03 -05:00
new ( RepoRedirect ) ,
2017-02-23 15:05:37 +08:00
new ( ExternalLoginUser ) ,
new ( ProtectedBranch ) ,
2017-03-17 15:16:08 +01:00
new ( UserOpenID ) ,
2017-03-19 16:54:12 -03:00
new ( IssueWatch ) ,
2017-04-21 13:32:31 +02:00
new ( CommitStatus ) ,
2017-09-12 08:48:13 +02:00
new ( Stopwatch ) ,
new ( TrackedTime ) ,
2017-10-26 02:49:16 +02:00
new ( DeletedBranch ) ,
2017-10-26 23:10:54 -07:00
new ( RepoIndexerStatus ) ,
2018-07-17 23:23:58 +02:00
new ( IssueDependency ) ,
2017-11-28 21:58:37 +01:00
new ( LFSLock ) ,
2017-12-04 01:14:26 +02:00
new ( Reaction ) ,
2018-05-09 18:29:04 +02:00
new ( IssueAssignees ) ,
2018-05-19 16:12:37 +02:00
new ( U2FRegistration ) ,
2018-06-21 18:00:13 +02:00
new ( TeamUnit ) ,
2018-08-06 07:43:22 +03:00
new ( Review ) ,
2019-03-08 17:42:50 +01:00
new ( OAuth2Application ) ,
new ( OAuth2AuthorizationCode ) ,
new ( OAuth2Grant ) ,
2019-10-13 21:23:14 +08:00
new ( Task ) ,
2020-02-11 11:34:17 +02:00
new ( LanguageStat ) ,
2020-03-27 12:34:39 +00:00
new ( EmailHash ) ,
2020-08-17 04:07:38 +01:00
new ( Project ) ,
new ( ProjectBoard ) ,
new ( ProjectIssue ) ,
2016-12-30 14:44:54 -02:00
)
2015-08-27 23:06:14 +08:00
2016-11-26 01:20:18 +01:00
gonicNames := [ ] string { "SSL" , "UID" }
2015-08-27 23:06:14 +08:00
for _ , name := range gonicNames {
2020-03-22 23:12:55 +08:00
names . LintGonicMapper [ name ] = true
2015-08-27 23:06:14 +08:00
}
2014-04-05 22:46:32 +08:00
}
2014-09-04 17:19:26 +02:00
func getEngine ( ) ( * xorm . Engine , error ) {
2019-08-24 17:24:45 +08:00
connStr , err := setting . DBConnStr ( )
if err != nil {
return nil , err
2014-03-30 10:47:08 -04:00
}
2017-02-20 16:11:13 +08:00
2020-01-20 12:45:14 -03:00
engine , err := xorm . NewEngine ( setting . Database . Type , connStr )
if err != nil {
return nil , err
}
2020-03-27 10:44:05 +02:00
if setting . Database . Type == "mysql" {
engine . Dialect ( ) . SetParams ( map [ string ] string { "rowFormat" : "DYNAMIC" } )
2020-07-21 20:28:27 +08:00
} else if setting . Database . Type == "mssql" {
engine . Dialect ( ) . SetParams ( map [ string ] string { "DEFAULT_VARCHAR" : "nvarchar" } )
2020-03-27 10:44:05 +02:00
}
2020-01-20 12:45:14 -03:00
engine . SetSchema ( setting . Database . Schema )
2020-09-11 10:30:19 +01:00
if setting . Database . UsePostgreSQL && len ( setting . Database . Schema ) > 0 {
// Add the schema to the search path
if _ , err := engine . Exec ( ` SELECT set_config (
' search_path ' ,
? || ',' || current_setting ( ' search_path ' ) ,
false ) ` ,
setting . Database . Schema ) ; err != nil {
return nil , err
}
}
2020-01-20 12:45:14 -03:00
return engine , nil
2014-09-04 17:19:26 +02:00
}
2016-11-26 01:20:18 +01:00
// NewTestEngine sets a new test xorm.Engine
2020-07-13 01:58:55 -07:00
func NewTestEngine ( ) ( err error ) {
2014-09-04 17:19:26 +02:00
x , err = getEngine ( )
2014-03-30 10:47:08 -04:00
if err != nil {
2015-08-02 12:36:35 +08:00
return fmt . Errorf ( "Connect to database: %v" , err )
2014-03-30 10:47:08 -04:00
}
2014-09-04 17:19:26 +02:00
2020-03-22 23:12:55 +08:00
x . SetMapper ( names . GonicMapper { } )
2019-05-14 15:04:07 +08:00
x . SetLogger ( NewXORMLogger ( ! setting . ProdMode ) )
2017-09-13 13:18:22 +08:00
x . ShowSQL ( ! setting . ProdMode )
2015-09-03 05:05:58 -04:00
return x . StoreEngine ( "InnoDB" ) . Sync2 ( tables ... )
2014-03-30 10:47:08 -04:00
}
2016-11-26 01:20:18 +01:00
// SetEngine sets the xorm.Engine
2014-03-29 17:50:51 -04:00
func SetEngine ( ) ( err error ) {
2014-09-04 17:19:26 +02:00
x , err = getEngine ( )
2014-02-18 17:48:02 -05:00
if err != nil {
2017-01-29 12:13:57 -08:00
return fmt . Errorf ( "Failed to connect to database: %v" , err )
2014-02-18 17:48:02 -05:00
}
2020-03-22 23:12:55 +08:00
x . SetMapper ( names . GonicMapper { } )
2014-12-06 20:22:48 -05:00
// WARNING: for serv command, MUST remove the output to os.stdout,
2014-03-20 16:04:56 -04:00
// so use log file to instead print to stdout.
2019-08-24 17:24:45 +08:00
x . SetLogger ( NewXORMLogger ( setting . Database . LogSQL ) )
x . ShowSQL ( setting . Database . LogSQL )
2019-10-21 22:20:47 +01:00
x . SetMaxOpenConns ( setting . Database . MaxOpenConns )
x . SetMaxIdleConns ( setting . Database . MaxIdleConns )
x . SetConnMaxLifetime ( setting . Database . ConnMaxLifetime )
2014-03-29 17:50:51 -04:00
return nil
2014-02-18 17:48:02 -05:00
}
2016-11-26 01:20:18 +01:00
// NewEngine initializes a new xorm.Engine
2020-05-29 15:24:15 +02: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.
2019-12-15 09:51:28 +00:00
func NewEngine ( ctx context . Context , migrateFunc func ( * xorm . Engine ) error ) ( err error ) {
2014-03-29 17:50:51 -04:00
if err = SetEngine ( ) ; err != nil {
return err
2014-04-05 22:46:32 +08:00
}
2015-01-22 14:49:52 +02:00
2019-12-15 09:51:28 +00:00
x . SetDefaultContext ( ctx )
2016-11-24 22:30:36 +08:00
if err = x . Ping ( ) ; err != nil {
return err
}
2017-07-02 16:50:57 +03:00
if err = migrateFunc ( x ) ; err != nil {
2015-02-11 21:58:37 -05:00
return fmt . Errorf ( "migrate: %v" , err )
2015-01-22 14:49:52 +02:00
}
2014-10-28 16:40:09 +01:00
if err = x . StoreEngine ( "InnoDB" ) . Sync2 ( tables ... ) ; err != nil {
2016-11-28 15:25:16 +08:00
return fmt . Errorf ( "sync database struct error: %v" , err )
2014-02-19 17:50:53 +08:00
}
2015-01-23 09:54:16 +02:00
2014-03-29 17:50:51 -04:00
return nil
2014-02-18 17:48:02 -05:00
}
2014-03-20 16:04:56 -04:00
2020-09-06 22:52:01 +01: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 01:20:18 +01:00
// Statistic contains the database statistics
2014-03-20 16:04:56 -04:00
type Statistic struct {
Counter struct {
2014-08-28 22:29:00 +08:00
User , Org , PublicKey ,
Repo , Watch , Star , Action , Access ,
Issue , Comment , Oauth , Follow ,
Mirror , Release , LoginSource , Webhook ,
Milestone , Label , HookTask ,
Team , UpdateTask , Attachment int64
2014-03-20 16:04:56 -04:00
}
}
2016-11-26 01:20:18 +01:00
// GetStatistic returns the database statistics
2014-03-20 16:04:56 -04:00
func GetStatistic ( ) ( stats Statistic ) {
2014-07-07 04:15:08 -04:00
stats . Counter . User = CountUsers ( )
2014-08-28 22:29:00 +08:00
stats . Counter . Org = CountOrganizations ( )
2014-06-21 00:51:41 -04:00
stats . Counter . PublicKey , _ = x . Count ( new ( PublicKey ) )
2016-07-24 14:32:46 +08:00
stats . Counter . Repo = CountRepositories ( true )
2014-06-21 00:51:41 -04:00
stats . Counter . Watch , _ = x . Count ( new ( Watch ) )
2014-08-28 22:29:00 +08:00
stats . Counter . Star , _ = x . Count ( new ( Star ) )
2014-06-21 00:51:41 -04:00
stats . Counter . Action , _ = x . Count ( new ( Action ) )
stats . Counter . Access , _ = x . Count ( new ( Access ) )
stats . Counter . Issue , _ = x . Count ( new ( Issue ) )
stats . Counter . Comment , _ = x . Count ( new ( Comment ) )
2015-09-17 16:11:44 -04:00
stats . Counter . Oauth = 0
2014-08-28 22:29:00 +08:00
stats . Counter . Follow , _ = x . Count ( new ( Follow ) )
stats . Counter . Mirror , _ = x . Count ( new ( Mirror ) )
2014-06-21 00:51:41 -04:00
stats . Counter . Release , _ = x . Count ( new ( Release ) )
2015-09-10 15:45:03 -04:00
stats . Counter . LoginSource = CountLoginSources ( )
2014-06-21 00:51:41 -04:00
stats . Counter . Webhook , _ = x . Count ( new ( Webhook ) )
stats . Counter . Milestone , _ = x . Count ( new ( Milestone ) )
2014-08-28 22:29:00 +08:00
stats . Counter . Label , _ = x . Count ( new ( Label ) )
stats . Counter . HookTask , _ = x . Count ( new ( HookTask ) )
stats . Counter . Team , _ = x . Count ( new ( Team ) )
stats . Counter . Attachment , _ = x . Count ( new ( Attachment ) )
2014-03-23 16:31:13 +08:00
return
2014-03-20 16:04:56 -04:00
}
2014-05-05 00:55:17 -04:00
2016-11-26 01:20:18 +01:00
// Ping tests if database is alive
2014-08-06 17:21:24 -04:00
func Ping ( ) error {
2018-04-04 02:34:27 -07:00
if x != nil {
return x . Ping ( )
}
return errors . New ( "database not configured" )
2014-08-06 17:21:24 -04:00
}
2017-01-03 16:20:28 +08:00
// DumpDatabase dumps all data from database according the special database SQL syntax to file system.
func DumpDatabase ( filePath string , dbType string ) error {
2020-03-22 23:12:55 +08:00
var tbs [ ] * schemas . Table
2017-01-03 16:20:28 +08:00
for _ , t := range tables {
2020-03-22 23:12:55 +08:00
t , err := x . TableInfo ( t )
if err != nil {
return err
}
tbs = append ( tbs , t )
2017-01-03 16:20:28 +08:00
}
2020-09-08 00:27:17 +02:00
type Version struct {
ID int64 ` xorm:"pk autoincr" `
Version int64
}
t , err := x . TableInfo ( Version { } )
if err != nil {
return err
}
tbs = append ( tbs , t )
2017-01-03 16:20:28 +08:00
if len ( dbType ) > 0 {
2020-03-22 23:12:55 +08:00
return x . DumpTablesToFile ( tbs , filePath , schemas . DBType ( dbType ) )
2017-01-03 16:20:28 +08:00
}
return x . DumpTablesToFile ( tbs , filePath )
2014-05-05 00:55:17 -04:00
}
2019-07-07 03:24:50 +08:00
// MaxBatchInsertSize returns the table's max batch insert size
func MaxBatchInsertSize ( bean interface { } ) int {
2020-03-22 23:12:55 +08:00
t , err := x . TableInfo ( bean )
if err != nil {
return 50
}
2019-07-07 03:24:50 +08:00
return 999 / len ( t . ColumnsSeq ( ) )
}
2019-09-06 10:20:09 +08: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-09 03:15:35 +08: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 )
}