2014-04-12 13:18:18 +08:00
|
|
|
// Beego (http://beego.me/)
|
|
|
|
// @description beego is an open-source, high-performance web framework for the Go programming language.
|
|
|
|
// @link http://github.com/astaxie/beego for the canonical source repository
|
|
|
|
// @license http://github.com/astaxie/beego/blob/master/LICENSE
|
|
|
|
// @authors slene
|
|
|
|
|
2013-08-19 22:37:39 +08:00
|
|
|
package orm
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2013-08-27 12:33:27 +08:00
|
|
|
type dbIndex struct {
|
|
|
|
Table string
|
|
|
|
Name string
|
|
|
|
Sql string
|
|
|
|
}
|
|
|
|
|
2014-01-17 17:04:15 +08:00
|
|
|
// create database drop sql.
|
2013-08-19 22:37:39 +08:00
|
|
|
func getDbDropSql(al *alias) (sqls []string) {
|
|
|
|
if len(modelCache.cache) == 0 {
|
|
|
|
fmt.Println("no Model found, need register your model")
|
|
|
|
os.Exit(2)
|
|
|
|
}
|
|
|
|
|
|
|
|
Q := al.DbBaser.TableQuote()
|
|
|
|
|
|
|
|
for _, mi := range modelCache.allOrdered() {
|
|
|
|
sqls = append(sqls, fmt.Sprintf(`DROP TABLE IF EXISTS %s%s%s`, Q, mi.table, Q))
|
|
|
|
}
|
|
|
|
return sqls
|
|
|
|
}
|
|
|
|
|
2014-01-17 17:04:15 +08:00
|
|
|
// get database column type string.
|
2013-08-27 12:33:27 +08:00
|
|
|
func getColumnTyp(al *alias, fi *fieldInfo) (col string) {
|
|
|
|
T := al.DbBaser.DbTypes()
|
|
|
|
fieldType := fi.fieldType
|
|
|
|
|
|
|
|
checkColumn:
|
|
|
|
switch fieldType {
|
|
|
|
case TypeBooleanField:
|
|
|
|
col = T["bool"]
|
|
|
|
case TypeCharField:
|
|
|
|
col = fmt.Sprintf(T["string"], fi.size)
|
|
|
|
case TypeTextField:
|
|
|
|
col = T["string-text"]
|
|
|
|
case TypeDateField:
|
|
|
|
col = T["time.Time-date"]
|
|
|
|
case TypeDateTimeField:
|
|
|
|
col = T["time.Time"]
|
|
|
|
case TypeBitField:
|
|
|
|
col = T["int8"]
|
|
|
|
case TypeSmallIntegerField:
|
|
|
|
col = T["int16"]
|
|
|
|
case TypeIntegerField:
|
|
|
|
col = T["int32"]
|
|
|
|
case TypeBigIntegerField:
|
|
|
|
if al.Driver == DR_Sqlite {
|
|
|
|
fieldType = TypeIntegerField
|
|
|
|
goto checkColumn
|
|
|
|
}
|
|
|
|
col = T["int64"]
|
|
|
|
case TypePositiveBitField:
|
|
|
|
col = T["uint8"]
|
|
|
|
case TypePositiveSmallIntegerField:
|
|
|
|
col = T["uint16"]
|
|
|
|
case TypePositiveIntegerField:
|
|
|
|
col = T["uint32"]
|
|
|
|
case TypePositiveBigIntegerField:
|
|
|
|
col = T["uint64"]
|
|
|
|
case TypeFloatField:
|
|
|
|
col = T["float64"]
|
|
|
|
case TypeDecimalField:
|
|
|
|
s := T["float64-decimal"]
|
|
|
|
if strings.Index(s, "%d") == -1 {
|
|
|
|
col = s
|
|
|
|
} else {
|
|
|
|
col = fmt.Sprintf(s, fi.digits, fi.decimals)
|
|
|
|
}
|
|
|
|
case RelForeignKey, RelOneToOne:
|
|
|
|
fieldType = fi.relModelInfo.fields.pk.fieldType
|
|
|
|
goto checkColumn
|
|
|
|
}
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2014-01-17 17:04:15 +08:00
|
|
|
// create alter sql string.
|
2013-08-27 12:33:27 +08:00
|
|
|
func getColumnAddQuery(al *alias, fi *fieldInfo) string {
|
|
|
|
Q := al.DbBaser.TableQuote()
|
|
|
|
typ := getColumnTyp(al, fi)
|
|
|
|
|
|
|
|
if fi.null == false {
|
|
|
|
typ += " " + "NOT NULL"
|
|
|
|
}
|
|
|
|
|
|
|
|
return fmt.Sprintf("ALTER TABLE %s%s%s ADD COLUMN %s%s%s %s", Q, fi.mi.table, Q, Q, fi.column, Q, typ)
|
|
|
|
}
|
|
|
|
|
2014-01-17 17:04:15 +08:00
|
|
|
// create database creation string.
|
2013-08-27 12:33:27 +08:00
|
|
|
func getDbCreateSql(al *alias) (sqls []string, tableIndexes map[string][]dbIndex) {
|
2013-08-19 22:37:39 +08:00
|
|
|
if len(modelCache.cache) == 0 {
|
|
|
|
fmt.Println("no Model found, need register your model")
|
|
|
|
os.Exit(2)
|
|
|
|
}
|
|
|
|
|
|
|
|
Q := al.DbBaser.TableQuote()
|
|
|
|
T := al.DbBaser.DbTypes()
|
2013-08-22 21:19:58 +08:00
|
|
|
sep := fmt.Sprintf("%s, %s", Q, Q)
|
2013-08-19 22:37:39 +08:00
|
|
|
|
2013-08-27 12:33:27 +08:00
|
|
|
tableIndexes = make(map[string][]dbIndex)
|
2013-08-25 11:31:07 +08:00
|
|
|
|
2013-08-19 22:37:39 +08:00
|
|
|
for _, mi := range modelCache.allOrdered() {
|
|
|
|
sql := fmt.Sprintf("-- %s\n", strings.Repeat("-", 50))
|
|
|
|
sql += fmt.Sprintf("-- Table Structure for `%s`\n", mi.fullName)
|
|
|
|
sql += fmt.Sprintf("-- %s\n", strings.Repeat("-", 50))
|
|
|
|
|
|
|
|
sql += fmt.Sprintf("CREATE TABLE IF NOT EXISTS %s%s%s (\n", Q, mi.table, Q)
|
|
|
|
|
|
|
|
columns := make([]string, 0, len(mi.fields.fieldsDB))
|
|
|
|
|
2013-08-22 21:19:58 +08:00
|
|
|
sqlIndexes := [][]string{}
|
|
|
|
|
2013-08-19 22:37:39 +08:00
|
|
|
for _, fi := range mi.fields.fieldsDB {
|
|
|
|
|
|
|
|
column := fmt.Sprintf(" %s%s%s ", Q, fi.column, Q)
|
2013-08-27 12:33:27 +08:00
|
|
|
col := getColumnTyp(al, fi)
|
2013-08-19 22:37:39 +08:00
|
|
|
|
|
|
|
if fi.auto {
|
2013-08-22 18:35:26 +08:00
|
|
|
switch al.Driver {
|
|
|
|
case DR_Sqlite, DR_Postgres:
|
2013-08-19 22:37:39 +08:00
|
|
|
column += T["auto"]
|
2013-08-22 18:35:26 +08:00
|
|
|
default:
|
2013-08-19 22:37:39 +08:00
|
|
|
column += col + " " + T["auto"]
|
|
|
|
}
|
|
|
|
} else if fi.pk {
|
|
|
|
column += col + " " + T["pk"]
|
|
|
|
} else {
|
|
|
|
column += col
|
|
|
|
|
|
|
|
if fi.null == false {
|
|
|
|
column += " " + "NOT NULL"
|
|
|
|
}
|
|
|
|
|
|
|
|
if fi.unique {
|
|
|
|
column += " " + "UNIQUE"
|
|
|
|
}
|
2013-08-22 21:19:58 +08:00
|
|
|
|
|
|
|
if fi.index {
|
2013-08-25 11:31:07 +08:00
|
|
|
sqlIndexes = append(sqlIndexes, []string{fi.column})
|
2013-08-22 21:19:58 +08:00
|
|
|
}
|
2013-08-19 22:37:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if strings.Index(column, "%COL%") != -1 {
|
|
|
|
column = strings.Replace(column, "%COL%", fi.column, -1)
|
|
|
|
}
|
|
|
|
|
|
|
|
columns = append(columns, column)
|
|
|
|
}
|
|
|
|
|
2013-08-22 21:19:58 +08:00
|
|
|
if mi.model != nil {
|
2013-10-14 22:31:35 +08:00
|
|
|
allnames := getTableUnique(mi.addrField)
|
|
|
|
if !mi.manual && len(mi.uniques) > 0 {
|
|
|
|
allnames = append(allnames, mi.uniques)
|
|
|
|
}
|
|
|
|
for _, names := range allnames {
|
2013-08-22 21:19:58 +08:00
|
|
|
cols := make([]string, 0, len(names))
|
|
|
|
for _, name := range names {
|
|
|
|
if fi, ok := mi.fields.GetByAny(name); ok && fi.dbcol {
|
|
|
|
cols = append(cols, fi.column)
|
|
|
|
} else {
|
|
|
|
panic(fmt.Errorf("cannot found column `%s` when parse UNIQUE in `%s.TableUnique`", name, mi.fullName))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
column := fmt.Sprintf(" UNIQUE (%s%s%s)", Q, strings.Join(cols, sep), Q)
|
|
|
|
columns = append(columns, column)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-19 22:37:39 +08:00
|
|
|
sql += strings.Join(columns, ",\n")
|
|
|
|
sql += "\n)"
|
|
|
|
|
|
|
|
if al.Driver == DR_MySQL {
|
2013-09-16 09:48:04 +08:00
|
|
|
var engine string
|
|
|
|
if mi.model != nil {
|
|
|
|
engine = getTableEngine(mi.addrField)
|
|
|
|
}
|
|
|
|
if engine == "" {
|
|
|
|
engine = al.Engine
|
|
|
|
}
|
|
|
|
sql += " ENGINE=" + engine
|
2013-08-19 22:37:39 +08:00
|
|
|
}
|
|
|
|
|
2013-08-22 21:19:58 +08:00
|
|
|
sql += ";"
|
2013-08-19 22:37:39 +08:00
|
|
|
sqls = append(sqls, sql)
|
2013-08-22 21:19:58 +08:00
|
|
|
|
|
|
|
if mi.model != nil {
|
|
|
|
for _, names := range getTableIndex(mi.addrField) {
|
|
|
|
cols := make([]string, 0, len(names))
|
|
|
|
for _, name := range names {
|
|
|
|
if fi, ok := mi.fields.GetByAny(name); ok && fi.dbcol {
|
|
|
|
cols = append(cols, fi.column)
|
|
|
|
} else {
|
|
|
|
panic(fmt.Errorf("cannot found column `%s` when parse INDEX in `%s.TableIndex`", name, mi.fullName))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
sqlIndexes = append(sqlIndexes, cols)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, names := range sqlIndexes {
|
2013-08-25 13:50:50 +08:00
|
|
|
name := mi.table + "_" + strings.Join(names, "_")
|
2013-08-22 21:19:58 +08:00
|
|
|
cols := strings.Join(names, sep)
|
|
|
|
sql := fmt.Sprintf("CREATE INDEX %s%s%s ON %s%s%s (%s%s%s);", Q, name, Q, Q, mi.table, Q, Q, cols, Q)
|
2013-08-27 12:33:27 +08:00
|
|
|
|
|
|
|
index := dbIndex{}
|
|
|
|
index.Table = mi.table
|
|
|
|
index.Name = name
|
|
|
|
index.Sql = sql
|
|
|
|
|
|
|
|
tableIndexes[mi.table] = append(tableIndexes[mi.table], index)
|
2013-08-22 21:19:58 +08:00
|
|
|
}
|
|
|
|
|
2013-08-19 22:37:39 +08:00
|
|
|
}
|
|
|
|
|
2013-08-25 11:31:07 +08:00
|
|
|
return
|
2013-08-19 22:37:39 +08:00
|
|
|
}
|