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 (
2014-10-19 09:35:24 +04:00
"database/sql"
2014-02-19 02:48:02 +04:00
"fmt"
2015-08-29 07:08:37 +03:00
"net/url"
2014-02-19 02:48:02 +04:00
"os"
2014-03-21 09:09:22 +04:00
"path"
2014-04-14 10:49:50 +04:00
"strings"
2015-08-19 19:56:12 +03:00
"time"
2014-02-19 02:48:02 +04:00
2015-08-19 19:56:12 +03:00
"github.com/Unknwon/com"
2014-02-19 02:48:02 +04:00
_ "github.com/go-sql-driver/mysql"
2015-01-23 10:54:16 +03:00
"github.com/go-xorm/core"
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
2015-02-23 10:15:53 +03:00
"github.com/gogits/gogs/models/migrations"
2015-10-10 06:05:20 +03:00
"github.com/gogits/gogs/modules/log"
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-10-19 09: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 08:58:46 +03:00
Find ( interface { } , ... interface { } ) error
2015-02-11 07:44:16 +03:00
Get ( interface { } ) ( bool , error )
2014-10-19 09:35:24 +04:00
Insert ( ... interface { } ) ( int64 , error )
2015-02-13 08:58:46 +03:00
InsertOne ( interface { } ) ( int64 , error )
2015-02-11 07:44:16 +03:00
Id ( interface { } ) * xorm . Session
2015-02-13 08:58:46 +03:00
Sql ( string , ... interface { } ) * xorm . Session
2015-02-11 07:44:16 +03:00
Where ( string , ... interface { } ) * xorm . Session
2014-10-19 09:35:24 +04:00
}
2015-02-13 08:58:46 +03:00
func sessionRelease ( sess * xorm . Session ) {
if ! sess . IsCommitedOrRollbacked {
sess . Rollback ( )
}
sess . Close ( )
}
2015-08-19 19:56:12 +03:00
// Note: get back time.Time from database Go sees it at UTC where they are really Local.
// So this function makes correct timezone offset.
func regulateTimeZone ( t time . Time ) time . Time {
2015-09-13 14:32:00 +03:00
if ! setting . UseMySQL {
2015-08-19 19:56:12 +03:00
return t
}
zone := t . Local ( ) . Format ( "-0700" )
if len ( zone ) != 5 {
2015-10-16 04:28:12 +03:00
log . Error ( 4 , "Unprocessable timezone: %s - %s" , t . Local ( ) , zone )
2015-08-19 19:56:12 +03:00
return t
}
2015-09-11 19:33:27 +03:00
hour := com . StrTo ( zone [ 2 : 3 ] ) . MustInt ( )
2015-09-12 22:31:36 +03:00
minutes := com . StrTo ( zone [ 3 : 5 ] ) . MustInt ( )
2015-08-19 19:56:12 +03:00
if zone [ 0 ] == '-' {
2015-09-11 19:33:27 +03:00
return t . Add ( time . Duration ( hour ) * time . Hour ) . Add ( time . Duration ( minutes ) * time . Minute )
2015-08-19 19:56:12 +03:00
}
2015-09-11 19:33:27 +03:00
return t . Add ( - 1 * time . Duration ( hour ) * time . Hour ) . Add ( - 1 * time . Duration ( minutes ) * time . Minute )
2015-08-19 19:56:12 +03:00
}
2014-03-21 09:48:10 +04:00
var (
2014-10-19 09:35:24 +04:00
x * 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 {
2015-02-01 20:41:03 +03:00
Type , Host , Name , User , Passwd , Path , SSLMode string
2014-03-21 09:48:10 +04:00
}
2014-03-31 00:01:50 +04:00
2014-04-13 00:24:09 +04:00
EnableSQLite3 bool
2015-09-06 23:31:22 +03:00
EnableTidb bool
2014-03-21 09:48:10 +04:00
)
2014-04-05 18:46:32 +04:00
func init ( ) {
2014-11-12 14:48:50 +03:00
tables = append ( tables ,
2015-09-17 23:11:44 +03:00
new ( User ) , new ( PublicKey ) , new ( AccessToken ) ,
2015-08-06 17:48:11 +03:00
new ( Repository ) , new ( DeployKey ) , new ( Collaboration ) , new ( Access ) ,
2015-09-02 02:07:02 +03:00
new ( Watch ) , new ( Star ) , new ( Follow ) , new ( Action ) ,
2015-09-02 12:09:12 +03:00
new ( Issue ) , new ( PullRequest ) , new ( Comment ) , new ( Attachment ) , new ( IssueUser ) ,
2015-08-10 09:42:50 +03:00
new ( Label ) , new ( IssueLabel ) , new ( Milestone ) ,
2014-10-09 02:29:18 +04:00
new ( Mirror ) , new ( Release ) , new ( LoginSource ) , new ( Webhook ) ,
2015-02-23 10:15:53 +03:00
new ( UpdateTask ) , new ( HookTask ) ,
new ( Team ) , new ( OrgUser ) , new ( TeamUser ) , new ( TeamRepo ) ,
2015-02-12 05:58:37 +03:00
new ( Notice ) , new ( EmailAddress ) )
2015-08-27 18:06:14 +03:00
2015-11-07 08:39:45 +03:00
gonicNames := [ ] string { "SSL" }
2015-08-27 18:06:14 +03:00
for _ , name := range gonicNames {
core . LintGonicMapper [ name ] = true
}
2014-04-05 18:46:32 +04:00
}
2015-09-17 06:08:46 +03:00
func LoadConfigs ( ) {
2014-12-31 13:37:29 +03:00
sec := setting . Cfg . Section ( "database" )
DbCfg . Type = sec . Key ( "DB_TYPE" ) . String ( )
2015-02-12 05:58:37 +03:00
switch DbCfg . Type {
case "sqlite3" :
setting . UseSQLite3 = true
case "mysql" :
setting . UseMySQL = true
case "postgres" :
setting . UsePostgreSQL = true
2015-09-12 22:31:36 +03:00
case "tidb" :
setting . UseTiDB = true
2014-03-31 00:01:50 +04:00
}
2014-12-31 13:37:29 +03:00
DbCfg . Host = sec . Key ( "HOST" ) . String ( )
DbCfg . Name = sec . Key ( "NAME" ) . String ( )
DbCfg . User = sec . Key ( "USER" ) . String ( )
2015-02-01 20:41:03 +03:00
if len ( DbCfg . Passwd ) == 0 {
DbCfg . Passwd = sec . Key ( "PASSWD" ) . String ( )
2014-06-11 03:11:53 +04:00
}
2015-02-01 20:41:03 +03:00
DbCfg . SSLMode = sec . Key ( "SSL_MODE" ) . String ( )
2014-12-31 13:37:29 +03:00
DbCfg . Path = sec . Key ( "PATH" ) . MustString ( "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" :
2015-03-13 21:21:47 +03:00
if DbCfg . Host [ 0 ] == '/' { // looks like a unix socket
2015-08-05 15:47:35 +03:00
cnnstr = fmt . Sprintf ( "%s:%s@unix(%s)/%s?charset=utf8&parseTime=true" ,
2015-03-13 21:21:47 +03:00
DbCfg . User , DbCfg . Passwd , DbCfg . Host , DbCfg . Name )
} else {
2015-08-05 15:47:35 +03:00
cnnstr = fmt . Sprintf ( "%s:%s@tcp(%s)/%s?charset=utf8&parseTime=true" ,
2015-03-13 21:21:47 +03:00
DbCfg . User , DbCfg . Passwd , 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 ]
}
2015-07-31 06:05:06 +03:00
cnnstr = fmt . Sprintf ( "postgres://%s:%s@%s:%s/%s?sslmode=%s" ,
2015-08-29 07:08:37 +03:00
url . QueryEscape ( DbCfg . User ) , url . QueryEscape ( DbCfg . Passwd ) , 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
}
2015-08-24 16:01:23 +03:00
if err := os . MkdirAll ( path . Dir ( DbCfg . Path ) , os . ModePerm ) ; err != nil {
return nil , fmt . Errorf ( "Fail to create directories: %v" , err )
}
2014-09-13 02:58:24 +04:00
cnnstr = "file:" + DbCfg . Path + "?cache=shared&mode=rwc"
2015-09-06 23:31:22 +03:00
case "tidb" :
if ! EnableTidb {
return nil , fmt . Errorf ( "Unknown database type: %s" , DbCfg . Type )
}
if err := os . MkdirAll ( path . Dir ( DbCfg . Path ) , os . ModePerm ) ; err != nil {
return nil , fmt . Errorf ( "Fail to create directories: %v" , err )
}
cnnstr = "goleveldb://" + DbCfg . Path
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 {
2015-08-02 07:36:35 +03:00
return fmt . Errorf ( "Connect to database: %v" , err )
2014-03-30 18:47:08 +04:00
}
2014-09-04 19:19:26 +04:00
2015-01-23 10:54:16 +03:00
x . SetMapper ( core . GonicMapper { } )
2015-09-03 12:05:58 +03:00
return x . StoreEngine ( "InnoDB" ) . Sync2 ( 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 {
2015-08-06 17:48:11 +03:00
return fmt . Errorf ( "Fail to connect to database: %v" , err )
2014-02-19 02:48:02 +04:00
}
2015-01-23 10:54:16 +03:00
x . SetMapper ( core . 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.
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 {
2015-08-02 07:36:35 +03:00
return fmt . Errorf ( "Fail to create xorm.log: %v" , err )
2014-03-17 22:03:58 +04:00
}
2015-02-15 01:01:33 +03:00
x . SetLogger ( 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
}
2015-01-22 15:49:52 +03:00
if err = migrations . Migrate ( 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
}
2014-10-28 18:40:09 +03:00
if err = x . StoreEngine ( "InnoDB" ) . 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
}
2015-01-23 10:54:16 +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
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 ) )
2015-09-17 23:11:44 +03:00
stats . Counter . Oauth = 0
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 ) )
2015-09-10 22:45:03 +03:00
stats . Counter . LoginSource = CountLoginSources ( )
2014-06-21 08:51:41 +04:00
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
}