@nlabs/reaktor 0.10.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 (314) hide show
  1. package/.env +1 -0
  2. package/.env.example +1 -0
  3. package/DATABASE_I18N_GUIDE.md +434 -0
  4. package/TEST_UTILITIES_GUIDE.md +360 -0
  5. package/coverage/index.html +46 -76
  6. package/index.js +1 -1
  7. package/jest.config.js +17 -0
  8. package/jest.setup.js +36 -0
  9. package/lex.config.cjs +2 -2
  10. package/lib/actions/apps.js +17 -239
  11. package/lib/actions/connections.js +6 -89
  12. package/lib/actions/content.js +17 -0
  13. package/lib/actions/conversations.js +19 -336
  14. package/lib/actions/dynamodb.js +2 -150
  15. package/lib/actions/email.js +2 -152
  16. package/lib/actions/files.js +5 -283
  17. package/lib/actions/groups.js +19 -259
  18. package/lib/actions/images.js +31 -700
  19. package/lib/actions/index.js +2 -66
  20. package/lib/actions/ios.js +9 -162
  21. package/lib/actions/locations.js +7 -122
  22. package/lib/actions/messages.js +21 -196
  23. package/lib/actions/notifications.js +2 -59
  24. package/lib/actions/payments.js +11 -464
  25. package/lib/actions/posts.js +75 -527
  26. package/lib/actions/profiles.js +8 -0
  27. package/lib/actions/reactions.js +25 -312
  28. package/lib/actions/s3.js +2 -133
  29. package/lib/actions/search.js +5 -90
  30. package/lib/actions/sms.js +2 -108
  31. package/lib/actions/statistics.js +6 -60
  32. package/lib/actions/subscriptions.js +12 -0
  33. package/lib/actions/tags.js +19 -287
  34. package/lib/actions/users.js +64 -764
  35. package/lib/actions/websockets.js +14 -158
  36. package/lib/adapters/arangoAdapter.js +2 -46
  37. package/lib/adapters/contentAdapter.js +2 -0
  38. package/lib/adapters/fileAdapter.js +2 -76
  39. package/lib/adapters/imageAdapter.js +2 -40
  40. package/lib/adapters/messageAdapter.js +2 -49
  41. package/lib/adapters/postAdapter.js +2 -70
  42. package/lib/adapters/reaktorAdapter.js +2 -44
  43. package/lib/adapters/tagAdapter.js +2 -50
  44. package/lib/adapters/userAdapter.js +2 -115
  45. package/lib/config.js +2 -125
  46. package/lib/handlers/graphqlHandler.js +2 -0
  47. package/lib/index.js +2 -66
  48. package/lib/lambdas/actions/websockets.js +14 -116
  49. package/lib/lambdas/authorizer.js +2 -67
  50. package/lib/lambdas/connection.js +2 -91
  51. package/lib/lambdas/utils/message.js +2 -42
  52. package/lib/lambdas/utils/websocket.js +2 -105
  53. package/lib/mocks/conversation.js +2 -35
  54. package/lib/mocks/file.js +2 -38
  55. package/lib/mocks/group.js +2 -47
  56. package/lib/mocks/image.js +2 -44
  57. package/lib/mocks/post.js +2 -55
  58. package/lib/mocks/tag.js +2 -37
  59. package/lib/mocks/user.js +2 -88
  60. package/lib/mutations/index.js +2 -26
  61. package/lib/mutations/locations.integration.js +2 -0
  62. package/lib/mutations/locations.js +2 -44
  63. package/lib/mutations/messages.integration.js +2 -0
  64. package/lib/mutations/messages.js +2 -86
  65. package/lib/mutations/posts.integration.js +2 -0
  66. package/lib/mutations/posts.js +2 -53
  67. package/lib/mutations/profiles.integration.js +2 -0
  68. package/lib/mutations/profiles.js +2 -0
  69. package/lib/mutations/reactions.integration.js +2 -0
  70. package/lib/mutations/reactions.js +2 -51
  71. package/lib/mutations/statistics.integration.js +2 -0
  72. package/lib/mutations/statistics.js +2 -39
  73. package/lib/mutations/subscriptions.integration.js +2 -0
  74. package/lib/mutations/subscriptions.js +2 -56
  75. package/lib/mutations/tags.integration.js +2 -0
  76. package/lib/mutations/tags.js +2 -120
  77. package/lib/mutations/users.integration.js +2 -0
  78. package/lib/mutations/users.js +2 -116
  79. package/lib/objectTypes/app.js +2 -173
  80. package/lib/objectTypes/bankAccount.js +2 -76
  81. package/lib/objectTypes/connection.js +2 -48
  82. package/lib/objectTypes/conversation.js +2 -77
  83. package/lib/objectTypes/creditCard.js +2 -86
  84. package/lib/objectTypes/document.js +2 -46
  85. package/lib/objectTypes/error.js +2 -46
  86. package/lib/objectTypes/external.js +2 -74
  87. package/lib/objectTypes/file.js +2 -100
  88. package/lib/objectTypes/filter.js +2 -43
  89. package/lib/objectTypes/group.js +2 -123
  90. package/lib/objectTypes/iapSubscription.js +2 -40
  91. package/lib/objectTypes/image.js +2 -129
  92. package/lib/objectTypes/index.js +2 -68
  93. package/lib/objectTypes/location.js +2 -109
  94. package/lib/objectTypes/message.js +2 -96
  95. package/lib/objectTypes/passcode.js +2 -42
  96. package/lib/objectTypes/plan.js +2 -95
  97. package/lib/objectTypes/post.js +2 -125
  98. package/lib/objectTypes/profile.js +2 -0
  99. package/lib/objectTypes/reaction.js +2 -61
  100. package/lib/objectTypes/relation.js +2 -49
  101. package/lib/objectTypes/search.js +2 -72
  102. package/lib/objectTypes/statistics.js +2 -39
  103. package/lib/objectTypes/subscription.js +2 -117
  104. package/lib/objectTypes/tag.js +2 -65
  105. package/lib/objectTypes/user.js +2 -144
  106. package/lib/queries/index.js +2 -33
  107. package/lib/queries/locations.integration.js +2 -0
  108. package/lib/queries/locations.js +2 -45
  109. package/lib/queries/messages.integration.js +2 -0
  110. package/lib/queries/messages.js +2 -52
  111. package/lib/queries/posts.integration.js +2 -0
  112. package/lib/queries/posts.js +2 -154
  113. package/lib/queries/reactions.integration.js +2 -0
  114. package/lib/queries/reactions.js +2 -56
  115. package/lib/queries/statistics.js +2 -39
  116. package/lib/queries/subscriptions.js +2 -44
  117. package/lib/queries/tags.integration.js +2 -0
  118. package/lib/queries/tags.js +2 -75
  119. package/lib/queries/users.integration.js +2 -0
  120. package/lib/queries/users.js +2 -64
  121. package/lib/templates/email/layout.js +3 -25
  122. package/lib/templates/email/passwordForgot.js +3 -25
  123. package/lib/templates/email/passwordRecovery.js +3 -25
  124. package/lib/templates/email/verifyEmail.js +3 -25
  125. package/lib/templates/email/welcome.js +3 -25
  126. package/lib/templates/sms/passwordForgot.js +2 -24
  127. package/lib/templates/sms/passwordRecovery.js +2 -24
  128. package/lib/templates/sms/verifyEmail.js +2 -24
  129. package/lib/templates/sms/verifyPhone.js +2 -24
  130. package/lib/templates/sms/welcome.js +2 -24
  131. package/lib/types/apps.types.js +2 -32
  132. package/lib/types/arangodb.types.js +1 -16
  133. package/lib/types/auth.types.js +1 -16
  134. package/lib/types/connections.types.js +1 -16
  135. package/lib/types/content.types.js +1 -0
  136. package/lib/types/conversations.types.js +1 -16
  137. package/lib/types/email.types.js +1 -16
  138. package/lib/types/error.types.js +2 -44
  139. package/lib/types/files.types.js +1 -16
  140. package/lib/types/google.types.js +1 -16
  141. package/lib/types/groups.types.js +1 -16
  142. package/lib/types/images.types.js +1 -16
  143. package/lib/types/index.js +2 -60
  144. package/lib/types/locations.types.js +1 -16
  145. package/lib/types/messages.types.js +1 -16
  146. package/lib/types/notifications.types.js +1 -16
  147. package/lib/types/payments.types.js +1 -16
  148. package/lib/types/posts.types.js +1 -16
  149. package/lib/types/profiles.types.js +1 -0
  150. package/lib/types/statistics.types.js +1 -16
  151. package/lib/types/tags.types.js +1 -16
  152. package/lib/types/users.types.js +1 -16
  153. package/lib/types/websockets.types.js +1 -16
  154. package/lib/utils/adapterUtils.js +2 -45
  155. package/lib/utils/analyticsUtils.js +2 -72
  156. package/lib/utils/arangodbUtils.js +5 -163
  157. package/lib/utils/authUtils.js +2 -0
  158. package/lib/utils/contextUtils.js +2 -0
  159. package/lib/utils/dbI18n.example.js +6 -0
  160. package/lib/utils/dbI18n.js +2 -0
  161. package/lib/utils/googleTranslate.js +2 -0
  162. package/lib/utils/graphqlUtils.js +2 -0
  163. package/lib/utils/index.js +2 -30
  164. package/lib/utils/languageDetection.js +2 -0
  165. package/lib/utils/localeUtils.example.js +2 -0
  166. package/lib/utils/localeUtils.js +2 -0
  167. package/lib/utils/middlewareUtils.js +2 -0
  168. package/lib/utils/sessionUtils.js +2 -0
  169. package/lib/utils/stripeUtils.js +2 -43
  170. package/lib/utils/templateUtils.js +2 -0
  171. package/lib/utils/testUtils.js +2 -0
  172. package/lib/utils/translationQueue.example.js +2 -0
  173. package/lib/utils/translationQueue.js +2 -0
  174. package/package.json +27 -24
  175. package/.prettierrc.js +0 -4
  176. package/lib/actions/apps.d.ts +0 -25
  177. package/lib/actions/connections.d.ts +0 -4
  178. package/lib/actions/conversations.d.ts +0 -14
  179. package/lib/actions/dynamodb.d.ts +0 -8
  180. package/lib/actions/email.d.ts +0 -5
  181. package/lib/actions/files.d.ts +0 -19
  182. package/lib/actions/groups.d.ts +0 -14
  183. package/lib/actions/images.d.ts +0 -26
  184. package/lib/actions/index.d.ts +0 -23
  185. package/lib/actions/ios.d.ts +0 -7
  186. package/lib/actions/locations.d.ts +0 -6
  187. package/lib/actions/messages.d.ts +0 -14
  188. package/lib/actions/notifications.d.ts +0 -5
  189. package/lib/actions/payments.d.ts +0 -10
  190. package/lib/actions/personas.d.ts +0 -3
  191. package/lib/actions/personas.js +0 -110
  192. package/lib/actions/posts.d.ts +0 -22
  193. package/lib/actions/reactions.d.ts +0 -30
  194. package/lib/actions/s3.d.ts +0 -7
  195. package/lib/actions/search.d.ts +0 -3
  196. package/lib/actions/sms.d.ts +0 -9
  197. package/lib/actions/statistics.d.ts +0 -3
  198. package/lib/actions/subscription.d.ts +0 -7
  199. package/lib/actions/subscription.js +0 -208
  200. package/lib/actions/tags.d.ts +0 -34
  201. package/lib/actions/users.d.ts +0 -72
  202. package/lib/actions/websockets.d.ts +0 -20
  203. package/lib/adapters/arangoAdapter.d.ts +0 -2
  204. package/lib/adapters/fileAdapter.d.ts +0 -3
  205. package/lib/adapters/imageAdapter.d.ts +0 -2
  206. package/lib/adapters/messageAdapter.d.ts +0 -2
  207. package/lib/adapters/postAdapter.d.ts +0 -2
  208. package/lib/adapters/reaktorAdapter.d.ts +0 -6
  209. package/lib/adapters/tagAdapter.d.ts +0 -2
  210. package/lib/adapters/userAdapter.d.ts +0 -2
  211. package/lib/config.d.ts +0 -20
  212. package/lib/index.d.ts +0 -12
  213. package/lib/lambdas/actions/websockets.d.ts +0 -7
  214. package/lib/lambdas/authorizer.d.ts +0 -20
  215. package/lib/lambdas/connection.d.ts +0 -12
  216. package/lib/lambdas/utils/message.d.ts +0 -1
  217. package/lib/lambdas/utils/websocket.d.ts +0 -7
  218. package/lib/mocks/conversation.d.ts +0 -8
  219. package/lib/mocks/file.d.ts +0 -11
  220. package/lib/mocks/group.d.ts +0 -17
  221. package/lib/mocks/image.d.ts +0 -3
  222. package/lib/mocks/post.d.ts +0 -38
  223. package/lib/mocks/tag.d.ts +0 -2
  224. package/lib/mocks/user.d.ts +0 -4
  225. package/lib/mutations/index.d.ts +0 -3
  226. package/lib/mutations/locations.d.ts +0 -2
  227. package/lib/mutations/messages.d.ts +0 -2
  228. package/lib/mutations/personas.d.ts +0 -2
  229. package/lib/mutations/personas.js +0 -100
  230. package/lib/mutations/posts.d.ts +0 -2
  231. package/lib/mutations/reactions.d.ts +0 -2
  232. package/lib/mutations/statistics.d.ts +0 -2
  233. package/lib/mutations/subscriptions.d.ts +0 -2
  234. package/lib/mutations/tags.d.ts +0 -2
  235. package/lib/mutations/users.d.ts +0 -1
  236. package/lib/objectTypes/app.d.ts +0 -3
  237. package/lib/objectTypes/bankAccount.d.ts +0 -1
  238. package/lib/objectTypes/connection.d.ts +0 -1
  239. package/lib/objectTypes/conversation.d.ts +0 -2
  240. package/lib/objectTypes/creditCard.d.ts +0 -1
  241. package/lib/objectTypes/document.d.ts +0 -1
  242. package/lib/objectTypes/error.d.ts +0 -1
  243. package/lib/objectTypes/external.d.ts +0 -1
  244. package/lib/objectTypes/file.d.ts +0 -2
  245. package/lib/objectTypes/filter.d.ts +0 -1
  246. package/lib/objectTypes/group.d.ts +0 -3
  247. package/lib/objectTypes/iapSubscription.d.ts +0 -1
  248. package/lib/objectTypes/image.d.ts +0 -2
  249. package/lib/objectTypes/index.d.ts +0 -24
  250. package/lib/objectTypes/location.d.ts +0 -2
  251. package/lib/objectTypes/message.d.ts +0 -2
  252. package/lib/objectTypes/passcode.d.ts +0 -1
  253. package/lib/objectTypes/persona.d.ts +0 -3
  254. package/lib/objectTypes/persona.js +0 -87
  255. package/lib/objectTypes/plan.d.ts +0 -2
  256. package/lib/objectTypes/post.d.ts +0 -2
  257. package/lib/objectTypes/reaction.d.ts +0 -2
  258. package/lib/objectTypes/relation.d.ts +0 -1
  259. package/lib/objectTypes/search.d.ts +0 -1
  260. package/lib/objectTypes/statistics.d.ts +0 -1
  261. package/lib/objectTypes/subscription.d.ts +0 -2
  262. package/lib/objectTypes/tag.d.ts +0 -2
  263. package/lib/objectTypes/user.d.ts +0 -4
  264. package/lib/queries/index.d.ts +0 -3
  265. package/lib/queries/locations.d.ts +0 -2
  266. package/lib/queries/messages.d.ts +0 -2
  267. package/lib/queries/posts.d.ts +0 -2
  268. package/lib/queries/reactions.d.ts +0 -2
  269. package/lib/queries/statistics.d.ts +0 -2
  270. package/lib/queries/subscriptions.d.ts +0 -2
  271. package/lib/queries/tags.d.ts +0 -2
  272. package/lib/queries/users.d.ts +0 -1
  273. package/lib/templates/email/layout.d.ts +0 -2
  274. package/lib/templates/email/passwordForgot.d.ts +0 -2
  275. package/lib/templates/email/passwordRecovery.d.ts +0 -2
  276. package/lib/templates/email/verifyEmail.d.ts +0 -2
  277. package/lib/templates/email/welcome.d.ts +0 -2
  278. package/lib/templates/sms/passwordForgot.d.ts +0 -2
  279. package/lib/templates/sms/passwordRecovery.d.ts +0 -2
  280. package/lib/templates/sms/verifyEmail.d.ts +0 -2
  281. package/lib/templates/sms/verifyPhone.d.ts +0 -2
  282. package/lib/templates/sms/welcome.d.ts +0 -2
  283. package/lib/types/apps.types.d.ts +0 -46
  284. package/lib/types/arangodb.types.d.ts +0 -34
  285. package/lib/types/auth.types.d.ts +0 -9
  286. package/lib/types/connections.types.d.ts +0 -5
  287. package/lib/types/conversations.types.d.ts +0 -27
  288. package/lib/types/email.types.d.ts +0 -13
  289. package/lib/types/error.types.d.ts +0 -20
  290. package/lib/types/files.types.d.ts +0 -23
  291. package/lib/types/google.types.d.ts +0 -29
  292. package/lib/types/groups.types.d.ts +0 -18
  293. package/lib/types/images.types.d.ts +0 -52
  294. package/lib/types/index.d.ts +0 -20
  295. package/lib/types/locations.types.d.ts +0 -18
  296. package/lib/types/messages.types.d.ts +0 -16
  297. package/lib/types/notifications.types.d.ts +0 -19
  298. package/lib/types/payments.types.d.ts +0 -109
  299. package/lib/types/personas.types.d.ts +0 -32
  300. package/lib/types/personas.types.js +0 -16
  301. package/lib/types/posts.types.d.ts +0 -28
  302. package/lib/types/statistics.types.d.ts +0 -3
  303. package/lib/types/tags.types.d.ts +0 -15
  304. package/lib/types/users.types.d.ts +0 -79
  305. package/lib/types/websockets.types.d.ts +0 -18
  306. package/lib/utils/adapterUtils.d.ts +0 -1
  307. package/lib/utils/analyticsUtils.d.ts +0 -21
  308. package/lib/utils/arangodbUtils.d.ts +0 -66
  309. package/lib/utils/auth.d.ts +0 -21
  310. package/lib/utils/auth.js +0 -57
  311. package/lib/utils/index.d.ts +0 -5
  312. package/lib/utils/session.d.ts +0 -18
  313. package/lib/utils/session.js +0 -60
  314. package/lib/utils/stripeUtils.d.ts +0 -3
@@ -1,358 +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_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.types");
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(
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
33
  FOR u, e IN OUTBOUND group._id hasConnection
143
- FILTER u._key == "${sessionId}"
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) => {
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}
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}
203
56
  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},
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},
224
60
  NOT_NULL(p.latitude, 0),
225
61
  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(
62
+ `),u.push("distance:distance"),T.push("distance")),r.length&&(T.push("matchedTags DESC"),d.push(`LET matchedReactions = LENGTH(
233
63
  FOR mr IN reactions
234
64
  FILTER mr.matched == true
235
65
  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
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
243
67
  LET reactions = (
244
68
  FOR reaction, hr IN 1..1 INBOUND p isTagged
245
- LET matched = LENGTH(${formatReactions}) > 0 && POSITION(${formatReactions}, reaction.name)
69
+ LET matched = LENGTH(${E}) > 0 && POSITION(${E}, reaction.name)
246
70
  SORT reaction.name
247
71
  RETURN MERGE(reaction, {matched:matched})
248
72
  )
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},
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},
276
80
  NOT_NULL(p.latitude, 0),
277
81
  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(
82
+ `),u.push("distance:distance"),E.push("distance")),r.length&&(E.push("matchedTags DESC"),d.push(`LET matchedTags = LENGTH(
285
83
  FOR t IN tags
286
84
  FILTER t.matched == true
287
85
  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
86
+ )`),u.push("matchedTags:matchedTags"),T.push("matchedTags > 0")),E.push("p.added DESC"),u.push("tags:tags");const g=`FOR p IN posts
295
87
  LET tags = (
296
88
  FOR tag, it IN 1..1 INBOUND p isTagged
297
- LET matched = LENGTH(${formatTagNames}) > 0 && POSITION(${formatTagNames}, tag.name)
89
+ LET matched = LENGTH(${I}) > 0 && POSITION(${I}, tag.name)
298
90
  SORT tag.name
299
91
  RETURN MERGE(tag, {matched:matched})
300
92
  )
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}
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}
325
103
  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}
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}
343
106
  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
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
354
108
  FOR user IN users
355
- FILTER p.parent == "${_key}" && user._key == p.userId
109
+ FILTER p.parent == "${u}" && user._key == p.userId
356
110
  LET reactions = (
357
111
  FOR post, r IN INBOUND p._id reactions
358
112
  COLLECT reactionName = r.value INTO reactionItems
@@ -361,235 +115,29 @@ const getPostComments = (context, postId, options) => {
361
115
  FOR group IN groups
362
116
  FILTER group._key == p.groupId
363
117
  FOR u, e IN OUTBOUND group._id hasConnection
364
- FILTER u._key == "${sessionId}"
118
+ FILTER u._key == "${n}"
365
119
  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
120
+ ${p.aql}
121
+ RETURN MERGE(p, {user: user, reactions: reactions})`:I==="public"&&(E=`FOR p IN posts
370
122
  FOR user IN users
371
- FILTER p.parent == "${_key}" && user._key == p.userId
123
+ FILTER p.parent == "${u}" && user._key == p.userId
372
124
  LET reactions = (
373
125
  FOR post, r IN INBOUND p._id reactions
374
126
  COLLECT reactionName = r.value INTO reactionItems
375
127
  RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
376
128
  )
377
129
  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}
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}
549
140
  LIMIT 1
550
141
  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.types';\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": ";;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAIA,mBAEO;AACP,sBAAkB;AAGlB,kBAA2D;AAC3D,yBAAwB;AACxB,mBAAyB;AACzB,4BAAqC;AACrC,2BAAuB;AAKvB,MAAM,qBAA6B;AACnC,MAAM,gBAAwB;AAEvB,MAAM,mBAAmB,CAAC,UAAuB,CAAC,MAAM;AAC7D,QAAM;AAAA,IACJ,OAAO;AAAA,IACP,WAAW;AAAA,IACX,YAAY;AAAA,IACZ,KAAK;AAAA,IACL,OAAO;AAAA,EACT,IAAI;AAEJ,SAAO;AAAA,IACL,cAAU,uBAAS,UAAU,EAAE;AAAA,IAC/B,WAAO,+BAAS,MAAM,EAAE;AAAA,IACxB,eAAW,uBAAS,WAAW,EAAE;AAAA,IACjC,UAAM,wBAAU,MAAM,EAAE;AAAA,EAC1B;AACF;AAEO,MAAM,kBAAkB,CAAC,QAAkB,eAC/C,UAAU,CAAC,GAAG,OAAO,CAAC,SAAS,UAAkB;AAChD,UAAO,OAAO;AAAA,IACZ,KAAK,WAAW;AACd,cAAQ,QAAQ,KAAK;AAAA;AAAA,8EAEiD,SAAS;AAAA;AAAA;AAAA,WAG5E;AACH,cAAQ,QAAQ,KAAK,iBAAiB;AACtC,aAAO;AAAA,IACT;AAAA,IACA,KAAK,WAAW;AACd,cAAQ,QAAQ,KAAK;AAAA;AAAA,6EAEgD,SAAS;AAAA;AAAA;AAAA,WAG3E;AACH,cAAQ,QAAQ,KAAK,iBAAiB;AACtC,aAAO;AAAA,IACT;AAAA,IACA,KAAK,aAAa;AAChB,cAAQ,QAAQ,KAAK;AAAA;AAAA;AAAA;AAAA,UAInB;AACF,cAAQ,QAAQ,KAAK,qBAAqB;AAC1C,aAAO;AAAA,IACT;AAAA,IACA,KAAK,aAAa;AAChB,cAAQ,QAAQ,KAAK;AAAA;AAAA;AAAA;AAAA;AAAA,UAKnB;AACF,cAAQ,QAAQ,KAAK,qBAAqB;AAC1C,aAAO;AAAA,IACT;AAAA,IACA,KAAK,aAAa;AAChB,cAAQ,QAAQ,KAAK;AAAA;AAAA;AAAA;AAAA;AAAA,UAKnB;AACF,cAAQ,QAAQ,KAAK,qBAAqB;AAC1C,aAAO;AAAA,IACT;AAAA,IACA,SAAS;AACP,aAAO;AAAA,IACT;AAAA,EACF;AACF,GAAG,EAAC,SAAS,CAAC,GAAG,SAAS,CAAC,EAAC,CAAC;AAExB,MAAM,UAAU,OACrB,SACA,QACA,YACsB;AACtB,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACzD,QAAM,mBAAuB,sBAAQ,MAAM;AAC3C,QAAM,EAAC,KAAI,IAAI,iBAAiB,OAAO;AACvC,QAAM,KAAK;AACX,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,QAAQ,SAAS;AAC1F,QAAM,SAAmB;AAAA,uBACJ,YAAY,iBAAiB,IAAI;AAAA;AAAA;AAItD,SAAO,GAAG,MAAM,MAAM,EACnB,KAAK,CAAC,WAAW,OAAO,KAAK,CAAC,EAC9B,KAAK,CAAC,SAAmB;AACxB,UAAM;AAAA,MACJ,KAAK;AAAA,MACL;AAAA,MACA;AAAA,MACA,UAAU;AAAA,IACZ,IAAc;AAEd,QAAI;AAEJ,QAAG,WAAW,WAAW;AACvB,aAAO;AAAA,IACT;AAEA,QAAG,WAAW,YAAY,SAAS;AACjC,sBAAgB,qBAAqB,SAAS;AAAA,YAC1C,cAAc,KAAK,IAAI,CAAC;AAAA;AAAA;AAAA;AAAA,8BAIN,SAAS;AAAA;AAAA,6BAEV,cAAc,KAAK,IAAI,CAAC;AAAA,IAC/C,WAAU,YAAY,UAAU;AAC9B,sBAAgB,qBAAqB,SAAS;AAAA,YAC1C,cAAc,KAAK,IAAI,CAAC;AAAA;AAAA,6BAEP,cAAc,KAAK,IAAI,CAAC;AAAA,IAC/C;AAEA,QAAG,eAAe;AAChB,aAAO,GAAG,MAAM,aAAa,EAC1B,KAAK,CAAC,WAAW,OAAO,KAAK,CAAC,EAC9B,MAAM,CAAC,cAAiB,gCAAS;AAAA,QAChC;AAAA,QACA,UAAU;AAAA,QACV,OAAO,wBAAW;AAAA,MACpB,GAAG,OAAO,OAAO,CAAC;AAAA,IACtB;AAEA,WAAO,CAAC;AAAA,EACV,CAAC,EACA,MAAM,CAAC,cAAiB,gCAAS;AAAA,IAChC;AAAA,IACA,UAAU;AAAA,IACV,OAAO,wBAAW;AAAA,EACpB,GAAG,OAAO,OAAO,CAAC;AACtB;AAsBO,MAAM,iBAAiB,CAC5B,SACA,UACA,WACA,YACwB;AACxB,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACzD,QAAM,EAAC,OAAO,KAAI,IAAI,iBAAiB,OAAO;AAC9C,QAAM,qBAAyB,uBAAS,QAAQ;AAChD,QAAM,sBAA0B,uBAAS,SAAS;AAClD,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,QAAQ,SAAS;AAC1F,gBAAc,KAAK;AAAA,MACf,cAAc;AAAA,MACd,eAAe;AAAA;AAAA;AAAA,GAGlB;AACD,gBAAc,KAAK,mBAAmB;AAEtC,QAAM,SAAiB;AAAA,MACnB,cAAc,KAAK,IAAI,CAAC;AAAA,wBACN,IAAI;AAAA,MACtB,MAAM,GAAG;AAAA;AAAA,gCAEiB,cAAc,KAAK,IAAI,CAAC;AAEtD,SAAO,SAAS,MAAM,MAAM,EACzB,KAAK,CAAC,WAAW,OAAO,IAAI,CAA0B,EACtD,MAAM,CAAC,UAAiB;AACvB,wCAAS;AAAA,MACP;AAAA,MACA,UAAU;AAAA,MACV,OAAO,wBAAW;AAAA,IACpB,GAAG,OAAO,OAAO;AAEjB,WAAO,CAAC;AAAA,EACV,CAAC;AACL;AA4CO,MAAM,mBAAmB,CAAC,SAAqB,YAA+C;AAEnG,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACzD,QAAM,EAAC,OAAO,KAAI,IAAI,iBAAiB,OAAO;AAC9C,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,QAAQ,SAAS;AAC1F,QAAM,SAAiB;AAAA,wBACD,IAAI;AAAA,MACtB,cAAc,KAAK,IAAI,CAAC;AAAA,MACxB,MAAM,GAAG;AAAA;AAAA,gCAEiB,cAAc,KAAK,IAAI,CAAC;AAEtD,SAAO,SAAS,MAAM,MAAM,EACzB,KAAK,CAAC,WAAW,OAAO,IAAI,CAAC,EAC7B,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA,EACR,CAAC;AACL;AAEO,MAAM,sBAAsB,CACjC,SACA,YAAsB,CAAC,GACvB,YACwB;AACxB,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACzD,QAAM,EAAC,UAAU,OAAO,WAAW,KAAI,IAAI,iBAAiB,OAAO;AACnE,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,QAAQ,SAAS;AAC1F,QAAM,kBAA0B,SAAS,SAAS;AAClD,QAAM,kBAA0B,KAAK,UAAU,UAAU,IAAI,CAAC,iBAAa,wBAAU,UAAU,EAAE,EAAE,YAAY,CAAC,CAAC;AACjH,QAAM,SAAmB,CAAC;AAC1B,QAAM,UAAoB,CAAC,cAAc,IAAI,KAAK,uBAAuB;AACzE,QAAM,qBAAyB,uBAAS,QAAQ;AAChD,QAAM,sBAA0B,uBAAS,SAAS;AAElD,MAAG,kBAAkB,iBAAiB;AACpC,kBAAc,KAAK;AAAA,QACf,cAAc;AAAA,QACd,eAAe;AAAA;AAAA;AAAA,KAGlB;AACD,kBAAc,KAAK,mBAAmB;AACtC,WAAO,KAAK,UAAU;AAAA,EACxB;AAEA,MAAG,UAAU,QAAQ;AACnB,WAAO,KAAK,kBAAkB;AAC9B,kBAAc,KAAK;AAAA;AAAA;AAAA;AAAA,MAIjB;AACF,kBAAc,KAAK,mCAAmC;AACtD,YAAQ,KAAK,sBAAsB;AAAA,EACrC;AAEA,SAAO,KAAK,cAAc;AAC1B,gBAAc,KAAK,qBAAqB;AAGxC,QAAM,SAAiB,yBAAyB,eAAe;AAAA;AAAA;AAAA,6BAGpC,eAAe,qBAAqB,eAAe;AAAA;AAAA;AAAA;AAAA,MAI1E,cAAc,KAAK,IAAI,CAAC;AAAA,aACjB,QAAQ,KAAK,MAAM,CAAC;AAAA,MAC3B,MAAM,GAAG;AAAA,gCACiB,cAAc,KAAK,IAAI,CAAC;AAEtD,SAAO,SAAS,MAAM,MAAM,EACzB,KAAK,CAAC,WAAW,OAAO,IAAI,CAA0B,EACtD,MAAM,CAAC,UAAiB;AACvB,wCAAS;AAAA,MACP;AAAA,MACA,UAAU;AAAA,MACV,OAAO,wBAAW;AAAA,IACpB,GAAG,OAAO,OAAO;AAEjB,WAAO,CAAC;AAAA,EACV,CAAC;AACL;AAEO,MAAM,iBAAiB,CAC5B,SACA,OAAiB,CAAC,GAClB,YACwB;AACxB,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACzD,QAAM,EAAC,UAAU,OAAO,WAAW,KAAI,IAAI,iBAAiB,OAAO;AACnE,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,QAAQ,SAAS;AAC1F,QAAM,iBAAyB,KAAK,UAAU,KAAK,IAAI,CAAC,YAAQ,wBAAU,KAAK,EAAE,EAAE,YAAY,CAAC,CAAC;AACjG,QAAM,SAAmB,CAAC;AAC1B,QAAM,UAAoB,CAAC,cAAc,IAAI,KAAK,uBAAuB;AACzE,QAAM,qBAAyB,uBAAS,QAAQ;AAChD,QAAM,sBAA0B,uBAAS,SAAS;AAElD,MAAG,kBAAkB,iBAAiB;AACpC,kBAAc,KAAK;AAAA,QACf,cAAc;AAAA,QACd,eAAe;AAAA;AAAA;AAAA,KAGlB;AACD,kBAAc,KAAK,mBAAmB;AACtC,WAAO,KAAK,UAAU;AAAA,EACxB;AAEA,MAAG,KAAK,QAAQ;AACd,WAAO,KAAK,kBAAkB;AAC9B,kBAAc,KAAK;AAAA;AAAA;AAAA;AAAA,MAIjB;AACF,kBAAc,KAAK,yBAAyB;AAC5C,YAAQ,KAAK,iBAAiB;AAAA,EAChC;AAEA,SAAO,KAAK,cAAc;AAC1B,gBAAc,KAAK,WAAW;AAE9B,QAAM,SAAiB;AAAA;AAAA;AAAA,6BAGI,cAAc,qBAAqB,cAAc;AAAA;AAAA;AAAA;AAAA,MAIxE,cAAc,KAAK,IAAI,CAAC;AAAA,aACjB,QAAQ,KAAK,MAAM,CAAC;AAAA,MAC3B,MAAM,GAAG;AAAA,WACJ,OAAO,KAAK,IAAI,CAAC;AAAA,gCACI,cAAc,KAAK,IAAI,CAAC;AAEtD,SAAO,SAAS,MAAM,MAAM,EACzB,KAAK,CAAC,WAAW,OAAO,IAAI,CAA0B,EACtD,MAAM,CAAC,UAAiB;AACvB,wCAAS;AAAA,MACP;AAAA,MACA,UAAU;AAAA,MACV,OAAO,wBAAW;AAAA,IACpB,GAAG,OAAO,OAAO;AAEjB,WAAO,CAAC;AAAA,EACV,CAAC;AACL;AAEO,MAAM,iBAAiB,CAAC,SAAqB,QAAgB,YAA+C;AACjH,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACzD,QAAM,EAAC,OAAO,KAAI,IAAI,iBAAiB,OAAO;AAC9C,QAAM,mBAAuB,sBAAQ,MAAM;AAC3C,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,QAAQ,SAAS;AAC1F,QAAM,SAAiB;AAAA,0BACC,YAAY,mBAAmB,IAAI;AAAA,MACvD,cAAc,KAAK,IAAI,CAAC;AAAA,MACxB,MAAM,GAAG;AAAA;AAAA,gCAEiB,cAAc,KAAK,IAAI,CAAC;AAEtD,SAAO,SAAS,MAAM,MAAM,EACzB,KAAK,CAAC,WAAW,OAAO,IAAI,CAA0B,EACtD,MAAM,CAAC,UAAiB;AACvB,wCAAS;AAAA,MACP;AAAA,MACA,UAAU;AAAA,MACV,OAAO,wBAAW;AAAA,IACpB,GAAG,OAAO,OAAO;AAEjB,WAAO,CAAC;AAAA,EACV,CAAC;AACL;AAEO,MAAM,kBAAkB,CAAC,SAAqB,QAAgB,YAA+C;AAClH,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACjD,QAAM,EAAC,OAAO,KAAI,IAAI,iBAAiB,OAAO;AAC9C,QAAM,mBAAuB,sBAAQ,MAAM;AAG3C,QAAM,SAAmB;AAAA,uBACJ,IAAI,iBAAiB,YAAY;AAAA;AAAA;AAItD,SAAO,SAAS,MAAM,MAAM,EACzB,KAAK,CAAC,WAAW,OAAO,KAAK,CAAC,EAC9B,KAAK,CAAC,SAAmB;AACxB,UAAM;AAAA,MACJ;AAAA,MACA;AAAA,MACA,UAAU;AAAA,IACZ,IAAc;AAGd,QAAI;AAEJ,QAAG,WAAW,YAAY,SAAS;AACjC,sBAAgB;AAAA;AAAA,gCAEQ,IAAI;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,8BASN,SAAS;AAAA;AAAA,YAE3B,MAAM,GAAG;AAAA;AAAA,IAEf,WAAU,YAAY,UAAU;AAC9B,sBAAgB;AAAA;AAAA,gCAEQ,IAAI;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,YAOxB,MAAM,GAAG;AAAA;AAAA,IAEf;AAEA,QAAG,eAAe;AAChB,aAAO,SAAS,MAAM,aAAa,EAChC,KAAK,CAAC,WAAW,OAAO,IAAI,CAA0B,EACtD,MAAM,CAAC,UAAiB;AACvB,4CAAS;AAAA,UACP;AAAA,UACA,UAAU;AAAA,UACV,OAAO,wBAAW;AAAA,QACpB,GAAG,OAAO,OAAO;AAEjB,eAAO,CAAC;AAAA,MACV,CAAC;AAAA,IACL;AAEA,WAAO,CAAC;AAAA,EACV,CAAC,EACA,MAAM,CAAC,UAAiB;AACvB,wCAAS;AAAA,MACP;AAAA,MACA,UAAU;AAAA,MACV,OAAO,wBAAW;AAAA,IACpB,GAAG,OAAO,OAAO;AAEjB,WAAO,CAAC;AAAA,EACV,CAAC;AACL;AAEO,MAAM,UAAU,OACrB,SACA;AAAA,EACE,UAAU;AAAA,EACV;AAAA,EACA,UAAU;AAAA,EACV;AAAA,EACA;AAAA,EACA;AAAA,EACA,OAAO;AAAA,EACP,WAAW;AAAA,EACX,UAAU;AAAA,EACV,OAAO,CAAC;AAAA,EACR;AAAA,EACA,OAAO;AACT,MACsB;AACtB,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACjD,QAAM,MAAM,KAAK,IAAI;AACrB,QAAM,aAAS,yBAAW,QAAQ,SAAS,EAAE;AAC7C,QAAM,SAAmB;AAAA,IACvB,KAAK,SAAS,MAAM;AAAA,IACpB,MAAM;AAAA,IACN,OAAO;AAAA,IACP,aAAS,0BAAY,SAAS,kBAAkB;AAAA,IAChD,SAAS,cAAU,uBAAS,SAAS,EAAE,IAAI;AAAA,IAC3C,SAAS,cAAU,sBAAQ,OAAO,IAAI;AAAA,IACtC,UAAU,aAAa,aAAY,uBAAS,QAAQ,IAAI;AAAA,IACxD,UAAU,eAAW,0BAAY,UAAU,GAAG,IAAI;AAAA,IAClD,WAAW,cAAc,aAAY,uBAAS,SAAS,IAAI;AAAA,IAC3D,UAAU;AAAA,IACV,UAAM,0BAAY,MAAM,GAAG;AAAA,IAC3B,UAAU,eAAW,sBAAQ,QAAQ,IAAI;AAAA,IACzC,aAAS,2BAAa,SAAS,EAAE;AAAA,IACjC,WAAW,gBAAY,uBAAS,WAAW,EAAE,IAAI;AAAA,IACjD,UAAM,wBAAU,MAAM,EAAE;AAAA,IACxB,QAAQ;AAAA,EACV;AACA,QAAM,SAAmB,6BAAa,MAAM;AAE5C,MAAI;AACF,UAAM,YAAsB,MAAM,SAAS,MAAM,MAAM,EACpD,KAAK,CAAC,WAAW,OAAO,KAAK,CAAC,EAC9B,MAAM,CAAC,cAAiB,gCAAS;AAAA,MAChC;AAAA,MACA,UAAU;AAAA,MACV,OAAO,wBAAW;AAAA,IACpB,GAAG,OAAO,OAAO,CAAC;AACpB,UAAM,EAAC,KAAK,UAAS,IAAI;AACzB,UAAM,kBAAkB,UAAM,yBAAY,OAAO,OAAO;AAExD,QAAG,KAAK,UAAU,gBAAgB,QAAQ;AACxC,YAAM,YAAY,UAAM,2BAAc,SAAS,KAAK,IAAI,CAAC,EAAC,MAAAA,MAAI,MAAMA,KAAI,CAAC,GACtE,IAAI,CAAC,SAAS,EAAC,GAAG,KAAK,OAAO,UAAS,EAAE;AAC5C,YAAM,eAAe,UAAM,2BAAc,SAAS,eAAe,GAC9D,IAAI,CAAC,SAAS,EAAC,GAAG,KAAK,OAAO,UAAS,EAAE;AAC5C,YAAM,cAAyB,UAAM;AAAA,QACnC;AAAA,QACA;AAAA,UACE,WAAW;AAAA,UACX,MAAM,CAAC,GAAG,aAAa,GAAG,QAAQ;AAAA,QACpC;AAAA,MACF;AAEA,aAAO;AAAA,QACL,GAAG;AAAA,QACH,MAAM;AAAA,MACR;AAAA,IACF;AAEA,WAAO;AAAA,EACT,SAAQ,OAAO;AACb,UAAM;AAAA,EACR;AACF;AAEO,MAAM,aAAa,OAAO,SAAqB,SAA2C;AAC/F,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACjD,QAAM,MAAc,KAAK,IAAI;AAC7B,QAAM,iBAAa,8BAAU,IAAI;AACjC,QAAM;AAAA,IACJ;AAAA,IACA,OAAO,CAAC;AAAA,EACV,IAAI;AAEJ,QAAM,SAAmB;AAAA,IACvB,GAAG;AAAA,IACH,UAAU;AAAA,EACZ;AAEA,MAAG,CAAC,QAAQ;AACV,eAAO,oCAAa;AAAA,MAClB;AAAA,MACA,UAAU;AAAA,MACV,OAAO,wBAAW;AAAA,IACpB,GAAG,CAAC,CAAC;AAAA,EACP;AAEA,QAAM,SAAS;AAAA,IACb,GAAG;AAAA,IACH,MAAM;AAAA,IACN,OAAO;AAAA,IACP,QAAQ;AAAA,EACV;AACA,QAAM,SAAmB,oCAAoB,MAAM,aAAa,SAAS;AAAA,aAC9D,MAAM;AAAA,aACN,MAAM;AAAA;AAGjB,MAAI;AACF,UAAM,cAAwB,MAAM,SACjC,MAAM,MAAM,EACZ,KAAK,CAAC,WAAW,OAAO,KAAK,CAAC,EAC9B,MAAM,CAAC,cAAiB,gCAAS;AAAA,MAChC;AAAA,MACA,UAAU;AAAA,MACV,OAAO,wBAAW;AAAA,IACpB,GAAG,OAAO,CAAC,CAAC,CAAC;AACf,UAAM,EAAC,KAAK,cAAa,IAAI;AAC7B,UAAM,kBAAkB,UAAM,yBAAY,OAAO,OAAO;AAExD,QAAG,MAAM,UAAU,iBAAiB,QAAQ;AAC1C,YAAM,WAAW,MAAM,UAAU,UAAM,2BAAc,SAAS,KAAK,IAAI,CAAC,EAAC,KAAI,MAAM,IAAI,CAAC,GACrF,IAAI,CAAC,SAAS,EAAC,GAAG,KAAK,OAAO,UAAS,EAAE,IAAI,CAAC;AACjD,YAAM,cAAc,iBAAiB,UAAU,UAAM,2BAAc,SAAS,eAAe,GACxF,IAAI,CAAC,SAAS,EAAC,GAAG,KAAK,OAAO,UAAS,EAAE,IAAI,CAAC;AACjD,YAAM,cAAyB,UAAM;AAAA,QACnC;AAAA,QACA;AAAA,UACE,WAAW;AAAA,UACX,MAAM,CAAC,GAAG,aAAa,GAAG,QAAQ;AAAA,QACpC;AAAA,MACF;AAEA,aAAO;AAAA,QACL,GAAG;AAAA,QACH,MAAM;AAAA,MACR;AAAA,IACF;AAEA,WAAO;AAAA,EACT,SAAQ,OAAO;AACb,UAAM;AAAA,EACR;AACF;AAEO,MAAM,aAAa,OAAO,SAAqB,cAAyC;AAC7F,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACjD,QAAM,mBAAuB,4BAAc,SAAS;AAEpD,MAAG,CAAC,cAAc;AAChB,eAAO,oCAAa;AAAA,MAClB;AAAA,MACA,UAAU;AAAA,MACV,OAAO,wBAAW;AAAA,IACpB,GAAG,CAAC,CAAC;AAAA,EACP;AAEA,QAAM,aAAuB;AAAA,oBACX,YAAY;AAAA;AAG9B,QAAM,SAAS,MAAM,UAAU,EAC5B,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA,EACR,CAAC;AAEH,QAAM,aAAuB;AAAA,sBACT,YAAY;AAAA;AAGhC,QAAM,SAAS,MAAM,UAAU,EAC5B,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA,EACR,CAAC;AAEH,QAAM,SAAS;AAAA,wBACO,YAAY,mBAAmB,SAAS;AAAA;AAAA;AAAA;AAK9D,SAAO,SAAS,MAAM,MAAM,EACzB,KAAK,CAAC,WAAW,OAAO,KAAK,CAAC,EAC9B,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA,EACR,CAAC;AACL;AAEO,MAAM,iBAAiB,CAC5B,SACA,WACA,WACA,WAAmB,YACnB,QAAQ,CAAC,MACa;AACtB,QAAM,SAAS;AACf,QAAM,EAAC,SAAQ,IAAI;AACnB,QAAM,iBAAiC,SAAS,WAAW,QAAQ;AACnE,QAAM,mBAAuB,4BAAc,SAAS;AACpD,QAAM,kBAAsB,4BAAc,SAAS;AAEnD,MAAG,CAAC,eAAe,CAAC,cAAc;AAChC,WAAO,QAAQ,OAAO,IAAI,MAAM,wBAAW,UAAU,CAAC;AAAA,EACxD;AAEA,QAAM,aAAiB,yBAAW,YAAY,YAAY,IAAI,WAAW,EAAE;AAC3E,QAAM,OAAO;AAAA,IACX,OAAO;AAAA,IACP,MAAM;AAAA,IACN,KAAK;AAAA,IACL,OAAO,KAAK,IAAI;AAAA,IAChB,GAAG;AAAA,EACL;AAEA,SAAO,eAAe,KAAK,MAAM,EAAC,WAAW,KAAI,CAAC,EAC/C,MAAM,CAAC,cACN,gCAAS;AAAA,IACP;AAAA,IACA,UAAU;AAAA,IACV,OAAO,wBAAW;AAAA,EACpB,GAAG,OAAO,CAAC,CAAC,CAAC;AACnB;",
  "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"]
}
