1
0
mirror of https://github.com/astaxie/beego.git synced 2025-01-11 08:37:13 +00:00
Beego/orm/db_alias.go

237 lines
5.1 KiB
Go
Raw Normal View History

2013-07-30 20:32:38 +08:00
package orm
import (
"database/sql"
"fmt"
"os"
"reflect"
2013-07-30 20:32:38 +08:00
"sync"
"time"
2013-07-30 20:32:38 +08:00
)
2014-01-17 17:04:15 +08:00
// database driver constant int.
2013-08-07 19:11:44 +08:00
type DriverType int
2013-07-30 20:32:38 +08:00
const (
2014-01-17 17:04:15 +08:00
_ DriverType = iota // int enum type
DR_MySQL // mysql
DR_Sqlite // sqlite
DR_Oracle // oracle
DR_Postgres // pgsql
2013-07-30 20:32:38 +08:00
)
2014-01-17 17:04:15 +08:00
// database driver string.
2013-08-07 19:11:44 +08:00
type driver string
2014-01-17 17:04:15 +08:00
// get type constant int of current driver..
2013-08-07 19:11:44 +08:00
func (d driver) Type() DriverType {
a, _ := dataBaseCache.get(string(d))
return a.Driver
}
2014-01-17 17:04:15 +08:00
// get name of current driver
2013-08-07 19:11:44 +08:00
func (d driver) Name() string {
return string(d)
}
2014-01-17 17:04:15 +08:00
// check driver iis implemented Driver interface or not.
2013-08-07 19:11:44 +08:00
var _ Driver = new(driver)
2013-07-30 20:32:38 +08:00
var (
dataBaseCache = &_dbCache{cache: make(map[string]*alias)}
2013-08-07 19:11:44 +08:00
drivers = map[string]DriverType{
2013-08-01 09:23:32 +08:00
"mysql": DR_MySQL,
"postgres": DR_Postgres,
"sqlite3": DR_Sqlite,
}
2013-08-07 19:11:44 +08:00
dbBasers = map[DriverType]dbBaser{
2013-07-30 20:32:38 +08:00
DR_MySQL: newdbBaseMysql(),
DR_Sqlite: newdbBaseSqlite(),
DR_Oracle: newdbBaseMysql(),
DR_Postgres: newdbBasePostgres(),
}
)
2014-01-17 17:04:15 +08:00
// database alias cacher.
2013-07-30 20:32:38 +08:00
type _dbCache struct {
mux sync.RWMutex
cache map[string]*alias
}
2014-01-17 17:04:15 +08:00
// add database alias with original name.
2013-07-30 20:32:38 +08:00
func (ac *_dbCache) add(name string, al *alias) (added bool) {
ac.mux.Lock()
defer ac.mux.Unlock()
if _, ok := ac.cache[name]; ok == false {
ac.cache[name] = al
added = true
}
return
}
2014-01-17 17:04:15 +08:00
// get database alias if cached.
2013-07-30 20:32:38 +08:00
func (ac *_dbCache) get(name string) (al *alias, ok bool) {
ac.mux.RLock()
defer ac.mux.RUnlock()
al, ok = ac.cache[name]
return
}
2014-01-17 17:04:15 +08:00
// get default alias.
2013-07-30 20:32:38 +08:00
func (ac *_dbCache) getDefault() (al *alias) {
al, _ = ac.get("default")
return
}
type alias struct {
Name string
Driver DriverType
DriverName string
DataSource string
MaxIdleConns int
MaxOpenConns int
DB *sql.DB
DbBaser dbBaser
TZ *time.Location
Engine string
2013-07-30 20:32:38 +08:00
}
// Setting the database connect params. Use the database driver self dataSource args.
func RegisterDataBase(aliasName, driverName, dataSource string, params ...int) {
2013-07-30 20:32:38 +08:00
al := new(alias)
al.Name = aliasName
2013-07-30 20:32:38 +08:00
al.DriverName = driverName
al.DataSource = dataSource
var (
err error
)
if dr, ok := drivers[driverName]; ok {
al.DbBaser = dbBasers[dr]
2013-08-07 19:11:44 +08:00
al.Driver = dr
2013-07-30 20:32:38 +08:00
} else {
err = fmt.Errorf("driver name `%s` have not registered", driverName)
goto end
}
if dataBaseCache.add(aliasName, al) == false {
err = fmt.Errorf("db name `%s` already registered, cannot reuse", aliasName)
2013-07-30 20:32:38 +08:00
goto end
}
al.DB, err = sql.Open(driverName, dataSource)
if err != nil {
err = fmt.Errorf("register db `%s`, %s", aliasName, err.Error())
2013-07-30 20:32:38 +08:00
goto end
}
// orm timezone system match database
// default use Local
al.TZ = time.Local
switch al.Driver {
case DR_MySQL:
2014-01-10 16:50:03 +08:00
row := al.DB.QueryRow("SELECT TIMEDIFF(NOW(), UTC_TIMESTAMP)")
var tz string
row.Scan(&tz)
2014-01-10 16:50:03 +08:00
if len(tz) >= 8 {
if tz[0] != '-' {
tz = "+" + tz
2013-12-21 21:00:29 +08:00
}
2014-01-10 16:50:03 +08:00
t, err := time.Parse("-07:00:00", tz)
if err == nil {
al.TZ = t.Location()
2014-01-10 16:50:03 +08:00
} else {
DebugLog.Printf("Detect DB timezone: %s %s\n", tz, err.Error())
}
}
// get default engine from current database
row = al.DB.QueryRow("SELECT ENGINE, TRANSACTIONS FROM information_schema.engines WHERE SUPPORT = 'DEFAULT'")
var engine string
var tx bool
row.Scan(&engine, &tx)
if engine != "" {
al.Engine = engine
} else {
engine = "INNODB"
}
case DR_Sqlite:
al.TZ = time.UTC
case DR_Postgres:
row := al.DB.QueryRow("SELECT current_setting('TIMEZONE')")
var tz string
row.Scan(&tz)
loc, err := time.LoadLocation(tz)
if err == nil {
al.TZ = loc
2014-01-10 16:50:03 +08:00
} else {
DebugLog.Printf("Detect DB timezone: %s %s\n", tz, err.Error())
}
}
for i, v := range params {
switch i {
case 0:
SetMaxIdleConns(al.Name, v)
case 1:
SetMaxOpenConns(al.Name, v)
}
}
2013-07-30 20:32:38 +08:00
err = al.DB.Ping()
if err != nil {
err = fmt.Errorf("register db `%s`, %s", aliasName, err.Error())
2013-07-30 20:32:38 +08:00
goto end
}
end:
if err != nil {
fmt.Println(err.Error())
os.Exit(2)
}
}
// Register a database driver use specify driver name, this can be definition the driver is which database type.
func RegisterDriver(driverName string, typ DriverType) {
if t, ok := drivers[driverName]; ok == false {
drivers[driverName] = typ
2013-07-30 20:32:38 +08:00
} else {
2013-07-31 22:11:22 +08:00
if t != typ {
2013-08-13 19:33:43 +08:00
fmt.Sprintf("driverName `%s` db driver already registered and is other type\n", driverName)
2013-07-31 22:11:22 +08:00
os.Exit(2)
}
2013-07-30 20:32:38 +08:00
}
}
// Change the database default used timezone
func SetDataBaseTZ(aliasName string, tz *time.Location) {
if al, ok := dataBaseCache.get(aliasName); ok {
al.TZ = tz
} else {
fmt.Sprintf("DataBase name `%s` not registered\n", aliasName)
2013-08-13 19:33:43 +08:00
os.Exit(2)
}
}
// Change the max idle conns for *sql.DB, use specify database alias name
func SetMaxIdleConns(aliasName string, maxIdleConns int) {
al := getDbAlias(aliasName)
al.MaxIdleConns = maxIdleConns
al.DB.SetMaxIdleConns(maxIdleConns)
}
// Change the max open conns for *sql.DB, use specify database alias name
func SetMaxOpenConns(aliasName string, maxOpenConns int) {
al := getDbAlias(aliasName)
al.MaxOpenConns = maxOpenConns
// for tip go 1.2
if fun := reflect.ValueOf(al.DB).MethodByName("SetMaxOpenConns"); fun.IsValid() {
fun.Call([]reflect.Value{reflect.ValueOf(maxOpenConns)})
}
}