2016-03-05 13:54:59 +03:00
package mssql
2016-03-05 14:22:29 +03:00
import (
"fmt"
2016-03-05 17:50:49 +03:00
"reflect"
2016-06-21 06:13:33 +03:00
"strconv"
2016-03-05 17:50:49 +03:00
"strings"
"time"
2016-03-05 14:22:29 +03:00
_ "github.com/denisenkom/go-mssqldb"
"github.com/jinzhu/gorm"
)
2017-03-22 17:57:13 +03:00
func setIdentityInsert ( scope * gorm . Scope ) {
if scope . Dialect ( ) . GetName ( ) == "mssql" {
for _ , field := range scope . PrimaryFields ( ) {
if _ , ok := field . TagSettings [ "AUTO_INCREMENT" ] ; ok && ! field . IsBlank {
scope . NewDB ( ) . Exec ( fmt . Sprintf ( "SET IDENTITY_INSERT %v ON" , scope . TableName ( ) ) )
scope . InstanceSet ( "mssql:identity_insert_on" , true )
}
}
}
}
func turnOffIdentityInsert ( scope * gorm . Scope ) {
if scope . Dialect ( ) . GetName ( ) == "mssql" {
if _ , ok := scope . InstanceGet ( "mssql:identity_insert_on" ) ; ok {
scope . NewDB ( ) . Exec ( fmt . Sprintf ( "SET IDENTITY_INSERT %v OFF" , scope . TableName ( ) ) )
}
}
}
2016-03-05 14:22:29 +03:00
func init ( ) {
2017-03-22 17:57:13 +03:00
gorm . DefaultCallback . Create ( ) . After ( "gorm:begin_transaction" ) . Register ( "mssql:set_identity_insert" , setIdentityInsert )
gorm . DefaultCallback . Create ( ) . Before ( "gorm:commit_or_rollback_transaction" ) . Register ( "mssql:turn_off_identity_insert" , turnOffIdentityInsert )
2016-03-05 17:50:49 +03:00
gorm . RegisterDialect ( "mssql" , & mssql { } )
}
type mssql struct {
2017-03-14 23:32:38 +03:00
db gorm . SQLCommon
2016-05-22 01:13:26 +03:00
gorm . DefaultForeignKeyNamer
2016-03-05 17:50:49 +03:00
}
func ( mssql ) GetName ( ) string {
return "mssql"
}
2017-03-14 23:32:38 +03:00
func ( s * mssql ) SetDB ( db gorm . SQLCommon ) {
2016-03-05 17:50:49 +03:00
s . db = db
}
func ( mssql ) BindVar ( i int ) string {
2017-04-18 11:13:02 +03:00
return "$$$" // ?
2016-03-05 17:50:49 +03:00
}
func ( mssql ) Quote ( key string ) string {
return fmt . Sprintf ( ` "%s" ` , key )
}
2017-01-15 16:24:53 +03:00
func ( s * mssql ) DataTypeOf ( field * gorm . StructField ) string {
var dataValue , sqlType , size , additionalType = gorm . ParseFieldStructForDialect ( field , s )
2016-03-05 17:50:49 +03:00
if sqlType == "" {
switch dataValue . Kind ( ) {
case reflect . Bool :
sqlType = "bit"
case reflect . Int , reflect . Int8 , reflect . Int16 , reflect . Int32 , reflect . Uint , reflect . Uint8 , reflect . Uint16 , reflect . Uint32 , reflect . Uintptr :
if _ , ok := field . TagSettings [ "AUTO_INCREMENT" ] ; ok || field . IsPrimaryKey {
2017-03-22 17:57:13 +03:00
field . TagSettings [ "AUTO_INCREMENT" ] = "AUTO_INCREMENT"
2016-03-05 17:50:49 +03:00
sqlType = "int IDENTITY(1,1)"
} else {
sqlType = "int"
}
case reflect . Int64 , reflect . Uint64 :
if _ , ok := field . TagSettings [ "AUTO_INCREMENT" ] ; ok || field . IsPrimaryKey {
2017-03-22 17:57:13 +03:00
field . TagSettings [ "AUTO_INCREMENT" ] = "AUTO_INCREMENT"
2016-03-05 17:50:49 +03:00
sqlType = "bigint IDENTITY(1,1)"
} else {
sqlType = "bigint"
}
case reflect . Float32 , reflect . Float64 :
sqlType = "float"
case reflect . String :
2017-04-19 10:21:56 +03:00
if size > 0 && size < 8000 {
2016-03-05 17:50:49 +03:00
sqlType = fmt . Sprintf ( "nvarchar(%d)" , size )
} else {
2017-04-19 10:21:56 +03:00
sqlType = "nvarchar(max)"
2016-03-05 17:50:49 +03:00
}
case reflect . Struct :
if _ , ok := dataValue . Interface ( ) . ( time . Time ) ; ok {
2017-08-11 12:18:49 +03:00
sqlType = "datetimeoffset"
2016-03-05 17:50:49 +03:00
}
default :
2017-04-19 10:21:56 +03:00
if gorm . IsByteArrayOrSlice ( dataValue ) {
if size > 0 && size < 8000 {
sqlType = fmt . Sprintf ( "varbinary(%d)" , size )
2016-03-05 17:50:49 +03:00
} else {
2017-04-19 10:21:56 +03:00
sqlType = "varbinary(max)"
2016-03-05 17:50:49 +03:00
}
}
}
}
if sqlType == "" {
panic ( fmt . Sprintf ( "invalid sql type %s (%s) for mssql" , dataValue . Type ( ) . Name ( ) , dataValue . Kind ( ) . String ( ) ) )
}
if strings . TrimSpace ( additionalType ) == "" {
return sqlType
}
return fmt . Sprintf ( "%v %v" , sqlType , additionalType )
}
func ( s mssql ) HasIndex ( tableName string , indexName string ) bool {
var count int
s . db . QueryRow ( "SELECT count(*) FROM sys.indexes WHERE name=? AND object_id=OBJECT_ID(?)" , indexName , tableName ) . Scan ( & count )
return count > 0
}
func ( s mssql ) RemoveIndex ( tableName string , indexName string ) error {
_ , err := s . db . Exec ( fmt . Sprintf ( "DROP INDEX %v ON %v" , indexName , s . Quote ( tableName ) ) )
return err
}
func ( s mssql ) HasForeignKey ( tableName string , foreignKeyName string ) bool {
return false
}
func ( s mssql ) HasTable ( tableName string ) bool {
var count int
2018-02-03 15:27:19 +03:00
currentDatabase , tableName := currentDatabaseAndTable ( & s , tableName )
s . db . QueryRow ( "SELECT count(*) FROM INFORMATION_SCHEMA.tables WHERE table_name = ? AND table_catalog = ?" , tableName , currentDatabase ) . Scan ( & count )
2016-03-05 17:50:49 +03:00
return count > 0
}
func ( s mssql ) HasColumn ( tableName string , columnName string ) bool {
var count int
2018-02-03 15:27:19 +03:00
currentDatabase , tableName := currentDatabaseAndTable ( & s , tableName )
s . db . QueryRow ( "SELECT count(*) FROM information_schema.columns WHERE table_catalog = ? AND table_name = ? AND column_name = ?" , currentDatabase , tableName , columnName ) . Scan ( & count )
2016-03-05 17:50:49 +03:00
return count > 0
}
2016-07-11 16:37:44 +03:00
func ( s mssql ) CurrentDatabase ( ) ( name string ) {
2016-03-05 17:50:49 +03:00
s . db . QueryRow ( "SELECT DB_NAME() AS [Current Database]" ) . Scan ( & name )
return
}
2016-06-21 06:13:33 +03:00
func ( mssql ) LimitAndOffsetSQL ( limit , offset interface { } ) ( sql string ) {
if offset != nil {
2017-04-18 18:31:56 +03:00
if parsedOffset , err := strconv . ParseInt ( fmt . Sprint ( offset ) , 0 , 0 ) ; err == nil && parsedOffset >= 0 {
2016-06-21 06:13:33 +03:00
sql += fmt . Sprintf ( " OFFSET %d ROWS" , parsedOffset )
2016-03-05 17:50:49 +03:00
}
}
2017-02-22 00:23:01 +03:00
if limit != nil {
2017-04-18 18:31:56 +03:00
if parsedLimit , err := strconv . ParseInt ( fmt . Sprint ( limit ) , 0 , 0 ) ; err == nil && parsedLimit >= 0 {
2017-02-22 00:23:01 +03:00
if sql == "" {
// add default zero offset
sql += " OFFSET 0 ROWS"
}
sql += fmt . Sprintf ( " FETCH NEXT %d ROWS ONLY" , parsedLimit )
}
}
2016-03-05 17:50:49 +03:00
return
}
func ( mssql ) SelectFromDummyTable ( ) string {
return ""
}
2016-03-07 09:54:20 +03:00
func ( mssql ) LastInsertIDReturningSuffix ( tableName , columnName string ) string {
2016-03-05 17:50:49 +03:00
return ""
2016-03-05 14:22:29 +03:00
}
2018-02-03 15:27:19 +03:00
func currentDatabaseAndTable ( dialect gorm . Dialect , tableName string ) ( string , string ) {
if strings . Contains ( tableName , "." ) {
splitStrings := strings . SplitN ( tableName , "." , 2 )
return splitStrings [ 0 ] , splitStrings [ 1 ]
}
return dialect . CurrentDatabase ( ) , tableName
}