123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164 |
- package gorm
- import (
- "fmt"
- "strings"
- )
- // Define callbacks for creating
- func init() {
- DefaultCallback.Create().Register("gorm:begin_transaction", beginTransactionCallback)
- DefaultCallback.Create().Register("gorm:before_create", beforeCreateCallback)
- DefaultCallback.Create().Register("gorm:save_before_associations", saveBeforeAssociationsCallback)
- DefaultCallback.Create().Register("gorm:update_time_stamp", updateTimeStampForCreateCallback)
- DefaultCallback.Create().Register("gorm:create", createCallback)
- DefaultCallback.Create().Register("gorm:force_reload_after_create", forceReloadAfterCreateCallback)
- DefaultCallback.Create().Register("gorm:save_after_associations", saveAfterAssociationsCallback)
- DefaultCallback.Create().Register("gorm:after_create", afterCreateCallback)
- DefaultCallback.Create().Register("gorm:commit_or_rollback_transaction", commitOrRollbackTransactionCallback)
- }
- // beforeCreateCallback will invoke `BeforeSave`, `BeforeCreate` method before creating
- func beforeCreateCallback(scope *Scope) {
- if !scope.HasError() {
- scope.CallMethod("BeforeSave")
- }
- if !scope.HasError() {
- scope.CallMethod("BeforeCreate")
- }
- }
- // updateTimeStampForCreateCallback will set `CreatedAt`, `UpdatedAt` when creating
- func updateTimeStampForCreateCallback(scope *Scope) {
- if !scope.HasError() {
- now := NowFunc()
- if createdAtField, ok := scope.FieldByName("CreatedAt"); ok {
- if createdAtField.IsBlank {
- createdAtField.Set(now)
- }
- }
- if updatedAtField, ok := scope.FieldByName("UpdatedAt"); ok {
- if updatedAtField.IsBlank {
- updatedAtField.Set(now)
- }
- }
- }
- }
- // createCallback the callback used to insert data into database
- func createCallback(scope *Scope) {
- if !scope.HasError() {
- defer scope.trace(NowFunc())
- var (
- columns, placeholders []string
- blankColumnsWithDefaultValue []string
- )
- for _, field := range scope.Fields() {
- if scope.changeableField(field) {
- if field.IsNormal {
- if field.IsBlank && field.HasDefaultValue {
- blankColumnsWithDefaultValue = append(blankColumnsWithDefaultValue, scope.Quote(field.DBName))
- scope.InstanceSet("gorm:blank_columns_with_default_value", blankColumnsWithDefaultValue)
- } else if !field.IsPrimaryKey || !field.IsBlank {
- columns = append(columns, scope.Quote(field.DBName))
- placeholders = append(placeholders, scope.AddToVars(field.Field.Interface()))
- }
- } else if field.Relationship != nil && field.Relationship.Kind == "belongs_to" {
- for _, foreignKey := range field.Relationship.ForeignDBNames {
- if foreignField, ok := scope.FieldByName(foreignKey); ok && !scope.changeableField(foreignField) {
- columns = append(columns, scope.Quote(foreignField.DBName))
- placeholders = append(placeholders, scope.AddToVars(foreignField.Field.Interface()))
- }
- }
- }
- }
- }
- var (
- returningColumn = "*"
- quotedTableName = scope.QuotedTableName()
- primaryField = scope.PrimaryField()
- extraOption string
- )
- if str, ok := scope.Get("gorm:insert_option"); ok {
- extraOption = fmt.Sprint(str)
- }
- if primaryField != nil {
- returningColumn = scope.Quote(primaryField.DBName)
- }
- lastInsertIDReturningSuffix := scope.Dialect().LastInsertIDReturningSuffix(quotedTableName, returningColumn)
- if len(columns) == 0 {
- scope.Raw(fmt.Sprintf(
- "INSERT INTO %v %v%v%v",
- quotedTableName,
- scope.Dialect().DefaultValueStr(),
- addExtraSpaceIfExist(extraOption),
- addExtraSpaceIfExist(lastInsertIDReturningSuffix),
- ))
- } else {
- scope.Raw(fmt.Sprintf(
- "INSERT INTO %v (%v) VALUES (%v)%v%v",
- scope.QuotedTableName(),
- strings.Join(columns, ","),
- strings.Join(placeholders, ","),
- addExtraSpaceIfExist(extraOption),
- addExtraSpaceIfExist(lastInsertIDReturningSuffix),
- ))
- }
- // execute create sql
- if lastInsertIDReturningSuffix == "" || primaryField == nil {
- if result, err := scope.SQLDB().Exec(scope.SQL, scope.SQLVars...); scope.Err(err) == nil {
- // set rows affected count
- scope.db.RowsAffected, _ = result.RowsAffected()
- // set primary value to primary field
- if primaryField != nil && primaryField.IsBlank {
- if primaryValue, err := result.LastInsertId(); scope.Err(err) == nil {
- scope.Err(primaryField.Set(primaryValue))
- }
- }
- }
- } else {
- if primaryField.Field.CanAddr() {
- if err := scope.SQLDB().QueryRow(scope.SQL, scope.SQLVars...).Scan(primaryField.Field.Addr().Interface()); scope.Err(err) == nil {
- primaryField.IsBlank = false
- scope.db.RowsAffected = 1
- }
- } else {
- scope.Err(ErrUnaddressable)
- }
- }
- }
- }
- // forceReloadAfterCreateCallback will reload columns that having default value, and set it back to current object
- func forceReloadAfterCreateCallback(scope *Scope) {
- if blankColumnsWithDefaultValue, ok := scope.InstanceGet("gorm:blank_columns_with_default_value"); ok {
- db := scope.DB().New().Table(scope.TableName()).Select(blankColumnsWithDefaultValue.([]string))
- for _, field := range scope.Fields() {
- if field.IsPrimaryKey && !field.IsBlank {
- db = db.Where(fmt.Sprintf("%v = ?", field.DBName), field.Field.Interface())
- }
- }
- db.Scan(scope.Value)
- }
- }
- // afterCreateCallback will invoke `AfterCreate`, `AfterSave` method after creating
- func afterCreateCallback(scope *Scope) {
- if !scope.HasError() {
- scope.CallMethod("AfterCreate")
- }
- if !scope.HasError() {
- scope.CallMethod("AfterSave")
- }
- }
|