schema.go 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252
  1. package schema
  2. import (
  3. "context"
  4. "errors"
  5. "fmt"
  6. "go/ast"
  7. "reflect"
  8. "sync"
  9. "gorm.io/gorm/clause"
  10. "gorm.io/gorm/logger"
  11. )
  12. // ErrUnsupportedDataType unsupported data type
  13. var ErrUnsupportedDataType = errors.New("unsupported data type")
  14. type Schema struct {
  15. Name string
  16. ModelType reflect.Type
  17. Table string
  18. PrioritizedPrimaryField *Field
  19. DBNames []string
  20. PrimaryFields []*Field
  21. PrimaryFieldDBNames []string
  22. Fields []*Field
  23. FieldsByName map[string]*Field
  24. FieldsByDBName map[string]*Field
  25. FieldsWithDefaultDBValue []*Field // fields with default value assigned by database
  26. Relationships Relationships
  27. CreateClauses []clause.Interface
  28. QueryClauses []clause.Interface
  29. UpdateClauses []clause.Interface
  30. DeleteClauses []clause.Interface
  31. BeforeCreate, AfterCreate bool
  32. BeforeUpdate, AfterUpdate bool
  33. BeforeDelete, AfterDelete bool
  34. BeforeSave, AfterSave bool
  35. AfterFind bool
  36. err error
  37. namer Namer
  38. cacheStore *sync.Map
  39. }
  40. func (schema Schema) String() string {
  41. if schema.ModelType.Name() == "" {
  42. return fmt.Sprintf("%v(%v)", schema.Name, schema.Table)
  43. }
  44. return fmt.Sprintf("%v.%v", schema.ModelType.PkgPath(), schema.ModelType.Name())
  45. }
  46. func (schema Schema) MakeSlice() reflect.Value {
  47. slice := reflect.MakeSlice(reflect.SliceOf(reflect.PtrTo(schema.ModelType)), 0, 0)
  48. results := reflect.New(slice.Type())
  49. results.Elem().Set(slice)
  50. return results
  51. }
  52. func (schema Schema) LookUpField(name string) *Field {
  53. if field, ok := schema.FieldsByDBName[name]; ok {
  54. return field
  55. }
  56. if field, ok := schema.FieldsByName[name]; ok {
  57. return field
  58. }
  59. return nil
  60. }
  61. type Tabler interface {
  62. TableName() string
  63. }
  64. // get data type from dialector
  65. func Parse(dest interface{}, cacheStore *sync.Map, namer Namer) (*Schema, error) {
  66. if dest == nil {
  67. return nil, fmt.Errorf("%w: %+v", ErrUnsupportedDataType, dest)
  68. }
  69. modelType := reflect.ValueOf(dest).Type()
  70. for modelType.Kind() == reflect.Slice || modelType.Kind() == reflect.Array || modelType.Kind() == reflect.Ptr {
  71. modelType = modelType.Elem()
  72. }
  73. if modelType.Kind() != reflect.Struct {
  74. if modelType.PkgPath() == "" {
  75. return nil, fmt.Errorf("%w: %+v", ErrUnsupportedDataType, dest)
  76. }
  77. return nil, fmt.Errorf("%w: %v.%v", ErrUnsupportedDataType, modelType.PkgPath(), modelType.Name())
  78. }
  79. if v, ok := cacheStore.Load(modelType); ok {
  80. return v.(*Schema), nil
  81. }
  82. modelValue := reflect.New(modelType)
  83. tableName := namer.TableName(modelType.Name())
  84. if tabler, ok := modelValue.Interface().(Tabler); ok {
  85. tableName = tabler.TableName()
  86. }
  87. if en, ok := namer.(embeddedNamer); ok {
  88. tableName = en.Table
  89. }
  90. schema := &Schema{
  91. Name: modelType.Name(),
  92. ModelType: modelType,
  93. Table: tableName,
  94. FieldsByName: map[string]*Field{},
  95. FieldsByDBName: map[string]*Field{},
  96. Relationships: Relationships{Relations: map[string]*Relationship{}},
  97. cacheStore: cacheStore,
  98. namer: namer,
  99. }
  100. defer func() {
  101. if schema.err != nil {
  102. logger.Default.Error(context.Background(), schema.err.Error())
  103. cacheStore.Delete(modelType)
  104. }
  105. }()
  106. for i := 0; i < modelType.NumField(); i++ {
  107. if fieldStruct := modelType.Field(i); ast.IsExported(fieldStruct.Name) {
  108. if field := schema.ParseField(fieldStruct); field.EmbeddedSchema != nil {
  109. schema.Fields = append(schema.Fields, field.EmbeddedSchema.Fields...)
  110. } else {
  111. schema.Fields = append(schema.Fields, field)
  112. }
  113. }
  114. }
  115. for _, field := range schema.Fields {
  116. if field.DBName == "" && field.DataType != "" {
  117. field.DBName = namer.ColumnName(schema.Table, field.Name)
  118. }
  119. if field.DBName != "" {
  120. // nonexistence or shortest path or first appear prioritized if has permission
  121. if v, ok := schema.FieldsByDBName[field.DBName]; !ok || ((field.Creatable || field.Updatable || field.Readable) && len(field.BindNames) < len(v.BindNames)) {
  122. if _, ok := schema.FieldsByDBName[field.DBName]; !ok {
  123. schema.DBNames = append(schema.DBNames, field.DBName)
  124. }
  125. schema.FieldsByDBName[field.DBName] = field
  126. schema.FieldsByName[field.Name] = field
  127. if v != nil && v.PrimaryKey {
  128. for idx, f := range schema.PrimaryFields {
  129. if f == v {
  130. schema.PrimaryFields = append(schema.PrimaryFields[0:idx], schema.PrimaryFields[idx+1:]...)
  131. }
  132. }
  133. }
  134. if field.PrimaryKey {
  135. schema.PrimaryFields = append(schema.PrimaryFields, field)
  136. }
  137. }
  138. }
  139. if _, ok := schema.FieldsByName[field.Name]; !ok {
  140. schema.FieldsByName[field.Name] = field
  141. }
  142. field.setupValuerAndSetter()
  143. }
  144. prioritizedPrimaryField := schema.LookUpField("id")
  145. if prioritizedPrimaryField == nil {
  146. prioritizedPrimaryField = schema.LookUpField("ID")
  147. }
  148. if prioritizedPrimaryField != nil {
  149. if prioritizedPrimaryField.PrimaryKey {
  150. schema.PrioritizedPrimaryField = prioritizedPrimaryField
  151. } else if len(schema.PrimaryFields) == 0 {
  152. prioritizedPrimaryField.PrimaryKey = true
  153. schema.PrioritizedPrimaryField = prioritizedPrimaryField
  154. schema.PrimaryFields = append(schema.PrimaryFields, prioritizedPrimaryField)
  155. }
  156. }
  157. if schema.PrioritizedPrimaryField == nil && len(schema.PrimaryFields) == 1 {
  158. schema.PrioritizedPrimaryField = schema.PrimaryFields[0]
  159. }
  160. for _, field := range schema.PrimaryFields {
  161. schema.PrimaryFieldDBNames = append(schema.PrimaryFieldDBNames, field.DBName)
  162. }
  163. for _, field := range schema.FieldsByDBName {
  164. if field.HasDefaultValue && field.DefaultValueInterface == nil {
  165. schema.FieldsWithDefaultDBValue = append(schema.FieldsWithDefaultDBValue, field)
  166. }
  167. }
  168. if field := schema.PrioritizedPrimaryField; field != nil {
  169. switch field.GORMDataType {
  170. case Int, Uint:
  171. if _, ok := field.TagSettings["AUTOINCREMENT"]; !ok {
  172. if !field.HasDefaultValue || field.DefaultValueInterface != nil {
  173. schema.FieldsWithDefaultDBValue = append(schema.FieldsWithDefaultDBValue, field)
  174. }
  175. field.HasDefaultValue = true
  176. field.AutoIncrement = true
  177. }
  178. }
  179. }
  180. callbacks := []string{"BeforeCreate", "AfterCreate", "BeforeUpdate", "AfterUpdate", "BeforeSave", "AfterSave", "BeforeDelete", "AfterDelete", "AfterFind"}
  181. for _, name := range callbacks {
  182. if methodValue := modelValue.MethodByName(name); methodValue.IsValid() {
  183. switch methodValue.Type().String() {
  184. case "func(*gorm.DB) error": // TODO hack
  185. reflect.Indirect(reflect.ValueOf(schema)).FieldByName(name).SetBool(true)
  186. default:
  187. logger.Default.Warn(context.Background(), "Model %v don't match %vInterface, should be %v(*gorm.DB)", schema, name, name)
  188. }
  189. }
  190. }
  191. if _, loaded := cacheStore.LoadOrStore(modelType, schema); !loaded {
  192. if _, embedded := schema.cacheStore.Load(embeddedCacheKey); !embedded {
  193. for _, field := range schema.Fields {
  194. if field.DataType == "" && (field.Creatable || field.Updatable || field.Readable) {
  195. if schema.parseRelation(field); schema.err != nil {
  196. return schema, schema.err
  197. }
  198. }
  199. fieldValue := reflect.New(field.IndirectFieldType)
  200. if fc, ok := fieldValue.Interface().(CreateClausesInterface); ok {
  201. field.Schema.CreateClauses = append(field.Schema.CreateClauses, fc.CreateClauses(field)...)
  202. }
  203. if fc, ok := fieldValue.Interface().(QueryClausesInterface); ok {
  204. field.Schema.QueryClauses = append(field.Schema.QueryClauses, fc.QueryClauses(field)...)
  205. }
  206. if fc, ok := fieldValue.Interface().(UpdateClausesInterface); ok {
  207. field.Schema.UpdateClauses = append(field.Schema.UpdateClauses, fc.UpdateClauses(field)...)
  208. }
  209. if fc, ok := fieldValue.Interface().(DeleteClausesInterface); ok {
  210. field.Schema.DeleteClauses = append(field.Schema.DeleteClauses, fc.DeleteClauses(field)...)
  211. }
  212. }
  213. }
  214. }
  215. return schema, schema.err
  216. }