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-03-08 02:22:15 +04:00
"github.com/gogits/gogs/modules/base"
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-04-05 18:46:32 +04:00
orm * xorm . Engine
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 ( ) {
tables = append ( tables , new ( User ) , new ( PublicKey ) , new ( Repository ) , new ( Watch ) ,
2014-04-13 04:35:35 +04:00
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 ) ,
new ( Milestone ) )
2014-04-05 18:46:32 +04:00
}
2014-03-21 09:48:10 +04:00
func LoadModelsConfig ( ) {
2014-03-21 11:27:59 +04:00
DbCfg . Type = base . Cfg . MustValue ( "database" , "DB_TYPE" )
2014-03-31 00:01:50 +04:00
if DbCfg . Type == "sqlite3" {
UseSQLite3 = true
}
2014-03-21 11:27:59 +04:00
DbCfg . Host = base . Cfg . MustValue ( "database" , "HOST" )
DbCfg . Name = base . Cfg . MustValue ( "database" , "NAME" )
DbCfg . User = base . Cfg . MustValue ( "database" , "USER" )
DbCfg . Pwd = base . Cfg . MustValue ( "database" , "PASSWD" )
DbCfg . SslMode = base . Cfg . MustValue ( "database" , "SSL_MODE" )
DbCfg . Path = base . Cfg . MustValue ( "database" , "PATH" , "data/gogs.db" )
2014-03-21 09:48:10 +04:00
}
2014-02-19 02:48:02 +04:00
2014-03-30 18:47:08 +04:00
func NewTestEngine ( x * xorm . Engine ) ( err error ) {
switch DbCfg . Type {
case "mysql" :
x , err = xorm . NewEngine ( "mysql" , fmt . Sprintf ( "%s:%s@tcp(%s)/%s?charset=utf8" ,
DbCfg . User , DbCfg . Pwd , DbCfg . Host , DbCfg . Name ) )
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 ]
}
cnnstr := fmt . Sprintf ( "user=%s password=%s host=%s port=%s dbname=%s sslmode=%s" ,
DbCfg . User , DbCfg . Pwd , host , port , DbCfg . Name , DbCfg . SslMode )
//fmt.Println(cnnstr)
x , err = xorm . NewEngine ( "postgres" , cnnstr )
2014-04-12 22:48:12 +04:00
case "sqlite3" :
2014-04-13 00:24:09 +04:00
if ! EnableSQLite3 {
return fmt . Errorf ( "Unknown database type: %s" , DbCfg . Type )
}
2014-04-12 22:48:12 +04:00
os . MkdirAll ( path . Dir ( DbCfg . Path ) , os . ModePerm )
x , err = xorm . NewEngine ( "sqlite3" , DbCfg . Path )
2014-03-30 18:47:08 +04:00
default :
2014-04-05 02:31:09 +04:00
return fmt . Errorf ( "Unknown database type: %s" , DbCfg . Type )
2014-03-30 18:47:08 +04:00
}
if err != nil {
2014-04-05 02:31:09 +04:00
return fmt . Errorf ( "models.init(fail to conntect database): %v" , err )
2014-03-30 18:47:08 +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-03-21 11:27:59 +04:00
switch DbCfg . Type {
2014-02-19 02:48:02 +04:00
case "mysql" :
2014-03-29 09:40:22 +04:00
orm , err = xorm . NewEngine ( "mysql" , fmt . Sprintf ( "%s:%s@tcp(%s)/%s?charset=utf8" ,
2014-03-21 11:27:59 +04:00
DbCfg . User , DbCfg . Pwd , DbCfg . Host , DbCfg . Name ) )
2014-03-17 22:03:58 +04:00
case "postgres" :
2014-04-15 05:07:21 +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-15 05:07:21 +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-15 05:07:21 +04:00
port = fields [ 1 ]
}
orm , err = xorm . NewEngine ( "postgres" , fmt . Sprintf ( "user=%s password=%s host=%s port=%s dbname=%s sslmode=%s" ,
DbCfg . User , DbCfg . Pwd , host , port , DbCfg . Name , DbCfg . SslMode ) )
2014-03-21 09:09:22 +04:00
case "sqlite3" :
2014-03-21 11:27:59 +04:00
os . MkdirAll ( path . Dir ( DbCfg . Path ) , os . ModePerm )
orm , err = xorm . NewEngine ( "sqlite3" , DbCfg . Path )
2014-02-19 02:48:02 +04:00
default :
2014-04-05 02:31:09 +04:00
return fmt . Errorf ( "Unknown database type: %s" , DbCfg . Type )
2014-02-19 02:48:02 +04:00
}
if err != nil {
2014-04-05 02:31:09 +04:00
return fmt . Errorf ( "models.init(fail to conntect 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-03-31 17:26:15 +04:00
execDir , _ := base . ExecDir ( )
logPath := execDir + "/log/xorm.log"
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-04-21 10:24:35 +04:00
orm . Logger = xorm . NewSimpleLogger ( f )
2014-03-31 17:26:15 +04:00
2014-02-25 11:28:04 +04:00
orm . ShowSQL = true
2014-03-31 17:26:15 +04:00
orm . ShowDebug = true
orm . ShowErr = 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
}
if err = orm . Sync ( 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-05-07 00:28:52 +04:00
User , PublicKey , Repo , Watch , Action , Access ,
Issue , Comment , Mirror , Oauth , Release ,
2014-05-12 22:06:42 +04:00
LoginSource , Webhook , Milestone int64
2014-03-21 00:04:56 +04:00
}
}
func GetStatistic ( ) ( stats Statistic ) {
stats . Counter . User , _ = orm . Count ( new ( User ) )
stats . Counter . PublicKey , _ = orm . Count ( new ( PublicKey ) )
stats . Counter . Repo , _ = orm . Count ( new ( Repository ) )
stats . Counter . Watch , _ = orm . Count ( new ( Watch ) )
stats . Counter . Action , _ = orm . Count ( new ( Action ) )
stats . Counter . Access , _ = orm . Count ( new ( Access ) )
2014-04-14 12:11:33 +04:00
stats . Counter . Issue , _ = orm . Count ( new ( Issue ) )
stats . Counter . Comment , _ = orm . Count ( new ( Comment ) )
stats . Counter . Mirror , _ = orm . Count ( new ( Mirror ) )
stats . Counter . Oauth , _ = orm . Count ( new ( Oauth2 ) )
stats . Counter . Release , _ = orm . Count ( new ( Release ) )
2014-05-07 00:28:52 +04:00
stats . Counter . LoginSource , _ = orm . Count ( new ( LoginSource ) )
stats . Counter . Webhook , _ = orm . Count ( new ( Webhook ) )
2014-05-12 22:06:42 +04:00
stats . Counter . Milestone , _ = orm . Count ( new ( Milestone ) )
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
// DumpDatabase dumps all data from database to file system.
func DumpDatabase ( filePath string ) error {
return orm . DumpAllToFile ( filePath )
}