2014-02-12 12:49:46 -05:00
// Copyright 2014 The Gogs Authors. All rights reserved.
// 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 (
2014-10-19 01:35:24 -04:00
"database/sql"
2014-02-18 17:48:02 -05:00
"fmt"
"os"
2014-03-21 01:09:22 -04:00
"path"
2014-04-14 14:49:50 +08:00
"strings"
2014-02-18 17:48:02 -05:00
_ "github.com/go-sql-driver/mysql"
2015-01-23 09:54:16 +02:00
"github.com/go-xorm/core"
2014-04-18 09:35:09 -04:00
"github.com/go-xorm/xorm"
2014-03-17 14:03:58 -04:00
_ "github.com/lib/pq"
2014-02-18 17:48:02 -05:00
2015-02-23 02:15:53 -05:00
"github.com/gogits/gogs/models/migrations"
2014-05-25 20:11:25 -04:00
"github.com/gogits/gogs/modules/setting"
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 {
Delete ( interface { } ) ( int64 , error )
Exec ( string , ... 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 )
2014-10-19 01:35:24 -04:00
Insert ( ... interface { } ) ( int64 , error )
2015-02-13 00:58:46 -05:00
InsertOne ( interface { } ) ( int64 , error )
2015-02-10 23:44:16 -05:00
Id ( interface { } ) * xorm . Session
2015-02-13 00:58:46 -05:00
Sql ( string , ... interface { } ) * xorm . Session
2015-02-10 23:44:16 -05:00
Where ( string , ... interface { } ) * xorm . Session
2014-10-19 01:35:24 -04:00
}
2015-02-13 00:58:46 -05:00
func sessionRelease ( sess * xorm . Session ) {
if ! sess . IsCommitedOrRollbacked {
sess . Rollback ( )
}
sess . Close ( )
}
2014-03-21 01:48:10 -04:00
var (
2014-10-19 01:35:24 -04:00
x * xorm . Engine
tables [ ] interface { }
2014-03-30 10:47:08 -04:00
HasEngine bool
2014-03-21 01:48:10 -04:00
2014-03-21 03:27:59 -04:00
DbCfg struct {
2015-02-01 12:41:03 -05:00
Type , Host , Name , User , Passwd , Path , SSLMode string
2014-03-21 01:48:10 -04:00
}
2014-03-30 16:01:50 -04:00
2014-04-12 16:24:09 -04:00
EnableSQLite3 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 ,
2015-02-11 21:58:37 -05:00
new ( User ) , new ( PublicKey ) , new ( Oauth2 ) , new ( AccessToken ) ,
new ( Repository ) , new ( Collaboration ) , new ( Access ) ,
new ( Watch ) , new ( Star ) , new ( Follow ) , new ( Action ) ,
2014-10-08 18:29:18 -04:00
new ( Issue ) , new ( Comment ) , new ( Attachment ) , new ( IssueUser ) , new ( Label ) , new ( Milestone ) ,
new ( Mirror ) , new ( Release ) , new ( LoginSource ) , new ( Webhook ) ,
2015-02-23 02:15:53 -05:00
new ( UpdateTask ) , new ( HookTask ) ,
new ( Team ) , new ( OrgUser ) , new ( TeamUser ) , new ( TeamRepo ) ,
2015-02-11 21:58:37 -05:00
new ( Notice ) , new ( EmailAddress ) )
2014-04-05 22:46:32 +08:00
}
2014-03-21 01:48:10 -04:00
func LoadModelsConfig ( ) {
2014-12-31 18:37:29 +08:00
sec := setting . Cfg . Section ( "database" )
DbCfg . Type = sec . Key ( "DB_TYPE" ) . String ( )
2015-02-11 21:58:37 -05:00
switch DbCfg . Type {
case "sqlite3" :
setting . UseSQLite3 = true
case "mysql" :
setting . UseMySQL = true
case "postgres" :
setting . UsePostgreSQL = true
2014-03-30 16:01:50 -04:00
}
2014-12-31 18:37:29 +08:00
DbCfg . Host = sec . Key ( "HOST" ) . String ( )
DbCfg . Name = sec . Key ( "NAME" ) . String ( )
DbCfg . User = sec . Key ( "USER" ) . String ( )
2015-02-01 12:41:03 -05:00
if len ( DbCfg . Passwd ) == 0 {
DbCfg . Passwd = sec . Key ( "PASSWD" ) . String ( )
2014-06-10 19:11:53 -04:00
}
2015-02-01 12:41:03 -05:00
DbCfg . SSLMode = sec . Key ( "SSL_MODE" ) . String ( )
2014-12-31 18:37:29 +08:00
DbCfg . Path = sec . Key ( "PATH" ) . MustString ( "data/gogs.db" )
2014-03-21 01:48:10 -04:00
}
2014-02-18 17:48:02 -05:00
2014-09-04 17:19:26 +02:00
func getEngine ( ) ( * xorm . Engine , error ) {
cnnstr := ""
2014-03-30 10:47:08 -04:00
switch DbCfg . Type {
case "mysql" :
2015-03-14 02:21:47 +08:00
if DbCfg . Host [ 0 ] == '/' { // looks like a unix socket
cnnstr = fmt . Sprintf ( "%s:%s@unix(%s)/%s?charset=utf8" ,
DbCfg . User , DbCfg . Passwd , DbCfg . Host , DbCfg . Name )
} else {
cnnstr = fmt . Sprintf ( "%s:%s@tcp(%s)/%s?charset=utf8" ,
DbCfg . User , DbCfg . Passwd , DbCfg . Host , DbCfg . Name )
}
2014-03-30 10:47:08 -04:00
case "postgres" :
2014-04-14 14:49:50 +08:00
var host , port = "127.0.0.1" , "5432"
fields := strings . Split ( DbCfg . Host , ":" )
2014-05-01 18:53:41 -04:00
if len ( fields ) > 0 && len ( strings . TrimSpace ( fields [ 0 ] ) ) > 0 {
2014-04-14 14:49:50 +08:00
host = fields [ 0 ]
}
2014-05-01 18:53:41 -04:00
if len ( fields ) > 1 && len ( strings . TrimSpace ( fields [ 1 ] ) ) > 0 {
2014-04-14 14:49:50 +08:00
port = fields [ 1 ]
}
2014-09-04 17:19:26 +02:00
cnnstr = fmt . Sprintf ( "user=%s password=%s host=%s port=%s dbname=%s sslmode=%s" ,
2015-02-01 12:41:03 -05:00
DbCfg . User , DbCfg . Passwd , host , port , DbCfg . Name , DbCfg . SSLMode )
2014-04-12 11:48:12 -07:00
case "sqlite3" :
2014-04-12 16:24:09 -04:00
if ! EnableSQLite3 {
2014-09-04 17:19:26 +02:00
return nil , fmt . Errorf ( "Unknown database type: %s" , DbCfg . Type )
2014-04-12 16:24:09 -04:00
}
2014-04-12 11:48:12 -07:00
os . MkdirAll ( path . Dir ( DbCfg . Path ) , os . ModePerm )
2014-09-12 18:58:24 -04:00
cnnstr = "file:" + DbCfg . Path + "?cache=shared&mode=rwc"
2014-03-30 10:47:08 -04:00
default :
2014-09-04 17:19:26 +02:00
return nil , fmt . Errorf ( "Unknown database type: %s" , DbCfg . Type )
2014-03-30 10:47:08 -04:00
}
2014-09-04 17:19:26 +02:00
return xorm . NewEngine ( DbCfg . Type , cnnstr )
}
func NewTestEngine ( x * xorm . Engine ) ( err error ) {
x , err = getEngine ( )
2014-03-30 10:47:08 -04:00
if err != nil {
2015-02-01 12:41:03 -05: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
2015-01-23 09:54:16 +02:00
x . SetMapper ( core . GonicMapper { } )
2014-04-05 22:46:32 +08:00
return x . Sync ( tables ... )
2014-03-30 10:47:08 -04:00
}
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 {
2015-02-01 12:41:03 -05:00
return fmt . Errorf ( "connect to database: %v" , err )
2014-02-18 17:48:02 -05:00
}
2015-01-23 09:54:16 +02:00
x . SetMapper ( core . 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.
2014-05-28 01:53:06 -04:00
logPath := path . Join ( setting . LogRootPath , "xorm.log" )
2014-03-31 21:26:15 +08:00
os . MkdirAll ( path . Dir ( logPath ) , os . ModePerm )
2014-02-25 14:01:52 +08:00
2014-03-31 21:26:15 +08:00
f , err := os . Create ( logPath )
2014-03-17 14:03:58 -04:00
if err != nil {
2014-04-04 18:31:09 -04:00
return fmt . Errorf ( "models.init(fail to create xorm.log): %v" , err )
2014-03-17 14:03:58 -04:00
}
2015-02-14 17:01:33 -05:00
x . SetLogger ( xorm . NewSimpleLogger ( f ) )
2014-03-31 21:26:15 +08:00
2014-06-21 00:51:41 -04:00
x . ShowSQL = true
2014-09-08 12:11:25 +08:00
x . ShowInfo = true
2014-06-21 00:51:41 -04:00
x . ShowDebug = true
x . ShowErr = true
2014-08-29 11:24:37 +08:00
x . ShowWarn = true
2014-03-29 17:50:51 -04:00
return nil
2014-02-18 17:48:02 -05:00
}
2014-03-29 17:50:51 -04:00
func NewEngine ( ) ( err error ) {
if err = SetEngine ( ) ; err != nil {
return err
2014-04-05 22:46:32 +08:00
}
2015-01-22 14:49:52 +02:00
if err = migrations . Migrate ( 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 {
2014-03-29 17:50:51 -04:00
return fmt . Errorf ( "sync database struct error: %v\n" , 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
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
}
}
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 ) )
2014-08-28 22:29:00 +08:00
stats . Counter . Repo = CountRepositories ( )
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 ) )
stats . Counter . Oauth , _ = x . Count ( new ( Oauth2 ) )
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 ) )
stats . Counter . LoginSource , _ = x . Count ( new ( LoginSource ) )
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 . UpdateTask , _ = x . Count ( new ( UpdateTask ) )
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
2014-08-06 17:21:24 -04:00
func Ping ( ) error {
return x . Ping ( )
}
2014-05-05 00:55:17 -04:00
// DumpDatabase dumps all data from database to file system.
func DumpDatabase ( filePath string ) error {
2014-06-21 00:51:41 -04:00
return x . DumpAllToFile ( filePath )
2014-05-05 00:55:17 -04:00
}