schema.go 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. package schema
  2. import (
  3. "errors"
  4. "fmt"
  5. "go/ast"
  6. "reflect"
  7. "sync"
  8. "github.com/jinzhu/gorm/logger"
  9. )
  10. // ErrUnsupportedDataType unsupported data type
  11. var ErrUnsupportedDataType = errors.New("unsupported data type")
  12. type Schema struct {
  13. Name string
  14. ModelType reflect.Type
  15. Table string
  16. PrioritizedPrimaryField *Field
  17. DBNames []string
  18. PrimaryFields []*Field
  19. Fields []*Field
  20. FieldsByName map[string]*Field
  21. FieldsByDBName map[string]*Field
  22. FieldsWithDefaultDBValue map[string]*Field // fields with default value assigned by database
  23. Relationships Relationships
  24. err error
  25. namer Namer
  26. cacheStore *sync.Map
  27. }
  28. func (schema Schema) String() string {
  29. if schema.ModelType.Name() == "" {
  30. return fmt.Sprintf("%v(%v)", schema.Name, schema.Table)
  31. }
  32. return fmt.Sprintf("%v.%v", schema.ModelType.PkgPath(), schema.ModelType.Name())
  33. }
  34. func (schema Schema) LookUpField(name string) *Field {
  35. if field, ok := schema.FieldsByDBName[name]; ok {
  36. return field
  37. }
  38. if field, ok := schema.FieldsByName[name]; ok {
  39. return field
  40. }
  41. return nil
  42. }
  43. // get data type from dialector
  44. func Parse(dest interface{}, cacheStore *sync.Map, namer Namer) (*Schema, error) {
  45. modelType := reflect.ValueOf(dest).Type()
  46. for modelType.Kind() == reflect.Slice || modelType.Kind() == reflect.Ptr {
  47. modelType = modelType.Elem()
  48. }
  49. if modelType.Kind() != reflect.Struct {
  50. if modelType.PkgPath() == "" {
  51. return nil, fmt.Errorf("%w: %+v", ErrUnsupportedDataType, dest)
  52. }
  53. return nil, fmt.Errorf("%w: %v.%v", ErrUnsupportedDataType, modelType.PkgPath(), modelType.Name())
  54. }
  55. if v, ok := cacheStore.Load(modelType); ok {
  56. return v.(*Schema), nil
  57. }
  58. schema := &Schema{
  59. Name: modelType.Name(),
  60. ModelType: modelType,
  61. Table: namer.TableName(modelType.Name()),
  62. FieldsByName: map[string]*Field{},
  63. FieldsByDBName: map[string]*Field{},
  64. Relationships: Relationships{Relations: map[string]*Relationship{}},
  65. cacheStore: cacheStore,
  66. namer: namer,
  67. }
  68. defer func() {
  69. if schema.err != nil {
  70. logger.Default.Error(schema.err.Error())
  71. cacheStore.Delete(modelType)
  72. }
  73. }()
  74. for i := 0; i < modelType.NumField(); i++ {
  75. if fieldStruct := modelType.Field(i); ast.IsExported(fieldStruct.Name) {
  76. if field := schema.ParseField(fieldStruct); field.EmbeddedSchema != nil {
  77. schema.Fields = append(schema.Fields, field.EmbeddedSchema.Fields...)
  78. } else {
  79. schema.Fields = append(schema.Fields, field)
  80. }
  81. }
  82. }
  83. for _, field := range schema.Fields {
  84. if field.DBName == "" && field.DataType != "" {
  85. field.DBName = namer.ColumnName(schema.Table, field.Name)
  86. }
  87. if field.DBName != "" {
  88. // nonexistence or shortest path or first appear prioritized if has permission
  89. if v, ok := schema.FieldsByDBName[field.DBName]; !ok || (field.Creatable && len(field.BindNames) < len(v.BindNames)) {
  90. if _, ok := schema.FieldsByDBName[field.DBName]; !ok {
  91. schema.DBNames = append(schema.DBNames, field.DBName)
  92. }
  93. schema.FieldsByDBName[field.DBName] = field
  94. schema.FieldsByName[field.Name] = field
  95. if v != nil && v.PrimaryKey {
  96. if schema.PrioritizedPrimaryField == v {
  97. schema.PrioritizedPrimaryField = nil
  98. }
  99. for idx, f := range schema.PrimaryFields {
  100. if f == v {
  101. schema.PrimaryFields = append(schema.PrimaryFields[0:idx], schema.PrimaryFields[idx+1:]...)
  102. } else if schema.PrioritizedPrimaryField == nil {
  103. schema.PrioritizedPrimaryField = f
  104. }
  105. }
  106. }
  107. if field.PrimaryKey {
  108. if schema.PrioritizedPrimaryField == nil {
  109. schema.PrioritizedPrimaryField = field
  110. }
  111. schema.PrimaryFields = append(schema.PrimaryFields, field)
  112. }
  113. }
  114. }
  115. if _, ok := schema.FieldsByName[field.Name]; !ok {
  116. schema.FieldsByName[field.Name] = field
  117. }
  118. field.setupValuerAndSetter()
  119. }
  120. if f := schema.LookUpField("id"); f != nil {
  121. if f.PrimaryKey {
  122. schema.PrioritizedPrimaryField = f
  123. } else if len(schema.PrimaryFields) == 0 {
  124. f.PrimaryKey = true
  125. schema.PrioritizedPrimaryField = f
  126. schema.PrimaryFields = append(schema.PrimaryFields, f)
  127. }
  128. }
  129. schema.FieldsWithDefaultDBValue = map[string]*Field{}
  130. for db, field := range schema.FieldsByDBName {
  131. if field.HasDefaultValue && field.DefaultValueInterface == nil {
  132. schema.FieldsWithDefaultDBValue[db] = field
  133. }
  134. }
  135. if schema.PrioritizedPrimaryField != nil {
  136. switch schema.PrioritizedPrimaryField.DataType {
  137. case Int, Uint:
  138. schema.FieldsWithDefaultDBValue[schema.PrioritizedPrimaryField.DBName] = schema.PrioritizedPrimaryField
  139. }
  140. }
  141. cacheStore.Store(modelType, schema)
  142. // parse relations for unidentified fields
  143. for _, field := range schema.Fields {
  144. if field.DataType == "" && field.Creatable {
  145. if schema.parseRelation(field); schema.err != nil {
  146. return schema, schema.err
  147. }
  148. }
  149. }
  150. return schema, schema.err
  151. }