@nlabs/reaktor 0.9.0 → 0.10.0

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