@nlabs/reaktor 0.4.0 → 0.4.2

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 (151) hide show
  1. package/lib/actions/conversations.d.ts +14 -0
  2. package/lib/actions/conversations.js +333 -0
  3. package/lib/actions/dynamodb.js +155 -0
  4. package/lib/actions/email.js +177 -0
  5. package/lib/actions/files.js +319 -0
  6. package/lib/{data → actions}/groups.d.ts +4 -3
  7. package/lib/actions/groups.js +282 -0
  8. package/lib/actions/images.d.ts +22 -0
  9. package/lib/actions/images.js +682 -0
  10. package/lib/actions/index.js +40 -0
  11. package/lib/{data → actions}/ios.d.ts +2 -1
  12. package/lib/actions/ios.js +179 -0
  13. package/lib/actions/locations.js +112 -0
  14. package/lib/actions/messages.d.ts +13 -0
  15. package/lib/actions/messages.js +216 -0
  16. package/lib/{data → actions}/notifications.d.ts +2 -2
  17. package/lib/actions/notifications.js +63 -0
  18. package/lib/{data → actions}/payments.d.ts +2 -2
  19. package/lib/actions/payments.js +491 -0
  20. package/lib/actions/posts.d.ts +19 -0
  21. package/lib/actions/posts.js +538 -0
  22. package/lib/actions/reactions.d.ts +30 -0
  23. package/lib/actions/reactions.js +340 -0
  24. package/lib/{data → actions}/s3.d.ts +1 -1
  25. package/lib/actions/s3.js +122 -0
  26. package/lib/{data → actions}/search.d.ts +2 -2
  27. package/lib/actions/search.js +99 -0
  28. package/lib/actions/sms.js +76 -0
  29. package/lib/actions/statistics.d.ts +2 -0
  30. package/lib/actions/statistics.js +63 -0
  31. package/lib/actions/subscription.js +209 -0
  32. package/lib/actions/tags.d.ts +26 -0
  33. package/lib/actions/tags.js +340 -0
  34. package/lib/actions/users.d.ts +44 -0
  35. package/lib/actions/users.js +571 -0
  36. package/lib/{data → actions}/websockets.d.ts +1 -1
  37. package/lib/actions/websockets.js +156 -0
  38. package/lib/config.d.ts +2 -3
  39. package/lib/config.js +116 -149
  40. package/lib/index.d.ts +1 -1
  41. package/lib/index.js +23 -45
  42. package/lib/templates/email/layout.d.ts +2 -0
  43. package/lib/templates/email/layout.js +292 -0
  44. package/lib/templates/email/passwordForgot.d.ts +2 -0
  45. package/lib/templates/email/passwordForgot.js +28 -0
  46. package/lib/templates/email/passwordRecovery.d.ts +2 -0
  47. package/lib/templates/email/passwordRecovery.js +25 -0
  48. package/lib/templates/email/verifyEmail.d.ts +2 -0
  49. package/lib/templates/email/verifyEmail.js +28 -0
  50. package/lib/templates/email/welcome.d.ts +2 -0
  51. package/lib/templates/email/welcome.js +28 -0
  52. package/lib/templates/sms/passwordForgot.d.ts +2 -0
  53. package/lib/templates/sms/passwordForgot.js +14 -0
  54. package/lib/templates/sms/passwordRecovery.d.ts +2 -0
  55. package/lib/templates/sms/passwordRecovery.js +14 -0
  56. package/lib/templates/sms/verifyEmail.d.ts +2 -0
  57. package/lib/templates/sms/verifyEmail.js +14 -0
  58. package/lib/templates/sms/verifyPhone.d.ts +2 -0
  59. package/lib/templates/sms/verifyPhone.js +14 -0
  60. package/lib/templates/sms/welcome.d.ts +2 -0
  61. package/lib/templates/sms/welcome.js +14 -0
  62. package/lib/types/apps.d.ts +2 -2
  63. package/lib/types/apps.js +4 -2
  64. package/lib/types/arangodb.js +4 -2
  65. package/lib/types/auth.d.ts +4 -8
  66. package/lib/types/auth.js +4 -2
  67. package/lib/types/conversations.d.ts +3 -3
  68. package/lib/types/conversations.js +4 -2
  69. package/lib/types/email.d.ts +2 -2
  70. package/lib/types/email.js +4 -2
  71. package/lib/types/files.js +4 -2
  72. package/lib/types/google.js +4 -2
  73. package/lib/types/groups.d.ts +2 -1
  74. package/lib/types/groups.js +4 -2
  75. package/lib/types/images.d.ts +8 -5
  76. package/lib/types/images.js +4 -2
  77. package/lib/types/index.d.ts +1 -1
  78. package/lib/types/index.js +37 -227
  79. package/lib/types/locations.js +4 -2
  80. package/lib/types/messages.d.ts +12 -2
  81. package/lib/types/messages.js +4 -2
  82. package/lib/types/notifications.d.ts +2 -2
  83. package/lib/types/notifications.js +4 -2
  84. package/lib/types/payments.js +4 -2
  85. package/lib/types/posts.d.ts +18 -1
  86. package/lib/types/posts.js +4 -2
  87. package/lib/types/statistics.d.ts +3 -0
  88. package/lib/types/statistics.js +4 -0
  89. package/lib/types/tags.d.ts +6 -0
  90. package/lib/types/tags.js +4 -2
  91. package/lib/types/users.d.ts +15 -11
  92. package/lib/types/users.js +4 -2
  93. package/lib/utils/analytics.d.ts +7 -0
  94. package/lib/utils/analytics.js +101 -77
  95. package/lib/utils/arangodb.d.ts +1 -1
  96. package/lib/utils/arangodb.js +93 -114
  97. package/lib/utils/auth.js +58 -55
  98. package/lib/utils/graphql.js +38 -19
  99. package/lib/utils/index.d.ts +1 -1
  100. package/lib/utils/index.js +26 -84
  101. package/lib/utils/objects.js +44 -53
  102. package/lib/utils/session.d.ts +18 -0
  103. package/lib/utils/session.js +42 -0
  104. package/package.json +32 -30
  105. package/lib/data/conversations.d.ts +0 -8
  106. package/lib/data/conversations.js +0 -311
  107. package/lib/data/dynamodb.js +0 -206
  108. package/lib/data/email.js +0 -222
  109. package/lib/data/files.js +0 -525
  110. package/lib/data/groups.js +0 -435
  111. package/lib/data/images.d.ts +0 -22
  112. package/lib/data/images.js +0 -1051
  113. package/lib/data/index.js +0 -266
  114. package/lib/data/ios.js +0 -355
  115. package/lib/data/locations.js +0 -172
  116. package/lib/data/messages.d.ts +0 -9
  117. package/lib/data/messages.js +0 -299
  118. package/lib/data/notifications.js +0 -59
  119. package/lib/data/payments.js +0 -771
  120. package/lib/data/posts.d.ts +0 -23
  121. package/lib/data/posts.js +0 -766
  122. package/lib/data/reactions.d.ts +0 -14
  123. package/lib/data/reactions.js +0 -529
  124. package/lib/data/s3.js +0 -155
  125. package/lib/data/search.js +0 -155
  126. package/lib/data/sms.js +0 -83
  127. package/lib/data/subscription.js +0 -337
  128. package/lib/data/tags.d.ts +0 -14
  129. package/lib/data/tags.js +0 -397
  130. package/lib/data/users.d.ts +0 -20
  131. package/lib/data/users.js +0 -470
  132. package/lib/data/websockets.js +0 -250
  133. package/lib/types/reactions.d.ts +0 -17
  134. package/lib/types/reactions.js +0 -2
  135. package/lib/utils/redis.d.ts +0 -1
  136. package/lib/utils/redis.js +0 -36
  137. package/templates/email/layout.html +0 -279
  138. package/templates/email/passwordForgot.html +0 -15
  139. package/templates/email/passwordRecovery.html +0 -12
  140. package/templates/email/verifyEmail.html +0 -15
  141. package/templates/sms/passwordForgot.txt +0 -1
  142. package/templates/sms/passwordRecovery.txt +0 -1
  143. package/templates/sms/verifyEmail.txt +0 -1
  144. package/templates/sms/verifyPhone.txt +0 -1
  145. /package/lib/{data → actions}/dynamodb.d.ts +0 -0
  146. /package/lib/{data → actions}/email.d.ts +0 -0
  147. /package/lib/{data → actions}/files.d.ts +0 -0
  148. /package/lib/{data → actions}/index.d.ts +0 -0
  149. /package/lib/{data → actions}/locations.d.ts +0 -0
  150. /package/lib/{data → actions}/sms.d.ts +0 -0
  151. /package/lib/{data → actions}/subscription.d.ts +0 -0
@@ -0,0 +1,19 @@
1
+ import { Database } from 'arangojs';
2
+ import { ApiContext, FileType, PostInputType, PostOptions, PostType } from '../types';
3
+ export declare const parsePostOptions: (options?: PostOptions) => {
4
+ latitude: number;
5
+ limit: import("../types").ArangoDBLimit;
6
+ longitude: number;
7
+ type: string;
8
+ };
9
+ export declare const getPostOptional: (fields: string[], sessionId: string) => any;
10
+ export declare const getPost: (context: ApiContext, itemId: string, options: PostOptions) => Promise<Partial<PostType>>;
11
+ export declare const getPostsByReactions: (context: ApiContext, reactions?: string[], options?: PostOptions) => Promise<PostType[]>;
12
+ export declare const getPostsByTags: (context: ApiContext, tags?: string[], options?: PostOptions) => Promise<PostType[]>;
13
+ export declare const getPostsByUser: (context: ApiContext, userId: string, options?: PostOptions) => Promise<PostType[]>;
14
+ export declare const getPostComments: (context: ApiContext, itemId: string, options?: PostOptions) => Promise<PostType[]>;
15
+ export declare const addPost: (context: ApiContext, post: PostInputType) => Promise<PostType>;
16
+ export declare const updatePost: (context: ApiContext, post: PostInputType) => Promise<PostType>;
17
+ export declare const deletePost: (context: ApiContext, postId: string) => Promise<PostType>;
18
+ export declare const cleanPosts: (database: Database) => Promise<number>;
19
+ export declare const createPostEdge: (db: Database, file: FileType, postId: string) => Promise<FileType>;
@@ -0,0 +1,538 @@
1
+ var __create = Object.create;
2
+ var __defProp = Object.defineProperty;
3
+ var __defProps = Object.defineProperties;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
6
+ var __getOwnPropNames = Object.getOwnPropertyNames;
7
+ var __getOwnPropSymbols = Object.getOwnPropertySymbols;
8
+ var __getProtoOf = Object.getPrototypeOf;
9
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
10
+ var __propIsEnum = Object.prototype.propertyIsEnumerable;
11
+ var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
12
+ var __spreadValues = (a, b) => {
13
+ for (var prop in b || (b = {}))
14
+ if (__hasOwnProp.call(b, prop))
15
+ __defNormalProp(a, prop, b[prop]);
16
+ if (__getOwnPropSymbols)
17
+ for (var prop of __getOwnPropSymbols(b)) {
18
+ if (__propIsEnum.call(b, prop))
19
+ __defNormalProp(a, prop, b[prop]);
20
+ }
21
+ return a;
22
+ };
23
+ var __spreadProps = (a, b) => __defProps(a, __getOwnPropDescs(b));
24
+ var __markAsModule = (target) => __defProp(target, "__esModule", { value: true });
25
+ var __export = (target, all) => {
26
+ __markAsModule(target);
27
+ for (var name in all)
28
+ __defProp(target, name, { get: all[name], enumerable: true });
29
+ };
30
+ var __reExport = (target, module2, desc) => {
31
+ if (module2 && typeof module2 === "object" || typeof module2 === "function") {
32
+ for (let key of __getOwnPropNames(module2))
33
+ if (!__hasOwnProp.call(target, key) && key !== "default")
34
+ __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
35
+ }
36
+ return target;
37
+ };
38
+ var __toModule = (module2) => {
39
+ return __reExport(__markAsModule(__defProp(module2 != null ? __create(__getProtoOf(module2)) : {}, "default", module2 && module2.__esModule && "default" in module2 ? { get: () => module2.default, enumerable: true } : { value: module2, enumerable: true })), module2);
40
+ };
41
+ __export(exports, {
42
+ addPost: () => addPost,
43
+ cleanPosts: () => cleanPosts,
44
+ createPostEdge: () => createPostEdge,
45
+ deletePost: () => deletePost,
46
+ getPost: () => getPost,
47
+ getPostComments: () => getPostComments,
48
+ getPostOptional: () => getPostOptional,
49
+ getPostsByReactions: () => getPostsByReactions,
50
+ getPostsByTags: () => getPostsByTags,
51
+ getPostsByUser: () => getPostsByUser,
52
+ parsePostOptions: () => parsePostOptions,
53
+ updatePost: () => updatePost
54
+ });
55
+ var import_utils = __toModule(require("@nlabs/utils"));
56
+ var import_arangojs = __toModule(require("arangojs"));
57
+ var import_utils2 = __toModule(require("../utils"));
58
+ var import_files = __toModule(require("./files"));
59
+ var import_tags = __toModule(require("./tags"));
60
+ const MAX_CONTENT_LENGTH = 1e5;
61
+ const eventCategory = "posts";
62
+ const parsePostOptions = (options = {}) => {
63
+ const {
64
+ from = 0,
65
+ latitude = 0,
66
+ longitude = 0,
67
+ to = 30,
68
+ type = "post"
69
+ } = options;
70
+ return {
71
+ latitude: (0, import_utils.parseNum)(latitude, 32),
72
+ limit: (0, import_utils2.getLimit)(from, to),
73
+ longitude: (0, import_utils.parseNum)(longitude, 32),
74
+ type: (0, import_utils.parseChar)(type, 32)
75
+ };
76
+ };
77
+ const getPostOptional = (fields, sessionId) => fields.reduce((selects, field) => {
78
+ switch (field) {
79
+ case "hasRsvp": {
80
+ selects.queries.push(`LET hasRsvp = TO_BOOL(FIRST(
81
+ FOR post, r IN INBOUND p._id hasReactions
82
+ FILTER r.name == "rsvp" && r.type == "posts" && r._from == "users/${sessionId}"
83
+ COLLECT WITH COUNT INTO count
84
+ RETURN count
85
+ ))`);
86
+ selects.objects.push("hasRsvp:hasRsvp");
87
+ return selects;
88
+ }
89
+ case "isSaved": {
90
+ selects.queries.push(`LET isSaved = TO_BOOL(FIRST(
91
+ FOR post, r IN INBOUND p._id hasReactions
92
+ FILTER r.name == "pin" && r.type == "posts" && r._from == "users/${sessionId}"
93
+ COLLECT WITH COUNT INTO count
94
+ RETURN count
95
+ ))`);
96
+ selects.objects.push("isSaved:isSaved");
97
+ return selects;
98
+ }
99
+ case "reactions": {
100
+ selects.queries.push(`LET reactions = (
101
+ FOR post, r IN INBOUND p._id hasReactions
102
+ COLLECT reactionName = r.value INTO reactionItems
103
+ RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
104
+ )`);
105
+ selects.objects.push("reactions:reactions");
106
+ return selects;
107
+ }
108
+ case "rsvpCount": {
109
+ selects.queries.push(`LET rsvpCount = FIRST(
110
+ FOR post, r IN INBOUND p._id hasReactions
111
+ FILTER r.name == "rsvp" && r.type == "posts"
112
+ COLLECT WITH COUNT INTO count
113
+ RETURN count
114
+ )`);
115
+ selects.objects.push("rsvpCount:rsvpCount");
116
+ return selects;
117
+ }
118
+ case "viewCount": {
119
+ selects.queries.push(`LET viewCount = FIRST(
120
+ FOR post, r IN INBOUND p._id hasReactions
121
+ FILTER r.name == "view" && r.type == "posts"
122
+ COLLECT WITH COUNT INTO count
123
+ RETURN count
124
+ )`);
125
+ selects.objects.push("viewCount:viewCount");
126
+ return selects;
127
+ }
128
+ default: {
129
+ return selects;
130
+ }
131
+ }
132
+ }, { objects: [], queries: [] });
133
+ const getPost = (context, itemId, options) => {
134
+ const action = "getPost";
135
+ const { database, fields, session: { userId: sessionId } } = context;
136
+ const formatItemId = (0, import_utils.parseId)(itemId);
137
+ const { type } = parsePostOptions(options);
138
+ const db = database;
139
+ const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
140
+ const aqlQry = import_arangojs.aql`FOR p IN posts
141
+ FILTER p._key == ${formatItemId} && p.type == ${type}
142
+ LIMIT 1
143
+ RETURN p`;
144
+ return db.query(aqlQry).then((cursor) => cursor.next()).then((post = {}) => {
145
+ const {
146
+ _id: postDocId,
147
+ groupId,
148
+ privacy = "default"
149
+ } = post;
150
+ let privacyAqlQry;
151
+ if (groupId && privacy === "group") {
152
+ privacyAqlQry = `LET p = DOCUMENT("${postDocId}")
153
+ ${selectQueries.join("\n")}
154
+ FOR group IN groups
155
+ FILTER group._key == p.groupId
156
+ FOR u, e IN OUTBOUND group._id isGrouped
157
+ FILTER u._key == "${sessionId}"
158
+ LIMIT 1
159
+ RETURN MERGE(p, {${selectObjects.join(", ")}})`;
160
+ } else if (privacy === "public") {
161
+ privacyAqlQry = `LET p = DOCUMENT("${postDocId}")
162
+ ${selectQueries.join("\n")}
163
+ LIMIT 1
164
+ RETURN MERGE(p, {${selectObjects.join(", ")}})`;
165
+ }
166
+ if (privacyAqlQry) {
167
+ return db.query(privacyAqlQry).then((cursor) => cursor.next() || {}).catch((error) => (0, import_utils2.logError)({
168
+ action,
169
+ category: eventCategory,
170
+ label: "db_error"
171
+ }, error, context).then(() => ({})));
172
+ }
173
+ return {};
174
+ }).catch((error) => (0, import_utils2.logError)({
175
+ action,
176
+ category: eventCategory,
177
+ label: "db_error"
178
+ }, error, context).then(() => ({})));
179
+ };
180
+ const getPostsByReactions = (context, reactions = [], options) => {
181
+ const action = "getPostsByReactions";
182
+ const { database, fields, session: { userId: sessionId } } = context;
183
+ const { latitude, limit, longitude, type } = parsePostOptions(options);
184
+ const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
185
+ const formatSessionId = `users/${sessionId}`;
186
+ const formatReactions = JSON.stringify(reactions.map((reaction) => (0, import_utils.parseChar)(reaction, 32).toLowerCase()));
187
+ const sortBy = [];
188
+ const filters = [`p.type == "${type}"`, 'p.privacy == "public"'];
189
+ const formatLatitude = (0, import_utils.parseNum)(latitude);
190
+ const formatLongitude = (0, import_utils.parseNum)(longitude);
191
+ if (formatLatitude && formatLongitude) {
192
+ selectQueries.push(`LET distance = DISTANCE(
193
+ ${formatLatitude},
194
+ ${formatLongitude},
195
+ NOT_NULL(p.latitude, 0),
196
+ NOT_NULL(p.longitude, 0))
197
+ `);
198
+ selectObjects.push("distance:distance");
199
+ sortBy.push("distance");
200
+ }
201
+ if (reactions.length) {
202
+ sortBy.push("matchedTags DESC");
203
+ selectQueries.push(`LET matchedReactions = LENGTH(
204
+ FOR mr IN reactions
205
+ FILTER mr.matched == true
206
+ RETURN mr
207
+ )`);
208
+ selectObjects.push("matchedReactions:matchedReactions");
209
+ filters.push("matchedReactions > 0");
210
+ }
211
+ sortBy.push("p.added DESC");
212
+ selectObjects.push("reactions:reactions");
213
+ const aqlQry = `FOR p, r IN OUTBOUND "${formatSessionId}" hasReactions
214
+ LET reactions = (
215
+ FOR reaction, hr IN 1..1 INBOUND p isTagged
216
+ LET matched = LENGTH(${formatReactions}) > 0 && POSITION(${formatReactions}, reaction.name)
217
+ SORT reaction.name
218
+ RETURN MERGE(reaction, {matched:matched})
219
+ )
220
+ ${selectQueries.join("\n")}
221
+ FILTER ${filters.join(" && ")}
222
+ ${limit.aql}
223
+ RETURN DISTINCT MERGE(p, {${selectObjects.join(", ")}})`;
224
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
225
+ action,
226
+ category: eventCategory,
227
+ label: "db_error"
228
+ }, error, context).then(() => []));
229
+ };
230
+ const getPostsByTags = (context, tags = [], options) => {
231
+ const { database, fields, session: { userId: sessionId } } = context;
232
+ const { latitude, limit, longitude, type } = parsePostOptions(options);
233
+ const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
234
+ const formatTagNames = JSON.stringify(tags.map((tag) => (0, import_utils.parseChar)(tag, 32).toLowerCase()));
235
+ const sortBy = [];
236
+ const filters = [`p.type == "${type}"`, 'p.privacy == "public"'];
237
+ const formatLatitude = (0, import_utils.parseNum)(latitude);
238
+ const formatLongitude = (0, import_utils.parseNum)(longitude);
239
+ if (formatLatitude && formatLongitude) {
240
+ selectQueries.push(`LET distance = DISTANCE(
241
+ ${formatLatitude},
242
+ ${formatLongitude},
243
+ NOT_NULL(p.latitude, 0),
244
+ NOT_NULL(p.longitude, 0))
245
+ `);
246
+ selectObjects.push("distance:distance");
247
+ sortBy.push("distance");
248
+ }
249
+ if (tags.length) {
250
+ sortBy.push("matchedTags DESC");
251
+ selectQueries.push(`LET matchedTags = LENGTH(
252
+ FOR t IN tags
253
+ FILTER t.matched == true
254
+ RETURN t
255
+ )`);
256
+ selectObjects.push("matchedTags:matchedTags");
257
+ filters.push("matchedTags > 0");
258
+ }
259
+ sortBy.push("p.added DESC");
260
+ selectObjects.push("tags:tags");
261
+ const aqlQry = `FOR p IN posts
262
+ LET tags = (
263
+ FOR tag, it IN 1..1 INBOUND p isTagged
264
+ LET matched = LENGTH(${formatTagNames}) > 0 && POSITION(${formatTagNames}, tag.name)
265
+ SORT tag.name
266
+ RETURN MERGE(tag, {matched:matched})
267
+ )
268
+ ${selectQueries.join("\n")}
269
+ FILTER ${filters.join(" && ")}
270
+ ${limit.aql}
271
+ SORT ${sortBy.join(", ")}
272
+ RETURN DISTINCT MERGE(p, {${selectObjects.join(", ")}})`;
273
+ console.log({ aqlQry, options });
274
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch(() => []);
275
+ };
276
+ const getPostsByUser = (context, userId, options) => {
277
+ const { database, fields, session: { userId: sessionId } } = context;
278
+ const { limit, type } = parsePostOptions(options);
279
+ const formatUserId = (0, import_utils.parseId)(userId);
280
+ const { objects: selectObjects, queries: selectQueries } = getPostOptional(fields, sessionId);
281
+ const aqlQry = `FOR p IN posts
282
+ FILTER p.userId == "${formatUserId}" && p.type == "${type}" && p.privacy == "public" && p.parent == null
283
+ ${selectQueries.join("\n")}
284
+ ${limit.aql}
285
+ SORT p.added
286
+ RETURN DISTINCT MERGE(p, {${selectObjects.join(", ")}})`;
287
+ return database.query(aqlQry).then((cursor) => cursor.all()).catch((error) => {
288
+ throw error;
289
+ });
290
+ };
291
+ const getPostComments = (context, itemId, options) => {
292
+ const { database, session: { userId: sessionId } } = context;
293
+ const { limit, type } = parsePostOptions(options);
294
+ const formatItemId = (0, import_utils.parseId)(itemId);
295
+ const db = database;
296
+ const aqlQry = import_arangojs.aql`FOR p IN posts
297
+ FILTER p.type == ${type} && p._key == ${formatItemId}
298
+ LIMIT 1
299
+ RETURN p`;
300
+ return db.query(aqlQry).then((cursor) => cursor.next()).then((post = {}) => {
301
+ const {
302
+ _key,
303
+ groupId,
304
+ privacy = "public"
305
+ } = post;
306
+ let privacyAqlQry;
307
+ if (groupId && privacy === "group") {
308
+ privacyAqlQry = `FOR p IN posts
309
+ FOR user IN users
310
+ FILTER p.parent == "${_key}" && user._key == p.userId
311
+ LET reactions = (
312
+ FOR post, r IN INBOUND p._id reactions
313
+ COLLECT reactionName = r.value INTO reactionItems
314
+ RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
315
+ )
316
+ FOR group IN groups
317
+ FILTER group._key == p.groupId
318
+ FOR u, e IN OUTBOUND group._id isGrouped
319
+ FILTER u._key == "${sessionId}"
320
+ SORT p.added
321
+ ${limit.aql}
322
+ RETURN MERGE(p, {user: user, reactions: reactions})`;
323
+ } else if (privacy === "public") {
324
+ privacyAqlQry = `FOR p IN posts
325
+ FOR user IN users
326
+ FILTER p.parent == "${_key}" && user._key == p.userId
327
+ LET reactions = (
328
+ FOR post, r IN INBOUND p._id reactions
329
+ COLLECT reactionName = r.value INTO reactionItems
330
+ RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
331
+ )
332
+ SORT p.added
333
+ ${limit.aql}
334
+ RETURN MERGE(p, {user: user, reactions: reactions})`;
335
+ }
336
+ if (privacyAqlQry) {
337
+ return db.query(privacyAqlQry).then((cursor) => cursor.all()).catch((error) => {
338
+ throw error;
339
+ });
340
+ }
341
+ return [];
342
+ }).catch((error) => {
343
+ throw error;
344
+ });
345
+ };
346
+ const addPost = async (context, post) => {
347
+ const { database, session: { userId: sessionId } } = context;
348
+ const {
349
+ content = "",
350
+ endDate,
351
+ groupId = "",
352
+ location,
353
+ latitude,
354
+ longitude,
355
+ name = "",
356
+ parentId = null,
357
+ privacy = "public",
358
+ tags = [],
359
+ startDate,
360
+ type = "default"
361
+ } = post;
362
+ const now = Date.now();
363
+ const insert = {
364
+ _key: (0, import_utils.createHash)(`post-${sessionId}`),
365
+ added: now,
366
+ content: (0, import_utils.parseString)(content, MAX_CONTENT_LENGTH),
367
+ endDate: endDate ? (0, import_utils.parseNum)(endDate, 13) : void 0,
368
+ groupId: groupId ? (0, import_utils.parseId)(groupId) : void 0,
369
+ latitude: latitude !== void 0 ? (0, import_utils.parseNum)(latitude) : void 0,
370
+ location: location ? (0, import_utils.parseString)(location, 160) : void 0,
371
+ longitude: longitude !== void 0 ? (0, import_utils.parseNum)(longitude) : void 0,
372
+ modified: now,
373
+ name: (0, import_utils.parseString)(name, 160),
374
+ parentId: parentId ? (0, import_utils.parseId)(parentId) : void 0,
375
+ privacy: privacy ? (0, import_utils.parseVarChar)(privacy, 16) : void 0,
376
+ startDate: startDate ? (0, import_utils.parseNum)(startDate, 13) : void 0,
377
+ type: (0, import_utils.parseChar)(type, 32),
378
+ userId: sessionId
379
+ };
380
+ const db = database;
381
+ const aqlQry = import_arangojs.aql`INSERT ${insert} IN posts RETURN NEW`;
382
+ try {
383
+ const savedPost = await db.query(aqlQry).then((cursor) => cursor.next() || {});
384
+ const { _id: postDocId } = savedPost;
385
+ console.log({ tags });
386
+ if (tags && tags.length) {
387
+ const tagNames = tags.map(({ name: name2 }) => (0, import_utils.parseChar)(name2, 32));
388
+ console.log({ tagNames, postDocId });
389
+ await (0, import_tags.linkTags)(db, tagNames, postDocId);
390
+ } else {
391
+ const tagList = await (0, import_tags.extractTags)(db, postDocId, insert.content);
392
+ savedPost.tags = tagList;
393
+ }
394
+ return savedPost;
395
+ } catch (error) {
396
+ throw error;
397
+ }
398
+ };
399
+ const updatePost = async (context, post) => {
400
+ const { database, session: { userId: sessionId } } = context;
401
+ const now = Date.now();
402
+ const {
403
+ content,
404
+ endDate,
405
+ groupId,
406
+ name,
407
+ parentId,
408
+ postId,
409
+ privacy,
410
+ startDate,
411
+ tags = [],
412
+ type
413
+ } = post;
414
+ const update = {
415
+ content: content ? (0, import_utils.parseString)(content, MAX_CONTENT_LENGTH) : void 0,
416
+ endDate: endDate ? (0, import_utils.parseNum)(endDate, 13) : void 0,
417
+ modified: now,
418
+ name: name ? (0, import_utils.parseString)(name, 160) : void 0,
419
+ parentId: parentId ? (0, import_utils.parseString)(parentId, 160) : void 0,
420
+ privacy: privacy ? (0, import_utils.parseVarChar)(privacy, 16) : void 0,
421
+ startDate: startDate ? (0, import_utils.parseNum)(startDate, 13) : void 0,
422
+ type: type !== void 0 ? (0, import_utils.parseChar)(type, 16) : void 0
423
+ };
424
+ let formatId = (0, import_utils.parseId)(postId);
425
+ formatId = formatId === "" ? (0, import_utils.createHash)(`post-${sessionId}`) : formatId;
426
+ const formatGroupId = (0, import_utils.parseId)(groupId);
427
+ const insert = __spreadProps(__spreadValues({}, update), {
428
+ _key: formatId,
429
+ added: now,
430
+ groupId: formatGroupId,
431
+ privacy,
432
+ userId: sessionId
433
+ });
434
+ const db = database;
435
+ const aqlQry = import_arangojs.aql`UPSERT {_key: ${formatId}, userId: ${sessionId}}
436
+ INSERT ${insert}
437
+ UPDATE ${update}
438
+ IN posts RETURN NEW`;
439
+ try {
440
+ const updatedPost = await db.query(aqlQry).then((cursor) => cursor.next() || {});
441
+ const { _id: updatedPostId } = updatedPost;
442
+ if (tags && tags.length) {
443
+ const tagNames = tags.map(({ name: name2 }) => (0, import_utils.parseChar)(name2, 32));
444
+ const tagQuery = import_arangojs.aql`FOR t IN tags
445
+ FILTER POSITION(${JSON.stringify(tagNames)}, t.name)
446
+ RETURN t`;
447
+ const existingTags = await db.query(tagQuery).then((cursor) => cursor.all());
448
+ const tagIds = existingTags.map(({ _id: tagId }) => tagId);
449
+ await (0, import_tags.linkTags)(db, tagIds, updatedPostId);
450
+ } else {
451
+ const tagList = await (0, import_tags.extractTags)(db, updatedPostId, update.content || "") || [];
452
+ updatedPost.tags = tagList;
453
+ }
454
+ const files = updatedPost.files || [];
455
+ if (files.length) {
456
+ const fileList = await (0, import_files.updateFiles)(db, formatId, files) || [];
457
+ updatedPost.files = fileList;
458
+ } else {
459
+ updatedPost.files = [];
460
+ }
461
+ return updatedPost;
462
+ } catch (error) {
463
+ throw error;
464
+ }
465
+ };
466
+ const deletePost = (context, postId) => {
467
+ const { database, session: { userId: sessionId } } = context;
468
+ const formatItemId = (0, import_utils.parseId)(postId);
469
+ const db = database;
470
+ const aqlQry = import_arangojs.aql`FOR p IN posts
471
+ FILTER p._key == ${formatItemId} && p.userId == ${sessionId}
472
+ LIMIT 1
473
+ REMOVE p IN posts
474
+ RETURN OLD`;
475
+ return db.query(aqlQry).then((cursor) => cursor.next()).then((post = {}) => {
476
+ if (post) {
477
+ const edgeAqlQry = import_arangojs.aql`FOR t IN isTagged
478
+ FILTER t._to == ${formatItemId}
479
+ REMOVE t IN isTagged`;
480
+ return db.query(edgeAqlQry).then(() => {
481
+ const fileAqlQry = import_arangojs.aql`FOR f IN hasFile
482
+ FILTER f._to == ${formatItemId}
483
+ REMOVE f IN hasFile`;
484
+ return db.query(fileAqlQry).then(() => post).catch((error) => {
485
+ throw error;
486
+ });
487
+ }).catch((error) => {
488
+ throw error;
489
+ });
490
+ }
491
+ return {};
492
+ }).catch((error) => {
493
+ throw error;
494
+ });
495
+ };
496
+ const cleanPosts = (database) => {
497
+ const aqlQry = import_arangojs.aql`FOR p IN posts
498
+ FILTER p.added < DATE_TIMESTAMP(DATE_SUBTRACT(DATE_NOW(), 60, 'day')) && p.type == "default"
499
+ REMOVE p IN posts
500
+ RETURN OLD`;
501
+ return database.query(aqlQry).then((cursor) => cursor.all()).then((results = []) => results.length).catch((error) => {
502
+ throw error;
503
+ });
504
+ };
505
+ const createPostEdge = (db, file, postId) => {
506
+ const edgeCollection = db.collection("isPosted");
507
+ const { fileId } = file;
508
+ const formatFileId = (0, import_utils.parseId)(fileId);
509
+ const edgeId = (0, import_utils.createHash)(`file-${postId}-${formatFileId}`);
510
+ const formatPostId = (0, import_utils.parseId)(postId);
511
+ const fileType = (0, import_utils.parseChar)(file.fileType, 16);
512
+ const edge = {
513
+ _from: `posts/${formatPostId}`,
514
+ _key: edgeId,
515
+ _to: `files/${formatFileId}`,
516
+ added: Date.now(),
517
+ type: fileType
518
+ };
519
+ return edgeCollection.save(edge, { returnNew: true }).then(() => file).catch((error) => {
520
+ throw error;
521
+ });
522
+ };
523
+ // Annotate the CommonJS export names for ESM import in node:
524
+ 0 && (module.exports = {
525
+ addPost,
526
+ cleanPosts,
527
+ createPostEdge,
528
+ deletePost,
529
+ getPost,
530
+ getPostComments,
531
+ getPostOptional,
532
+ getPostsByReactions,
533
+ getPostsByTags,
534
+ getPostsByUser,
535
+ parsePostOptions,
536
+ updatePost
537
+ });
538
+ //# 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 {createHash, parseChar, parseId, parseNum, parseString, parseVarChar} from '@nlabs/utils';\nimport {aql, Database} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\n\nimport {ApiContext, FileType, PostInputType, PostOptions, PostType, TagType} from '../types';\nimport {getLimit, logError} from '../utils';\nimport {updateFiles} from './files';\nimport {extractTags, linkTags} from './tags';\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),\n    longitude: parseNum(longitude, 32),\n    type: parseChar(type, 32)\n  };\n};\n\nexport const getPostOptional = (fields: string[], sessionId: string) =>\n  fields.reduce((selects: any, field: string) => {\n    switch(field) {\n      case 'hasRsvp': {\n        selects.queries.push(`LET hasRsvp = TO_BOOL(FIRST(\n          FOR post, r IN INBOUND p._id hasReactions\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 hasReactions\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 hasReactions\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 hasReactions\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 hasReactions\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 = (context: ApiContext, itemId: string, options: PostOptions): Promise<Partial<PostType>> => {\n  const action: string = 'getPost';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(itemId);\n  const {type} = parsePostOptions(options);\n  const db = database;\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n    FILTER p._key == ${formatItemId} && p.type == ${type}\n    LIMIT 1\n    RETURN p`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType = {}) => {\n      const {\n        _id: postDocId,\n        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 = `LET p = DOCUMENT(\"${postDocId}\")\n          ${selectQueries.join('\\n')}\n          FOR group IN groups\n          FILTER group._key == p.groupId\n          FOR u, e IN OUTBOUND group._id isGrouped\n          FILTER u._key == \"${sessionId}\"\n          LIMIT 1\n          RETURN MERGE(p, {${selectObjects.join(', ')}})`;\n      } else if(privacy === 'public') {\n        privacyAqlQry = `LET p = DOCUMENT(\"${postDocId}\")\n          ${selectQueries.join('\\n')}\n          LIMIT 1\n          RETURN MERGE(p, {${selectObjects.join(', ')}})`;\n      }\n\n      if(privacyAqlQry) {\n        return db.query(privacyAqlQry)\n          .then((cursor: ArrayCursor) => cursor.next() || {})\n          .catch((error: Error) => logError({\n            action,\n            category: eventCategory,\n            label: 'db_error'\n          }, error, context).then(() => ({})));\n      }\n\n      return {};\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => ({})));\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//   console.log('getPostList::context', context);\n//   const {limit, type} = parsePostOptions(options);\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n//   const aqlQry: string = `FOR p IN posts\n//     FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n//     ${selectQueries.join('\\n')}\n//     ${limit.aql}\n//     SORT p.added\n//     RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//   return database.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .catch((error: Error) => {\n//       console.log('getPostList::error', error);\n//       throw error;\n//     });\n// };\n\n// export const getPostsByArea = (\n//   context: ApiContext,\n//   latitude: number,\n//   longitude: number,\n//   options?: PostOptions\n// ): Promise<PostType[]> => {\n//   // const action: string = 'getListByUser';\n//   const {database, fields, session: {userId: sessionId}} = context;\n//   const {limit, type} = parsePostOptions(options);\n//   const formatLatitude: number = parseNum(latitude);\n//   const formatLongitude: number = parseNum(longitude);\n//   const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n//   selectQueries.push(`LET distance = DISTANCE(\n//     ${formatLatitude},\n//     ${formatLongitude},\n//     NOT_NULL(p.latitude, 0),\n//     NOT_NULL(p.longitude, 0))\n//   `);\n//   selectObjects.push('distance:distance');\n\n//   const aqlQry: string = `FOR p IN posts\n//     ${selectQueries.join('\\n')}\n//     FILTER p.type == \"${type}\" && p.privacy == \"public\" && p.parentId == null\n//     ${limit.aql}\n//     SORT distance, p.added\n//     RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//   return database.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .catch((error: Error) => {\n//       throw error;\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: ArrayCursor) => cursor.all())\n//     .then((groups: GroupType[] = []) => {\n//       if(groups.length) {\n//         const {limit, type} = parsePostOptions(options);\n//         const postAqlQry: string = `FOR p IN posts\n//           FILTER p.type == \"${type}\" && p.groupId == \"${formatGroupId}\" && p.parent == null\n//           ${selectQueries.join('\\n')}\n//           ${limit.aql}\n//           SORT p.added\n//           RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n//         return db.query(postAqlQry)\n//           .then((cursor: ArrayCursor) => cursor.all())\n//           .catch((error: Error) => {\n//             throw error;\n//           });\n//       }\n\n//       return [];\n//     })\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\n// export const getPostsByLatest = (context: ApiContext, options?: PostOptions): Promise<PostType[]> => {\n//   // const action: string = 'getListByLatest';\n//   console.log('getPostsByLatest::options', options);\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//   console.log('getPostsByLatest::aqlQry', aqlQry);\n//   return database.query(aqlQry)\n//     .then((cursor: ArrayCursor) => cursor.all())\n//     .catch((error: Error) => {\n//       throw error;\n//     });\n// };\n\nexport const getPostsByReactions = (\n  context: ApiContext,\n  reactions: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  const action: string = 'getPostsByReactions';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {latitude, limit, longitude, type} = parsePostOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatReactions: string = JSON.stringify(reactions.map((reaction) => parseChar(reaction, 32).toLowerCase()));\n  const sortBy: string[] = [];\n  const filters: string[] = [`p.type == \"${type}\"`, 'p.privacy == \"public\"'];\n  const formatLatitude: number = parseNum(latitude);\n  const formatLongitude: number = parseNum(longitude);\n\n  if(formatLatitude && formatLongitude) {\n    selectQueries.push(`LET distance = DISTANCE(\n      ${formatLatitude},\n      ${formatLongitude},\n      NOT_NULL(p.latitude, 0),\n      NOT_NULL(p.longitude, 0))\n    `);\n    selectObjects.push('distance:distance');\n    sortBy.push('distance');\n  }\n\n  if(reactions.length) {\n    sortBy.push('matchedTags DESC');\n    selectQueries.push(`LET matchedReactions = LENGTH(\n      FOR mr IN reactions\n      FILTER mr.matched == true\n      RETURN mr\n    )`);\n    selectObjects.push('matchedReactions:matchedReactions');\n    filters.push('matchedReactions > 0');\n  }\n\n  sortBy.push('p.added DESC');\n  selectObjects.push('reactions:reactions');\n\n  // Get data from database\n  const aqlQry: string = `FOR p, r IN OUTBOUND \"${formatSessionId}\" hasReactions\n    LET reactions = (\n      FOR reaction, hr IN 1..1 INBOUND p isTagged\n      LET matched = LENGTH(${formatReactions}) > 0 && POSITION(${formatReactions}, reaction.name)\n      SORT reaction.name\n      RETURN MERGE(reaction, {matched:matched})\n    )\n    ${selectQueries.join('\\n')}\n    FILTER ${filters.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: 'db_error'\n    }, error, context).then(() => []));\n};\n\nexport const getPostsByTags = (\n  context: ApiContext,\n  tags: string[] = [],\n  options?: PostOptions\n): Promise<PostType[]> => {\n  // const action: string = 'getListByTags';\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  console.log({aqlQry, options});\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch(() => []);\n};\n\nexport const getPostsByUser = (context: ApiContext, userId: string, options?: PostOptions): Promise<PostType[]> => {\n  // const action: string = 'getListByUser';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatUserId: string = parseId(userId);\n  const {objects: selectObjects, queries: selectQueries} = getPostOptional(fields, sessionId);\n  const aqlQry: string = `FOR p IN posts\n    FILTER p.userId == \"${formatUserId}\" && p.type == \"${type}\" && p.privacy == \"public\" && p.parent == null\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT p.added\n    RETURN DISTINCT MERGE(p, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getPostComments = (context: ApiContext, itemId: string, options?: PostOptions): Promise<PostType[]> => {\n  // const action: string = 'getComments';\n  const {database, session: {userId: sessionId}} = context;\n  const {limit, type} = parsePostOptions(options);\n  const formatItemId: string = parseId(itemId);\n\n  // Get the parent post to get restrictions\n  const db = database;\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 db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType = {}) => {\n      const {\n        _key,\n        groupId,\n        privacy = 'public'\n      }: PostType = post;\n\n      // Query based on privacy level\n      let privacyAqlQry: string;\n\n      if(groupId && privacy === 'group') {\n        privacyAqlQry = `FOR p IN posts\n          FOR user IN users\n          FILTER p.parent == \"${_key}\" && user._key == p.userId\n          LET reactions = (\n            FOR post, r IN INBOUND p._id reactions\n            COLLECT reactionName = r.value INTO reactionItems\n            RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n          )\n          FOR group IN groups\n          FILTER group._key == p.groupId\n          FOR u, e IN OUTBOUND group._id isGrouped\n          FILTER u._key == \"${sessionId}\"\n          SORT p.added\n          ${limit.aql}\n          RETURN MERGE(p, {user: user, reactions: reactions})`;\n      } else if(privacy === 'public') {\n        privacyAqlQry = `FOR p IN posts\n          FOR user IN users\n          FILTER p.parent == \"${_key}\" && user._key == p.userId\n          LET reactions = (\n            FOR post, r IN INBOUND p._id reactions\n            COLLECT reactionName = r.value INTO reactionItems\n            RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n          )\n          SORT p.added\n          ${limit.aql}\n          RETURN MERGE(p, {user: user, reactions: reactions})`;\n      }\n\n      if(privacyAqlQry) {\n        return db.query(privacyAqlQry)\n          .then((cursor: ArrayCursor) => cursor.all())\n          .catch((error: Error) => {\n            throw error;\n          });\n      }\n\n      return [];\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const addPost = async (context: ApiContext, post: PostInputType): Promise<PostType> => {\n  // const action: string = 'add';\n  const {database, session: {userId: sessionId}} = context;\n\n  const {\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  } = post;\n\n  const now: number = Date.now();\n\n  const insert: PostType = {\n    _key: createHash(`post-${sessionId}`),\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: privacy ? parseVarChar(privacy, 16) : undefined,\n    startDate: startDate ? parseNum(startDate, 13) : undefined,\n    type: parseChar(type, 32),\n    userId: sessionId\n  };\n\n  const db: Database = database;\n  const aqlQry: AqlQuery = aql`INSERT ${insert} IN posts RETURN NEW`;\n\n  try {\n    const savedPost: PostType = await db.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next() || {});\n\n    const {_id: postDocId} = savedPost;\n\n    console.log({tags});\n    if(tags && tags.length) {\n      const tagNames: string[] = tags.map(({name}) => parseChar(name, 32));\n      console.log({tagNames, postDocId});\n      await linkTags(db, tagNames, postDocId);\n    } else {\n      // Update linked tags within posts\n      const tagList: TagType[] = await extractTags(db, postDocId, insert.content);\n      savedPost.tags = tagList;\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 = 'update';\n  const {database, session: {userId: sessionId}} = context;\n  const now: number = Date.now();\n  const {\n    content,\n    endDate,\n    groupId,\n    name,\n    parentId,\n    postId,\n    privacy,\n    startDate,\n    tags = [],\n    type\n  } = post;\n\n  const update: PostType = {\n    content: content ? parseString(content, MAX_CONTENT_LENGTH) : undefined,\n    endDate: endDate ? parseNum(endDate, 13) : undefined,\n    modified: now,\n    name: name ? parseString(name, 160) : undefined,\n    parentId: parentId ? parseString(parentId, 160) : undefined,\n    privacy: privacy ? parseVarChar(privacy, 16) : undefined,\n    startDate: startDate ? parseNum(startDate, 13) : undefined,\n    type: type !== undefined ? parseChar(type, 16) : undefined\n  };\n\n  let formatId: string = parseId(postId);\n  formatId = formatId === '' ? createHash(`post-${sessionId}`) : formatId;\n  const formatGroupId: string = parseId(groupId);\n  const insert: any = {\n    ...update,\n    _key: formatId,\n    added: now,\n    groupId: formatGroupId,\n    privacy,\n    userId: sessionId\n  };\n  const db: Database = database;\n  const aqlQry: AqlQuery = aql`UPSERT {_key: ${formatId}, userId: ${sessionId}}\n    INSERT ${insert}\n    UPDATE ${update}\n    IN posts RETURN NEW`;\n\n  try {\n    const updatedPost: PostType = await db.query(aqlQry)\n      .then((cursor: ArrayCursor) => cursor.next() || {});\n\n    const {_id: updatedPostId} = updatedPost;\n\n    if(tags && tags.length) {\n      const tagNames: string[] = tags.map(({name}) => parseChar(name, 32));\n      const tagQuery: AqlQuery = aql`FOR t IN tags\n        FILTER POSITION(${JSON.stringify(tagNames)}, t.name)\n        RETURN t`;\n\n      const existingTags = await db.query(tagQuery).then((cursor: ArrayCursor) => cursor.all());\n      const tagIds: string[] = existingTags.map(({_id: tagId}) => tagId);\n      await linkTags(db, tagIds, updatedPostId);\n    } else {\n      // Update linked tags\n      const tagList = await extractTags(db, updatedPostId, update.content || '') || [];\n      updatedPost.tags = tagList;\n    }\n\n    // Update linked files\n    const files: FileType[] = updatedPost.files || [];\n\n    if(files.length) {\n      const fileList = await updateFiles(db, formatId, files) || [];\n      updatedPost.files = fileList;\n    } else {\n      updatedPost.files = [];\n    }\n\n    return updatedPost;\n  } catch(error) {\n    throw error;\n  }\n};\n\nexport const deletePost = (context: ApiContext, postId: string): Promise<PostType> => {\n  // const action: string = 'delete';\n  const {database, session: {userId: sessionId}} = context;\n  const formatItemId: string = parseId(postId);\n  const db: Database = database;\n  const aqlQry = aql`FOR p IN posts\n      FILTER p._key == ${formatItemId} && p.userId == ${sessionId}\n      LIMIT 1\n      REMOVE p IN posts\n      RETURN OLD`;\n\n  return db.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((post: PostType = {}) => {\n      if(post) {\n        // Remove tag links\n        const edgeAqlQry: AqlQuery = aql`FOR t IN isTagged\n            FILTER t._to == ${formatItemId}\n            REMOVE t IN isTagged`;\n\n        return db.query(edgeAqlQry)\n          .then(() => {\n            // Remove attached files\n            const fileAqlQry: AqlQuery = aql`FOR f IN hasFile\n                FILTER f._to == ${formatItemId}\n                REMOVE f IN hasFile`;\n\n            return db.query(fileAqlQry)\n              .then(() => post)\n              .catch((error: Error) => {\n                throw error;\n              });\n          })\n          .catch((error: Error) => {\n            throw error;\n          });\n      }\n      return {};\n    })\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const cleanPosts = (database: Database): Promise<number> => {\n  // Remove all messages that are over 60 days and not saved\n  const aqlQry: AqlQuery = aql`FOR p IN posts\n      FILTER p.added < DATE_TIMESTAMP(DATE_SUBTRACT(DATE_NOW(), 60, 'day')) && p.type == \"default\"\n      REMOVE p IN posts\n      RETURN OLD`;\n\n  return database.query(aqlQry)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .then((results: PostType[] = []) => results.length)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const createPostEdge = (db: Database, file: FileType, postId: string): Promise<FileType> => {\n  const edgeCollection: EdgeCollection = db.collection('isPosted');\n  const {fileId} = file;\n  const formatFileId: string = parseId(fileId);\n  const edgeId: string = createHash(`file-${postId}-${formatFileId}`);\n  const formatPostId: string = parseId(postId);\n  const fileType: string = parseChar(file.fileType, 16);\n\n  const edge: any = {\n    _from: `posts/${formatPostId}`,\n    _key: edgeId,\n    _to: `files/${formatFileId}`,\n    added: Date.now(),\n    type: fileType\n  };\n\n  return edgeCollection.save(edge, {returnNew: true})\n    .then(() => file)\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n"],
  "mappings": "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",
  "names": []
}
