crud.go 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390
  1. package rest
  2. import (
  3. "context"
  4. "fmt"
  5. "git.nspix.com/golang/micro/gateway/http"
  6. "git.nspix.com/golang/micro/log"
  7. "git.nspix.com/golang/rest/v2/pkg/logger"
  8. "gorm.io/driver/mysql"
  9. "gorm.io/driver/sqlite"
  10. "gorm.io/gorm"
  11. "reflect"
  12. "sort"
  13. "sync"
  14. "sync/atomic"
  15. )
  16. const (
  17. TypeModule = "module"
  18. TypeTable = "table"
  19. )
  20. type (
  21. CRUD struct {
  22. index int32
  23. db *gorm.DB
  24. entities sync.Map
  25. httpSvr *http.Server
  26. httpMiddleware []http.Middleware
  27. callback *Callback
  28. moduleLabels map[string]string
  29. }
  30. treeValue struct {
  31. Label string `json:"label"` //标签
  32. Value string `json:"value"` //值
  33. Type string `json:"type"` //类型
  34. Children []*treeValue `json:"children"`
  35. }
  36. contextKey struct{}
  37. )
  38. var (
  39. ctxKey = contextKey{}
  40. )
  41. func (t *treeValue) Append(v *treeValue) {
  42. if t.Children == nil {
  43. t.Children = make([]*treeValue, 0)
  44. }
  45. t.Children = append(t.Children, v)
  46. }
  47. //DB 获取当前数据库实例
  48. func (crud *CRUD) DB() *gorm.DB {
  49. return crud.db
  50. }
  51. //WithDB 设置当前数据库实例
  52. func (crud *CRUD) WithDB(db *gorm.DB) {
  53. crud.db = db
  54. }
  55. //Callback 获取回调管理器
  56. func (crud *CRUD) Callback() *Callback {
  57. return crud.callback
  58. }
  59. //handleQueryCrudModules 获取模块的信息
  60. func (crud *CRUD) handleQueryCrudModules(ctx *http.Context) (err error) {
  61. ts := make([]*treeValue, 0)
  62. entities := make(Entities, 0)
  63. crud.entities.Range(func(key, value interface{}) bool {
  64. entities = append(entities, value.(*Entity))
  65. return true
  66. })
  67. sort.Sort(entities)
  68. isHandled := false
  69. for _, e := range entities {
  70. isHandled = false
  71. for _, tv := range ts {
  72. if tv.Value == e.model.ModuleName() {
  73. if viewer, ok := e.model.(ModelViewer); ok {
  74. tv.Append(&treeValue{Value: e.model.ModuleName() + "-" + e.model.TableName(), Label: viewer.TableLabel(), Type: TypeTable})
  75. } else {
  76. tv.Append(&treeValue{Value: e.model.ModuleName() + "-" + e.model.TableName(), Label: e.model.TableName(), Type: TypeTable})
  77. }
  78. isHandled = true
  79. break
  80. }
  81. }
  82. if isHandled {
  83. continue
  84. }
  85. moduleLabel := crud.moduleLabels[e.model.ModuleName()]
  86. if moduleLabel == "" {
  87. moduleLabel = e.model.ModuleName()
  88. }
  89. tv := &treeValue{Label: moduleLabel, Value: e.model.ModuleName(), Type: TypeModule}
  90. if viewer, ok := e.model.(ModelViewer); ok {
  91. tv.Append(&treeValue{Value: e.model.ModuleName() + "-" + e.model.TableName(), Label: viewer.TableLabel(), Type: TypeTable})
  92. } else {
  93. tv.Append(&treeValue{Value: e.model.ModuleName() + "-" + e.model.TableName(), Label: e.model.TableName(), Type: TypeTable})
  94. }
  95. ts = append(ts, tv)
  96. }
  97. return ctx.Success(ts)
  98. }
  99. //handleQuerySchema 处理http查询schema请求
  100. func (crud *CRUD) handleQuerySchema(ctx *http.Context) (err error) {
  101. var (
  102. schemas []*Schema
  103. modelEntity *Entity
  104. )
  105. moduleName := ctx.ParamValue("module")
  106. tableName := ctx.ParamValue("table")
  107. crud.entities.Range(func(key, value interface{}) bool {
  108. entity := value.(*Entity)
  109. if entity.model.ModuleName() == moduleName {
  110. if entity.model.TableName() == tableName {
  111. modelEntity = entity
  112. return false
  113. }
  114. if entity.opts.RemoveTablePrefix {
  115. for _, prefix := range entity.opts.TablePrefixes {
  116. if prefix+tableName == entity.model.TableName() {
  117. modelEntity = entity
  118. return false
  119. }
  120. }
  121. }
  122. }
  123. return true
  124. })
  125. if modelEntity == nil {
  126. return ctx.Error(10011, fmt.Sprintf("module %s table %s schema not found", moduleName, tableName))
  127. }
  128. if schemas, err = getSchemasNoCache(crud.db, ctx.ParamValue(NamespaceVariable), modelEntity.model.ModuleName(), modelEntity.model.TableName()); err == nil {
  129. return ctx.Success(schemas)
  130. }
  131. return ctx.Error(10011, err.Error())
  132. }
  133. //handleSaveSchema 保存schema
  134. func (crud *CRUD) handleSaveSchema(ctx *http.Context) (err error) {
  135. var (
  136. modelEntity *Entity
  137. )
  138. moduleName := ctx.ParamValue("module")
  139. tableName := ctx.ParamValue("table")
  140. crud.entities.Range(func(key, value interface{}) bool {
  141. entity := value.(*Entity)
  142. if entity.model.ModuleName() == moduleName {
  143. if entity.model.TableName() == tableName {
  144. modelEntity = entity
  145. return false
  146. }
  147. if entity.opts.RemoveTablePrefix {
  148. for _, prefix := range entity.opts.TablePrefixes {
  149. if prefix+tableName == entity.model.TableName() {
  150. modelEntity = entity
  151. return false
  152. }
  153. }
  154. }
  155. }
  156. return true
  157. })
  158. if modelEntity == nil {
  159. return ctx.Error(10011, fmt.Sprintf("module %s table %s schema not found", moduleName, tableName))
  160. }
  161. schemas := make([]*Schema, 0)
  162. if err = ctx.Bind(&schemas); err != nil {
  163. return ctx.Error(HttpInvalidPayload, err.Error())
  164. }
  165. if err = crud.db.Transaction(func(tx *gorm.DB) error {
  166. for _, scm := range schemas {
  167. if err2 := tx.Save(scm).Error; err2 != nil {
  168. return err2
  169. }
  170. }
  171. return nil
  172. }); err == nil {
  173. invalidCache(ctx.ParamValue(NamespaceVariable), modelEntity.model.ModuleName(), modelEntity.model.TableName())
  174. return ctx.Success(map[string]interface{}{
  175. "count": len(schemas),
  176. "state": "success",
  177. })
  178. }
  179. return ctx.Error(10014, err.Error())
  180. }
  181. //handleDeleteSchema 删除表的schema
  182. func (crud *CRUD) handleDeleteSchema(ctx *http.Context) (err error) {
  183. id := ctx.ParamValue("id")
  184. model := &Schema{}
  185. if err = crud.db.Where("id=?", id).First(model).Error; err == nil {
  186. invalidCache(ctx.ParamValue(NamespaceVariable), model.Module, model.Table)
  187. if err = crud.db.Where("id=?", id).Delete(&Schema{}).Error; err == nil {
  188. return ctx.Success(map[string]string{
  189. "id": id,
  190. "state": "success",
  191. })
  192. } else {
  193. return ctx.Error(10012, err.Error())
  194. }
  195. } else {
  196. return ctx.Error(10012, err.Error())
  197. }
  198. }
  199. //router 绑定路由
  200. func (crud *CRUD) router() {
  201. if crud.httpSvr == nil {
  202. return
  203. }
  204. //获取注册上来的模块
  205. crud.httpSvr.Handle("GET", "/crud/modules", crud.handleQueryCrudModules, crud.httpMiddleware...)
  206. //获取所有schema
  207. crud.httpSvr.Handle("GET", "/schema/:module/:table", crud.handleQuerySchema, crud.httpMiddleware...)
  208. //更新schema
  209. crud.httpSvr.Handle("POST", "/schema/:module/:table", crud.handleSaveSchema, crud.httpMiddleware...)
  210. //删除schema
  211. crud.httpSvr.Handle("DELETE", "/schema/:id", crud.handleDeleteSchema, crud.httpMiddleware...)
  212. }
  213. // Attach 附加一个模型数据
  214. func (crud *CRUD) Attach(model Model, ops ...Option) (err error) {
  215. opts := NewOptions()
  216. for _, op := range ops {
  217. op(opts)
  218. }
  219. if opts.DB == nil {
  220. opts.DB = crud.db
  221. }
  222. //auto migrate database struct
  223. tx := opts.DB.Session(&gorm.Session{NewDB: true})
  224. if opts.MigrateOptions != nil && opts.MigrateOptions.TableOptions != "" {
  225. tx.Set("gorm:table_options", opts.MigrateOptions.TableOptions)
  226. }
  227. if err = tx.AutoMigrate(model); err != nil {
  228. return
  229. }
  230. //migrate table schema
  231. if err = migrateUp(opts.Namespace, model, opts.MigrateOptions); err != nil {
  232. return
  233. }
  234. scenarios := model.Scenario()
  235. entity := newEntity(atomic.AddInt32(&crud.index, 1), model, opts)
  236. entity.callback = crud.callback
  237. if len(scenarios) == 0 {
  238. entity.scenarios = []string{ScenarioList, ScenarioCreate, ScenarioUpdate, ScenarioDelete, ScenarioExport, ScenarioView}
  239. } else {
  240. entity.scenarios = model.Scenario()
  241. }
  242. crud.entities.Store(entity.ID(), entity)
  243. return
  244. }
  245. // Detach 删除一个模型数据
  246. func (crud *CRUD) Detach(model Model) {
  247. id := model.TableName() + "@" + model.ModuleName()
  248. crud.entities.Delete(id)
  249. }
  250. //Routes 生成增删改查的路由
  251. func (crud *CRUD) Routes(svr *http.Server, ms ...http.Middleware) {
  252. var (
  253. method string
  254. uri string
  255. )
  256. if crud.httpSvr == nil {
  257. crud.httpSvr = svr
  258. }
  259. if crud.httpSvr == nil {
  260. return
  261. }
  262. crud.httpMiddleware = ms
  263. crud.entities.Range(func(key, value interface{}) bool {
  264. entity := value.(*Entity)
  265. if entity.opts.Middleware == nil || len(entity.opts.Middleware) == 0 {
  266. entity.opts.Middleware = crud.httpMiddleware
  267. }
  268. for _, scenario := range entity.scenarios {
  269. method = entity.getScenarioMethod(scenario)
  270. uri = entity.getScenarioUrl(scenario)
  271. log.Debugf("CRUD: register %s %s", method, uri)
  272. crud.httpSvr.Handle(
  273. method,
  274. uri,
  275. entity.getScenarioHandle(scenario),
  276. entity.opts.Middleware...,
  277. )
  278. }
  279. //注册获取键值对数据格式
  280. if entity.isKvMapping() {
  281. uri = entity.getScenarioUrl("mapping")
  282. log.Debugf("CRUD: register %s %s", "GET", uri)
  283. crud.httpSvr.Handle("GET", uri, entity.getScenarioHandle("mapping"), entity.opts.Middleware...)
  284. }
  285. return true
  286. })
  287. crud.router()
  288. }
  289. //Schemas 获取一个模型的显示字段
  290. func (crud *CRUD) Schemas(namespace, moduleName, tableName, scenario string) []*Schema {
  291. if v, ok := crud.entities.Load(tableName + "@" + moduleName); ok {
  292. e := v.(*Entity)
  293. return visibleSchemas(namespace, e.model.ModuleName(), e.model.TableName(), scenario)
  294. }
  295. return nil
  296. }
  297. // IsNewRecord 判断该模型是不是一条新记录
  298. func (crud *CRUD) IsNewRecord(value reflect.Value, stmt *gorm.Statement) bool {
  299. for _, pf := range stmt.Schema.PrimaryFields {
  300. if _, isZero := pf.ValueOf(value); isZero {
  301. return true
  302. }
  303. }
  304. return false
  305. }
  306. // SetModuleLabel 设置模块标签
  307. func (crud *CRUD) SetModuleLabel(moduleName string, moduleLabel string) {
  308. if crud.moduleLabels == nil {
  309. crud.moduleLabels = make(map[string]string)
  310. }
  311. crud.moduleLabels[moduleName] = moduleLabel
  312. }
  313. //NewCRUD 创建一个新的CRUD模型
  314. func NewCRUD(db *gorm.DB, svr *http.Server) (crud *CRUD, err error) {
  315. if err = initSchema(db); err != nil {
  316. return
  317. }
  318. crud = &CRUD{
  319. db: db,
  320. callback: newCallback(),
  321. httpSvr: svr,
  322. moduleLabels: make(map[string]string),
  323. }
  324. return
  325. }
  326. func openDatabase(cfg *Config) (db *gorm.DB, err error) {
  327. dbCfg := &gorm.Config{Logger: logger.NewGormLogger()}
  328. switch cfg.Driver {
  329. case "sqlite3":
  330. db, err = gorm.Open(sqlite.Open(cfg.ParseDSN()), dbCfg)
  331. case "mysql":
  332. db, err = gorm.Open(mysql.Open(cfg.ParseDSN()), dbCfg)
  333. default:
  334. err = fmt.Errorf("unsupported sql driver %s", cfg.Driver)
  335. }
  336. return
  337. }
  338. //Dialer 连接一个数据库
  339. func Dialer(cfg *Config) (crud *CRUD, err error) {
  340. var (
  341. db *gorm.DB
  342. )
  343. if db, err = openDatabase(cfg); err != nil {
  344. return
  345. }
  346. if err = initSchema(db); err != nil {
  347. return
  348. }
  349. crud = &CRUD{
  350. db: db,
  351. moduleLabels: make(map[string]string),
  352. callback: newCallback(),
  353. }
  354. return
  355. }
  356. func FromContext(ctx context.Context) *CRUD {
  357. if v := ctx.Value(ctxKey); v != nil {
  358. return v.(*CRUD)
  359. }
  360. return nil
  361. }
  362. func WithContext(ctx context.Context, r *CRUD) context.Context {
  363. return context.WithValue(ctx, ctxKey, r)
  364. }