models.js 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161
  1. const ObjectID = require("mongodb").ObjectID;
  2. const {connect} = require('mm')
  3. module.exports = async (dbName='shop') => {
  4. const {Savable, slice} = await connect(dbName)
  5. async function getModels({id}){
  6. const SlicedSavable = slice([id, 'user'])
  7. class User extends SlicedSavable {
  8. constructor(...params){
  9. super(...params)
  10. //TODO: calc likes count by getter (no two-way relation for this to avoid overflow on many Kilos of likes
  11. //cached like count, which incremented and decremented
  12. //
  13. //following and followers array
  14. }
  15. static get relations(){ //don't needed due to ___owner in most cases
  16. return {
  17. avatar : "userAvatar",
  18. }
  19. }
  20. }
  21. SlicedSavable.addClass(User)
  22. class OwnerSlicedSavable extends SlicedSavable {
  23. get owner(){
  24. if (!this.___owner) return this.___owner
  25. return SlicedSavable.m.User.findOne({_id: ObjectID(this.___owner)})
  26. }
  27. }
  28. class Image extends OwnerSlicedSavable {
  29. constructor(...params){
  30. super(...params)
  31. }
  32. static async fromFileData(fileData){
  33. let image = new Image({})
  34. image.fileData = fileData
  35. image.url = `images/${fileData.filename}`
  36. image.originalFileName = fileData.originalname
  37. await image.save()
  38. return image;
  39. }
  40. async save(...params){
  41. if (this.userAvatar){
  42. if (this.userAvatar._id.toString() !== id){
  43. throw new ReferenceError(`You can't set ava for other user`)
  44. }
  45. }
  46. return await super.save(...params)
  47. }
  48. static get relations(){
  49. return {
  50. userAvatar: "avatar", //if it is ava
  51. good: ["images"], //if it is ava
  52. category: "image", //if it is ava
  53. }
  54. }
  55. }
  56. SlicedSavable.addClass(Image)
  57. class Good extends SlicedSavable {
  58. constructor(...params){
  59. super(...params)
  60. }
  61. static get relations(){
  62. return {
  63. categories: ["goods"],
  64. orderGoods: "good",
  65. images: "good"
  66. }
  67. }
  68. }
  69. SlicedSavable.addClass(Good)
  70. class Category extends SlicedSavable {
  71. constructor(...params){
  72. super(...params)
  73. }
  74. static get relations(){
  75. return {
  76. goods: ["categories"],
  77. image: "category",
  78. }
  79. }
  80. }
  81. SlicedSavable.addClass(Category)
  82. class Order extends SlicedSavable {
  83. constructor(...params){
  84. super(...params)
  85. }
  86. get total(){
  87. return (async() => (await Promise.all(this.orderGoods)).reduce((a,b) => (a.total || a) + b.total, 0))()
  88. }
  89. static get relations(){
  90. return {
  91. user: ["orders"],
  92. orderGoods: "order"
  93. }
  94. }
  95. }
  96. SlicedSavable.addClass(Order)
  97. class OrderGood extends SlicedSavable {
  98. constructor(...params){
  99. super(...params)
  100. }
  101. get total(){
  102. return this.price*this.count
  103. }
  104. static get relations(){
  105. return {
  106. good: ["orderGoods"],
  107. order: ["orderGoods"]
  108. }
  109. }
  110. }
  111. SlicedSavable.addClass(OrderGood)
  112. const thisUser = await Savable.m.User.findOne({_id: ObjectID(id)})
  113. return {models: {
  114. SlicedSavable, ...SlicedSavable.classes
  115. },
  116. thisUser}
  117. }
  118. return {
  119. Savable,
  120. slice,
  121. getModels
  122. }
  123. }