
Protect the `.fields' variable state from partial-initialization timing issues. As gorm warms state and makes `.GetStructFields' calls, they lead to a long deferred function in model_struct.go. Then the deferred function calls `Scope.Fields', leading to another `.GetStructFields' call.. cyclically. Once information is cached the cycling ends. This extra rule evicts incorrect/suspiciously short fields information. Symptom of incorrect internal state ocurring can manifest as invalid insert SQL statement errors generated during `DB.Save', e.g.: INSERT INTO "your_table" DEFAULT VALUES RETURNING "your_table"."id" This is why we refresh if nil or only a single field (usually is the ID field). Included is a unit-test which triggers the scenario and prove that it is fixed.
103 lines
2.7 KiB
Go
103 lines
2.7 KiB
Go
package gorm
|
|
|
|
import (
|
|
"database/sql"
|
|
"errors"
|
|
"reflect"
|
|
)
|
|
|
|
type Field struct {
|
|
*StructField
|
|
IsBlank bool
|
|
Field reflect.Value
|
|
}
|
|
|
|
func (field *Field) Set(value interface{}) error {
|
|
if !field.Field.IsValid() {
|
|
return errors.New("field value not valid")
|
|
}
|
|
|
|
if !field.Field.CanAddr() {
|
|
return errors.New("unaddressable value")
|
|
}
|
|
|
|
if rvalue, ok := value.(reflect.Value); ok {
|
|
value = rvalue.Interface()
|
|
}
|
|
|
|
if scanner, ok := field.Field.Addr().Interface().(sql.Scanner); ok {
|
|
if v, ok := value.(reflect.Value); ok {
|
|
if err := scanner.Scan(v.Interface()); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
if err := scanner.Scan(value); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
} else {
|
|
reflectValue, ok := value.(reflect.Value)
|
|
if !ok {
|
|
reflectValue = reflect.ValueOf(value)
|
|
}
|
|
|
|
if reflectValue.Type().ConvertibleTo(field.Field.Type()) {
|
|
field.Field.Set(reflectValue.Convert(field.Field.Type()))
|
|
} else {
|
|
return errors.New("could not convert argument")
|
|
}
|
|
}
|
|
|
|
field.IsBlank = isBlank(field.Field)
|
|
return nil
|
|
}
|
|
|
|
// Fields get value's fields
|
|
func (scope *Scope) Fields() map[string]*Field {
|
|
// Recalculate if fields is empty (nil) or number of fields is LTE 1.
|
|
//
|
|
// Protect the `.fields' variable state from partial-initialization timing
|
|
// issues.
|
|
//
|
|
// As gorm warms state and makes `.GetStructFields' calls, they lead to a long
|
|
// deferred function in model_struct.go. Then the deferred function calls
|
|
// `Scope.Fields', leading to another `.GetStructFields' call.. cyclically.
|
|
// Once information is cached the cycling ends.
|
|
//
|
|
// This extra rule evicts incorrect/suspiciously short fields information.
|
|
//
|
|
// Symptom of incorrect internal state ocurring can manifest as invalid
|
|
// insert SQL statement errors generated during `DB.Save', e.g.:
|
|
//
|
|
// INSERT INTO "your_table" DEFAULT VALUES RETURNING "your_table"."id"
|
|
//
|
|
// This is why we refresh if nil or only a single field (usually is the ID field).
|
|
if scope.fields == nil || len(scope.fields) <= 1 {
|
|
fields := map[string]*Field{}
|
|
structFields := scope.GetStructFields()
|
|
|
|
indirectValue := scope.IndirectValue()
|
|
isStruct := indirectValue.Kind() == reflect.Struct
|
|
for _, structField := range structFields {
|
|
if isStruct {
|
|
fields[structField.DBName] = getField(indirectValue, structField)
|
|
} else {
|
|
fields[structField.DBName] = &Field{StructField: structField, IsBlank: true}
|
|
}
|
|
}
|
|
|
|
scope.fields = fields
|
|
}
|
|
return scope.fields
|
|
}
|
|
|
|
func getField(indirectValue reflect.Value, structField *StructField) *Field {
|
|
field := &Field{StructField: structField}
|
|
for _, name := range structField.Names {
|
|
indirectValue = reflect.Indirect(indirectValue).FieldByName(name)
|
|
}
|
|
field.Field = indirectValue
|
|
field.IsBlank = isBlank(indirectValue)
|
|
return field
|
|
}
|