mirror of https://github.com/go-gorm/gorm.git
Support merge batch data some having primary values
This commit is contained in:
parent
3d8f6f9cf9
commit
a1e35bdc94
|
@ -52,21 +52,19 @@ func SaveBeforeAssociations(db *gorm.DB) {
|
||||||
obj := db.Statement.ReflectValue.Index(i)
|
obj := db.Statement.ReflectValue.Index(i)
|
||||||
if _, zero := rel.Field.ValueOf(obj); !zero { // check belongs to relation value
|
if _, zero := rel.Field.ValueOf(obj); !zero { // check belongs to relation value
|
||||||
rv := rel.Field.ReflectValueOf(obj) // relation reflect value
|
rv := rel.Field.ReflectValueOf(obj) // relation reflect value
|
||||||
if _, isZero := rel.FieldSchema.PrioritizedPrimaryField.ValueOf(rv); isZero {
|
objs = append(objs, obj)
|
||||||
objs = append(objs, obj)
|
if isPtr {
|
||||||
if isPtr {
|
elems = reflect.Append(elems, rv)
|
||||||
elems = reflect.Append(elems, rv)
|
|
||||||
} else {
|
|
||||||
elems = reflect.Append(elems, rv.Addr())
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
setupReferences(obj, rv)
|
elems = reflect.Append(elems, rv.Addr())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if elems.Len() > 0 {
|
if elems.Len() > 0 {
|
||||||
if db.AddError(db.Session(&gorm.Session{}).Create(elems.Interface()).Error) == nil {
|
if db.AddError(db.Session(&gorm.Session{}).Clauses(clause.OnConflict{
|
||||||
|
DoNothing: true,
|
||||||
|
}).Create(elems.Interface()).Error) == nil {
|
||||||
for i := 0; i < elems.Len(); i++ {
|
for i := 0; i < elems.Len(); i++ {
|
||||||
setupReferences(objs[i], elems.Index(i))
|
setupReferences(objs[i], elems.Index(i))
|
||||||
}
|
}
|
||||||
|
@ -79,10 +77,11 @@ func SaveBeforeAssociations(db *gorm.DB) {
|
||||||
rv = rv.Addr()
|
rv = rv.Addr()
|
||||||
}
|
}
|
||||||
|
|
||||||
if _, isZero := rel.FieldSchema.PrioritizedPrimaryField.ValueOf(rv); isZero {
|
if db.AddError(db.Session(&gorm.Session{}).Clauses(clause.OnConflict{
|
||||||
db.Session(&gorm.Session{}).Create(rv.Interface())
|
DoNothing: true,
|
||||||
|
}).Create(rv.Interface()).Error) == nil {
|
||||||
|
setupReferences(db.Statement.ReflectValue, rv)
|
||||||
}
|
}
|
||||||
setupReferences(db.Statement.ReflectValue, rv)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -130,16 +129,20 @@ func SaveAfterAssociations(db *gorm.DB) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if _, isZero := rel.FieldSchema.PrioritizedPrimaryField.ValueOf(rv); isZero {
|
elems = reflect.Append(elems, rv)
|
||||||
elems = reflect.Append(elems, rv)
|
|
||||||
} else {
|
|
||||||
db.Session(&gorm.Session{}).Save(rv.Addr().Interface())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if elems.Len() > 0 {
|
if elems.Len() > 0 {
|
||||||
db.Session(&gorm.Session{}).Create(elems.Interface())
|
assignmentColumns := []string{}
|
||||||
|
for _, ref := range rel.References {
|
||||||
|
assignmentColumns = append(assignmentColumns, ref.ForeignKey.DBName)
|
||||||
|
}
|
||||||
|
|
||||||
|
db.Session(&gorm.Session{}).Clauses(clause.OnConflict{
|
||||||
|
Columns: []clause.Column{{Name: rel.FieldSchema.PrioritizedPrimaryField.DBName}},
|
||||||
|
DoUpdates: clause.AssignmentColumns(assignmentColumns),
|
||||||
|
}).Create(elems.Interface())
|
||||||
}
|
}
|
||||||
case reflect.Struct:
|
case reflect.Struct:
|
||||||
if _, zero := rel.Field.ValueOf(db.Statement.ReflectValue); !zero {
|
if _, zero := rel.Field.ValueOf(db.Statement.ReflectValue); !zero {
|
||||||
|
@ -148,6 +151,7 @@ func SaveAfterAssociations(db *gorm.DB) {
|
||||||
f = f.Addr()
|
f = f.Addr()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
assignmentColumns := []string{}
|
||||||
for _, ref := range rel.References {
|
for _, ref := range rel.References {
|
||||||
if ref.OwnPrimaryKey {
|
if ref.OwnPrimaryKey {
|
||||||
fv, _ := ref.PrimaryKey.ValueOf(db.Statement.ReflectValue)
|
fv, _ := ref.PrimaryKey.ValueOf(db.Statement.ReflectValue)
|
||||||
|
@ -155,13 +159,13 @@ func SaveAfterAssociations(db *gorm.DB) {
|
||||||
} else if ref.PrimaryValue != "" {
|
} else if ref.PrimaryValue != "" {
|
||||||
ref.ForeignKey.Set(f, ref.PrimaryValue)
|
ref.ForeignKey.Set(f, ref.PrimaryValue)
|
||||||
}
|
}
|
||||||
|
assignmentColumns = append(assignmentColumns, ref.ForeignKey.DBName)
|
||||||
}
|
}
|
||||||
|
|
||||||
if _, isZero := rel.FieldSchema.PrioritizedPrimaryField.ValueOf(f); isZero {
|
db.Session(&gorm.Session{}).Clauses(clause.OnConflict{
|
||||||
db.Session(&gorm.Session{}).Create(f.Interface())
|
Columns: []clause.Column{{Name: rel.FieldSchema.PrioritizedPrimaryField.DBName}},
|
||||||
} else {
|
DoUpdates: clause.AssignmentColumns(assignmentColumns),
|
||||||
db.Session(&gorm.Session{}).Save(f.Interface())
|
}).Create(f.Interface())
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -193,14 +197,10 @@ func SaveAfterAssociations(db *gorm.DB) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if _, isZero := rel.FieldSchema.PrioritizedPrimaryField.ValueOf(elem); isZero {
|
if isPtr {
|
||||||
if isPtr {
|
elems = reflect.Append(elems, elem)
|
||||||
elems = reflect.Append(elems, elem)
|
|
||||||
} else {
|
|
||||||
elems = reflect.Append(elems, elem.Addr())
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
db.Session(&gorm.Session{}).Save(elem.Addr().Interface())
|
elems = reflect.Append(elems, elem.Addr())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -216,7 +216,15 @@ func SaveAfterAssociations(db *gorm.DB) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if elems.Len() > 0 {
|
if elems.Len() > 0 {
|
||||||
db.Session(&gorm.Session{}).Create(elems.Interface())
|
assignmentColumns := []string{}
|
||||||
|
for _, ref := range rel.References {
|
||||||
|
assignmentColumns = append(assignmentColumns, ref.ForeignKey.DBName)
|
||||||
|
}
|
||||||
|
|
||||||
|
db.Session(&gorm.Session{}).Clauses(clause.OnConflict{
|
||||||
|
Columns: []clause.Column{{Name: rel.FieldSchema.PrioritizedPrimaryField.DBName}},
|
||||||
|
DoUpdates: clause.AssignmentColumns(assignmentColumns),
|
||||||
|
}).Create(elems.Interface())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -258,15 +266,11 @@ func SaveAfterAssociations(db *gorm.DB) {
|
||||||
for i := 0; i < f.Len(); i++ {
|
for i := 0; i < f.Len(); i++ {
|
||||||
elem := f.Index(i)
|
elem := f.Index(i)
|
||||||
|
|
||||||
if _, isZero := rel.FieldSchema.PrioritizedPrimaryField.ValueOf(elem); isZero {
|
objs = append(objs, v)
|
||||||
objs = append(objs, v)
|
if isPtr {
|
||||||
if isPtr {
|
elems = reflect.Append(elems, elem)
|
||||||
elems = reflect.Append(elems, elem)
|
|
||||||
} else {
|
|
||||||
elems = reflect.Append(elems, elem.Addr())
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
appendToJoins(v, elem)
|
elems = reflect.Append(elems, elem.Addr())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -282,7 +286,7 @@ func SaveAfterAssociations(db *gorm.DB) {
|
||||||
}
|
}
|
||||||
|
|
||||||
if elems.Len() > 0 {
|
if elems.Len() > 0 {
|
||||||
db.Session(&gorm.Session{}).Create(elems.Interface())
|
db.Session(&gorm.Session{}).Clauses(clause.OnConflict{DoNothing: true}).Create(elems.Interface())
|
||||||
|
|
||||||
for i := 0; i < elems.Len(); i++ {
|
for i := 0; i < elems.Len(); i++ {
|
||||||
appendToJoins(objs[i], elems.Index(i))
|
appendToJoins(objs[i], elems.Index(i))
|
||||||
|
|
|
@ -55,29 +55,44 @@ func Create(config *Config) func(db *gorm.DB) {
|
||||||
result, err := db.Statement.ConnPool.ExecContext(db.Statement.Context, db.Statement.SQL.String(), db.Statement.Vars...)
|
result, err := db.Statement.ConnPool.ExecContext(db.Statement.Context, db.Statement.SQL.String(), db.Statement.Vars...)
|
||||||
|
|
||||||
if err == nil {
|
if err == nil {
|
||||||
if db.Statement.Schema != nil && db.Statement.Schema.PrioritizedPrimaryField != nil && db.Statement.Schema.PrioritizedPrimaryField.HasDefaultValue {
|
db.RowsAffected, _ = result.RowsAffected()
|
||||||
if insertID, err := result.LastInsertId(); err == nil {
|
if db.RowsAffected > 0 {
|
||||||
switch db.Statement.ReflectValue.Kind() {
|
if db.Statement.Schema != nil && db.Statement.Schema.PrioritizedPrimaryField != nil && db.Statement.Schema.PrioritizedPrimaryField.HasDefaultValue {
|
||||||
case reflect.Slice, reflect.Array:
|
if insertID, err := result.LastInsertId(); err == nil {
|
||||||
if config.LastInsertIDReversed {
|
switch db.Statement.ReflectValue.Kind() {
|
||||||
for i := db.Statement.ReflectValue.Len() - 1; i >= 0; i-- {
|
case reflect.Slice, reflect.Array:
|
||||||
db.Statement.Schema.PrioritizedPrimaryField.Set(db.Statement.ReflectValue.Index(i), insertID)
|
if config.LastInsertIDReversed {
|
||||||
insertID--
|
for i := db.Statement.ReflectValue.Len() - 1; i >= 0; i-- {
|
||||||
}
|
_, isZero := db.Statement.Schema.PrioritizedPrimaryField.ValueOf(db.Statement.ReflectValue.Index(i))
|
||||||
} else {
|
if isZero {
|
||||||
for i := 0; i < db.Statement.ReflectValue.Len(); i++ {
|
db.Statement.Schema.PrioritizedPrimaryField.Set(db.Statement.ReflectValue.Index(i), insertID)
|
||||||
db.Statement.Schema.PrioritizedPrimaryField.Set(db.Statement.ReflectValue.Index(i), insertID)
|
insertID--
|
||||||
insertID++
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
allUpdated := int(db.RowsAffected) == db.Statement.ReflectValue.Len()
|
||||||
|
isZero := true
|
||||||
|
|
||||||
|
for i := 0; i < db.Statement.ReflectValue.Len(); i++ {
|
||||||
|
|
||||||
|
if !allUpdated {
|
||||||
|
_, isZero = db.Statement.Schema.PrioritizedPrimaryField.ValueOf(db.Statement.ReflectValue.Index(i))
|
||||||
|
}
|
||||||
|
|
||||||
|
if isZero {
|
||||||
|
db.Statement.Schema.PrioritizedPrimaryField.Set(db.Statement.ReflectValue.Index(i), insertID)
|
||||||
|
insertID++
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
case reflect.Struct:
|
||||||
|
db.Statement.Schema.PrioritizedPrimaryField.Set(db.Statement.ReflectValue, insertID)
|
||||||
}
|
}
|
||||||
case reflect.Struct:
|
} else {
|
||||||
db.Statement.Schema.PrioritizedPrimaryField.Set(db.Statement.ReflectValue, insertID)
|
db.AddError(err)
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
db.AddError(err)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
db.RowsAffected, _ = result.RowsAffected()
|
|
||||||
} else {
|
} else {
|
||||||
db.AddError(err)
|
db.AddError(err)
|
||||||
}
|
}
|
||||||
|
@ -129,9 +144,19 @@ func CreateWithReturning(db *gorm.DB) {
|
||||||
|
|
||||||
switch db.Statement.ReflectValue.Kind() {
|
switch db.Statement.ReflectValue.Kind() {
|
||||||
case reflect.Slice, reflect.Array:
|
case reflect.Slice, reflect.Array:
|
||||||
|
c := db.Statement.Clauses["ON CONFLICT"]
|
||||||
|
onConflict, _ := c.Expression.(clause.OnConflict)
|
||||||
|
|
||||||
for rows.Next() {
|
for rows.Next() {
|
||||||
|
BEGIN:
|
||||||
for idx, field := range fields {
|
for idx, field := range fields {
|
||||||
values[idx] = field.ReflectValueOf(db.Statement.ReflectValue.Index(int(db.RowsAffected))).Addr().Interface()
|
fieldValue := field.ReflectValueOf(db.Statement.ReflectValue.Index(int(db.RowsAffected)))
|
||||||
|
if onConflict.DoNothing && !fieldValue.IsZero() {
|
||||||
|
db.RowsAffected++
|
||||||
|
goto BEGIN
|
||||||
|
}
|
||||||
|
|
||||||
|
values[idx] = fieldValue.Addr().Interface()
|
||||||
}
|
}
|
||||||
|
|
||||||
db.RowsAffected++
|
db.RowsAffected++
|
||||||
|
@ -211,7 +236,7 @@ func ConvertToCreateValues(stmt *gorm.Statement) (values clause.Values) {
|
||||||
case reflect.Slice, reflect.Array:
|
case reflect.Slice, reflect.Array:
|
||||||
stmt.SQL.Grow(stmt.ReflectValue.Len() * 15)
|
stmt.SQL.Grow(stmt.ReflectValue.Len() * 15)
|
||||||
values.Values = make([][]interface{}, stmt.ReflectValue.Len())
|
values.Values = make([][]interface{}, stmt.ReflectValue.Len())
|
||||||
defaultValueFieldsHavingValue := map[string][]interface{}{}
|
defaultValueFieldsHavingValue := map[*schema.Field][]interface{}{}
|
||||||
for i := 0; i < stmt.ReflectValue.Len(); i++ {
|
for i := 0; i < stmt.ReflectValue.Len(); i++ {
|
||||||
rv := reflect.Indirect(stmt.ReflectValue.Index(i))
|
rv := reflect.Indirect(stmt.ReflectValue.Index(i))
|
||||||
values.Values[i] = make([]interface{}, len(values.Columns))
|
values.Values[i] = make([]interface{}, len(values.Columns))
|
||||||
|
@ -231,20 +256,20 @@ func ConvertToCreateValues(stmt *gorm.Statement) (values clause.Values) {
|
||||||
for _, field := range stmt.Schema.FieldsWithDefaultDBValue {
|
for _, field := range stmt.Schema.FieldsWithDefaultDBValue {
|
||||||
if v, ok := selectColumns[field.DBName]; (ok && v) || (!ok && !restricted) {
|
if v, ok := selectColumns[field.DBName]; (ok && v) || (!ok && !restricted) {
|
||||||
if v, isZero := field.ValueOf(rv); !isZero {
|
if v, isZero := field.ValueOf(rv); !isZero {
|
||||||
if len(defaultValueFieldsHavingValue[field.DBName]) == 0 {
|
if len(defaultValueFieldsHavingValue[field]) == 0 {
|
||||||
defaultValueFieldsHavingValue[field.DBName] = make([]interface{}, stmt.ReflectValue.Len())
|
defaultValueFieldsHavingValue[field] = make([]interface{}, stmt.ReflectValue.Len())
|
||||||
}
|
}
|
||||||
defaultValueFieldsHavingValue[field.DBName][i] = v
|
defaultValueFieldsHavingValue[field][i] = v
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
for db, vs := range defaultValueFieldsHavingValue {
|
for field, vs := range defaultValueFieldsHavingValue {
|
||||||
values.Columns = append(values.Columns, clause.Column{Name: db})
|
values.Columns = append(values.Columns, clause.Column{Name: field.DBName})
|
||||||
for idx := range values.Values {
|
for idx := range values.Values {
|
||||||
if vs[idx] == nil {
|
if vs[idx] == nil {
|
||||||
values.Values[idx] = append(values.Values[idx], clause.Expr{SQL: "DEFAULT"})
|
values.Values[idx] = append(values.Values[idx], stmt.Dialector.DefaultValueOf(field))
|
||||||
} else {
|
} else {
|
||||||
values.Values[idx] = append(values.Values[idx], vs[idx])
|
values.Values[idx] = append(values.Values[idx], vs[idx])
|
||||||
}
|
}
|
||||||
|
|
|
@ -64,6 +64,7 @@ func (c Clause) Build(builder Builder) {
|
||||||
const (
|
const (
|
||||||
PrimaryKey string = "@@@py@@@" // primary key
|
PrimaryKey string = "@@@py@@@" // primary key
|
||||||
CurrentTable string = "@@@ct@@@" // current table
|
CurrentTable string = "@@@ct@@@" // current table
|
||||||
|
Associations string = "@@@as@@@" // associations
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
|
|
@ -14,6 +14,7 @@ type Dialector interface {
|
||||||
Initialize(*DB) error
|
Initialize(*DB) error
|
||||||
Migrator(db *DB) Migrator
|
Migrator(db *DB) Migrator
|
||||||
DataTypeOf(*schema.Field) string
|
DataTypeOf(*schema.Field) string
|
||||||
|
DefaultValueOf(*schema.Field) clause.Expression
|
||||||
BindVarTo(writer clause.Writer, stmt *Statement, v interface{})
|
BindVarTo(writer clause.Writer, stmt *Statement, v interface{})
|
||||||
QuoteTo(clause.Writer, string)
|
QuoteTo(clause.Writer, string)
|
||||||
Explain(sql string, vars ...interface{}) string
|
Explain(sql string, vars ...interface{}) string
|
||||||
|
|
|
@ -57,10 +57,6 @@ func (m Migrator) DataTypeOf(field *schema.Field) string {
|
||||||
func (m Migrator) FullDataTypeOf(field *schema.Field) (expr clause.Expr) {
|
func (m Migrator) FullDataTypeOf(field *schema.Field) (expr clause.Expr) {
|
||||||
expr.SQL = m.DataTypeOf(field)
|
expr.SQL = m.DataTypeOf(field)
|
||||||
|
|
||||||
if field.AutoIncrement {
|
|
||||||
expr.SQL += " AUTO_INCREMENT"
|
|
||||||
}
|
|
||||||
|
|
||||||
if field.NotNull {
|
if field.NotNull {
|
||||||
expr.SQL += " NOT NULL"
|
expr.SQL += " NOT NULL"
|
||||||
}
|
}
|
||||||
|
|
|
@ -235,7 +235,7 @@ func TestParseFieldWithPermission(t *testing.T) {
|
||||||
}
|
}
|
||||||
|
|
||||||
fields := []schema.Field{
|
fields := []schema.Field{
|
||||||
{Name: "ID", DBName: "id", BindNames: []string{"ID"}, DataType: schema.Uint, PrimaryKey: true, Size: 64, Creatable: true, Updatable: true, Readable: true, HasDefaultValue: true},
|
{Name: "ID", DBName: "id", BindNames: []string{"ID"}, DataType: schema.Uint, PrimaryKey: true, Size: 64, Creatable: true, Updatable: true, Readable: true, HasDefaultValue: true, AutoIncrement: true},
|
||||||
{Name: "Name", DBName: "", BindNames: []string{"Name"}, DataType: "", Tag: `gorm:"-"`, Creatable: false, Updatable: false, Readable: false},
|
{Name: "Name", DBName: "", BindNames: []string{"Name"}, DataType: "", Tag: `gorm:"-"`, Creatable: false, Updatable: false, Readable: false},
|
||||||
{Name: "Name2", DBName: "name2", BindNames: []string{"Name2"}, DataType: schema.String, Tag: `gorm:"->"`, Creatable: false, Updatable: false, Readable: true},
|
{Name: "Name2", DBName: "name2", BindNames: []string{"Name2"}, DataType: schema.String, Tag: `gorm:"->"`, Creatable: false, Updatable: false, Readable: true},
|
||||||
{Name: "Name3", DBName: "name3", BindNames: []string{"Name3"}, DataType: schema.String, Tag: `gorm:"<-"`, Creatable: true, Updatable: true, Readable: true},
|
{Name: "Name3", DBName: "name3", BindNames: []string{"Name3"}, DataType: schema.String, Tag: `gorm:"<-"`, Creatable: true, Updatable: true, Readable: true},
|
||||||
|
|
|
@ -251,11 +251,13 @@ func (schema *Schema) buildMany2ManyRelation(relation *Relationship, field *Fiel
|
||||||
}
|
}
|
||||||
relation.JoinTable.Name = many2many
|
relation.JoinTable.Name = many2many
|
||||||
relation.JoinTable.Table = schema.namer.JoinTableName(many2many)
|
relation.JoinTable.Table = schema.namer.JoinTableName(many2many)
|
||||||
|
relation.JoinTable.PrimaryFields = make([]*Field, len(relation.JoinTable.Fields))
|
||||||
|
|
||||||
// build references
|
// build references
|
||||||
for _, f := range relation.JoinTable.Fields {
|
for idx, f := range relation.JoinTable.Fields {
|
||||||
// use same data type for foreign keys
|
// use same data type for foreign keys
|
||||||
f.DataType = fieldsMap[f.Name].DataType
|
f.DataType = fieldsMap[f.Name].DataType
|
||||||
|
relation.JoinTable.PrimaryFields[idx] = f
|
||||||
|
|
||||||
relation.References = append(relation.References, &Reference{
|
relation.References = append(relation.References, &Reference{
|
||||||
PrimaryKey: fieldsMap[f.Name],
|
PrimaryKey: fieldsMap[f.Name],
|
||||||
|
|
|
@ -188,6 +188,7 @@ func Parse(dest interface{}, cacheStore *sync.Map, namer Namer) (*Schema, error)
|
||||||
schema.FieldsWithDefaultDBValue = append(schema.FieldsWithDefaultDBValue, field)
|
schema.FieldsWithDefaultDBValue = append(schema.FieldsWithDefaultDBValue, field)
|
||||||
}
|
}
|
||||||
field.HasDefaultValue = true
|
field.HasDefaultValue = true
|
||||||
|
field.AutoIncrement = true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -32,7 +32,7 @@ func checkUserSchema(t *testing.T, user *schema.Schema) {
|
||||||
|
|
||||||
// check fields
|
// check fields
|
||||||
fields := []schema.Field{
|
fields := []schema.Field{
|
||||||
{Name: "ID", DBName: "id", BindNames: []string{"Model", "ID"}, DataType: schema.Uint, PrimaryKey: true, Tag: `gorm:"primarykey"`, TagSettings: map[string]string{"PRIMARYKEY": "PRIMARYKEY"}, Size: 64, HasDefaultValue: true},
|
{Name: "ID", DBName: "id", BindNames: []string{"Model", "ID"}, DataType: schema.Uint, PrimaryKey: true, Tag: `gorm:"primarykey"`, TagSettings: map[string]string{"PRIMARYKEY": "PRIMARYKEY"}, Size: 64, HasDefaultValue: true, AutoIncrement: true},
|
||||||
{Name: "CreatedAt", DBName: "created_at", BindNames: []string{"Model", "CreatedAt"}, DataType: schema.Time},
|
{Name: "CreatedAt", DBName: "created_at", BindNames: []string{"Model", "CreatedAt"}, DataType: schema.Time},
|
||||||
{Name: "UpdatedAt", DBName: "updated_at", BindNames: []string{"Model", "UpdatedAt"}, DataType: schema.Time},
|
{Name: "UpdatedAt", DBName: "updated_at", BindNames: []string{"Model", "UpdatedAt"}, DataType: schema.Time},
|
||||||
{Name: "DeletedAt", DBName: "deleted_at", BindNames: []string{"Model", "DeletedAt"}, Tag: `gorm:"index"`, DataType: schema.Time},
|
{Name: "DeletedAt", DBName: "deleted_at", BindNames: []string{"Model", "DeletedAt"}, Tag: `gorm:"index"`, DataType: schema.Time},
|
||||||
|
@ -125,7 +125,7 @@ func TestParseSchemaWithAdvancedDataType(t *testing.T) {
|
||||||
|
|
||||||
// check fields
|
// check fields
|
||||||
fields := []schema.Field{
|
fields := []schema.Field{
|
||||||
{Name: "ID", DBName: "id", BindNames: []string{"ID"}, DataType: schema.Int, PrimaryKey: true, Size: 64, HasDefaultValue: true},
|
{Name: "ID", DBName: "id", BindNames: []string{"ID"}, DataType: schema.Int, PrimaryKey: true, Size: 64, HasDefaultValue: true, AutoIncrement: true},
|
||||||
{Name: "Name", DBName: "name", BindNames: []string{"Name"}, DataType: schema.String},
|
{Name: "Name", DBName: "name", BindNames: []string{"Name"}, DataType: schema.String},
|
||||||
{Name: "Birthday", DBName: "birthday", BindNames: []string{"Birthday"}, DataType: schema.Time},
|
{Name: "Birthday", DBName: "birthday", BindNames: []string{"Birthday"}, DataType: schema.Time},
|
||||||
{Name: "RegisteredAt", DBName: "registered_at", BindNames: []string{"RegisteredAt"}, DataType: schema.Time},
|
{Name: "RegisteredAt", DBName: "registered_at", BindNames: []string{"RegisteredAt"}, DataType: schema.Time},
|
||||||
|
|
|
@ -68,6 +68,7 @@ func TestHasOneAssociation(t *testing.T) {
|
||||||
AssertAssociationCount(t, user2, "Account", 0, "after delete")
|
AssertAssociationCount(t, user2, "Account", 0, "after delete")
|
||||||
|
|
||||||
// Prepare Data for Clear
|
// Prepare Data for Clear
|
||||||
|
account = Account{Number: "account-has-one-append"}
|
||||||
if err := DB.Model(&user2).Association("Account").Append(&account); err != nil {
|
if err := DB.Model(&user2).Association("Account").Append(&account); err != nil {
|
||||||
t.Fatalf("Error happened when append Account, got %v", err)
|
t.Fatalf("Error happened when append Account, got %v", err)
|
||||||
}
|
}
|
||||||
|
@ -185,6 +186,7 @@ func TestPolymorphicHasOneAssociation(t *testing.T) {
|
||||||
AssertAssociationCount(t, pet2, "Toy", 0, "after delete")
|
AssertAssociationCount(t, pet2, "Toy", 0, "after delete")
|
||||||
|
|
||||||
// Prepare Data for Clear
|
// Prepare Data for Clear
|
||||||
|
toy = Toy{Name: "toy-has-one-append"}
|
||||||
if err := DB.Model(&pet2).Association("Toy").Append(&toy); err != nil {
|
if err := DB.Model(&pet2).Association("Toy").Append(&toy); err != nil {
|
||||||
t.Fatalf("Error happened when append Toy, got %v", err)
|
t.Fatalf("Error happened when append Toy, got %v", err)
|
||||||
}
|
}
|
||||||
|
|
10
tests/go.mod
10
tests/go.mod
|
@ -6,11 +6,11 @@ require (
|
||||||
github.com/google/uuid v1.1.1
|
github.com/google/uuid v1.1.1
|
||||||
github.com/jinzhu/now v1.1.1
|
github.com/jinzhu/now v1.1.1
|
||||||
github.com/lib/pq v1.6.0
|
github.com/lib/pq v1.6.0
|
||||||
gorm.io/driver/mysql v0.2.1
|
gorm.io/driver/mysql v0.2.2
|
||||||
gorm.io/driver/postgres v0.2.1
|
gorm.io/driver/postgres v0.2.2
|
||||||
gorm.io/driver/sqlite v1.0.4
|
gorm.io/driver/sqlite v1.0.5
|
||||||
gorm.io/driver/sqlserver v0.2.1
|
gorm.io/driver/sqlserver v0.2.2
|
||||||
gorm.io/gorm v0.2.7
|
gorm.io/gorm v0.2.9
|
||||||
)
|
)
|
||||||
|
|
||||||
replace gorm.io/gorm => ../
|
replace gorm.io/gorm => ../
|
||||||
|
|
|
@ -58,7 +58,6 @@ func GetUser(name string, config Config) *User {
|
||||||
for i := 0; i < config.Languages; i++ {
|
for i := 0; i < config.Languages; i++ {
|
||||||
name := name + "_locale_" + strconv.Itoa(i+1)
|
name := name + "_locale_" + strconv.Itoa(i+1)
|
||||||
language := Language{Code: name, Name: name}
|
language := Language{Code: name, Name: name}
|
||||||
DB.Create(&language)
|
|
||||||
user.Languages = append(user.Languages, language)
|
user.Languages = append(user.Languages, language)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -4,6 +4,7 @@ import (
|
||||||
"database/sql"
|
"database/sql"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"reflect"
|
"reflect"
|
||||||
|
"sort"
|
||||||
"testing"
|
"testing"
|
||||||
|
|
||||||
"gorm.io/gorm"
|
"gorm.io/gorm"
|
||||||
|
@ -735,7 +736,6 @@ func TestManyToManyPreloadWithMultiPrimaryKeys(t *testing.T) {
|
||||||
if err := DB.Preload("Level1s").Find(&got, "value = ?", "Bob").Error; err != nil {
|
if err := DB.Preload("Level1s").Find(&got, "value = ?", "Bob").Error; err != nil {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
return
|
|
||||||
|
|
||||||
if !reflect.DeepEqual(got, want) {
|
if !reflect.DeepEqual(got, want) {
|
||||||
t.Errorf("got %s; want %s", toJSONString(got), toJSONString(want))
|
t.Errorf("got %s; want %s", toJSONString(got), toJSONString(want))
|
||||||
|
@ -1459,6 +1459,12 @@ func TestPrefixedPreloadDuplication(t *testing.T) {
|
||||||
t.Error(err)
|
t.Error(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
for _, level1 := range append(got, want...) {
|
||||||
|
sort.Slice(level1.Level2.Level3.Level4s, func(i, j int) bool {
|
||||||
|
return level1.Level2.Level3.Level4s[i].ID > level1.Level2.Level3.Level4s[j].ID
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
if !reflect.DeepEqual(got, want) {
|
if !reflect.DeepEqual(got, want) {
|
||||||
t.Errorf("got %s; want %s", toJSONString(got), toJSONString(want))
|
t.Errorf("got %s; want %s", toJSONString(got), toJSONString(want))
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,7 +8,7 @@ if [ -d tests ]
|
||||||
then
|
then
|
||||||
cd tests
|
cd tests
|
||||||
cp go.mod go.mod.bak
|
cp go.mod go.mod.bak
|
||||||
sed '/gorm.io\/driver/d' go.mod.bak > go.mod
|
sed '/$[[:space:]]*gorm.io\/driver/d' go.mod.bak > go.mod
|
||||||
cd ..
|
cd ..
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
|
|
@ -34,6 +34,9 @@ func init() {
|
||||||
}
|
}
|
||||||
|
|
||||||
RunMigrations()
|
RunMigrations()
|
||||||
|
if DB.Dialector.Name() == "sqlite" {
|
||||||
|
DB.Exec("PRAGMA foreign_keys = ON")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -66,7 +69,6 @@ func OpenTestConnection() (db *gorm.DB, err error) {
|
||||||
default:
|
default:
|
||||||
log.Println("testing sqlite3...")
|
log.Println("testing sqlite3...")
|
||||||
db, err = gorm.Open(sqlite.Open(filepath.Join(os.TempDir(), "gorm.db")), &gorm.Config{})
|
db, err = gorm.Open(sqlite.Open(filepath.Join(os.TempDir(), "gorm.db")), &gorm.Config{})
|
||||||
db.Exec("PRAGMA foreign_keys = ON")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if debug := os.Getenv("DEBUG"); debug == "true" {
|
if debug := os.Getenv("DEBUG"); debug == "true" {
|
||||||
|
|
|
@ -18,6 +18,10 @@ func (DummyDialector) Initialize(*gorm.DB) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (DummyDialector) DefaultValueOf(field *schema.Field) clause.Expression {
|
||||||
|
return clause.Expr{SQL: "DEFAULT"}
|
||||||
|
}
|
||||||
|
|
||||||
func (DummyDialector) Migrator(*gorm.DB) gorm.Migrator {
|
func (DummyDialector) Migrator(*gorm.DB) gorm.Migrator {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue