repo.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651
  1. package repo
  2. import (
  3. "context"
  4. "fmt"
  5. "mats/db"
  6. "mats/log"
  7. "go.mongodb.org/mongo-driver/bson"
  8. "go.mongodb.org/mongo-driver/bson/primitive"
  9. "go.mongodb.org/mongo-driver/mongo"
  10. "go.mongodb.org/mongo-driver/mongo/options"
  11. )
  12. type RepoSession struct {
  13. Ctx context.Context
  14. Client *db.MongoDB
  15. }
  16. const (
  17. CollectionUser = "users"
  18. CollectionProject = "projects"
  19. CollectionAdmin = "admins"
  20. CollectionTexture = "textures"
  21. CollectionCategories = "categories"
  22. CollectionDbAssetCategory = "categories-asset"
  23. CollectionLibCategory = "libCategory"
  24. CollectionLib2Category = "lib2Category"
  25. CollectionTeam = "team"
  26. CollectionStickers = "sticker"
  27. CollectionMesh = "meshes"
  28. CollectionEvn3d = "Env3ds"
  29. CollectionTask = "tasks"
  30. CollectionTaskMeshConv = "task-meshconv"
  31. CollectionTaskHdrConv = "task-hdrconv"
  32. CollectionTaskShadowCreator = "task-shadowCreator"
  33. CollectionTaskShadow = "task-shadow"
  34. CollectionProductTpls = "productTpls"
  35. CollectionMaterials = "materials"
  36. CollectionBaseMats = "basemats"
  37. CollectionDecorates = "decorate-mesh"
  38. CollectionInfoDesigns = "infoDesigns"
  39. CollectionInfoTrends = "infoTrends"
  40. CollectionDesignProducts = "designProducts"
  41. CollectionShoes = "shoes"
  42. CollectionImageMat = "imgmats"
  43. CollectionFabric = "fabrics"
  44. CollectionDesigns = "designs"
  45. CollectionBackground = "backgrounds"
  46. CollectionLasts = "last-mesh"
  47. CollectionHeels = "heel-mesh"
  48. CollectionSoles = "sole-mesh"
  49. CollectionPerms = "perms"
  50. CollectionRoles = "roles"
  51. CollectionDatabase = "database"
  52. CollectionQueenters = "queenters"
  53. CollectionQueenterOutputs = "queenter-outputs"
  54. CollectionHubMats = "hubmats"
  55. )
  56. type Map map[string]interface{}
  57. type PageResult struct {
  58. List []map[string]interface{} `json:"list"`
  59. Total int64 `json:"total"`
  60. Page int64 `json:"page"`
  61. Size int64 `json:"size"`
  62. }
  63. type PageSearchOptions struct {
  64. Db string
  65. CollectName string
  66. Page int64
  67. Size int64
  68. Query map[string]interface{}
  69. Project []string
  70. Sort interface{}
  71. }
  72. type DocSearchOptions struct {
  73. Db string
  74. CollectName string
  75. Query Map
  76. Project []string
  77. }
  78. func NewDocSearchOptions(filter Map, project []string) *DocSearchOptions {
  79. return &DocSearchOptions{
  80. Query: filter,
  81. Project: project,
  82. }
  83. }
  84. func RepoAddDoc(ctx *RepoSession, collectName string, doc interface{}) (string, error) {
  85. users := ctx.Client.GetCollection(collectName)
  86. result, err := users.InsertOne(ctx.Ctx, doc)
  87. if err != nil {
  88. return "", err
  89. }
  90. return result.InsertedID.(primitive.ObjectID).Hex(), nil
  91. }
  92. func RepoDbAddDoc(ctx *RepoSession, dbName string, collectName string, doc interface{}) (string, error) {
  93. users := ctx.Client.GetDbCollection(dbName, collectName)
  94. result, err := users.InsertOne(ctx.Ctx, doc)
  95. if err != nil {
  96. return "", err
  97. }
  98. return result.InsertedID.(primitive.ObjectID).Hex(), nil
  99. }
  100. func RepoDeleteDoc(ctx *RepoSession, collectName string, id string) (interface{}, error) {
  101. uid, _ := primitive.ObjectIDFromHex(id)
  102. colls := ctx.Client.GetCollection(collectName)
  103. return colls.DeleteOne(ctx.Ctx, &bson.M{"_id": uid})
  104. }
  105. func RepoDeleteDbDoc(ctx *RepoSession, dbName string, collectName string, id string) (interface{}, error) {
  106. uid, _ := primitive.ObjectIDFromHex(id)
  107. colls := ctx.Client.GetDbCollection(dbName, collectName)
  108. return colls.DeleteOne(ctx.Ctx, &bson.M{"_id": uid})
  109. }
  110. func RepoDeleteDocs(ctx *RepoSession, collectName string, query interface{}) (interface{}, error) {
  111. colls := ctx.Client.GetCollection(collectName)
  112. return colls.DeleteMany(ctx.Ctx, query)
  113. }
  114. func RepoUpdateSetDoc(ctx *RepoSession, collectName string, idstr string, model interface{}) (*mongo.UpdateResult, error) {
  115. colls := ctx.Client.GetCollection(collectName)
  116. update := bson.M{"$set": model}
  117. uid, _ := primitive.ObjectIDFromHex(idstr)
  118. return colls.UpdateByID(ctx.Ctx, uid, update)
  119. }
  120. func RepoUpdateSeDbDoc(ctx *RepoSession, db string, collectName string, idstr string, model interface{}) (*mongo.UpdateResult, error) {
  121. colls := ctx.Client.GetDbCollection(db, collectName)
  122. update := bson.M{"$set": model}
  123. uid, _ := primitive.ObjectIDFromHex(idstr)
  124. return colls.UpdateByID(ctx.Ctx, uid, update)
  125. }
  126. func RepoUpdateSetDocProps(ctx *RepoSession, collectName string, idstr string, update interface{}) (*mongo.UpdateResult, error) {
  127. colls := ctx.Client.GetCollection(collectName)
  128. // update := bson.M{"$set": model}
  129. uid, _ := primitive.ObjectIDFromHex(idstr)
  130. return colls.UpdateByID(ctx.Ctx, uid, update)
  131. }
  132. func RepoUpdateSetDbDocProps(ctx *RepoSession, db string, collectName string, idstr string, update interface{}) (*mongo.UpdateResult, error) {
  133. colls := ctx.Client.GetDbCollection(db, collectName)
  134. // update := bson.M{"$set": model}
  135. uid, _ := primitive.ObjectIDFromHex(idstr)
  136. return colls.UpdateByID(ctx.Ctx, uid, update)
  137. }
  138. func RepoSeachDoc2(ctx *RepoSession, param *DocSearchOptions, v interface{}) error {
  139. colls := ctx.Client.GetDbCollection(param.Db, param.CollectName)
  140. opt := &options.FindOneOptions{}
  141. if len(param.Project) > 0 {
  142. prj := bson.M{}
  143. for _, v := range param.Project {
  144. prj[v] = 1
  145. }
  146. opt.SetProjection(prj)
  147. }
  148. filter := bson.M{}
  149. if len(param.Query) > 0 {
  150. for k, v := range param.Query {
  151. if k == "_id" {
  152. if uid, ok := v.(string); ok {
  153. docId, _ := primitive.ObjectIDFromHex(uid)
  154. filter["_id"] = docId
  155. continue
  156. }
  157. }
  158. filter[k] = v
  159. }
  160. }
  161. err := colls.FindOne(ctx.Ctx, filter, opt).Decode(v)
  162. if err == mongo.ErrNoDocuments {
  163. return fmt.Errorf("数据已被删除!")
  164. }
  165. if err != nil {
  166. return err
  167. }
  168. return nil
  169. }
  170. func RepoSeachDoc(ctx *RepoSession, param *DocSearchOptions, v interface{}) (bool, error) {
  171. colls := ctx.Client.GetDbCollection(param.Db, param.CollectName)
  172. opt := &options.FindOneOptions{}
  173. if len(param.Project) > 0 {
  174. prj := bson.M{}
  175. for _, v := range param.Project {
  176. prj[v] = 1
  177. }
  178. opt.SetProjection(prj)
  179. }
  180. filter := bson.M{}
  181. if len(param.Query) > 0 {
  182. for k, v := range param.Query {
  183. if k == "_id" {
  184. if uid, ok := v.(string); ok {
  185. docId, _ := primitive.ObjectIDFromHex(uid)
  186. filter["_id"] = docId
  187. continue
  188. }
  189. }
  190. filter[k] = v
  191. }
  192. }
  193. err := colls.FindOne(ctx.Ctx, filter, opt).Decode(v)
  194. if err == mongo.ErrNoDocuments {
  195. return false, nil
  196. }
  197. if err != nil {
  198. return false, err
  199. }
  200. return true, nil
  201. }
  202. func RepoSeachDocMap(ctx *RepoSession, param *DocSearchOptions) (bool, map[string]interface{}) {
  203. ret := map[string]interface{}{}
  204. ok := true
  205. colls := ctx.Client.GetDbCollection(param.Db, param.CollectName)
  206. opt := &options.FindOneOptions{}
  207. if len(param.Project) > 0 {
  208. prj := bson.M{}
  209. for _, v := range param.Project {
  210. prj[v] = 1
  211. }
  212. opt.SetProjection(prj)
  213. }
  214. filter := bson.M{}
  215. if len(param.Query) > 0 {
  216. for k, v := range param.Query {
  217. if k == "_id" {
  218. if uid, ok := v.(string); ok {
  219. docId, _ := primitive.ObjectIDFromHex(uid)
  220. filter["_id"] = docId
  221. continue
  222. }
  223. }
  224. filter[k] = v
  225. }
  226. }
  227. ok = true
  228. err := colls.FindOne(ctx.Ctx, filter, opt).Decode(ret)
  229. if err == mongo.ErrNoDocuments {
  230. ok = false
  231. }
  232. if err != nil {
  233. ok = false
  234. }
  235. return ok, ret
  236. }
  237. // PageSearch 单表分页查询
  238. func RepoPageSearch(ctx *RepoSession, para *PageSearchOptions) (out *PageResult, err error) {
  239. var colls *mongo.Collection
  240. if len(para.Db) > 0 {
  241. colls = ctx.Client.GetDbCollection(para.Db, para.CollectName)
  242. } else {
  243. colls = ctx.Client.GetCollection(para.CollectName)
  244. }
  245. findoptions := &options.FindOptions{}
  246. if para.Size > 0 {
  247. findoptions.SetLimit(para.Size)
  248. findoptions.SetSkip(para.Size * (para.Page - 1))
  249. }
  250. if para.Sort != nil {
  251. findoptions.SetSort(para.Sort)
  252. }
  253. if len(para.Project) > 0 {
  254. prj := bson.M{}
  255. for _, v := range para.Project {
  256. prj[v] = 1
  257. }
  258. findoptions.SetProjection(prj)
  259. }
  260. filter := bson.M{}
  261. if len(para.Query) > 0 {
  262. for k, v := range para.Query {
  263. if value, ok := v.(string); ok {
  264. if len(value) > 0 {
  265. filter[k] = v
  266. continue
  267. }
  268. } else if v != nil {
  269. filter[k] = v
  270. }
  271. }
  272. }
  273. cur, err := colls.Find(ctx.Ctx, filter, findoptions)
  274. out = &PageResult{
  275. List: []map[string]interface{}{},
  276. Total: 0,
  277. Page: para.Page,
  278. Size: para.Size,
  279. }
  280. if err != nil {
  281. return out, err
  282. }
  283. defer cur.Close(ctx.Ctx)
  284. err = cur.All(ctx.Ctx, &out.List)
  285. out.Total, _ = colls.CountDocuments(ctx.Ctx, filter)
  286. return
  287. }
  288. // PageSearch 单表分页查询
  289. func RepoCountDoc(ctx *RepoSession, collectionName string, Query Map) (int64, error) {
  290. colls := ctx.Client.GetCollection(collectionName)
  291. filter := bson.M{}
  292. if len(Query) > 0 {
  293. for k, v := range Query {
  294. if value, ok := v.(string); ok {
  295. if len(value) > 0 {
  296. filter[k] = v
  297. continue
  298. }
  299. } else {
  300. filter[k] = v
  301. }
  302. }
  303. }
  304. return colls.CountDocuments(ctx.Ctx, filter)
  305. }
  306. // PageSearch 单表分页查询
  307. func RepoDocsSearch(ctx *RepoSession, para *PageSearchOptions, out interface{}) (err error) {
  308. colls := ctx.Client.GetCollection(para.CollectName)
  309. findoptions := &options.FindOptions{}
  310. if para.Size > 0 {
  311. findoptions.SetLimit(para.Size)
  312. findoptions.SetSkip(para.Size * (para.Page - 1))
  313. }
  314. if para.Sort != nil {
  315. findoptions.SetSort(para.Sort)
  316. }
  317. if len(para.Project) > 0 {
  318. prj := bson.M{}
  319. for _, v := range para.Project {
  320. prj[v] = 1
  321. }
  322. findoptions.SetProjection(prj)
  323. }
  324. filter := bson.M{}
  325. if len(para.Query) > 0 {
  326. for k, v := range para.Query {
  327. if value, ok := v.(string); ok {
  328. if len(value) > 0 {
  329. filter[k] = v
  330. continue
  331. }
  332. } else {
  333. filter[k] = v
  334. }
  335. }
  336. }
  337. cur, err := colls.Find(ctx.Ctx, filter, findoptions)
  338. if err != nil {
  339. return err
  340. }
  341. defer cur.Close(ctx.Ctx)
  342. err = cur.All(ctx.Ctx, out)
  343. return
  344. }
  345. func RepoDocArrayAppend(ctx *RepoSession, collectName string, idstr string, fieldpath string, arrayItem interface{}) (*mongo.UpdateResult, error) {
  346. colls := ctx.Client.GetCollection(collectName)
  347. arrayOp := bson.M{}
  348. arrayOp[fieldpath] = arrayItem
  349. update := bson.M{"$push": arrayOp}
  350. uid, _ := primitive.ObjectIDFromHex(idstr)
  351. return colls.UpdateByID(ctx.Ctx, uid, update)
  352. }
  353. // { _id: 4, "grades.grade": 85 },
  354. // { $set: { "grades.$.std" : 6 } }
  355. type ArrayOneUpdateOption struct {
  356. Query Map
  357. Set Map
  358. CollectName string
  359. Id string
  360. }
  361. // if len(scene.Stickers) > 0 {
  362. // optSet["scenes.$.stickers"] = scene.Stickers
  363. // }
  364. // option := &repo.ArrayOneUpdateOption{
  365. // CollectName: repo.CollectionDesigns,
  366. // Id: id,
  367. // Query: repo.Map{"scenes.id": scene.Id},
  368. // Set: optSet,
  369. // }
  370. func RepoDocArrayOneUpdate(ctx *RepoSession, options *ArrayOneUpdateOption) (*mongo.UpdateResult, error) {
  371. colls := ctx.Client.GetCollection(options.CollectName)
  372. docId, _ := primitive.ObjectIDFromHex(options.Id)
  373. query := bson.M{"_id": docId}
  374. if len(options.Query) > 0 {
  375. for k, v := range options.Query {
  376. query[k] = v
  377. }
  378. }
  379. setOp := bson.M{}
  380. for k, v := range options.Set {
  381. setOp[k] = v
  382. }
  383. update := bson.M{"$set": setOp}
  384. return colls.UpdateOne(ctx.Ctx, query, update)
  385. }
  386. type ArrayOneRemoveOption struct {
  387. ArrayQuery Map
  388. CollectName string
  389. Id string
  390. }
  391. // { $pull: { "items" : { id: 23 } } }
  392. func RepoDocArrayOneRemove(ctx *RepoSession, options *ArrayOneRemoveOption) (*mongo.UpdateResult, error) {
  393. colls := ctx.Client.GetCollection(options.CollectName)
  394. docId, _ := primitive.ObjectIDFromHex(options.Id)
  395. query := bson.M{"_id": docId}
  396. arrayQuery := bson.M{}
  397. if len(options.ArrayQuery) > 0 {
  398. for k, v := range options.ArrayQuery {
  399. arrayQuery[k] = v
  400. }
  401. }
  402. update := bson.M{"$pull": arrayQuery}
  403. return colls.UpdateOne(ctx.Ctx, query, update)
  404. }
  405. type ArrayOneSearchOption struct {
  406. ArrayQuery Map
  407. CollectName string
  408. Id string
  409. Field string
  410. }
  411. func RepoDocArraySearch(ctx *RepoSession, options *ArrayOneSearchOption) (ok bool, ret map[string]interface{}) {
  412. colls := ctx.Client.GetCollection(options.CollectName)
  413. docId, _ := primitive.ObjectIDFromHex(options.Id)
  414. match := []bson.E{}
  415. match = append(match, bson.E{"_id", docId})
  416. if len(options.ArrayQuery) > 0 {
  417. for k, v := range options.ArrayQuery {
  418. match = append(match, bson.E{k, v})
  419. }
  420. }
  421. matchStage := bson.D{
  422. {"$match", match},
  423. }
  424. unwindStage := bson.D{
  425. {"$unwind", fmt.Sprintf("%s%s", "$", options.Field)},
  426. }
  427. curr, err := colls.Aggregate(ctx.Ctx, mongo.Pipeline{matchStage, unwindStage})
  428. if err != nil {
  429. ok = false
  430. return
  431. }
  432. defer curr.Close(ctx.Ctx)
  433. var list []map[string]interface{}
  434. err = curr.All(ctx.Ctx, list)
  435. if err != nil {
  436. ok = false
  437. return
  438. }
  439. ok = true
  440. ret = list[0]
  441. return
  442. }
  443. type DocsSearchOptions struct {
  444. CollectName string
  445. Query map[string]interface{}
  446. Project []string
  447. Sort interface{} //bson.D{ bson.E{"update_time", -1}, bson.E{"goods_id", -1},}
  448. }
  449. func RepoSeachDocsMap(ctx *RepoSession, param *DocsSearchOptions) (ok bool, list []map[string]interface{}) {
  450. colls := ctx.Client.GetCollection(param.CollectName)
  451. findoptions := &options.FindOptions{}
  452. if len(param.Project) > 0 {
  453. prj := bson.M{}
  454. for _, v := range param.Project {
  455. prj[v] = 1
  456. }
  457. findoptions.SetProjection(prj)
  458. }
  459. if param.Sort != nil {
  460. findoptions.SetSort(param.Sort)
  461. }
  462. filter := bson.M{}
  463. if len(param.Query) > 0 {
  464. for k, v := range param.Query {
  465. if value, ok := v.(string); ok {
  466. if len(value) > 0 {
  467. filter[k] = v
  468. continue
  469. }
  470. } else {
  471. filter[k] = v
  472. }
  473. }
  474. }
  475. cur, err := colls.Find(ctx.Ctx, filter, findoptions)
  476. if err != nil {
  477. ok = false
  478. return
  479. }
  480. defer cur.Close(ctx.Ctx)
  481. listRes := []map[string]interface{}{}
  482. err = cur.All(ctx.Ctx, &listRes)
  483. if err != nil {
  484. log.Error(err)
  485. ok = false
  486. return
  487. }
  488. list = listRes
  489. ok = true
  490. return
  491. }
  492. func RepoSeachDocs(ctx *RepoSession, param *DocsSearchOptions, list interface{}) error {
  493. colls := ctx.Client.GetCollection(param.CollectName)
  494. findoptions := &options.FindOptions{}
  495. if len(param.Project) > 0 {
  496. prj := bson.M{}
  497. for _, v := range param.Project {
  498. prj[v] = 1
  499. }
  500. findoptions.SetProjection(prj)
  501. }
  502. if param.Sort != nil {
  503. findoptions.SetSort(param.Sort)
  504. }
  505. filter := bson.M{}
  506. if len(param.Query) > 0 {
  507. for k, v := range param.Query {
  508. if value, ok := v.(string); ok {
  509. if len(value) > 0 {
  510. filter[k] = v
  511. continue
  512. }
  513. } else {
  514. filter[k] = v
  515. }
  516. }
  517. }
  518. cur, err := colls.Find(ctx.Ctx, filter, findoptions)
  519. if err != nil {
  520. return err
  521. }
  522. defer cur.Close(ctx.Ctx)
  523. err = cur.All(ctx.Ctx, list)
  524. if err != nil {
  525. return err
  526. }
  527. return nil
  528. }