crud.go 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231
  1. package rest
  2. import (
  3. "reflect"
  4. "sync"
  5. "git.nspix.com/golang/micro/gateway/http"
  6. "git.nspix.com/golang/micro/log"
  7. "gorm.io/gorm"
  8. )
  9. type (
  10. CRUD struct {
  11. db *gorm.DB
  12. entities sync.Map
  13. httpSvr *http.Server
  14. hooks []Hook
  15. httpMiddleware []http.Middleware
  16. }
  17. treeValue struct {
  18. Label string `json:"label"`
  19. Value string `json:"value"`
  20. Children []*treeValue `json:"children"`
  21. }
  22. )
  23. func (t *treeValue) Append(v *treeValue) {
  24. if t.Children == nil {
  25. t.Children = make([]*treeValue, 0)
  26. }
  27. t.Children = append(t.Children, v)
  28. }
  29. func (crud *CRUD) DB() *gorm.DB {
  30. return crud.db
  31. }
  32. //Use 附加一个钩子函数
  33. func (crud *CRUD) Use(h Hook) {
  34. crud.hooks = append(crud.hooks, h)
  35. }
  36. //handleQueryCrudModules 处理
  37. func (crud *CRUD) handleQueryCrudModules(ctx *http.Context) (err error) {
  38. ts := make([]*treeValue, 0)
  39. crud.entities.Range(func(key, value interface{}) bool {
  40. e := value.(*Entity)
  41. for _, tv := range ts {
  42. if tv.Value == e.model.ModuleName() {
  43. tv.Append(&treeValue{Value: e.model.ModuleName() + "-" + e.model.TableName(), Label: e.model.TableName()})
  44. return true
  45. }
  46. }
  47. tv := &treeValue{Label: e.model.ModuleName(), Value: e.model.ModuleName()}
  48. tv.Append(&treeValue{Value: e.model.ModuleName() + "-" + e.model.TableName(), Label: e.model.TableName()})
  49. ts = append(ts, tv)
  50. return true
  51. })
  52. return ctx.Success(ts)
  53. }
  54. //handleQuerySchema 处理http查询schema请求
  55. func (crud *CRUD) handleQuerySchema(ctx *http.Context) (err error) {
  56. var (
  57. schemas []*Schema
  58. )
  59. if schemas, err = getSchemasNoCache(crud.db, ctx.ParamValue("@namespace"), ctx.ParamValue("module"), ctx.ParamValue("table")); err == nil {
  60. return ctx.Success(schemas)
  61. }
  62. return ctx.Error(10011, err.Error())
  63. }
  64. //handleSaveSchema 保存schema
  65. func (crud *CRUD) handleSaveSchema(ctx *http.Context) (err error) {
  66. schemas := make([]*Schema, 0)
  67. if err = ctx.Bind(&schemas); err != nil {
  68. return ctx.Error(HttpInvalidPayload, err.Error())
  69. }
  70. if err = crud.db.Transaction(func(tx *gorm.DB) error {
  71. for _, scm := range schemas {
  72. if err2 := tx.Save(scm).Error; err2 != nil {
  73. return err2
  74. }
  75. }
  76. return nil
  77. }); err == nil {
  78. invalidCache(ctx.ParamValue("@namespace"), ctx.ParamValue("module"), ctx.ParamValue("table"))
  79. return ctx.Success(map[string]interface{}{
  80. "count": len(schemas),
  81. "state": "success",
  82. })
  83. }
  84. return ctx.Error(10014, err.Error())
  85. }
  86. //handleDeleteSchema 删除表的schema
  87. func (crud *CRUD) handleDeleteSchema(ctx *http.Context) (err error) {
  88. id := ctx.ParamValue("id")
  89. model := &Schema{}
  90. if err = crud.db.Where("id=?", id).First(model).Error; err == nil {
  91. invalidCache(ctx.ParamValue("@namespace"), model.Module, model.Table)
  92. if err = crud.db.Where("id=?", id).Delete(&Schema{}).Error; err == nil {
  93. return ctx.Success(map[string]string{
  94. "id": id,
  95. "state": "success",
  96. })
  97. } else {
  98. return ctx.Error(10012, err.Error())
  99. }
  100. } else {
  101. return ctx.Error(10012, err.Error())
  102. }
  103. }
  104. //router 绑定路由
  105. func (crud *CRUD) router() {
  106. if crud.httpSvr == nil {
  107. return
  108. }
  109. //获取注册上来的模块
  110. crud.httpSvr.Handle("GET", "/crud/modules", crud.handleQueryCrudModules, crud.httpMiddleware...)
  111. //获取所有schema
  112. crud.httpSvr.Handle("GET", "/schema/:module/:table", crud.handleQuerySchema, crud.httpMiddleware...)
  113. //更新schema
  114. crud.httpSvr.Handle("POST", "/schema/:module/:table", crud.handleSaveSchema, crud.httpMiddleware...)
  115. //删除schema
  116. crud.httpSvr.Handle("DELETE", "/schema/:id", crud.handleDeleteSchema, crud.httpMiddleware...)
  117. }
  118. // Attach 附加一个模型数据
  119. func (crud *CRUD) Attach(model Model, ops ...Option) (err error) {
  120. opts := NewOptions()
  121. for _, op := range ops {
  122. op(opts)
  123. }
  124. if opts.DB == nil {
  125. opts.DB = crud.db
  126. }
  127. //auto migrate database struct
  128. if err = crud.db.AutoMigrate(model); err != nil {
  129. return
  130. }
  131. //migrate table schema
  132. if err = migrateUp("", model); err != nil {
  133. return
  134. }
  135. scenarios := model.Scenario()
  136. entity := newEntity(model, opts)
  137. entity.hooks = crud.hooks
  138. if len(scenarios) == 0 {
  139. entity.scenarios = []string{ScenarioList, ScenarioCreate, ScenarioUpdate, ScenarioDelete, ScenarioExport, ScenarioView}
  140. } else {
  141. entity.scenarios = model.Scenario()
  142. }
  143. crud.entities.Store(entity.ID(), entity)
  144. return
  145. }
  146. // Detach 删除一个模型数据
  147. func (crud *CRUD) Detach(model Model) {
  148. id := model.TableName() + "@" + model.ModuleName()
  149. crud.entities.Delete(id)
  150. }
  151. //Routes 生成增删改查的路由
  152. func (crud *CRUD) Routes(ms ...http.Middleware) {
  153. var (
  154. method string
  155. uri string
  156. )
  157. if crud.httpSvr == nil {
  158. return
  159. }
  160. crud.httpMiddleware = ms
  161. crud.entities.Range(func(key, value interface{}) bool {
  162. entity := value.(*Entity)
  163. if entity.opts.Middleware == nil || len(entity.opts.Middleware) == 0 {
  164. entity.opts.Middleware = crud.httpMiddleware
  165. }
  166. for _, scenario := range entity.scenarios {
  167. method = entity.getScenarioMethod(scenario)
  168. uri = entity.getScenarioUrl(scenario)
  169. log.Debugf("CRUD: register %s %s", method, uri)
  170. crud.httpSvr.Handle(
  171. method,
  172. uri,
  173. entity.getScenarioHandle(scenario),
  174. entity.opts.Middleware...,
  175. )
  176. }
  177. //注册获取键值对数据格式
  178. if entity.isKvMapping() {
  179. crud.httpSvr.Handle("GET", entity.getScenarioUrl("mapping"), entity.getScenarioHandle("mapping"), entity.opts.Middleware...)
  180. }
  181. return true
  182. })
  183. crud.router()
  184. }
  185. //Schemas 获取一个模型的显示字段
  186. func (crud *CRUD) Schemas(namespace, moduleName, tableName, scenario string) []*Schema {
  187. if v, ok := crud.entities.Load(tableName + "@" + moduleName); ok {
  188. e := v.(*Entity)
  189. return visibleSchemas(namespace, e.model.ModuleName(), e.model.TableName(), scenario)
  190. }
  191. return nil
  192. }
  193. // IsNewRecord 判断该模型是不是一条新记录
  194. func (crud *CRUD) IsNewRecord(value reflect.Value, stmt *gorm.Statement) bool {
  195. for _, pf := range stmt.Schema.PrimaryFields {
  196. if _, isZero := pf.ValueOf(value); isZero {
  197. return true
  198. }
  199. }
  200. return false
  201. }
  202. //NewCRUD 创建一个新的CRUD模型
  203. func NewCRUD(db *gorm.DB, svr *http.Server) (crud *CRUD, err error) {
  204. if err = initSchema(db); err != nil {
  205. return
  206. }
  207. crud = &CRUD{
  208. db: db,
  209. hooks: make([]Hook, 0),
  210. httpSvr: svr,
  211. }
  212. return
  213. }