crud.go 7.0 KB

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