@nlabs/reaktor 0.9.0 → 0.10.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (400) hide show
  1. package/.env +1 -0
  2. package/.env.example +1 -0
  3. package/DATABASE_I18N_GUIDE.md +434 -0
  4. package/README.md +9 -0
  5. package/TEST_UTILITIES_GUIDE.md +360 -0
  6. package/coverage/index.html +61 -46
  7. package/dist/actions/apps.js +242 -0
  8. package/dist/actions/connections.js +90 -0
  9. package/dist/actions/conversations.js +350 -0
  10. package/dist/actions/dynamodb.js +150 -0
  11. package/dist/actions/email.js +152 -0
  12. package/dist/actions/files.js +283 -0
  13. package/dist/actions/groups.js +292 -0
  14. package/dist/actions/images.js +735 -0
  15. package/dist/actions/index.js +66 -0
  16. package/dist/actions/ios.js +164 -0
  17. package/dist/actions/locations.js +122 -0
  18. package/dist/actions/messages.js +208 -0
  19. package/dist/actions/notifications.js +59 -0
  20. package/dist/actions/payments.js +497 -0
  21. package/dist/actions/personas.js +110 -0
  22. package/dist/actions/posts.js +595 -0
  23. package/dist/actions/reactions.js +322 -0
  24. package/dist/actions/s3.js +133 -0
  25. package/dist/actions/search.js +90 -0
  26. package/dist/actions/sms.js +108 -0
  27. package/dist/actions/statistics.js +62 -0
  28. package/dist/actions/subscription.js +220 -0
  29. package/dist/actions/tags.js +292 -0
  30. package/dist/actions/users.js +784 -0
  31. package/dist/actions/websockets.js +174 -0
  32. package/dist/adapters/arangoAdapter.js +46 -0
  33. package/dist/adapters/fileAdapter.js +76 -0
  34. package/dist/adapters/imageAdapter.js +40 -0
  35. package/dist/adapters/messageAdapter.js +49 -0
  36. package/dist/adapters/postAdapter.js +70 -0
  37. package/dist/adapters/reaktorAdapter.js +44 -0
  38. package/dist/adapters/tagAdapter.js +50 -0
  39. package/dist/adapters/userAdapter.js +115 -0
  40. package/dist/config.js +125 -0
  41. package/dist/index.js +66 -0
  42. package/dist/lambdas/actions/websockets.js +132 -0
  43. package/dist/lambdas/authorizer.js +67 -0
  44. package/dist/lambdas/connection.js +91 -0
  45. package/dist/lambdas/utils/message.js +42 -0
  46. package/dist/lambdas/utils/websocket.js +105 -0
  47. package/dist/mocks/conversation.js +35 -0
  48. package/dist/mocks/file.js +38 -0
  49. package/dist/mocks/group.js +47 -0
  50. package/dist/mocks/image.js +44 -0
  51. package/dist/mocks/nlabs.png +0 -0
  52. package/dist/mocks/post.js +55 -0
  53. package/dist/mocks/tag.js +37 -0
  54. package/dist/mocks/user.js +88 -0
  55. package/dist/mutations/index.js +26 -0
  56. package/dist/mutations/locations.js +44 -0
  57. package/dist/mutations/messages.js +86 -0
  58. package/dist/mutations/personas.js +100 -0
  59. package/dist/mutations/posts.js +53 -0
  60. package/dist/mutations/reactions.js +51 -0
  61. package/dist/mutations/statistics.js +39 -0
  62. package/dist/mutations/subscriptions.js +56 -0
  63. package/dist/mutations/tags.js +120 -0
  64. package/dist/mutations/users.js +116 -0
  65. package/dist/objectTypes/app.js +173 -0
  66. package/dist/objectTypes/bankAccount.js +76 -0
  67. package/dist/objectTypes/connection.js +48 -0
  68. package/dist/objectTypes/conversation.js +77 -0
  69. package/dist/objectTypes/creditCard.js +86 -0
  70. package/dist/objectTypes/document.js +46 -0
  71. package/dist/objectTypes/error.js +46 -0
  72. package/dist/objectTypes/external.js +74 -0
  73. package/dist/objectTypes/file.js +100 -0
  74. package/dist/objectTypes/filter.js +43 -0
  75. package/dist/objectTypes/group.js +123 -0
  76. package/dist/objectTypes/iapSubscription.js +40 -0
  77. package/dist/objectTypes/image.js +129 -0
  78. package/dist/objectTypes/index.js +68 -0
  79. package/dist/objectTypes/location.js +109 -0
  80. package/dist/objectTypes/message.js +96 -0
  81. package/dist/objectTypes/passcode.js +42 -0
  82. package/dist/objectTypes/persona.js +87 -0
  83. package/dist/objectTypes/plan.js +95 -0
  84. package/dist/objectTypes/post.js +125 -0
  85. package/dist/objectTypes/reaction.js +61 -0
  86. package/dist/objectTypes/relation.js +49 -0
  87. package/dist/objectTypes/search.js +72 -0
  88. package/dist/objectTypes/statistics.js +39 -0
  89. package/dist/objectTypes/subscription.js +117 -0
  90. package/dist/objectTypes/tag.js +65 -0
  91. package/dist/objectTypes/user.js +144 -0
  92. package/dist/queries/index.js +33 -0
  93. package/dist/queries/locations.js +45 -0
  94. package/dist/queries/messages.js +52 -0
  95. package/dist/queries/posts.js +154 -0
  96. package/dist/queries/reactions.js +56 -0
  97. package/dist/queries/statistics.js +39 -0
  98. package/dist/queries/subscriptions.js +44 -0
  99. package/dist/queries/tags.js +75 -0
  100. package/dist/queries/users.js +64 -0
  101. package/dist/templates/email/layout.js +302 -0
  102. package/dist/templates/email/passwordForgot.js +38 -0
  103. package/dist/templates/email/passwordRecovery.js +35 -0
  104. package/dist/templates/email/verifyEmail.js +38 -0
  105. package/dist/templates/email/welcome.js +38 -0
  106. package/dist/templates/sms/passwordForgot.js +24 -0
  107. package/dist/templates/sms/passwordRecovery.js +24 -0
  108. package/dist/templates/sms/verifyEmail.js +24 -0
  109. package/dist/templates/sms/verifyPhone.js +24 -0
  110. package/dist/templates/sms/welcome.js +24 -0
  111. package/dist/types/apps.js +32 -0
  112. package/{lib → dist}/types/arangodb.js +1 -1
  113. package/{lib → dist}/types/auth.js +1 -1
  114. package/{lib → dist}/types/connections.js +1 -1
  115. package/dist/types/conversations.js +16 -0
  116. package/{lib → dist}/types/email.js +1 -1
  117. package/dist/types/files.js +16 -0
  118. package/dist/types/google.js +16 -0
  119. package/{lib → dist}/types/groups.js +1 -1
  120. package/dist/types/images.js +16 -0
  121. package/dist/types/index.js +60 -0
  122. package/{lib → dist}/types/locations.js +1 -1
  123. package/{lib → dist}/types/messages.js +1 -1
  124. package/{lib → dist}/types/notifications.js +1 -1
  125. package/dist/types/payments.js +16 -0
  126. package/dist/types/personas.js +16 -0
  127. package/dist/types/posts.js +16 -0
  128. package/{lib → dist}/types/tags.js +1 -1
  129. package/dist/types/users.js +16 -0
  130. package/dist/types/websockets.js +16 -0
  131. package/dist/utils/adapterUtils.js +45 -0
  132. package/dist/utils/analyticsUtils.js +72 -0
  133. package/dist/utils/arangodbUtils.js +165 -0
  134. package/dist/utils/auth.js +57 -0
  135. package/dist/utils/index.js +30 -0
  136. package/{lib → dist}/utils/session.js +10 -7
  137. package/index.js +1 -1
  138. package/jest.config.js +17 -0
  139. package/jest.setup.js +36 -0
  140. package/lex.config.cjs +13 -0
  141. package/lib/actions/apps.js +17 -249
  142. package/lib/actions/connections.js +7 -0
  143. package/lib/actions/content.js +17 -0
  144. package/lib/actions/conversations.js +19 -325
  145. package/lib/actions/dynamodb.js +2 -150
  146. package/lib/actions/email.js +2 -152
  147. package/lib/actions/files.js +5 -287
  148. package/lib/actions/groups.js +23 -263
  149. package/lib/actions/images.js +31 -646
  150. package/lib/actions/index.js +2 -62
  151. package/lib/actions/ios.js +9 -162
  152. package/lib/actions/locations.js +7 -110
  153. package/lib/actions/messages.js +21 -193
  154. package/lib/actions/notifications.js +2 -59
  155. package/lib/actions/payments.js +11 -461
  156. package/lib/actions/posts.js +77 -515
  157. package/lib/actions/profiles.js +8 -0
  158. package/lib/actions/reactions.js +25 -312
  159. package/lib/actions/s3.js +2 -133
  160. package/lib/actions/search.js +5 -90
  161. package/lib/actions/sms.js +2 -106
  162. package/lib/actions/statistics.js +6 -60
  163. package/lib/actions/subscriptions.js +12 -0
  164. package/lib/actions/tags.js +19 -262
  165. package/lib/actions/users.js +67 -537
  166. package/lib/actions/websockets.js +16 -158
  167. package/lib/adapters/arangoAdapter.js +2 -46
  168. package/lib/adapters/contentAdapter.js +2 -0
  169. package/lib/adapters/fileAdapter.js +2 -76
  170. package/lib/adapters/imageAdapter.js +2 -0
  171. package/lib/adapters/messageAdapter.js +2 -0
  172. package/lib/adapters/postAdapter.js +2 -70
  173. package/lib/adapters/reaktorAdapter.js +2 -44
  174. package/lib/adapters/tagAdapter.js +2 -50
  175. package/lib/adapters/userAdapter.js +2 -110
  176. package/lib/config.js +2 -125
  177. package/lib/handlers/graphqlHandler.js +2 -0
  178. package/lib/index.js +2 -30
  179. package/lib/lambdas/actions/websockets.js +14 -112
  180. package/lib/lambdas/authorizer.js +2 -67
  181. package/lib/lambdas/connection.js +2 -92
  182. package/lib/lambdas/utils/message.js +2 -42
  183. package/lib/lambdas/utils/websocket.js +2 -105
  184. package/lib/mocks/conversation.js +2 -35
  185. package/lib/mocks/file.js +2 -38
  186. package/lib/mocks/group.js +2 -47
  187. package/lib/mocks/image.js +2 -43
  188. package/lib/mocks/post.js +2 -55
  189. package/lib/mocks/tag.js +2 -37
  190. package/lib/mocks/user.js +2 -88
  191. package/lib/mutations/index.js +2 -0
  192. package/lib/mutations/locations.integration.js +2 -0
  193. package/lib/mutations/locations.js +2 -0
  194. package/lib/mutations/messages.integration.js +2 -0
  195. package/lib/mutations/messages.js +2 -0
  196. package/lib/mutations/posts.integration.js +2 -0
  197. package/lib/mutations/posts.js +2 -0
  198. package/lib/mutations/profiles.integration.js +2 -0
  199. package/lib/mutations/profiles.js +2 -0
  200. package/lib/mutations/reactions.integration.js +2 -0
  201. package/lib/mutations/reactions.js +2 -0
  202. package/lib/mutations/statistics.integration.js +2 -0
  203. package/lib/mutations/statistics.js +2 -0
  204. package/lib/mutations/subscriptions.integration.js +2 -0
  205. package/lib/mutations/subscriptions.js +2 -0
  206. package/lib/mutations/tags.integration.js +2 -0
  207. package/lib/mutations/tags.js +2 -0
  208. package/lib/mutations/users.integration.js +2 -0
  209. package/lib/mutations/users.js +2 -0
  210. package/lib/objectTypes/app.js +2 -0
  211. package/lib/objectTypes/bankAccount.js +2 -0
  212. package/lib/objectTypes/connection.js +2 -0
  213. package/lib/objectTypes/conversation.js +2 -0
  214. package/lib/objectTypes/creditCard.js +2 -0
  215. package/lib/objectTypes/document.js +2 -0
  216. package/lib/objectTypes/error.js +2 -0
  217. package/lib/objectTypes/external.js +2 -0
  218. package/lib/objectTypes/file.js +2 -0
  219. package/lib/objectTypes/filter.js +2 -0
  220. package/lib/objectTypes/group.js +2 -0
  221. package/lib/objectTypes/iapSubscription.js +2 -0
  222. package/lib/objectTypes/image.js +2 -0
  223. package/lib/objectTypes/index.js +2 -0
  224. package/lib/objectTypes/location.js +2 -0
  225. package/lib/objectTypes/message.js +2 -0
  226. package/lib/objectTypes/passcode.js +2 -0
  227. package/lib/objectTypes/plan.js +2 -0
  228. package/lib/objectTypes/post.js +2 -0
  229. package/lib/objectTypes/profile.js +2 -0
  230. package/lib/objectTypes/reaction.js +2 -0
  231. package/lib/objectTypes/relation.js +2 -0
  232. package/lib/objectTypes/search.js +2 -0
  233. package/lib/objectTypes/statistics.js +2 -0
  234. package/lib/objectTypes/subscription.js +2 -0
  235. package/lib/objectTypes/tag.js +2 -0
  236. package/lib/objectTypes/user.js +2 -0
  237. package/lib/queries/index.js +2 -0
  238. package/lib/queries/locations.integration.js +2 -0
  239. package/lib/queries/locations.js +2 -0
  240. package/lib/queries/messages.integration.js +2 -0
  241. package/lib/queries/messages.js +2 -0
  242. package/lib/queries/posts.integration.js +2 -0
  243. package/lib/queries/posts.js +2 -0
  244. package/lib/queries/reactions.integration.js +2 -0
  245. package/lib/queries/reactions.js +2 -0
  246. package/lib/queries/statistics.js +2 -0
  247. package/lib/queries/subscriptions.js +2 -0
  248. package/lib/queries/tags.integration.js +2 -0
  249. package/lib/queries/tags.js +2 -0
  250. package/lib/queries/users.integration.js +2 -0
  251. package/lib/queries/users.js +2 -0
  252. package/lib/templates/email/layout.js +3 -25
  253. package/lib/templates/email/passwordForgot.js +3 -25
  254. package/lib/templates/email/passwordRecovery.js +3 -25
  255. package/lib/templates/email/verifyEmail.js +3 -25
  256. package/lib/templates/email/welcome.js +3 -25
  257. package/lib/templates/sms/passwordForgot.js +2 -24
  258. package/lib/templates/sms/passwordRecovery.js +2 -24
  259. package/lib/templates/sms/verifyEmail.js +2 -24
  260. package/lib/templates/sms/verifyPhone.js +2 -24
  261. package/lib/templates/sms/welcome.js +2 -24
  262. package/lib/types/apps.types.js +2 -0
  263. package/lib/types/arangodb.types.js +1 -0
  264. package/lib/types/auth.types.js +1 -0
  265. package/lib/types/connections.types.js +1 -0
  266. package/lib/types/content.types.js +1 -0
  267. package/lib/types/conversations.types.js +1 -0
  268. package/lib/types/email.types.js +1 -0
  269. package/lib/types/error.types.js +2 -0
  270. package/lib/types/files.types.js +1 -0
  271. package/lib/types/google.types.js +1 -0
  272. package/lib/types/groups.types.js +1 -0
  273. package/lib/types/images.types.js +1 -0
  274. package/lib/types/index.js +2 -56
  275. package/lib/types/locations.types.js +1 -0
  276. package/lib/types/messages.types.js +1 -0
  277. package/lib/types/notifications.types.js +1 -0
  278. package/lib/types/payments.types.js +1 -0
  279. package/lib/types/posts.types.js +1 -0
  280. package/lib/types/profiles.types.js +1 -0
  281. package/lib/types/statistics.types.js +1 -0
  282. package/lib/types/tags.types.js +1 -0
  283. package/lib/types/users.types.js +1 -0
  284. package/lib/types/websockets.types.js +1 -0
  285. package/lib/utils/adapterUtils.js +2 -45
  286. package/lib/utils/analyticsUtils.js +2 -72
  287. package/lib/utils/arangodbUtils.js +5 -142
  288. package/lib/utils/authUtils.js +2 -0
  289. package/lib/utils/contextUtils.js +2 -0
  290. package/lib/utils/dbI18n.example.js +6 -0
  291. package/lib/utils/dbI18n.js +2 -0
  292. package/lib/utils/googleTranslate.js +2 -0
  293. package/lib/utils/graphqlUtils.js +2 -0
  294. package/lib/utils/index.js +2 -30
  295. package/lib/utils/languageDetection.js +2 -0
  296. package/lib/utils/localeUtils.example.js +2 -0
  297. package/lib/utils/localeUtils.js +2 -0
  298. package/lib/utils/middlewareUtils.js +2 -0
  299. package/lib/utils/sessionUtils.js +2 -0
  300. package/lib/utils/stripeUtils.js +2 -0
  301. package/lib/utils/templateUtils.js +2 -0
  302. package/lib/utils/testUtils.js +2 -0
  303. package/lib/utils/translationQueue.example.js +2 -0
  304. package/lib/utils/translationQueue.js +2 -0
  305. package/package.json +41 -37
  306. package/.eslintrc +0 -10
  307. package/.prettierrc.js +0 -4
  308. package/lib/actions/apps.d.ts +0 -25
  309. package/lib/actions/conversations.d.ts +0 -14
  310. package/lib/actions/dynamodb.d.ts +0 -8
  311. package/lib/actions/email.d.ts +0 -5
  312. package/lib/actions/files.d.ts +0 -19
  313. package/lib/actions/groups.d.ts +0 -14
  314. package/lib/actions/images.d.ts +0 -26
  315. package/lib/actions/index.d.ts +0 -21
  316. package/lib/actions/ios.d.ts +0 -7
  317. package/lib/actions/locations.d.ts +0 -5
  318. package/lib/actions/messages.d.ts +0 -13
  319. package/lib/actions/notifications.d.ts +0 -5
  320. package/lib/actions/payments.d.ts +0 -10
  321. package/lib/actions/posts.d.ts +0 -19
  322. package/lib/actions/reactions.d.ts +0 -30
  323. package/lib/actions/s3.d.ts +0 -7
  324. package/lib/actions/search.d.ts +0 -3
  325. package/lib/actions/sms.d.ts +0 -3
  326. package/lib/actions/statistics.d.ts +0 -3
  327. package/lib/actions/subscription.d.ts +0 -7
  328. package/lib/actions/subscription.js +0 -218
  329. package/lib/actions/tags.d.ts +0 -29
  330. package/lib/actions/users.d.ts +0 -47
  331. package/lib/actions/websockets.d.ts +0 -19
  332. package/lib/adapters/arangoAdapter.d.ts +0 -2
  333. package/lib/adapters/fileAdapter.d.ts +0 -3
  334. package/lib/adapters/postAdapter.d.ts +0 -2
  335. package/lib/adapters/reaktorAdapter.d.ts +0 -6
  336. package/lib/adapters/tagAdapter.d.ts +0 -2
  337. package/lib/adapters/userAdapter.d.ts +0 -2
  338. package/lib/config.d.ts +0 -20
  339. package/lib/index.d.ts +0 -5
  340. package/lib/lambdas/actions/websockets.d.ts +0 -6
  341. package/lib/lambdas/authorizer.d.ts +0 -20
  342. package/lib/lambdas/connection.d.ts +0 -12
  343. package/lib/lambdas/utils/message.d.ts +0 -1
  344. package/lib/lambdas/utils/websocket.d.ts +0 -7
  345. package/lib/mocks/conversation.d.ts +0 -8
  346. package/lib/mocks/file.d.ts +0 -11
  347. package/lib/mocks/group.d.ts +0 -17
  348. package/lib/mocks/image.d.ts +0 -3
  349. package/lib/mocks/post.d.ts +0 -38
  350. package/lib/mocks/tag.d.ts +0 -2
  351. package/lib/mocks/user.d.ts +0 -4
  352. package/lib/templates/email/layout.d.ts +0 -2
  353. package/lib/templates/email/passwordForgot.d.ts +0 -2
  354. package/lib/templates/email/passwordRecovery.d.ts +0 -2
  355. package/lib/templates/email/verifyEmail.d.ts +0 -2
  356. package/lib/templates/email/welcome.d.ts +0 -2
  357. package/lib/templates/sms/passwordForgot.d.ts +0 -2
  358. package/lib/templates/sms/passwordRecovery.d.ts +0 -2
  359. package/lib/templates/sms/verifyEmail.d.ts +0 -2
  360. package/lib/templates/sms/verifyPhone.d.ts +0 -2
  361. package/lib/templates/sms/welcome.d.ts +0 -2
  362. package/lib/types/apps.d.ts +0 -46
  363. package/lib/types/apps.js +0 -32
  364. package/lib/types/arangodb.d.ts +0 -30
  365. package/lib/types/auth.d.ts +0 -7
  366. package/lib/types/connections.d.ts +0 -7
  367. package/lib/types/conversations.d.ts +0 -29
  368. package/lib/types/conversations.js +0 -16
  369. package/lib/types/email.d.ts +0 -13
  370. package/lib/types/error.d.ts +0 -20
  371. package/lib/types/files.d.ts +0 -26
  372. package/lib/types/files.js +0 -16
  373. package/lib/types/google.d.ts +0 -29
  374. package/lib/types/google.js +0 -16
  375. package/lib/types/groups.d.ts +0 -21
  376. package/lib/types/images.d.ts +0 -51
  377. package/lib/types/images.js +0 -16
  378. package/lib/types/index.d.ts +0 -18
  379. package/lib/types/locations.d.ts +0 -20
  380. package/lib/types/messages.d.ts +0 -16
  381. package/lib/types/notifications.d.ts +0 -17
  382. package/lib/types/payments.d.ts +0 -112
  383. package/lib/types/payments.js +0 -16
  384. package/lib/types/posts.d.ts +0 -31
  385. package/lib/types/posts.js +0 -16
  386. package/lib/types/statistics.d.ts +0 -3
  387. package/lib/types/tags.d.ts +0 -10
  388. package/lib/types/users.d.ts +0 -76
  389. package/lib/types/users.js +0 -16
  390. package/lib/types/websocket.d.ts +0 -13
  391. package/lib/types/websocket.js +0 -16
  392. package/lib/utils/adapterUtils.d.ts +0 -1
  393. package/lib/utils/analyticsUtils.d.ts +0 -21
  394. package/lib/utils/arangodbUtils.d.ts +0 -65
  395. package/lib/utils/auth.d.ts +0 -20
  396. package/lib/utils/auth.js +0 -50
  397. package/lib/utils/index.d.ts +0 -5
  398. package/lib/utils/session.d.ts +0 -16
  399. /package/{lib → dist}/types/error.js +0 -0
  400. /package/{lib → dist}/types/statistics.js +0 -0
@@ -1,346 +1,112 @@
1
- var __defProp = Object.defineProperty;
2
- var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
3
- var __getOwnPropNames = Object.getOwnPropertyNames;
4
- var __hasOwnProp = Object.prototype.hasOwnProperty;
5
- var __export = (target, all) => {
6
- for (var name in all)
7
- __defProp(target, name, { get: all[name], enumerable: true });
8
- };
9
- var __copyProps = (to, from, except, desc) => {
10
- if (from && typeof from === "object" || typeof from === "function") {
11
- for (let key of __getOwnPropNames(from))
12
- if (!__hasOwnProp.call(to, key) && key !== except)
13
- __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
14
- }
15
- return to;
16
- };
17
- var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
18
- var posts_exports = {};
19
- __export(posts_exports, {
20
- addPost: () => addPost,
21
- createPostEdge: () => createPostEdge,
22
- deletePost: () => deletePost,
23
- getPost: () => getPost,
24
- getPostComments: () => getPostComments,
25
- getPostOptional: () => getPostOptional,
26
- getPostsByArea: () => getPostsByArea,
27
- getPostsByLatest: () => getPostsByLatest,
28
- getPostsByReactions: () => getPostsByReactions,
29
- getPostsByTags: () => getPostsByTags,
30
- getPostsByUser: () => getPostsByUser,
31
- parsePostOptions: () => parsePostOptions,
32
- updatePost: () => updatePost
33
- });
34
- module.exports = __toCommonJS(posts_exports);
35
- var import_arangojs = require("arangojs");
36
- var import_utils = require("@nlabs/utils");
37
- var import_postAdapter = require("../adapters/postAdapter");
38
- var import_error = require("../types/error");
39
- var import_analyticsUtils = require("../utils/analyticsUtils");
40
- var import_arangodbUtils = require("../utils/arangodbUtils");
41
- var import_tags = require("./tags");
42
- const MAX_CONTENT_LENGTH = 1e5;
43
- const eventCategory = "posts";
44
- const parsePostOptions = (options = {}) => {
45
- const {
46
- from = 0,
47
- latitude = 0,
48
- longitude = 0,
49
- to = 30,
50
- type = "post"
51
- } = options;
52
- return {
53
- latitude: (0, import_utils.parseNum)(latitude, 32),
54
- limit: (0, import_arangodbUtils.getLimit)(from, to),
55
- longitude: (0, import_utils.parseNum)(longitude, 32),
56
- type: (0, import_utils.parseChar)(type, 32)
57
- };
58
- };
59
- const getPostOptional = (fields, sessionId) => (fields || []).reduce((selects, field) => {
60
- switch (field) {
61
- case "hasRsvp": {
62
- selects.queries.push(`LET hasRsvp = TO_BOOL(FIRST(
1
+ import{createHash as B,parseArangoId as S,parseChar as _,parseId as D,parseNum as L,parseString as q,parseVarChar as k}from"@nlabs/utils";import{aql as A}from"arangojs";import{extractTags as w,getTagsByName as U,updateTagsInItem as j}from"./tags";import{parsePost as H}from"../adapters/postAdapter";import{ErrorTypes as N}from"../types/error.types";import{logError as f,logException as Q}from"../utils/analyticsUtils";import{getLimit as V,useDb as y}from"../utils/arangodbUtils";const W=1e5,O="posts",$=(e={})=>{const{from:r=0,latitude:t=0,longitude:o=0,to:i=30,type:n="post"}=e;return{latitude:L(t,32),limit:V(r,i),longitude:L(o,32),type:_(n,32)}},C=(e,r)=>(e||[]).reduce((t,o)=>{switch(o){case"hasRsvp":return t.queries.push(`LET hasRsvp = TO_BOOL(FIRST(
63
2
  FOR post, r IN INBOUND p._id hasReaction
64
- FILTER r.name == "rsvp" && r.type == "posts" && r._from == "users/${sessionId}"
3
+ FILTER r.name == "rsvp" && r.type == "posts" && r._from == "users/${r}"
65
4
  COLLECT WITH COUNT INTO count
66
5
  RETURN count
67
- ))`);
68
- selects.objects.push("hasRsvp:hasRsvp");
69
- return selects;
70
- }
71
- case "isSaved": {
72
- selects.queries.push(`LET isSaved = TO_BOOL(FIRST(
6
+ ))`),t.objects.push("hasRsvp:hasRsvp"),t;case"isSaved":return t.queries.push(`LET isSaved = TO_BOOL(FIRST(
73
7
  FOR post, r IN INBOUND p._id hasReaction
74
- FILTER r.name == "pin" && r.type == "posts" && r._from == "users/${sessionId}"
8
+ FILTER r.name == "pin" && r.type == "posts" && r._from == "users/${r}"
75
9
  COLLECT WITH COUNT INTO count
76
10
  RETURN count
77
- ))`);
78
- selects.objects.push("isSaved:isSaved");
79
- return selects;
80
- }
81
- case "reactions": {
82
- selects.queries.push(`LET reactions = (
11
+ ))`),t.objects.push("isSaved:isSaved"),t;case"reactions":return t.queries.push(`LET reactions = (
83
12
  FOR post, r IN INBOUND p._id hasReaction
84
13
  COLLECT reactionName = r.value INTO reactionItems
85
14
  RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
86
- )`);
87
- selects.objects.push("reactions:reactions");
88
- return selects;
89
- }
90
- case "rsvpCount": {
91
- selects.queries.push(`LET rsvpCount = FIRST(
15
+ )`),t.objects.push("reactions:reactions"),t;case"rsvpCount":return t.queries.push(`LET rsvpCount = FIRST(
92
16
  FOR post, r IN INBOUND p._id hasReaction
93
17
  FILTER r.name == "rsvp" && r.type == "posts"
94
18
  COLLECT WITH COUNT INTO count
95
19
  RETURN count
96
- )`);
97
- selects.objects.push("rsvpCount:rsvpCount");
98
- return selects;
99
- }
100
- case "viewCount": {
101
- selects.queries.push(`LET viewCount = FIRST(
20
+ )`),t.objects.push("rsvpCount:rsvpCount"),t;case"viewCount":return t.queries.push(`LET viewCount = FIRST(
102
21
  FOR post, r IN INBOUND p._id hasReaction
103
22
  FILTER r.name == "view" && r.type == "posts"
104
23
  COLLECT WITH COUNT INTO count
105
24
  RETURN count
106
- )`);
107
- selects.objects.push("viewCount:viewCount");
108
- return selects;
109
- }
110
- default: {
111
- return selects;
112
- }
113
- }
114
- }, { objects: [], queries: [] });
115
- const getPost = async (context, postId, options) => {
116
- const action = "getPost";
117
- const { database, fields, session: { userId: sessionId } } = context;
118
- const formatItemId = (0, import_utils.parseId)(postId);
119
- const { type } = parsePostOptions(options);
120
- const db = database;
121
- const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
122
- const aqlQry = import_arangojs.aql`FOR p IN posts
123
- FILTER p._key == ${formatItemId} && p.type == ${type}
25
+ )`),t.objects.push("viewCount:viewCount"),t;default:return t}},{objects:[],queries:[]}),et=async(e,r,t)=>{const o="getPost",{databaseName:i,fields:n,session:{userId:p}}=e,R=D(r),{type:c}=$(t),s=y(i),{objects:a,queries:u}=C(n,p),d=A`FOR p IN posts
26
+ FILTER p._key == ${R} && p.type == ${c}
124
27
  LIMIT 1
125
- RETURN p`;
126
- return db.query(aqlQry).then((cursor) => cursor.next()).then((post) => {
127
- const {
128
- _id: postDocId,
129
- userId,
130
- groupId,
131
- privacy = "default"
132
- } = post;
133
- let privacyAqlQry;
134
- if (userId === sessionId) {
135
- return post;
136
- }
137
- if (groupId && privacy === "group") {
138
- privacyAqlQry = `LET p = DOCUMENT("${postDocId}")
139
- ${selectQueries.join("\n")}
28
+ RETURN p`;return s.query(d).then(I=>I.next()).then(I=>{const{_id:E,userId:T,groupId:h,privacy:m="default"}=I;let g;return T===p?I:(h&&m==="group"?g=`LET p = DOCUMENT("${E}")
29
+ ${u.join(`
30
+ `)}
140
31
  FOR group IN groups
141
32
  FILTER group._key == p.groupId
142
- FOR u, e IN OUTBOUND group._id isGrouped
143
- FILTER u._key == "${sessionId}"
33
+ FOR u, e IN OUTBOUND group._id hasConnection
34
+ FILTER u._key == "${p}"
144
35
  LIMIT 1
145
- RETURN MERGE(p, {${selectObjects.join(", ")}})`;
146
- } else if (privacy === "public") {
147
- privacyAqlQry = `LET p = DOCUMENT("${postDocId}")
148
- ${selectQueries.join("\n")}
36
+ RETURN MERGE(p, {${a.join(", ")}})`:m==="public"&&(g=`LET p = DOCUMENT("${E}")
37
+ ${u.join(`
38
+ `)}
149
39
  LIMIT 1
150
- RETURN MERGE(p, {${selectObjects.join(", ")}})`;
151
- }
152
- if (privacyAqlQry) {
153
- return db.query(privacyAqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
154
- action,
155
- category: eventCategory,
156
- label: import_error.ErrorTypes.DATABASE_ERROR
157
- }, error, context));
158
- }
159
- return {};
160
- }).catch((error) => (0, import_analyticsUtils.logError)({
161
- action,
162
- category: eventCategory,
163
- label: import_error.ErrorTypes.DATABASE_ERROR
164
- }, error, context));
165
- };
166
- const getPostsByArea = (context, latitude, longitude, options) => {
167
- const action = "getPostsByArea";
168
- const { database, fields, session: { userId: sessionId } } = context;
169
- const { limit, type } = parsePostOptions(options);
170
- const formatLatitude = (0, import_utils.parseNum)(latitude);
171
- const formatLongitude = (0, import_utils.parseNum)(longitude);
172
- const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
173
- selectQueries.push(`LET distance = DISTANCE(
174
- ${formatLatitude},
175
- ${formatLongitude},
40
+ RETURN MERGE(p, {${a.join(", ")}})`),g?s.query(g).then(l=>l.next()).catch(l=>f({action:o,category:O,label:N.DATABASE_ERROR},l,e)):{})}).catch(I=>f({action:o,category:O,label:N.DATABASE_ERROR},I,e))},st=(e,r,t,o)=>{const i="getPostsByArea",{databaseName:n,fields:p,session:{userId:R}}=e,{limit:c,type:s}=$(o),a=L(r),u=L(t),{objects:d,queries:I}=C(p,R);I.push(`LET distance = DISTANCE(
41
+ ${a},
42
+ ${u},
176
43
  NOT_NULL(p.latitude, 0),
177
44
  NOT_NULL(p.longitude, 0))
178
- `);
179
- selectObjects.push("distance:distance");
180
- const aqlQry = `FOR p IN posts
181
- ${selectQueries.join("\n")}
182
- FILTER p.type == "${type}" && p.privacy == "public" && p.parentId == null
183
- ${limit.aql}
45
+ `),d.push("distance:distance");const E=`FOR p IN posts
46
+ ${I.join(`
47
+ `)}
48
+ FILTER p.type == "${s}" && p.privacy == "public" && p.parentId == null
49
+ ${c.aql}
184
50
  SORT distance, p.added
185
- RETURN DISTINCT MERGE(p, {${selectObjects.join(", ")}})`;
186
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
187
- action,
188
- category: eventCategory,
189
- label: import_error.ErrorTypes.DATABASE_ERROR
190
- }, error, context));
191
- };
192
- const getPostsByLatest = (context, options) => {
193
- const { database, fields, session: { userId: sessionId } } = context;
194
- const { limit, type } = parsePostOptions(options);
195
- const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
196
- const aqlQry = `FOR p IN posts
197
- FILTER p.type == "${type}" && p.privacy == "public" && p.parent == null
198
- ${selectQueries.join("\n")}
199
- ${limit.aql}
51
+ RETURN DISTINCT MERGE(p, {${d.join(", ")}})`;return y(n).query(E).then(T=>T.all()).catch(T=>(f({action:i,category:O,label:N.DATABASE_ERROR},T,e),[]))},ot=(e,r)=>{const{databaseName:t,fields:o,session:{userId:i}}=e,{limit:n,type:p}=$(r),{objects:R,queries:c}=C(o,i),s=`FOR p IN posts
52
+ FILTER p.type == "${p}" && p.privacy == "public" && p.parent == null
53
+ ${c.join(`
54
+ `)}
55
+ ${n.aql}
200
56
  SORT p.added DESC
201
- RETURN DISTINCT MERGE(p, {${selectObjects.join(", ")}})`;
202
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => {
203
- throw error;
204
- });
205
- };
206
- const getPostsByReactions = (context, reactions = [], options) => {
207
- const action = "getPostsByReactions";
208
- const { database, fields, session: { userId: sessionId } } = context;
209
- const { latitude, limit, longitude, type } = parsePostOptions(options);
210
- const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
211
- const formatSessionId = `users/${sessionId}`;
212
- const formatReactions = JSON.stringify(reactions.map((reaction) => (0, import_utils.parseChar)(reaction, 32).toLowerCase()));
213
- const sortBy = [];
214
- const filters = [`p.type == "${type}"`, 'p.privacy == "public"'];
215
- const formatLatitude = (0, import_utils.parseNum)(latitude);
216
- const formatLongitude = (0, import_utils.parseNum)(longitude);
217
- if (formatLatitude && formatLongitude) {
218
- selectQueries.push(`LET distance = DISTANCE(
219
- ${formatLatitude},
220
- ${formatLongitude},
57
+ RETURN DISTINCT MERGE(p, {${R.join(", ")}})`;return y(t).query(s).then(a=>a.all()).catch(a=>{throw a})},rt=(e,r=[],t)=>{const o="getPostsByReactions",{databaseName:i,fields:n,session:{userId:p}}=e,{latitude:R,limit:c,longitude:s,type:a}=$(t),{objects:u,queries:d}=C(n,p),I=`users/${p}`,E=JSON.stringify(r.map(P=>_(P,32).toLowerCase())),T=[],h=[`p.type == "${a}"`,'p.privacy == "public"'],m=L(R),g=L(s);m&&g&&(d.push(`LET distance = DISTANCE(
58
+ ${m},
59
+ ${g},
221
60
  NOT_NULL(p.latitude, 0),
222
61
  NOT_NULL(p.longitude, 0))
223
- `);
224
- selectObjects.push("distance:distance");
225
- sortBy.push("distance");
226
- }
227
- if (reactions.length) {
228
- sortBy.push("matchedTags DESC");
229
- selectQueries.push(`LET matchedReactions = LENGTH(
62
+ `),u.push("distance:distance"),T.push("distance")),r.length&&(T.push("matchedTags DESC"),d.push(`LET matchedReactions = LENGTH(
230
63
  FOR mr IN reactions
231
64
  FILTER mr.matched == true
232
65
  RETURN mr
233
- )`);
234
- selectObjects.push("matchedReactions:matchedReactions");
235
- filters.push("matchedReactions > 0");
236
- }
237
- sortBy.push("p.added DESC");
238
- selectObjects.push("reactions:reactions");
239
- const aqlQry = `FOR p, r IN OUTBOUND "${formatSessionId}" hasReaction
66
+ )`),u.push("matchedReactions:matchedReactions"),h.push("matchedReactions > 0")),T.push("p.added DESC"),u.push("reactions:reactions");const l=`FOR p, r IN OUTBOUND "${I}" hasReaction
240
67
  LET reactions = (
241
68
  FOR reaction, hr IN 1..1 INBOUND p isTagged
242
- LET matched = LENGTH(${formatReactions}) > 0 && POSITION(${formatReactions}, reaction.name)
69
+ LET matched = LENGTH(${E}) > 0 && POSITION(${E}, reaction.name)
243
70
  SORT reaction.name
244
71
  RETURN MERGE(reaction, {matched:matched})
245
72
  )
246
- ${selectQueries.join("\n")}
247
- FILTER ${filters.join(" && ")}
248
- ${limit.aql}
249
- RETURN DISTINCT MERGE(p, {${selectObjects.join(", ")}})`;
250
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
251
- action,
252
- category: eventCategory,
253
- label: import_error.ErrorTypes.DATABASE_ERROR
254
- }, error, context));
255
- };
256
- const getPostsByTags = (context, tags = [], options) => {
257
- const action = "getPostsByTags";
258
- const { database, fields, session: { userId: sessionId } } = context;
259
- const { latitude, limit, longitude, type } = parsePostOptions(options);
260
- const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
261
- const formatTagNames = JSON.stringify(tags.map((tag) => (0, import_utils.parseChar)(tag, 32).toLowerCase()));
262
- const sortBy = [];
263
- const filters = [`p.type == "${type}"`, 'p.privacy == "public"'];
264
- const formatLatitude = (0, import_utils.parseNum)(latitude);
265
- const formatLongitude = (0, import_utils.parseNum)(longitude);
266
- if (formatLatitude && formatLongitude) {
267
- selectQueries.push(`LET distance = DISTANCE(
268
- ${formatLatitude},
269
- ${formatLongitude},
73
+ ${d.join(`
74
+ `)}
75
+ FILTER ${h.join(" && ")}
76
+ ${c.aql}
77
+ RETURN DISTINCT MERGE(p, {${u.join(", ")}})`;return y(i).query(l).then(P=>P.all()).catch(P=>(f({action:o,category:O,label:N.DATABASE_ERROR},P,e),[]))},nt=(e,r=[],t)=>{const o="getPostsByTags",{databaseName:i,fields:n,session:{userId:p}}=e,{latitude:R,limit:c,longitude:s,type:a}=$(t),{objects:u,queries:d}=C(n,p),I=JSON.stringify(r.map(l=>_(l,32).toLowerCase())),E=[],T=[`p.type == "${a}"`,'p.privacy == "public"'],h=L(R),m=L(s);h&&m&&(d.push(`LET distance = DISTANCE(
78
+ ${h},
79
+ ${m},
270
80
  NOT_NULL(p.latitude, 0),
271
81
  NOT_NULL(p.longitude, 0))
272
- `);
273
- selectObjects.push("distance:distance");
274
- sortBy.push("distance");
275
- }
276
- if (tags.length) {
277
- sortBy.push("matchedTags DESC");
278
- selectQueries.push(`LET matchedTags = LENGTH(
82
+ `),u.push("distance:distance"),E.push("distance")),r.length&&(E.push("matchedTags DESC"),d.push(`LET matchedTags = LENGTH(
279
83
  FOR t IN tags
280
84
  FILTER t.matched == true
281
85
  RETURN t
282
- )`);
283
- selectObjects.push("matchedTags:matchedTags");
284
- filters.push("matchedTags > 0");
285
- }
286
- sortBy.push("p.added DESC");
287
- selectObjects.push("tags:tags");
288
- const aqlQry = `FOR p IN posts
86
+ )`),u.push("matchedTags:matchedTags"),T.push("matchedTags > 0")),E.push("p.added DESC"),u.push("tags:tags");const g=`FOR p IN posts
289
87
  LET tags = (
290
88
  FOR tag, it IN 1..1 INBOUND p isTagged
291
- LET matched = LENGTH(${formatTagNames}) > 0 && POSITION(${formatTagNames}, tag.name)
89
+ LET matched = LENGTH(${I}) > 0 && POSITION(${I}, tag.name)
292
90
  SORT tag.name
293
91
  RETURN MERGE(tag, {matched:matched})
294
92
  )
295
- ${selectQueries.join("\n")}
296
- FILTER ${filters.join(" && ")}
297
- ${limit.aql}
298
- SORT ${sortBy.join(", ")}
299
- RETURN DISTINCT MERGE(p, {${selectObjects.join(", ")}})`;
300
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
301
- action,
302
- category: eventCategory,
303
- label: import_error.ErrorTypes.DATABASE_ERROR
304
- }, error, context));
305
- };
306
- const getPostsByUser = (context, userId, options) => {
307
- const action = "getPostsByUser";
308
- const { database, fields, session: { userId: sessionId } } = context;
309
- const { limit, type } = parsePostOptions(options);
310
- const formatUserId = (0, import_utils.parseId)(userId);
311
- const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
312
- const aqlQry = `FOR p IN posts
313
- FILTER p.userId == "${formatUserId}" && p.type == "${type}" && p.privacy == "public" && p.parent == null
314
- ${selectQueries.join("\n")}
315
- ${limit.aql}
93
+ ${d.join(`
94
+ `)}
95
+ FILTER ${T.join(" && ")}
96
+ ${c.aql}
97
+ SORT ${E.join(", ")}
98
+ RETURN DISTINCT MERGE(p, {${u.join(", ")}})`;return y(i).query(g).then(l=>l.all()).catch(l=>(f({action:o,category:O,label:N.DATABASE_ERROR},l,e),[]))},at=(e,r,t)=>{const o="getPostsByUser",{databaseName:i,fields:n,session:{userId:p}}=e,{limit:R,type:c}=$(t),s=D(r),{objects:a,queries:u}=C(n,p),d=`FOR p IN posts
99
+ FILTER p.userId == "${s}" && p.type == "${c}" && p.privacy == "public" && p.parent == null
100
+ ${u.join(`
101
+ `)}
102
+ ${R.aql}
316
103
  SORT p.added
317
- RETURN DISTINCT MERGE(p, {${selectObjects.join(", ")}})`;
318
- return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
319
- action,
320
- category: eventCategory,
321
- label: import_error.ErrorTypes.DATABASE_ERROR
322
- }, error, context));
323
- };
324
- const getPostComments = (context, postId, options) => {
325
- const action = "getPostComments";
326
- const { database, session: { userId: sessionId } } = context;
327
- const { limit, type } = parsePostOptions(options);
328
- const formatItemId = (0, import_utils.parseId)(postId);
329
- const aqlQry = import_arangojs.aql`FOR p IN posts
330
- FILTER p.type == ${type} && p._key == ${formatItemId}
104
+ RETURN DISTINCT MERGE(p, {${a.join(", ")}})`;return y(i).query(d).then(I=>I.all()).catch(I=>(f({action:o,category:O,label:N.DATABASE_ERROR},I,e),[]))},it=(e,r,t)=>{const o="getPostComments",{databaseName:i,session:{userId:n}}=e,{limit:p,type:R}=$(t),c=D(r),s=A`FOR p IN posts
105
+ FILTER p.type == ${R} && p._key == ${c}
331
106
  LIMIT 1
332
- RETURN p`;
333
- return database.query(aqlQry).then((cursor) => cursor.next()).then((post) => {
334
- const {
335
- _key,
336
- groupId,
337
- privacy = "default"
338
- } = post;
339
- let privacyAqlQry;
340
- if (groupId && privacy === "group") {
341
- privacyAqlQry = `FOR p IN posts
107
+ RETURN p`;return y(i).query(s).then(a=>a.next()).then(a=>{const{_key:u,groupId:d,privacy:I="default"}=a;let E;return d&&I==="group"?E=`FOR p IN posts
342
108
  FOR user IN users
343
- FILTER p.parent == "${_key}" && user._key == p.userId
109
+ FILTER p.parent == "${u}" && user._key == p.userId
344
110
  LET reactions = (
345
111
  FOR post, r IN INBOUND p._id reactions
346
112
  COLLECT reactionName = r.value INTO reactionItems
@@ -348,234 +114,30 @@ const getPostComments = (context, postId, options) => {
348
114
  )
349
115
  FOR group IN groups
350
116
  FILTER group._key == p.groupId
351
- FOR u, e IN OUTBOUND group._id isGrouped
352
- FILTER u._key == "${sessionId}"
117
+ FOR u, e IN OUTBOUND group._id hasConnection
118
+ FILTER u._key == "${n}"
353
119
  SORT p.added
354
- ${limit.aql}
355
- RETURN MERGE(p, {user: user, reactions: reactions})`;
356
- } else if (privacy === "public") {
357
- privacyAqlQry = `FOR p IN posts
120
+ ${p.aql}
121
+ RETURN MERGE(p, {user: user, reactions: reactions})`:I==="public"&&(E=`FOR p IN posts
358
122
  FOR user IN users
359
- FILTER p.parent == "${_key}" && user._key == p.userId
123
+ FILTER p.parent == "${u}" && user._key == p.userId
360
124
  LET reactions = (
361
125
  FOR post, r IN INBOUND p._id reactions
362
126
  COLLECT reactionName = r.value INTO reactionItems
363
127
  RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
364
128
  )
365
129
  SORT p.added
366
- ${limit.aql}
367
- RETURN MERGE(p, {user: user, reactions: reactions})`;
368
- }
369
- if (privacyAqlQry) {
370
- return database.query(privacyAqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
371
- action,
372
- category: eventCategory,
373
- label: import_error.ErrorTypes.DATABASE_ERROR
374
- }, error, context));
375
- }
376
- return [];
377
- }).catch((error) => (0, import_analyticsUtils.logError)({
378
- action,
379
- category: eventCategory,
380
- label: import_error.ErrorTypes.DATABASE_ERROR
381
- }, error, context));
382
- };
383
- const addPost = async (context, {
384
- content = "",
385
- endDate,
386
- groupId = "",
387
- location,
388
- latitude,
389
- longitude,
390
- name = "",
391
- parentId = null,
392
- privacy = "public",
393
- tags = [],
394
- startDate,
395
- type = "default"
396
- }) => {
397
- const action = "addPost";
398
- const { database, session: { userId: sessionId } } = context;
399
- const now = Date.now();
400
- const postId = (0, import_utils.createHash)(`post-${sessionId}`);
401
- const insert = {
402
- _id: `posts/${postId}`,
403
- _key: postId,
404
- added: now,
405
- content: (0, import_utils.parseString)(content, MAX_CONTENT_LENGTH),
406
- endDate: endDate ? (0, import_utils.parseNum)(endDate, 13) : void 0,
407
- groupId: groupId ? (0, import_utils.parseId)(groupId) : void 0,
408
- latitude: latitude !== void 0 ? (0, import_utils.parseNum)(latitude) : void 0,
409
- location: location ? (0, import_utils.parseString)(location, 160) : void 0,
410
- longitude: longitude !== void 0 ? (0, import_utils.parseNum)(longitude) : void 0,
411
- modified: now,
412
- name: (0, import_utils.parseString)(name, 160),
413
- parentId: parentId ? (0, import_utils.parseId)(parentId) : void 0,
414
- privacy: (0, import_utils.parseVarChar)(privacy, 16),
415
- startDate: startDate ? (0, import_utils.parseNum)(startDate, 13) : void 0,
416
- type: (0, import_utils.parseChar)(type, 32),
417
- userId: sessionId
418
- };
419
- const aqlQry = import_arangojs.aql`INSERT ${insert} IN posts RETURN NEW`;
420
- try {
421
- const savedPost = await database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
422
- action,
423
- category: eventCategory,
424
- label: import_error.ErrorTypes.DATABASE_ERROR
425
- }, error, context));
426
- const { _id: postDocId } = savedPost;
427
- const contentTagNames = await (0, import_tags.extractTags)(insert.content);
428
- if (tags.length || contentTagNames.length) {
429
- const userTags = (await (0, import_tags.getTagsByName)(context, tags.map(({ name: name2 }) => name2))).map((tag) => ({ ...tag, tagBy: sessionId }));
430
- const contentTags = (await (0, import_tags.getTagsByName)(context, contentTagNames)).map((tag) => ({ ...tag, tagBy: "extract" }));
431
- const updatedTags = await (0, import_tags.updateTagsInItem)(
432
- context,
433
- {
434
- itemDocId: postDocId,
435
- tags: [...contentTags, ...userTags]
436
- }
437
- );
438
- return {
439
- ...savedPost,
440
- tags: updatedTags
441
- };
442
- }
443
- return savedPost;
444
- } catch (error) {
445
- throw error;
446
- }
447
- };
448
- const updatePost = async (context, post) => {
449
- const action = "updatePost";
450
- const { database, session: { userId: sessionId } } = context;
451
- const now = Date.now();
452
- const parsedPost = (0, import_postAdapter.parsePost)(post);
453
- const {
454
- postId,
455
- tags = []
456
- } = parsedPost;
457
- const update = {
458
- ...parsedPost,
459
- modified: now
460
- };
461
- if (!postId) {
462
- return (0, import_analyticsUtils.logException)({
463
- action,
464
- category: eventCategory,
465
- value: import_error.ErrorTypes.INVALID_ID
466
- }, {});
467
- }
468
- const insert = {
469
- ...update,
470
- _key: postId,
471
- added: now,
472
- userId: sessionId
473
- };
474
- const aqlQry = import_arangojs.aql`UPSERT {_key: ${postId}, userId: ${sessionId}}
475
- INSERT ${insert}
476
- UPDATE ${update}
477
- IN posts RETURN NEW`;
478
- try {
479
- const updatedPost = await database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
480
- action,
481
- category: eventCategory,
482
- value: import_error.ErrorTypes.DATABASE_ERROR
483
- }, error, {}));
484
- const { _id: updatedPostId } = updatedPost;
485
- const contentTagNames = await (0, import_tags.extractTags)(insert.content);
486
- if (tags?.length || contentTagNames?.length) {
487
- const userTags = tags?.length ? (await (0, import_tags.getTagsByName)(context, tags.map(({ name }) => name))).map((tag) => ({ ...tag, tagBy: sessionId })) : [];
488
- const contentTags = contentTagNames?.length ? (await (0, import_tags.getTagsByName)(context, contentTagNames)).map((tag) => ({ ...tag, tagBy: "extract" })) : [];
489
- const updatedTags = await (0, import_tags.updateTagsInItem)(
490
- context,
491
- {
492
- itemDocId: updatedPostId,
493
- tags: [...contentTags, ...userTags]
494
- }
495
- );
496
- return {
497
- ...updatedPost,
498
- tags: updatedTags
499
- };
500
- }
501
- return updatedPost;
502
- } catch (error) {
503
- throw error;
504
- }
505
- };
506
- const deletePost = async (context, postDocId) => {
507
- const action = "deletePost";
508
- const { database, session: { userId: sessionId } } = context;
509
- const formatPostId = (0, import_utils.parseArangoId)(postDocId);
510
- if (!formatPostId) {
511
- return (0, import_analyticsUtils.logException)({
512
- action,
513
- category: eventCategory,
514
- value: import_error.ErrorTypes.INVALID_ID
515
- }, {});
516
- }
517
- const edgeAqlQry = import_arangojs.aql`FOR t IN isTagged
518
- FILTER t._to == ${formatPostId}
519
- REMOVE t IN isTagged`;
520
- await database.query(edgeAqlQry).catch((error) => {
521
- throw error;
522
- });
523
- const fileAqlQry = import_arangojs.aql`FOR f IN hasFile
524
- FILTER f._to == ${formatPostId}
525
- REMOVE f IN hasFile`;
526
- await database.query(fileAqlQry).catch((error) => {
527
- throw error;
528
- });
529
- const aqlQry = import_arangojs.aql`FOR p IN posts
530
- FILTER p._id == ${formatPostId} && p.userId == ${sessionId}
130
+ ${p.aql}
131
+ RETURN MERGE(p, {user: user, reactions: reactions})`),E?y(i).query(E).then(T=>T.all()).catch(T=>(f({action:o,category:O,label:N.DATABASE_ERROR},T,e),[])):[]}).catch(a=>(f({action:o,category:O,label:N.DATABASE_ERROR},a,e),[]))},pt=async(e,{content:r="",endDate:t,groupId:o="",location:i,latitude:n,longitude:p,name:R="",parentId:c=null,privacy:s="public",tags:a=[],startDate:u,type:d="default"})=>{const I="addPost",{databaseName:E,session:{userId:T}}=e,h=Date.now(),m=B(`post-${T}`),g={_id:`posts/${m}`,_key:m,added:h,content:q(r,W),endDate:t?L(t,13):void 0,groupId:o?D(o):void 0,latitude:n!==void 0?L(n):void 0,location:i?q(i,160):void 0,longitude:p!==void 0?L(p):void 0,modified:h,name:q(R,160),parentId:c?D(c):void 0,privacy:k(s,16),startDate:u?L(u,13):void 0,type:_(d,32),userId:T},l=A`INSERT ${g} IN posts RETURN NEW`;try{const P=await y(E).query(l).then(b=>b.next()).catch(b=>f({action:I,category:O,label:N.DATABASE_ERROR},b,e)),{_id:M}=P,v=await w(g.content);if(a.length||v.length){const b=(await U(e,a.map(({name:F})=>F))).map(F=>({...F,tagBy:T})),x=(await U(e,v)).map(F=>({...F,tagBy:"extract"})),G=await j(e,{itemDocId:M,tags:[...x,...b]});return{...P,tags:G}}return P}catch(P){throw P}},ct=async(e,r)=>{const t="updatePost",{databaseName:o,session:{userId:i}}=e,n=Date.now(),p=H(r),{postId:R,tags:c=[]}=p,s={...p,modified:n};if(!R)return Q({action:t,category:O,value:N.INVALID_ID},{});const a={...s,_key:R,added:n,userId:i},u=A`UPSERT {_key: ${R}, userId: ${i}}
132
+ INSERT ${a}
133
+ UPDATE ${s}
134
+ IN posts RETURN NEW`;try{const d=await y(o).query(u).then(T=>T.next()).catch(T=>f({action:t,category:O,value:N.DATABASE_ERROR},T,{})),{_id:I}=d,E=await w(a.content);if(c?.length||E?.length){const T=c?.length?(await U(e,c.map(({name:g})=>g))).map(g=>({...g,tagBy:i})):[],h=E?.length?(await U(e,E)).map(g=>({...g,tagBy:"extract"})):[],m=await j(e,{itemDocId:I,tags:[...h,...T]});return{...d,tags:m}}return d}catch(d){throw d}},ut=async(e,r)=>{const t="deletePost",{databaseName:o,session:{userId:i}}=e,n=S(r);if(!n)return Q({action:t,category:O,value:N.INVALID_ID},{});const p=A`FOR t IN isTagged
135
+ FILTER t._to == ${n}
136
+ REMOVE t IN isTagged`;await y(o).query(p).catch(s=>{throw s});const R=A`FOR f IN hasFile
137
+ FILTER f._to == ${n}
138
+ REMOVE f IN hasFile`;await y(o).query(R).catch(s=>{throw s});const c=A`FOR p IN posts
139
+ FILTER p._id == ${n} && p.userId == ${i}
531
140
  LIMIT 1
532
141
  REMOVE p IN posts
533
- RETURN OLD`;
534
- return database.query(aqlQry).then((cursor) => cursor.next()).catch((error) => {
535
- throw error;
536
- });
537
- };
538
- const createPostEdge = (context, postDocId, itemDocId, edgeType = "isPosted", props = {}) => {
539
- const action = "createPostEdge";
540
- const { database } = context;
541
- const edgeCollection = database.collection(edgeType);
542
- const formatPostId = (0, import_utils.parseArangoId)(postDocId);
543
- const formatDocId = (0, import_utils.parseArangoId)(itemDocId);
544
- if (!formatDocId || !formatPostId) {
545
- return (0, import_analyticsUtils.logException)({
546
- action,
547
- category: eventCategory,
548
- value: import_error.ErrorTypes.INVALID_ID
549
- }, {});
550
- }
551
- const edgeId = (0, import_utils.createHash)(`postEdge-${formatPostId}-${formatDocId}`);
552
- const edge = {
553
- _from: formatPostId,
554
- _key: edgeId,
555
- _to: formatDocId,
556
- added: Date.now(),
557
- ...props
558
- };
559
- return edgeCollection.save(edge, { returnNew: true }).catch((error) => (0, import_analyticsUtils.logError)({
560
- action,
561
- category: eventCategory,
562
- value: import_error.ErrorTypes.DATABASE_ERROR
563
- }, error, {}));
564
- };
565
- // Annotate the CommonJS export names for ESM import in node:
566
- 0 && (module.exports = {
567
- addPost,
568
- createPostEdge,
569
- deletePost,
570
- getPost,
571
- getPostComments,
572
- getPostOptional,
573
- getPostsByArea,
574
- getPostsByLatest,
575
- getPostsByReactions,
576
- getPostsByTags,
577
- getPostsByUser,
578
- parsePostOptions,
579
- updatePost
580
- });
581
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/posts.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport {\n  createHash, parseArangoId, parseChar, parseId, parseNum, parseString, parseVarChar\n} from '@nlabs/utils';\n\nimport {parsePost} from '../adapters/postAdapter';\nimport {ErrorTypes} from '../types/error';\nimport {logError, logException} from '../utils/analyticsUtils';\nimport {getLimit} from '../utils/arangodbUtils';\nimport {extractTags, getTagsByName, updateTagsInItem} from './tags';\nimport type {ApiContext, ArangoDbLimit, FileType, PostInputType, PostOptions, PostType, TagType} from '../types';\n\nconst MAX_CONTENT_LENGTH: number = 100000;\nconst eventCategory: string = 'posts';\n\nexport const parsePostOptions = (options: PostOptions = {}) => {\n  const {\n    from = 0,\n    latitude = 0,\n    longitude = 0,\n    to = 30,\n    type = 'post'\n  } = options;\n\n  return {\n    latitude: parseNum(latitude, 32),\n    limit: getLimit(from, to) as ArangoDbLimit,\n    longitude: parseNum(longitude, 32),\n    type: parseChar(type, 32)\n  };\n};\n\nexport const getPostOptional = (fields: string[], sessionId: string) =>\n  (fields || []).reduce((selects: any, field: string) => {\n    switch(field) {\n      case 'hasRsvp': {\n        selects.queries.push(`LET hasRsvp = TO_BOOL(FIRST(\n          FOR post, r IN INBOUND p._id hasReaction\n          FILTER r.name == \"rsvp\" && r.type == \"posts\" && r._from == \"users/${sessionId}\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        ))`);\n        selects.objects.push('hasRsvp:hasRsvp');\n        return selects;\n      }\n      case 'isSaved': {\n        selects.queries.push(`LET isSaved = TO_BOOL(FIRST(\n          FOR post, r IN INBOUND p._id hasReaction\n          FILTER r.name == \"pin\" && r.type == \"posts\" && r._from == \"users/${sessionId}\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        ))`);\n        selects.objects.push('isSaved:isSaved');\n        return selects;\n      }\n      case 'reactions': {\n        selects.queries.push(`LET reactions = (\n          FOR post, r IN INBOUND p._id hasReaction\n          COLLECT reactionName = r.value INTO reactionItems\n          RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n        )`);\n        selects.objects.push('reactions:reactions');\n        return selects;\n      }\n      case 'rsvpCount': {\n        selects.queries.push(`LET rsvpCount = FIRST(\n          FOR post, r IN INBOUND p._id hasReaction\n          FILTER r.name == \"rsvp\" && r.type == \"posts\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        )`);\n        selects.objects.push('rsvpCount:rsvpCount');\n        return selects;\n      }\n      case 'viewCount': {\n        selects.queries.push(`LET viewCount = FIRST(\n          FOR post, r IN INBOUND p._id hasReaction\n          FILTER r.name == \"view\" && r.type == \"posts\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        )`);\n        selects.objects.push('viewCount:viewCount');\n        return selects;\n      }\n      default: {\n        return selects;\n      }\n    }\n  }, {objects: [], queries: []});\n\nexport const getPost = async (\n  context: ApiContext,\n  postId: string,\n  options?: PostOptions\n): Promise<PostType> => {\n  const action: string = 'getPost';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(postId);\n  const {type} = parsePostOptions(options);\n  const db = database;\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n    FILTER p._key == ${formatItemId} && p.type == ${type}\n    LIMIT 1\n    RETURN p`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType) => {\n      const {\n        _id: postDocId,\n        userId,\n        groupId,\n        privacy = 'default'\n      }: PostType = post;\n\n      let privacyAqlQry: string;\n\n      if(userId === sessionId) {\n        return post;\n      }\n\n      if(groupId && privacy === 'group') {\n        privacyAqlQry = `LET p = DOCUMENT(\"${postDocId}\")\n          ${selectQueries.join('\\n')}\n          FOR group IN groups\n          FILTER group._key == p.groupId\n          FOR u, e IN OUTBOUND group._id isGrouped\n          FILTER u._key == \"${sessionId}\"\n          LIMIT 1\n          RETURN MERGE(p, {${selectObjects.join(', ')}})`;\n      } else if(privacy === 'public') {\n        privacyAqlQry = `LET p = DOCUMENT(\"${postDocId}\")\n          ${selectQueries.join('\\n')}\n          LIMIT 1\n          RETURN MERGE(p, {${selectObjects.join(', ')}})`;\n      }\n\n      if(privacyAqlQry) {\n        return db.query(privacyAqlQry)\n          .then((cursor: ArrayCursor) => cursor.next())\n          .catch((error: Error) => logError({\n            action,\n            category: eventCategory,\n            label: ErrorTypes.DATABASE_ERROR\n          }, error, context));\n      }\n\n      return {};\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\n// export const getPostList = (context: ApiContext, options?: PostOptions): Promise<PostType[]> => {\n//   // const action: string = 'getListByApp';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {limit, type} = parsePostOptions(options);\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n//   const aqlQry: string = `FOR p IN posts\n//     FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n//     ${selectQueries.join('\\n')}\n//     ${limit.aql}\n//     SORT p.added\n//     RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//   return database.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\nexport const getPostsByArea = (\n  context: ApiContext,\n  latitude: number,\n  longitude: number,\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByArea';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatLatitude: number = parseNum(latitude);\n  const formatLongitude: number = parseNum(longitude);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  selectQueries.push(`LET distance = DISTANCE(\n    ${formatLatitude},\n    ${formatLongitude},\n    NOT_NULL(p.latitude, 0),\n    NOT_NULL(p.longitude, 0))\n  `);\n  selectObjects.push('distance:distance');\n\n  const aqlQry: string = `FOR p IN posts\n    ${selectQueries.join('\\n')}\n    FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parentId == null\n    ${limit.aql}\n    SORT distance, p.added\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\n// export const getPostsByGroup = (\n//   context: ApiContext,\n//   groupId: string,\n//   options?: PostOptions\n// ): Promise<PostType[]> => {\n//   // const action: string = 'getListByGroup';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n\n//   // Group id\n//   const formatGroupId: string = parseId(groupId);\n//   const db = database;\n//   const aqlQry: string = `FOR u, g IN INBOUND ${formatGroupId} hasGroup\n//       FILTER u._key == ${sessionId}\n//       RETURN g`;\n\n//   return db.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .then((groups: GroupType[] = []) => {\n//       if(groups.length) {\n//         const {limit, type} = parsePostOptions(options);\n//         const postAqlQry: string = `FOR p IN posts\n//           FILTER p.type == \"${type}\" && p.groupId == \"${formatGroupId}\" && p.parent == null\n//           ${selectQueries.join('\\n')}\n//           ${limit.aql}\n//           SORT p.added\n//           RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//         return db.query(postAqlQry)\n//           .then((cursor: ArrayCursor) => cursor.all())\n//           .catch((error: Error) => {\n//             throw error;\n//           });\n//       }\n\n//       return [];\n//     })\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\nexport const getPostsByLatest = (context: ApiContext, options?: PostOptions): Promise<PostType[]> => {\n  // const action: string = 'getListByLatest';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: string = `FOR p IN posts\n    FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT p.added DESC\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getPostsByReactions = (\n  context: ApiContext,\n  reactions: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByReactions';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {latitude, limit, longitude, type} = parsePostOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatReactions: string = JSON.stringify(reactions.map((reaction) => parseChar(reaction, 32).toLowerCase()));\n  const sortBy: string[] = [];\n  const filters: string[] = [`p.type == \"${type}\"`, 'p.privacy == \"public\"'];\n  const formatLatitude: number = parseNum(latitude);\n  const formatLongitude: number = parseNum(longitude);\n\n  if(formatLatitude && formatLongitude) {\n    selectQueries.push(`LET distance = DISTANCE(\n      ${formatLatitude},\n      ${formatLongitude},\n      NOT_NULL(p.latitude, 0),\n      NOT_NULL(p.longitude, 0))\n    `);\n    selectObjects.push('distance:distance');\n    sortBy.push('distance');\n  }\n\n  if(reactions.length) {\n    sortBy.push('matchedTags DESC');\n    selectQueries.push(`LET matchedReactions = LENGTH(\n      FOR mr IN reactions\n      FILTER mr.matched == true\n      RETURN mr\n    )`);\n    selectObjects.push('matchedReactions:matchedReactions');\n    filters.push('matchedReactions > 0');\n  }\n\n  sortBy.push('p.added DESC');\n  selectObjects.push('reactions:reactions');\n\n  // Get data from database\n  const aqlQry: string = `FOR p, r IN OUTBOUND \"${formatSessionId}\" hasReaction\n    LET reactions = (\n      FOR reaction, hr IN 1..1 INBOUND p isTagged\n      LET matched = LENGTH(${formatReactions}) > 0 && POSITION(${formatReactions}, reaction.name)\n      SORT reaction.name\n      RETURN MERGE(reaction, {matched:matched})\n    )\n    ${selectQueries.join('\\n')}\n    FILTER ${filters.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getPostsByTags = (\n  context: ApiContext,\n  tags: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByTags';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {latitude, limit, longitude, type} = parsePostOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const formatTagNames: string = JSON.stringify(tags.map((tag) => parseChar(tag, 32).toLowerCase()));\n  const sortBy: string[] = [];\n  const filters: string[] = [`p.type == \"${type}\"`, 'p.privacy == \"public\"'];\n  const formatLatitude: number = parseNum(latitude);\n  const formatLongitude: number = parseNum(longitude);\n\n  if(formatLatitude && formatLongitude) {\n    selectQueries.push(`LET distance = DISTANCE(\n      ${formatLatitude},\n      ${formatLongitude},\n      NOT_NULL(p.latitude, 0),\n      NOT_NULL(p.longitude, 0))\n    `);\n    selectObjects.push('distance:distance');\n    sortBy.push('distance');\n  }\n\n  if(tags.length) {\n    sortBy.push('matchedTags DESC');\n    selectQueries.push(`LET matchedTags = LENGTH(\n      FOR t IN tags\n      FILTER t.matched == true\n      RETURN t\n    )`);\n    selectObjects.push('matchedTags:matchedTags');\n    filters.push('matchedTags > 0');\n  }\n\n  sortBy.push('p.added DESC');\n  selectObjects.push('tags:tags');\n\n  const aqlQry: string = `FOR p IN posts\n    LET tags = (\n      FOR tag, it IN 1..1 INBOUND p isTagged\n      LET matched = LENGTH(${formatTagNames}) > 0 && POSITION(${formatTagNames}, tag.name)\n      SORT tag.name\n      RETURN MERGE(tag, {matched:matched})\n    )\n    ${selectQueries.join('\\n')}\n    FILTER ${filters.join(' && ')}\n    ${limit.aql}\n    SORT ${sortBy.join(', ')}\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getPostsByUser = (context: ApiContext, userId: string, options?: PostOptions): Promise<PostType[]> => {\n  const action: string = 'getPostsByUser';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatUserId: string = parseId(userId);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: string = `FOR p IN posts\n    FILTER p.userId == \"${formatUserId}\" && p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT p.added\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getPostComments = (context: ApiContext, postId: string, options?: PostOptions): Promise<PostType[]> => {\n  const action: string = 'getPostComments';\n  const {database, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatItemId: string = parseId(postId);\n\n  // Get the parent post to get restrictions\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n    FILTER p.type == ${type} && p._key == ${formatItemId}\n    LIMIT 1\n    RETURN p`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType) => {\n      const {\n        _key,\n        groupId,\n        privacy = 'default'\n      }: PostType = post;\n\n      // Query based on privacy level\n      let privacyAqlQry: string;\n\n      if(groupId && privacy === 'group') {\n        privacyAqlQry = `FOR p IN posts\n          FOR user IN users\n          FILTER p.parent == \"${_key}\" && user._key == p.userId\n          LET reactions = (\n            FOR post, r IN INBOUND p._id reactions\n            COLLECT reactionName = r.value INTO reactionItems\n            RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n          )\n          FOR group IN groups\n          FILTER group._key == p.groupId\n          FOR u, e IN OUTBOUND group._id isGrouped\n          FILTER u._key == \"${sessionId}\"\n          SORT p.added\n          ${limit.aql}\n          RETURN MERGE(p, {user: user, reactions: reactions})`;\n      } else if(privacy === 'public') {\n        privacyAqlQry = `FOR p IN posts\n          FOR user IN users\n          FILTER p.parent == \"${_key}\" && user._key == p.userId\n          LET reactions = (\n            FOR post, r IN INBOUND p._id reactions\n            COLLECT reactionName = r.value INTO reactionItems\n            RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n          )\n          SORT p.added\n          ${limit.aql}\n          RETURN MERGE(p, {user: user, reactions: reactions})`;\n      }\n\n      if(privacyAqlQry) {\n        return database.query(privacyAqlQry)\n          .then((cursor: ArrayCursor) => cursor.all())\n          .catch((error: Error) => logError({\n            action,\n            category: eventCategory,\n            label: ErrorTypes.DATABASE_ERROR\n          }, error, context));\n      }\n\n      return [];\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const addPost = async (\n  context: ApiContext,\n  {\n    content = '',\n    endDate,\n    groupId = '',\n    location,\n    latitude,\n    longitude,\n    name = '',\n    parentId = null,\n    privacy = 'public',\n    tags = [],\n    startDate,\n    type = 'default'\n  }: PostInputType\n): Promise<PostType> => {\n  const action: string = 'addPost';\n  const {database, session: {userId: sessionId}} = context;\n  const now = Date.now();\n  const postId = createHash(`post-${sessionId}`);\n  const insert: PostType = {\n    _id: `posts/${postId}`,\n    _key: postId,\n    added: now,\n    content: parseString(content, MAX_CONTENT_LENGTH),\n    endDate: endDate ? parseNum(endDate, 13) : undefined,\n    groupId: groupId ? parseId(groupId) : undefined,\n    latitude: latitude !== undefined ? parseNum(latitude) : undefined,\n    location: location ? parseString(location, 160) : undefined,\n    longitude: longitude !== undefined ? parseNum(longitude) : undefined,\n    modified: now,\n    name: parseString(name, 160),\n    parentId: parentId ? parseId(parentId) : undefined,\n    privacy: parseVarChar(privacy, 16),\n    startDate: startDate ? parseNum(startDate, 13) : undefined,\n    type: parseChar(type, 32),\n    userId: sessionId\n  };\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN posts RETURN NEW`;\n\n  try {\n    const savedPost: PostType = await database.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context));\n    const {_id: postDocId} = savedPost;\n    const contentTagNames = await extractTags(insert.content);\n\n    if(tags.length || contentTagNames.length) {\n      const userTags = (await getTagsByName(context, tags.map(({name}) => name)))\n        .map((tag) => ({...tag, tagBy: sessionId}));\n      const contentTags = (await getTagsByName(context, contentTagNames))\n        .map((tag) => ({...tag, tagBy: 'extract'}));\n      const updatedTags: TagType[] = await updateTagsInItem(\n        context,\n        {\n          itemDocId: postDocId,\n          tags: [...contentTags, ...userTags]\n        }\n      );\n\n      return {\n        ...savedPost,\n        tags: updatedTags\n      };\n    }\n\n    return savedPost;\n  } catch(error) {\n    throw error;\n  }\n};\n\nexport const updatePost = async (context: ApiContext, post: PostInputType): Promise<PostType> => {\n  const action: string = 'updatePost';\n  const {database, session: {userId: sessionId}} = context;\n  const now: number = Date.now();\n  const parsedPost = parsePost(post);\n  const {\n    postId,\n    tags = []\n  } = parsedPost;\n\n  const update: PostType = {\n    ...parsedPost,\n    modified: now\n  };\n\n  if(!postId) {\n    return logException({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ID\n    }, {});\n  }\n\n  const insert: any = {\n    ...update,\n    _key: postId,\n    added: now,\n    userId: sessionId\n  };\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${postId}, userId: ${sessionId}}\n    INSERT ${insert}\n    UPDATE ${update}\n    IN posts RETURN NEW`;\n\n  try {\n    const updatedPost: PostType = await database\n      .query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, {}));\n    const {_id: updatedPostId} = updatedPost;\n    const contentTagNames = await extractTags(insert.content);\n\n    if(tags?.length || contentTagNames?.length) {\n      const userTags = tags?.length ? (await getTagsByName(context, tags.map(({name}) => name)))\n        .map((tag) => ({...tag, tagBy: sessionId})) : [];\n      const contentTags = contentTagNames?.length ? (await getTagsByName(context, contentTagNames))\n        .map((tag) => ({...tag, tagBy: 'extract'})) : [];\n      const updatedTags: TagType[] = await updateTagsInItem(\n        context,\n        {\n          itemDocId: updatedPostId,\n          tags: [...contentTags, ...userTags]\n        }\n      );\n\n      return {\n        ...updatedPost,\n        tags: updatedTags\n      };\n    }\n\n    return updatedPost;\n  } catch(error) {\n    throw error;\n  }\n};\n\nexport const deletePost = async (context: ApiContext, postDocId: string): Promise<PostType> => {\n  const action: string = 'deletePost';\n  const {database, session: {userId: sessionId}} = context;\n  const formatPostId: string = parseArangoId(postDocId);\n\n  if(!formatPostId) {\n    return logException({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ID\n    }, {});\n  }\n\n  const edgeAqlQry: AqlQuery = aql`FOR t IN isTagged\n  FILTER t._to == ${formatPostId}\n  REMOVE t IN isTagged`;\n\n  await database.query(edgeAqlQry)\n    .catch((error: Error) => {\n      throw error;\n    });\n\n  const fileAqlQry: AqlQuery = aql`FOR f IN hasFile\n    FILTER f._to == ${formatPostId}\n    REMOVE f IN hasFile`;\n\n  await database.query(fileAqlQry)\n    .catch((error: Error) => {\n      throw error;\n    });\n\n  const aqlQry = aql`FOR p IN posts\n      FILTER p._id == ${formatPostId} && p.userId == ${sessionId}\n      LIMIT 1\n      REMOVE p IN posts\n      RETURN OLD`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const createPostEdge = (\n  context: ApiContext,\n  postDocId: string,\n  itemDocId: string,\n  edgeType: string = 'isPosted',\n  props: any = {}\n): Promise<FileType> => {\n  const action = 'createPostEdge';\n  const {database} = context;\n  const edgeCollection: EdgeCollection = database.collection(edgeType);\n  const formatPostId: string = parseArangoId(postDocId);\n  const formatDocId: string = parseArangoId(itemDocId);\n\n  if(!formatDocId || !formatPostId) {\n    return logException({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ID\n    }, {});\n  }\n\n  const edgeId: string = createHash(`postEdge-${formatPostId}-${formatDocId}`);\n  const edge: any = {\n    _from: formatPostId,\n    _key: edgeId,\n    _to: formatDocId,\n    added: Date.now(),\n    ...props\n  };\n\n  return edgeCollection.save(edge, {returnNew: true})\n    .catch((error: Error) =>\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, {}));\n};\n"],
  "mappings": "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",
  "names": ["name"]
}

142
+ RETURN OLD`;return y(o).query(c).then(s=>s.next()).catch(s=>{throw s})},dt=(e,r,t,o="isPosted",i={})=>{const n="createPostEdge",{databaseName:p}=e,R=y(p).collection(o),c=S(r),s=S(t);if(!s||!c)return Promise.reject(new Error(N.INVALID_ID));const a=B(`postEdge-${c}-${s}`),u={_from:c,_key:a,_to:s,added:Date.now(),...i};return R.save(u,{returnNew:!0}).catch(d=>f({action:n,category:O,value:N.DATABASE_ERROR},d,{}))};export{pt as addPost,dt as createPostEdge,ut as deletePost,et as getPost,it as getPostComments,C as getPostOptional,st as getPostsByArea,ot as getPostsByLatest,rt as getPostsByReactions,nt as getPostsByTags,at as getPostsByUser,$ as parsePostOptions,ct as updatePost};
143
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/posts.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {\n  createHash, parseArangoId, parseChar, parseId, parseNum, parseString, parseVarChar\n} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\n\nimport {extractTags, getTagsByName, updateTagsInItem} from './tags';\nimport {parsePost} from '../adapters/postAdapter';\nimport {ErrorTypes} from '../types/error.types';\nimport {logError, logException} from '../utils/analyticsUtils';\nimport {getLimit, useDb} from '../utils/arangodbUtils';\n\nimport type {ApiContext, ArangoDbLimit, FileType, PostInputType, PostOptions, PostType, TagType} from '../types';\nimport type {EdgeCollection} from 'arangojs/collections';\n\nconst MAX_CONTENT_LENGTH: number = 100000;\nconst eventCategory: string = 'posts';\n\nexport const parsePostOptions = (options: PostOptions = {}) => {\n  const {\n    from = 0,\n    latitude = 0,\n    longitude = 0,\n    to = 30,\n    type = 'post'\n  } = options;\n\n  return {\n    latitude: parseNum(latitude, 32),\n    limit: getLimit(from, to) as ArangoDbLimit,\n    longitude: parseNum(longitude, 32),\n    type: parseChar(type, 32)\n  };\n};\n\nexport const getPostOptional = (fields: string[], sessionId: string) =>\n  (fields || []).reduce((selects, field: string) => {\n    switch(field) {\n      case 'hasRsvp': {\n        selects.queries.push(`LET hasRsvp = TO_BOOL(FIRST(\n          FOR post, r IN INBOUND p._id hasReaction\n          FILTER r.name == \"rsvp\" && r.type == \"posts\" && r._from == \"users/${sessionId}\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        ))`);\n        selects.objects.push('hasRsvp:hasRsvp');\n        return selects;\n      }\n      case 'isSaved': {\n        selects.queries.push(`LET isSaved = TO_BOOL(FIRST(\n          FOR post, r IN INBOUND p._id hasReaction\n          FILTER r.name == \"pin\" && r.type == \"posts\" && r._from == \"users/${sessionId}\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        ))`);\n        selects.objects.push('isSaved:isSaved');\n        return selects;\n      }\n      case 'reactions': {\n        selects.queries.push(`LET reactions = (\n          FOR post, r IN INBOUND p._id hasReaction\n          COLLECT reactionName = r.value INTO reactionItems\n          RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n        )`);\n        selects.objects.push('reactions:reactions');\n        return selects;\n      }\n      case 'rsvpCount': {\n        selects.queries.push(`LET rsvpCount = FIRST(\n          FOR post, r IN INBOUND p._id hasReaction\n          FILTER r.name == \"rsvp\" && r.type == \"posts\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        )`);\n        selects.objects.push('rsvpCount:rsvpCount');\n        return selects;\n      }\n      case 'viewCount': {\n        selects.queries.push(`LET viewCount = FIRST(\n          FOR post, r IN INBOUND p._id hasReaction\n          FILTER r.name == \"view\" && r.type == \"posts\"\n          COLLECT WITH COUNT INTO count\n          RETURN count\n        )`);\n        selects.objects.push('viewCount:viewCount');\n        return selects;\n      }\n      default: {\n        return selects;\n      }\n    }\n  }, {objects: [], queries: []});\n\nexport const getPost = async (\n  context: ApiContext,\n  postId: string,\n  options?: PostOptions\n): Promise<PostType> => {\n  const action: string = 'getPost';\n  const {databaseName, fields, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(postId);\n  const {type} = parsePostOptions(options);\n  const database = useDb(databaseName);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n    FILTER p._key == ${formatItemId} && p.type == ${type}\n    LIMIT 1\n    RETURN p`;\n\n  return database.query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((post: PostType) => {\n      const {\n        _id: postDocId,\n        userId,\n        groupId,\n        privacy = 'default'\n      }: PostType = post;\n\n      let privacyAqlQry: string;\n\n      if(userId === sessionId) {\n        return post;\n      }\n\n      if(groupId && privacy === 'group') {\n        privacyAqlQry = `LET p = DOCUMENT(\"${postDocId}\")\n          ${selectQueries.join('\\n')}\n          FOR group IN groups\n          FILTER group._key == p.groupId\n          FOR u, e IN OUTBOUND group._id hasConnection\n          FILTER u._key == \"${sessionId}\"\n          LIMIT 1\n          RETURN MERGE(p, {${selectObjects.join(', ')}})`;\n      } else if(privacy === 'public') {\n        privacyAqlQry = `LET p = DOCUMENT(\"${postDocId}\")\n          ${selectQueries.join('\\n')}\n          LIMIT 1\n          RETURN MERGE(p, {${selectObjects.join(', ')}})`;\n      }\n\n      if(privacyAqlQry) {\n        return database.query(privacyAqlQry)\n          .then((cursor) => cursor.next())\n          .catch((error: Error) => logError({\n            action,\n            category: eventCategory,\n            label: ErrorTypes.DATABASE_ERROR\n          }, error, context));\n      }\n\n      return {};\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\n\n// export const getPostList = (context: ApiContext, options?: PostOptions): Promise<PostType[]> => {\n//   // const action: string = 'getListByApp';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {limit, type} = parsePostOptions(options);\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n//   const aqlQry: string = `FOR p IN posts\n//     FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n//     ${selectQueries.join('\\n')}\n//     ${limit.aql}\n//     SORT p.added\n//     RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//   return database.query(aqlQry)\n//     .then((cursor) => cursor.all())\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\nexport const getPostsByArea = (\n  context: ApiContext,\n  latitude: number,\n  longitude: number,\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByArea';\n  const {databaseName, fields, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatLatitude: number = parseNum(latitude);\n  const formatLongitude: number = parseNum(longitude);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  selectQueries.push(`LET distance = DISTANCE(\n    ${formatLatitude},\n    ${formatLongitude},\n    NOT_NULL(p.latitude, 0),\n    NOT_NULL(p.longitude, 0))\n  `);\n  selectObjects.push('distance:distance');\n\n  const aqlQry: string = `FOR p IN posts\n    ${selectQueries.join('\\n')}\n    FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parentId == null\n    ${limit.aql}\n    SORT distance, p.added\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as PostType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as PostType[];\n    });\n};\n\n// export const getPostsByGroup = (\n//   context: ApiContext,\n//   groupId: string,\n//   options?: PostOptions\n// ): Promise<PostType[]> => {\n//   // const action: string = 'getListByGroup';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n\n//   // Group id\n//   const formatGroupId: string = parseId(groupId);\n//   const db = database;\n//   const aqlQry: string = `FOR u, g IN INBOUND ${formatGroupId} hasGroup\n//       FILTER u._key == ${sessionId}\n//       RETURN g`;\n\n//   return db.query(aqlQry)\n//     .then((cursor) => cursor.all())\n//     .then((groups: GroupType[] = []) => {\n//       if(groups.length) {\n//         const {limit, type} = parsePostOptions(options);\n//         const postAqlQry: string = `FOR p IN posts\n//           FILTER p.type == \"${type}\" && p.groupId == \"${formatGroupId}\" && p.parent == null\n//           ${selectQueries.join('\\n')}\n//           ${limit.aql}\n//           SORT p.added\n//           RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//         return db.query(postAqlQry)\n//           .then((cursor) => cursor.all())\n//           .catch((error: Error) => {\n//             throw error;\n//           });\n//       }\n\n//       return [];\n//     })\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\nexport const getPostsByLatest = (context: ApiContext, options?: PostOptions): Promise<PostType[]> => {\n  // const action: string = 'getListByLatest';\n  const {databaseName, fields, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: string = `FOR p IN posts\n    FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT p.added DESC\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getPostsByReactions = (\n  context: ApiContext,\n  reactions: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByReactions';\n  const {databaseName, fields, session: {userId: sessionId}} = context;\n  const {latitude, limit, longitude, type} = parsePostOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatReactions: string = JSON.stringify(reactions.map((reaction) => parseChar(reaction, 32).toLowerCase()));\n  const sortBy: string[] = [];\n  const filters: string[] = [`p.type == \"${type}\"`, 'p.privacy == \"public\"'];\n  const formatLatitude: number = parseNum(latitude);\n  const formatLongitude: number = parseNum(longitude);\n\n  if(formatLatitude && formatLongitude) {\n    selectQueries.push(`LET distance = DISTANCE(\n      ${formatLatitude},\n      ${formatLongitude},\n      NOT_NULL(p.latitude, 0),\n      NOT_NULL(p.longitude, 0))\n    `);\n    selectObjects.push('distance:distance');\n    sortBy.push('distance');\n  }\n\n  if(reactions.length) {\n    sortBy.push('matchedTags DESC');\n    selectQueries.push(`LET matchedReactions = LENGTH(\n      FOR mr IN reactions\n      FILTER mr.matched == true\n      RETURN mr\n    )`);\n    selectObjects.push('matchedReactions:matchedReactions');\n    filters.push('matchedReactions > 0');\n  }\n\n  sortBy.push('p.added DESC');\n  selectObjects.push('reactions:reactions');\n\n  // Get data from database\n  const aqlQry: string = `FOR p, r IN OUTBOUND \"${formatSessionId}\" hasReaction\n    LET reactions = (\n      FOR reaction, hr IN 1..1 INBOUND p isTagged\n      LET matched = LENGTH(${formatReactions}) > 0 && POSITION(${formatReactions}, reaction.name)\n      SORT reaction.name\n      RETURN MERGE(reaction, {matched:matched})\n    )\n    ${selectQueries.join('\\n')}\n    FILTER ${filters.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as PostType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as PostType[];\n    });\n};\n\nexport const getPostsByTags = (\n  context: ApiContext,\n  tags: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByTags';\n  const {databaseName, fields, session: {userId: sessionId}} = context;\n  const {latitude, limit, longitude, type} = parsePostOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const formatTagNames: string = JSON.stringify(tags.map((tag) => parseChar(tag, 32).toLowerCase()));\n  const sortBy: string[] = [];\n  const filters: string[] = [`p.type == \"${type}\"`, 'p.privacy == \"public\"'];\n  const formatLatitude: number = parseNum(latitude);\n  const formatLongitude: number = parseNum(longitude);\n\n  if(formatLatitude && formatLongitude) {\n    selectQueries.push(`LET distance = DISTANCE(\n      ${formatLatitude},\n      ${formatLongitude},\n      NOT_NULL(p.latitude, 0),\n      NOT_NULL(p.longitude, 0))\n    `);\n    selectObjects.push('distance:distance');\n    sortBy.push('distance');\n  }\n\n  if(tags.length) {\n    sortBy.push('matchedTags DESC');\n    selectQueries.push(`LET matchedTags = LENGTH(\n      FOR t IN tags\n      FILTER t.matched == true\n      RETURN t\n    )`);\n    selectObjects.push('matchedTags:matchedTags');\n    filters.push('matchedTags > 0');\n  }\n\n  sortBy.push('p.added DESC');\n  selectObjects.push('tags:tags');\n\n  const aqlQry: string = `FOR p IN posts\n    LET tags = (\n      FOR tag, it IN 1..1 INBOUND p isTagged\n      LET matched = LENGTH(${formatTagNames}) > 0 && POSITION(${formatTagNames}, tag.name)\n      SORT tag.name\n      RETURN MERGE(tag, {matched:matched})\n    )\n    ${selectQueries.join('\\n')}\n    FILTER ${filters.join(' && ')}\n    ${limit.aql}\n    SORT ${sortBy.join(', ')}\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as PostType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as PostType[];\n    });\n};\n\nexport const getPostsByUser = (context: ApiContext, userId: string, options?: PostOptions): Promise<PostType[]> => {\n  const action: string = 'getPostsByUser';\n  const {databaseName, fields, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatUserId: string = parseId(userId);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: string = `FOR p IN posts\n    FILTER p.userId == \"${formatUserId}\" && p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT p.added\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as PostType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as PostType[];\n    });\n};\n\nexport const getPostComments = (context: ApiContext, postId: string, options?: PostOptions): Promise<PostType[]> => {\n  const action: string = 'getPostComments';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatItemId: string = parseId(postId);\n\n  // Get the parent post to get restrictions\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n    FILTER p.type == ${type} && p._key == ${formatItemId}\n    LIMIT 1\n    RETURN p`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((post: PostType) => {\n      const {\n        _key,\n        groupId,\n        privacy = 'default'\n      }: PostType = post;\n\n      // Query based on privacy level\n      let privacyAqlQry: string;\n\n      if(groupId && privacy === 'group') {\n        privacyAqlQry = `FOR p IN posts\n          FOR user IN users\n          FILTER p.parent == \"${_key}\" && user._key == p.userId\n          LET reactions = (\n            FOR post, r IN INBOUND p._id reactions\n            COLLECT reactionName = r.value INTO reactionItems\n            RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n          )\n          FOR group IN groups\n          FILTER group._key == p.groupId\n          FOR u, e IN OUTBOUND group._id hasConnection\n          FILTER u._key == \"${sessionId}\"\n          SORT p.added\n          ${limit.aql}\n          RETURN MERGE(p, {user: user, reactions: reactions})`;\n      } else if(privacy === 'public') {\n        privacyAqlQry = `FOR p IN posts\n          FOR user IN users\n          FILTER p.parent == \"${_key}\" && user._key == p.userId\n          LET reactions = (\n            FOR post, r IN INBOUND p._id reactions\n            COLLECT reactionName = r.value INTO reactionItems\n            RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n          )\n          SORT p.added\n          ${limit.aql}\n          RETURN MERGE(p, {user: user, reactions: reactions})`;\n      }\n\n      if(privacyAqlQry) {\n        return useDb(databaseName).query(privacyAqlQry)\n          .then((cursor) => cursor.all() as unknown as PostType[])\n          .catch((error: Error) => {\n            logError({\n              action,\n              category: eventCategory,\n              label: ErrorTypes.DATABASE_ERROR\n            }, error, context);\n\n            return [] as PostType[];\n          });\n      }\n\n      return [] as PostType[];\n    })\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as PostType[];\n    });\n};\n\nexport const addPost = async (\n  context: ApiContext,\n  {\n    content = '',\n    endDate,\n    groupId = '',\n    location,\n    latitude,\n    longitude,\n    name = '',\n    parentId = null,\n    privacy = 'public',\n    tags = [],\n    startDate,\n    type = 'default'\n  }: PostInputType\n): Promise<PostType> => {\n  const action: string = 'addPost';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const now = Date.now();\n  const postId = createHash(`post-${sessionId}`);\n  const insert: PostType = {\n    _id: `posts/${postId}`,\n    _key: postId,\n    added: now,\n    content: parseString(content, MAX_CONTENT_LENGTH),\n    endDate: endDate ? parseNum(endDate, 13) : undefined,\n    groupId: groupId ? parseId(groupId) : undefined,\n    latitude: latitude !== undefined ? parseNum(latitude) : undefined,\n    location: location ? parseString(location, 160) : undefined,\n    longitude: longitude !== undefined ? parseNum(longitude) : undefined,\n    modified: now,\n    name: parseString(name, 160),\n    parentId: parentId ? parseId(parentId) : undefined,\n    privacy: parseVarChar(privacy, 16),\n    startDate: startDate ? parseNum(startDate, 13) : undefined,\n    type: parseChar(type, 32),\n    userId: sessionId\n  };\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN posts RETURN NEW`;\n\n  try {\n    const savedPost: PostType = await useDb(databaseName).query(aqlQry)\n      .then((cursor) => cursor.next())\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context));\n    const {_id: postDocId} = savedPost;\n    const contentTagNames = await extractTags(insert.content);\n\n    if(tags.length || contentTagNames.length) {\n      const userTags = (await getTagsByName(context, tags.map(({name}) => name)))\n        .map((tag) => ({...tag, tagBy: sessionId}));\n      const contentTags = (await getTagsByName(context, contentTagNames))\n        .map((tag) => ({...tag, tagBy: 'extract'}));\n      const updatedTags: TagType[] = await updateTagsInItem(\n        context,\n        {\n          itemDocId: postDocId,\n          tags: [...contentTags, ...userTags]\n        }\n      );\n\n      return {\n        ...savedPost,\n        tags: updatedTags\n      };\n    }\n\n    return savedPost;\n  } catch(error) {\n    throw error;\n  }\n};\n\nexport const updatePost = async (context: ApiContext, post: PostInputType): Promise<PostType> => {\n  const action: string = 'updatePost';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const now: number = Date.now();\n  const parsedPost = parsePost(post);\n  const {\n    postId,\n    tags = []\n  } = parsedPost;\n\n  const update: PostType = {\n    ...parsedPost,\n    modified: now\n  };\n\n  if(!postId) {\n    return logException({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ID\n    }, {});\n  }\n\n  const insert = {\n    ...update,\n    _key: postId,\n    added: now,\n    userId: sessionId\n  };\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${postId}, userId: ${sessionId}}\n    INSERT ${insert}\n    UPDATE ${update}\n    IN posts RETURN NEW`;\n\n  try {\n    const updatedPost: PostType = await useDb(databaseName)\n      .query(aqlQry)\n      .then((cursor) => cursor.next())\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, {}));\n    const {_id: updatedPostId} = updatedPost;\n    const contentTagNames = await extractTags(insert.content);\n\n    if(tags?.length || contentTagNames?.length) {\n      const userTags = tags?.length ? (await getTagsByName(context, tags.map(({name}) => name)))\n        .map((tag) => ({...tag, tagBy: sessionId})) : [];\n      const contentTags = contentTagNames?.length ? (await getTagsByName(context, contentTagNames))\n        .map((tag) => ({...tag, tagBy: 'extract'})) : [];\n      const updatedTags: TagType[] = await updateTagsInItem(\n        context,\n        {\n          itemDocId: updatedPostId,\n          tags: [...contentTags, ...userTags]\n        }\n      );\n\n      return {\n        ...updatedPost,\n        tags: updatedTags\n      };\n    }\n\n    return updatedPost;\n  } catch(error) {\n    throw error;\n  }\n};\n\nexport const deletePost = async (context: ApiContext, postDocId: string): Promise<PostType> => {\n  const action: string = 'deletePost';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const formatPostId: string = parseArangoId(postDocId);\n\n  if(!formatPostId) {\n    return logException({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ID\n    }, {});\n  }\n\n  const edgeAqlQry: AqlQuery = aql`FOR t IN isTagged\n  FILTER t._to == ${formatPostId}\n  REMOVE t IN isTagged`;\n\n  await useDb(databaseName).query(edgeAqlQry)\n    .catch((error: Error) => {\n      throw error;\n    });\n\n  const fileAqlQry: AqlQuery = aql`FOR f IN hasFile\n    FILTER f._to == ${formatPostId}\n    REMOVE f IN hasFile`;\n\n  await useDb(databaseName).query(fileAqlQry)\n    .catch((error: Error) => {\n      throw error;\n    });\n\n  const aqlQry = aql`FOR p IN posts\n      FILTER p._id == ${formatPostId} && p.userId == ${sessionId}\n      LIMIT 1\n      REMOVE p IN posts\n      RETURN OLD`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const createPostEdge = (\n  context: ApiContext,\n  postDocId: string,\n  itemDocId: string,\n  edgeType: string = 'isPosted',\n  props = {}\n): Promise<FileType> => {\n  const action = 'createPostEdge';\n  const {databaseName} = context;\n  const edgeCollection: EdgeCollection = useDb(databaseName).collection(edgeType);\n  const formatPostId: string = parseArangoId(postDocId);\n  const formatDocId: string = parseArangoId(itemDocId);\n\n  if(!formatDocId || !formatPostId) {\n    return Promise.reject(new Error(ErrorTypes.INVALID_ID));\n  }\n\n  const edgeId: string = createHash(`postEdge-${formatPostId}-${formatDocId}`);\n  const edge = {\n    _from: formatPostId,\n    _key: edgeId,\n    _to: formatDocId,\n    added: Date.now(),\n    ...props\n  };\n\n  return edgeCollection.save(edge, {returnNew: true})\n    .catch((error: Error) =>\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, {}));\n};\n"],
  "mappings": "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",
  "names": ["createHash", "parseArangoId", "parseChar", "parseId", "parseNum", "parseString", "parseVarChar", "aql", "extractTags", "getTagsByName", "updateTagsInItem", "parsePost", "ErrorTypes", "logError", "logException", "getLimit", "useDb", "MAX_CONTENT_LENGTH", "eventCategory", "parsePostOptions", "options", "from", "latitude", "longitude", "to", "type", "getPostOptional", "fields", "sessionId", "selects", "field", "getPost", "context", "postId", "action", "databaseName", "formatItemId", "database", "selectObjects", "selectQueries", "aqlQry", "cursor", "post", "postDocId", "userId", "groupId", "privacy", "privacyAqlQry", "error", "getPostsByArea", "limit", "formatLatitude", "formatLongitude", "getPostsByLatest", "getPostsByReactions", "reactions", "formatSessionId", "formatReactions", "reaction", "sortBy", "filters", "getPostsByTags", "tags", "formatTagNames", "tag", "getPostsByUser", "formatUserId", "getPostComments", "_key", "addPost", "content", "endDate", "location", "name", "parentId", "startDate", "now", "insert", "savedPost", "contentTagNames", "userTags", "contentTags", "updatedTags", "updatePost", "parsedPost", "update", "updatedPost", "updatedPostId", "deletePost", "formatPostId", "edgeAqlQry", "fileAqlQry", "createPostEdge", "itemDocId", "edgeType", "props", "edgeCollection", "formatDocId", "edgeId", "edge"]
}
