crud.go 5.7 KB

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