Fix schema.ParseWithSchemaTable method for only use schemaTable in migrator and improve test.
This commit is contained in:
		
							parent
							
								
									09483e8928
								
							
						
					
					
						commit
						b1626b1b46
					
				@ -43,7 +43,7 @@ func (m Migrator) RunWithValue(value interface{}, fc func(*gorm.Statement) error
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	if table, ok := value.(string); ok {
 | 
						if table, ok := value.(string); ok {
 | 
				
			||||||
		stmt.Table = table
 | 
							stmt.Table = table
 | 
				
			||||||
	} else if err := stmt.Parse(value); err != nil {
 | 
						} else if err := stmt.ParseWithTableName(value, stmt.Table); err != nil {
 | 
				
			||||||
		return err
 | 
							return err
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
				
			|||||||
@ -107,7 +107,17 @@ func parse(dest interface{}, cacheStore *sync.Map, namer Namer, schemaTable stri
 | 
				
			|||||||
		return nil, fmt.Errorf("%w: %s.%s", ErrUnsupportedDataType, modelType.PkgPath(), modelType.Name())
 | 
							return nil, fmt.Errorf("%w: %s.%s", ErrUnsupportedDataType, modelType.PkgPath(), modelType.Name())
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if v, ok := cacheStore.Load(modelType); ok {
 | 
						// Cache the Schema for performance,
 | 
				
			||||||
 | 
						// Use the modelType or modelType + schemaTable (if it present) as cache key.
 | 
				
			||||||
 | 
						var schemaCacheKey interface{}
 | 
				
			||||||
 | 
						if schemaTable != "" {
 | 
				
			||||||
 | 
							schemaCacheKey = fmt.Sprintf("%p-%s", &modelType, schemaTable)
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							schemaCacheKey = modelType
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// Load exist schmema cache, return if exists
 | 
				
			||||||
 | 
						if v, ok := cacheStore.Load(schemaCacheKey); ok {
 | 
				
			||||||
		s := v.(*Schema)
 | 
							s := v.(*Schema)
 | 
				
			||||||
		// Wait for the initialization of other goroutines to complete
 | 
							// Wait for the initialization of other goroutines to complete
 | 
				
			||||||
		<-s.initialized
 | 
							<-s.initialized
 | 
				
			||||||
@ -115,18 +125,15 @@ func parse(dest interface{}, cacheStore *sync.Map, namer Namer, schemaTable stri
 | 
				
			|||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	modelValue := reflect.New(modelType)
 | 
						modelValue := reflect.New(modelType)
 | 
				
			||||||
 | 
						tableName := namer.TableName(modelType.Name())
 | 
				
			||||||
	// schemaTable for assignment table name directly
 | 
						if tabler, ok := modelValue.Interface().(Tabler); ok {
 | 
				
			||||||
	tableName := schemaTable
 | 
							tableName = tabler.TableName()
 | 
				
			||||||
	if schemaTable == "" {
 | 
						}
 | 
				
			||||||
		tableName = namer.TableName(modelType.Name())
 | 
						if en, ok := namer.(embeddedNamer); ok {
 | 
				
			||||||
 | 
							tableName = en.Table
 | 
				
			||||||
		if tabler, ok := modelValue.Interface().(Tabler); ok {
 | 
						}
 | 
				
			||||||
			tableName = tabler.TableName()
 | 
						if schemaTable != "" && schemaTable != tableName {
 | 
				
			||||||
		}
 | 
							tableName = schemaTable
 | 
				
			||||||
		if en, ok := namer.(embeddedNamer); ok {
 | 
					 | 
				
			||||||
			tableName = en.Table
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	schema := &Schema{
 | 
						schema := &Schema{
 | 
				
			||||||
@ -143,7 +150,8 @@ func parse(dest interface{}, cacheStore *sync.Map, namer Namer, schemaTable stri
 | 
				
			|||||||
	// When the schema initialization is completed, the channel will be closed
 | 
						// When the schema initialization is completed, the channel will be closed
 | 
				
			||||||
	defer close(schema.initialized)
 | 
						defer close(schema.initialized)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if v, loaded := cacheStore.Load(modelType); loaded {
 | 
						// Load exist schmema cache, return if exists
 | 
				
			||||||
 | 
						if v, ok := cacheStore.Load(schemaCacheKey); ok {
 | 
				
			||||||
		s := v.(*Schema)
 | 
							s := v.(*Schema)
 | 
				
			||||||
		// Wait for the initialization of other goroutines to complete
 | 
							// Wait for the initialization of other goroutines to complete
 | 
				
			||||||
		<-s.initialized
 | 
							<-s.initialized
 | 
				
			||||||
@ -250,13 +258,12 @@ func parse(dest interface{}, cacheStore *sync.Map, namer Namer, schemaTable stri
 | 
				
			|||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	if schemaTable == "" {
 | 
						// Cache the schema
 | 
				
			||||||
		if v, loaded := cacheStore.LoadOrStore(modelType, schema); loaded {
 | 
						if v, loaded := cacheStore.LoadOrStore(schemaCacheKey, schema); loaded {
 | 
				
			||||||
			s := v.(*Schema)
 | 
							s := v.(*Schema)
 | 
				
			||||||
			// Wait for the initialization of other goroutines to complete
 | 
							// Wait for the initialization of other goroutines to complete
 | 
				
			||||||
			<-s.initialized
 | 
							<-s.initialized
 | 
				
			||||||
			return s, s.err
 | 
							return s, s.err
 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	defer func() {
 | 
						defer func() {
 | 
				
			||||||
 | 
				
			|||||||
@ -456,7 +456,11 @@ func (stmt *Statement) Build(clauses ...string) {
 | 
				
			|||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func (stmt *Statement) Parse(value interface{}) (err error) {
 | 
					func (stmt *Statement) Parse(value interface{}) (err error) {
 | 
				
			||||||
	if stmt.Schema, err = schema.ParseWithSchemaTable(value, stmt.DB.cacheStore, stmt.DB.NamingStrategy, stmt.DB.Statement.Table); err == nil && stmt.Table == "" {
 | 
						return stmt.ParseWithTableName(value, "")
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func (stmt *Statement) ParseWithTableName(value interface{}, schemaTable string) (err error) {
 | 
				
			||||||
 | 
						if stmt.Schema, err = schema.ParseWithSchemaTable(value, stmt.DB.cacheStore, stmt.DB.NamingStrategy, schemaTable); err == nil && stmt.Table == "" {
 | 
				
			||||||
		if tables := strings.Split(stmt.Schema.Table, "."); len(tables) == 2 {
 | 
							if tables := strings.Split(stmt.Schema.Table, "."); len(tables) == 2 {
 | 
				
			||||||
			stmt.TableExpr = &clause.Expr{SQL: stmt.Quote(stmt.Schema.Table)}
 | 
								stmt.TableExpr = &clause.Expr{SQL: stmt.Quote(stmt.Schema.Table)}
 | 
				
			||||||
			stmt.Table = tables[1]
 | 
								stmt.Table = tables[1]
 | 
				
			||||||
 | 
				
			|||||||
@ -4,10 +4,10 @@ import (
 | 
				
			|||||||
	"database/sql/driver"
 | 
						"database/sql/driver"
 | 
				
			||||||
	"encoding/json"
 | 
						"encoding/json"
 | 
				
			||||||
	"errors"
 | 
						"errors"
 | 
				
			||||||
	"testing"
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
	"gorm.io/gorm"
 | 
						"gorm.io/gorm"
 | 
				
			||||||
	. "gorm.io/gorm/utils/tests"
 | 
						. "gorm.io/gorm/utils/tests"
 | 
				
			||||||
 | 
						"testing"
 | 
				
			||||||
)
 | 
					)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func TestEmbeddedStruct(t *testing.T) {
 | 
					func TestEmbeddedStruct(t *testing.T) {
 | 
				
			||||||
 | 
				
			|||||||
@ -6,6 +6,7 @@ require (
 | 
				
			|||||||
	github.com/google/uuid v1.3.0
 | 
						github.com/google/uuid v1.3.0
 | 
				
			||||||
	github.com/jinzhu/now v1.1.2
 | 
						github.com/jinzhu/now v1.1.2
 | 
				
			||||||
	github.com/lib/pq v1.10.3
 | 
						github.com/lib/pq v1.10.3
 | 
				
			||||||
 | 
						github.com/mattn/go-sqlite3 v1.14.9 // indirect
 | 
				
			||||||
	gorm.io/driver/mysql v1.1.2
 | 
						gorm.io/driver/mysql v1.1.2
 | 
				
			||||||
	gorm.io/driver/postgres v1.1.2
 | 
						gorm.io/driver/postgres v1.1.2
 | 
				
			||||||
	gorm.io/driver/sqlite v1.1.6
 | 
						gorm.io/driver/sqlite v1.1.6
 | 
				
			||||||
 | 
				
			|||||||
@ -398,7 +398,7 @@ func TestMigrateIndexesWithDynamicTableName(t *testing.T) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
	// Create sub tables
 | 
						// Create sub tables
 | 
				
			||||||
	for _, v := range []string{"01", "02", "03"} {
 | 
						for _, v := range []string{"01", "02", "03"} {
 | 
				
			||||||
		tableName := "users_" + v
 | 
							tableName := "dynamic_users_" + v
 | 
				
			||||||
		m := DB.Scopes(func(db *gorm.DB) *gorm.DB {
 | 
							m := DB.Scopes(func(db *gorm.DB) *gorm.DB {
 | 
				
			||||||
			return db.Table(tableName)
 | 
								return db.Table(tableName)
 | 
				
			||||||
		}).Migrator()
 | 
							}).Migrator()
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user