2014-02-12 21:49:46 +04: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 19:23:23 +04:00
2014-02-19 02:48:02 +04:00
import (
"fmt"
"os"
2014-03-21 09:09:22 +04:00
"path"
2014-04-14 10:49:50 +04:00
"strings"
2014-02-19 02:48:02 +04:00
_ "github.com/go-sql-driver/mysql"
2014-04-18 17:35:09 +04:00
"github.com/go-xorm/xorm"
2014-03-17 22:03:58 +04:00
_ "github.com/lib/pq"
2014-02-19 02:48:02 +04:00
2014-05-26 04:11:25 +04:00
"github.com/gogits/gogs/modules/setting"
2014-02-19 02:48:02 +04:00
)
2014-02-13 19:23:23 +04:00
2014-03-21 09:48:10 +04:00
var (
2014-06-21 08:51:41 +04:00
x * xorm . Engine
2014-04-05 18:46:32 +04:00
tables [ ] interface { }
2014-03-30 18:47:08 +04:00
HasEngine bool
2014-03-21 09:48:10 +04:00
2014-03-21 11:27:59 +04:00
DbCfg struct {
2014-03-21 09:48:10 +04:00
Type , Host , Name , User , Pwd , Path , SslMode string
}
2014-03-31 00:01:50 +04:00
2014-04-13 00:24:09 +04:00
EnableSQLite3 bool
UseSQLite3 bool
2014-03-21 09:48:10 +04:00
)
2014-04-05 18:46:32 +04:00
func init ( ) {
2014-08-11 07:11:18 +04:00
tables = append ( tables , new ( User ) , new ( PublicKey ) ,
new ( Repository ) , new ( Watch ) , new ( Star ) , new ( Action ) , new ( Access ) ,
new ( Issue ) , new ( Comment ) , new ( Oauth2 ) , new ( Follow ) ,
2014-05-12 22:06:42 +04:00
new ( Mirror ) , new ( Release ) , new ( LoginSource ) , new ( Webhook ) , new ( IssueUser ) ,
2014-06-28 19:56:41 +04:00
new ( Milestone ) , new ( Label ) , new ( HookTask ) , new ( Team ) , new ( OrgUser ) , new ( TeamUser ) ,
2014-07-23 23:15:47 +04:00
new ( UpdateTask ) , new ( Attachment ) )
2014-04-05 18:46:32 +04:00
}
2014-03-21 09:48:10 +04:00
func LoadModelsConfig ( ) {
2014-05-26 04:11:25 +04:00
DbCfg . Type = setting . Cfg . MustValue ( "database" , "DB_TYPE" )
2014-03-31 00:01:50 +04:00
if DbCfg . Type == "sqlite3" {
UseSQLite3 = true
}
2014-05-26 04:11:25 +04:00
DbCfg . Host = setting . Cfg . MustValue ( "database" , "HOST" )
DbCfg . Name = setting . Cfg . MustValue ( "database" , "NAME" )
DbCfg . User = setting . Cfg . MustValue ( "database" , "USER" )
2014-06-11 03:11:53 +04:00
if len ( DbCfg . Pwd ) == 0 {
DbCfg . Pwd = setting . Cfg . MustValue ( "database" , "PASSWD" )
}
2014-05-26 04:11:25 +04:00
DbCfg . SslMode = setting . Cfg . MustValue ( "database" , "SSL_MODE" )
DbCfg . Path = setting . Cfg . MustValue ( "database" , "PATH" , "data/gogs.db" )
2014-03-21 09:48:10 +04:00
}
2014-02-19 02:48:02 +04:00
2014-09-04 19:19:26 +04:00
func getEngine ( ) ( * xorm . Engine , error ) {
cnnstr := ""
2014-03-30 18:47:08 +04:00
switch DbCfg . Type {
case "mysql" :
2014-09-04 19:19:26 +04:00
cnnstr = fmt . Sprintf ( "%s:%s@tcp(%s)/%s?charset=utf8" ,
DbCfg . User , DbCfg . Pwd , DbCfg . Host , DbCfg . Name )
2014-03-30 18:47:08 +04:00
case "postgres" :
2014-04-14 10:49:50 +04:00
var host , port = "127.0.0.1" , "5432"
fields := strings . Split ( DbCfg . Host , ":" )
2014-05-02 02:53:41 +04:00
if len ( fields ) > 0 && len ( strings . TrimSpace ( fields [ 0 ] ) ) > 0 {
2014-04-14 10:49:50 +04:00
host = fields [ 0 ]
}
2014-05-02 02:53:41 +04:00
if len ( fields ) > 1 && len ( strings . TrimSpace ( fields [ 1 ] ) ) > 0 {
2014-04-14 10:49:50 +04:00
port = fields [ 1 ]
}
2014-09-04 19:19:26 +04:00
cnnstr = fmt . Sprintf ( "user=%s password=%s host=%s port=%s dbname=%s sslmode=%s" ,
2014-04-14 10:49:50 +04:00
DbCfg . User , DbCfg . Pwd , host , port , DbCfg . Name , DbCfg . SslMode )
2014-04-12 22:48:12 +04:00
case "sqlite3" :
2014-04-13 00:24:09 +04:00
if ! EnableSQLite3 {
2014-09-04 19:19:26 +04:00
return nil , fmt . Errorf ( "Unknown database type: %s" , DbCfg . Type )
2014-04-13 00:24:09 +04:00
}
2014-04-12 22:48:12 +04:00
os . MkdirAll ( path . Dir ( DbCfg . Path ) , os . ModePerm )
2014-09-13 02:58:24 +04:00
cnnstr = "file:" + DbCfg . Path + "?cache=shared&mode=rwc"
2014-03-30 18:47:08 +04:00
default :
2014-09-04 19:19:26 +04:00
return nil , fmt . Errorf ( "Unknown database type: %s" , DbCfg . Type )
2014-03-30 18:47:08 +04:00
}
2014-09-04 19:19:26 +04:00
return xorm . NewEngine ( DbCfg . Type , cnnstr )
}
func NewTestEngine ( x * xorm . Engine ) ( err error ) {
x , err = getEngine ( )
2014-03-30 18:47:08 +04:00
if err != nil {
2014-09-29 01:01:05 +04:00
return fmt . Errorf ( "models.init(fail to connect to database): %v" , err )
2014-03-30 18:47:08 +04:00
}
2014-09-04 19:19:26 +04:00
2014-04-05 18:46:32 +04:00
return x . Sync ( tables ... )
2014-03-30 18:47:08 +04:00
}
2014-03-30 01:50:51 +04:00
func SetEngine ( ) ( err error ) {
2014-09-04 19:19:26 +04:00
x , err = getEngine ( )
2014-02-19 02:48:02 +04:00
if err != nil {
2014-09-29 01:01:05 +04:00
return fmt . Errorf ( "models.init(fail to connect to database): %v" , err )
2014-02-19 02:48:02 +04:00
}
2014-03-21 00:04:56 +04:00
// WARNNING: for serv command, MUST remove the output to os.stdout,
// so use log file to instead print to stdout.
2014-05-28 09:53:06 +04:00
logPath := path . Join ( setting . LogRootPath , "xorm.log" )
2014-03-31 17:26:15 +04:00
os . MkdirAll ( path . Dir ( logPath ) , os . ModePerm )
2014-02-25 10:01:52 +04:00
2014-03-31 17:26:15 +04:00
f , err := os . Create ( logPath )
2014-03-17 22:03:58 +04:00
if err != nil {
2014-04-05 02:31:09 +04:00
return fmt . Errorf ( "models.init(fail to create xorm.log): %v" , err )
2014-03-17 22:03:58 +04:00
}
2014-06-21 08:51:41 +04:00
x . Logger = xorm . NewSimpleLogger ( f )
2014-03-31 17:26:15 +04:00
2014-06-21 08:51:41 +04:00
x . ShowSQL = true
2014-09-08 08:11:25 +04:00
x . ShowInfo = true
2014-06-21 08:51:41 +04:00
x . ShowDebug = true
x . ShowErr = true
2014-08-29 07:24:37 +04:00
x . ShowWarn = true
2014-03-30 01:50:51 +04:00
return nil
2014-02-19 02:48:02 +04:00
}
2014-03-30 01:50:51 +04:00
func NewEngine ( ) ( err error ) {
if err = SetEngine ( ) ; err != nil {
return err
2014-04-05 18:46:32 +04:00
}
2014-06-21 08:51:41 +04:00
if err = x . Sync2 ( tables ... ) ; err != nil {
2014-03-30 01:50:51 +04:00
return fmt . Errorf ( "sync database struct error: %v\n" , err )
2014-02-19 13:50:53 +04: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
type Statistic struct {
Counter struct {
2014-08-28 18:29:00 +04: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-21 00:04:56 +04:00
}
}
func GetStatistic ( ) ( stats Statistic ) {
2014-07-07 12:15:08 +04:00
stats . Counter . User = CountUsers ( )
2014-08-28 18:29:00 +04:00
stats . Counter . Org = CountOrganizations ( )
2014-06-21 08:51:41 +04:00
stats . Counter . PublicKey , _ = x . Count ( new ( PublicKey ) )
2014-08-28 18:29:00 +04:00
stats . Counter . Repo = CountRepositories ( )
2014-06-21 08:51:41 +04:00
stats . Counter . Watch , _ = x . Count ( new ( Watch ) )
2014-08-28 18:29:00 +04:00
stats . Counter . Star , _ = x . Count ( new ( Star ) )
2014-06-21 08: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 18:29:00 +04:00
stats . Counter . Follow , _ = x . Count ( new ( Follow ) )
stats . Counter . Mirror , _ = x . Count ( new ( Mirror ) )
2014-06-21 08: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 18:29:00 +04: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 12:31:13 +04:00
return
2014-03-21 00:04:56 +04:00
}
2014-05-05 08:55:17 +04:00
2014-08-07 01:21:24 +04:00
func Ping ( ) error {
return x . Ping ( )
}
2014-05-05 08:55:17 +04:00
// DumpDatabase dumps all data from database to file system.
func DumpDatabase ( filePath string ) error {
2014-06-21 08:51:41 +04:00
return x . DumpAllToFile ( filePath )
2014-05-05 08:55:17 +04:00
}