@nlabs/reaktor 0.3.0 → 0.4.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 (128) 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/actions/notifications.js +63 -0
  17. package/lib/{data → actions}/payments.d.ts +2 -2
  18. package/lib/actions/payments.js +491 -0
  19. package/lib/actions/posts.d.ts +19 -0
  20. package/lib/actions/posts.js +538 -0
  21. package/lib/actions/reactions.d.ts +30 -0
  22. package/lib/actions/reactions.js +340 -0
  23. package/lib/{data → actions}/s3.d.ts +1 -1
  24. package/lib/actions/s3.js +122 -0
  25. package/lib/{data → actions}/search.d.ts +2 -2
  26. package/lib/actions/search.js +99 -0
  27. package/lib/actions/sms.js +76 -0
  28. package/lib/actions/statistics.d.ts +2 -0
  29. package/lib/actions/statistics.js +63 -0
  30. package/lib/actions/subscription.js +209 -0
  31. package/lib/actions/tags.d.ts +26 -0
  32. package/lib/actions/tags.js +340 -0
  33. package/lib/actions/users.d.ts +44 -0
  34. package/lib/actions/users.js +571 -0
  35. package/lib/{data → actions}/websockets.d.ts +1 -1
  36. package/lib/actions/websockets.js +156 -0
  37. package/lib/config.d.ts +2 -3
  38. package/lib/config.js +120 -0
  39. package/lib/index.d.ts +1 -1
  40. package/lib/index.js +23 -0
  41. package/lib/templates/email/layout.d.ts +2 -0
  42. package/lib/templates/email/layout.js +292 -0
  43. package/lib/templates/email/passwordForgot.d.ts +2 -0
  44. package/lib/templates/email/passwordForgot.js +28 -0
  45. package/lib/templates/email/passwordRecovery.d.ts +2 -0
  46. package/lib/templates/email/passwordRecovery.js +25 -0
  47. package/lib/templates/email/verifyEmail.d.ts +2 -0
  48. package/lib/templates/email/verifyEmail.js +28 -0
  49. package/lib/templates/email/welcome.d.ts +2 -0
  50. package/lib/templates/email/welcome.js +28 -0
  51. package/lib/templates/sms/passwordForgot.d.ts +2 -0
  52. package/lib/templates/sms/passwordForgot.js +14 -0
  53. package/lib/templates/sms/passwordRecovery.d.ts +2 -0
  54. package/lib/templates/sms/passwordRecovery.js +14 -0
  55. package/lib/templates/sms/verifyEmail.d.ts +2 -0
  56. package/lib/templates/sms/verifyEmail.js +14 -0
  57. package/lib/templates/sms/verifyPhone.d.ts +2 -0
  58. package/lib/templates/sms/verifyPhone.js +14 -0
  59. package/lib/templates/sms/welcome.d.ts +2 -0
  60. package/lib/templates/sms/welcome.js +14 -0
  61. package/lib/types/apps.d.ts +2 -2
  62. package/lib/types/apps.js +4 -0
  63. package/lib/types/arangodb.js +4 -0
  64. package/lib/types/auth.d.ts +4 -8
  65. package/lib/types/auth.js +4 -0
  66. package/lib/types/conversations.d.ts +5 -3
  67. package/lib/types/conversations.js +4 -0
  68. package/lib/types/email.d.ts +2 -2
  69. package/lib/types/email.js +4 -0
  70. package/lib/types/files.js +4 -0
  71. package/lib/types/google.js +4 -0
  72. package/lib/types/groups.d.ts +2 -1
  73. package/lib/types/groups.js +4 -0
  74. package/lib/types/images.d.ts +8 -5
  75. package/lib/types/images.js +4 -0
  76. package/lib/types/index.d.ts +1 -1
  77. package/lib/types/index.js +37 -0
  78. package/lib/types/locations.js +4 -0
  79. package/lib/types/messages.d.ts +12 -2
  80. package/lib/types/messages.js +4 -0
  81. package/lib/types/notifications.d.ts +2 -2
  82. package/lib/types/notifications.js +4 -0
  83. package/lib/types/payments.js +4 -0
  84. package/lib/types/posts.d.ts +18 -1
  85. package/lib/types/posts.js +4 -0
  86. package/lib/types/statistics.d.ts +3 -0
  87. package/lib/types/statistics.js +4 -0
  88. package/lib/types/tags.d.ts +6 -0
  89. package/lib/types/tags.js +4 -0
  90. package/lib/types/users.d.ts +15 -10
  91. package/lib/types/users.js +4 -0
  92. package/lib/utils/analytics.d.ts +7 -0
  93. package/lib/utils/analytics.js +107 -0
  94. package/lib/utils/arangodb.d.ts +1 -1
  95. package/lib/utils/arangodb.js +122 -0
  96. package/lib/utils/auth.js +78 -0
  97. package/lib/utils/graphql.js +40 -0
  98. package/lib/utils/index.d.ts +1 -1
  99. package/lib/utils/index.js +26 -0
  100. package/lib/utils/objects.js +53 -0
  101. package/lib/utils/session.d.ts +18 -0
  102. package/lib/utils/session.js +42 -0
  103. package/package.json +35 -33
  104. package/lib/data/conversations.d.ts +0 -8
  105. package/lib/data/images.d.ts +0 -21
  106. package/lib/data/messages.d.ts +0 -9
  107. package/lib/data/posts.d.ts +0 -23
  108. package/lib/data/reactions.d.ts +0 -14
  109. package/lib/data/tags.d.ts +0 -14
  110. package/lib/data/users.d.ts +0 -17
  111. package/lib/types/reactions.d.ts +0 -15
  112. package/lib/utils/redis.d.ts +0 -1
  113. package/templates/email/layout.html +0 -279
  114. package/templates/email/passwordForgot.html +0 -15
  115. package/templates/email/passwordRecovery.html +0 -12
  116. package/templates/email/verifyEmail.html +0 -15
  117. package/templates/sms/passwordForgot.txt +0 -1
  118. package/templates/sms/passwordRecovery.txt +0 -1
  119. package/templates/sms/verifyEmail.txt +0 -1
  120. package/templates/sms/verifyPhone.txt +0 -1
  121. /package/lib/{data → actions}/dynamodb.d.ts +0 -0
  122. /package/lib/{data → actions}/email.d.ts +0 -0
  123. /package/lib/{data → actions}/files.d.ts +0 -0
  124. /package/lib/{data → actions}/index.d.ts +0 -0
  125. /package/lib/{data → actions}/locations.d.ts +0 -0
  126. /package/lib/{data → actions}/notifications.d.ts +0 -0
  127. /package/lib/{data → actions}/sms.d.ts +0 -0
  128. /package/lib/{data → actions}/subscription.d.ts +0 -0
@@ -0,0 +1,571 @@
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 __objRest = (source, exclude) => {
26
+ var target = {};
27
+ for (var prop in source)
28
+ if (__hasOwnProp.call(source, prop) && exclude.indexOf(prop) < 0)
29
+ target[prop] = source[prop];
30
+ if (source != null && __getOwnPropSymbols)
31
+ for (var prop of __getOwnPropSymbols(source)) {
32
+ if (exclude.indexOf(prop) < 0 && __propIsEnum.call(source, prop))
33
+ target[prop] = source[prop];
34
+ }
35
+ return target;
36
+ };
37
+ var __export = (target, all) => {
38
+ __markAsModule(target);
39
+ for (var name in all)
40
+ __defProp(target, name, { get: all[name], enumerable: true });
41
+ };
42
+ var __reExport = (target, module2, desc) => {
43
+ if (module2 && typeof module2 === "object" || typeof module2 === "function") {
44
+ for (let key of __getOwnPropNames(module2))
45
+ if (!__hasOwnProp.call(target, key) && key !== "default")
46
+ __defProp(target, key, { get: () => module2[key], enumerable: !(desc = __getOwnPropDesc(module2, key)) || desc.enumerable });
47
+ }
48
+ return target;
49
+ };
50
+ var __toModule = (module2) => {
51
+ 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);
52
+ };
53
+ __export(exports, {
54
+ UserAccess: () => UserAccess,
55
+ addUser: () => addUser,
56
+ confirmCode: () => confirmCode,
57
+ createToken: () => createToken,
58
+ deactivateUser: () => deactivateUser,
59
+ deleteUser: () => deleteUser,
60
+ getActiveUserCount: () => getActiveUserCount,
61
+ getDisplayName: () => getDisplayName,
62
+ getSessionUser: () => getSessionUser,
63
+ getUser: () => getUser,
64
+ getUserOptional: () => getUserOptional,
65
+ getUsers: () => getUsers,
66
+ getUsersByLatest: () => getUsersByLatest,
67
+ getUsersByReactions: () => getUsersByReactions,
68
+ getUsersByTags: () => getUsersByTags,
69
+ parseUserOptions: () => parseUserOptions,
70
+ refreshSession: () => refreshSession,
71
+ signIn: () => signIn,
72
+ signOut: () => signOut,
73
+ updateUser: () => updateUser
74
+ });
75
+ var import_utils = __toModule(require("@nlabs/utils"));
76
+ var import_arangojs = __toModule(require("arangojs"));
77
+ var import_isEmpty = __toModule(require("lodash/isEmpty"));
78
+ var import_luxon = __toModule(require("luxon"));
79
+ var import_stripe = __toModule(require("stripe"));
80
+ var import_config = __toModule(require("../config"));
81
+ var import_utils2 = __toModule(require("../utils"));
82
+ var import_session = __toModule(require("../utils/session"));
83
+ const eventCategory = "users";
84
+ const apiVersion = "2020-03-02";
85
+ var UserAccess;
86
+ (function(UserAccess2) {
87
+ UserAccess2[UserAccess2["DEACTIVATED"] = 0] = "DEACTIVATED";
88
+ UserAccess2[UserAccess2["ACTIVE"] = 1] = "ACTIVE";
89
+ UserAccess2[UserAccess2["PREMIUM"] = 2] = "PREMIUM";
90
+ UserAccess2[UserAccess2["CONTENT_ADMIN"] = 3] = "CONTENT_ADMIN";
91
+ UserAccess2[UserAccess2["ADMIN"] = 4] = "ADMIN";
92
+ })(UserAccess || (UserAccess = {}));
93
+ const createToken = (userId, username, userAccess, expires = 15) => {
94
+ const now = import_luxon.DateTime.local();
95
+ const sessionExpires = now.plus({ minutes: expires });
96
+ const iat = Math.floor(now.toSeconds());
97
+ const exp = Math.floor(sessionExpires.toSeconds());
98
+ const token = (0, import_session.setSession)({
99
+ exp,
100
+ iat,
101
+ username,
102
+ userAccess,
103
+ userId
104
+ });
105
+ return {
106
+ expires: sessionExpires.toMillis(),
107
+ issued: now.toMillis(),
108
+ token
109
+ };
110
+ };
111
+ const getUserOptional = (fields = []) => fields.reduce((selects, field) => {
112
+ if (field.includes("Count")) {
113
+ return (0, import_utils2.selectReactionCountByType)("users", "u", field, selects);
114
+ }
115
+ return selects;
116
+ }, { objects: [], queries: [] });
117
+ const parseUserOptions = (options = {}) => {
118
+ const {
119
+ from = 0,
120
+ to = 30
121
+ } = options;
122
+ return __spreadProps(__spreadValues({}, options), {
123
+ limit: (0, import_utils2.getLimit)(from, to)
124
+ });
125
+ };
126
+ const addUser = async (context, args) => {
127
+ const action = "addUser";
128
+ const { database } = context;
129
+ const { user } = args;
130
+ const { email, password, phone, username } = user;
131
+ const salt = (0, import_utils.createHash)(`${username}${password}`, null);
132
+ const encryptedPassword = (0, import_utils.createPassword)(password, salt);
133
+ const formatUsername = (0, import_utils.parseUsername)(username);
134
+ const formatEmail = (0, import_utils.parseEmail)(email);
135
+ const formatPhone = (0, import_utils.parsePhone)(phone);
136
+ if ((0, import_isEmpty.default)(formatUsername) || (0, import_isEmpty.default)(password) || !(0, import_isEmpty.default)(formatPhone) && !(0, import_isEmpty.default)(formatEmail)) {
137
+ return (0, import_utils2.logException)({
138
+ action,
139
+ args: { username },
140
+ category: eventCategory,
141
+ value: import_utils2.ErrorTypes.INVALID_ARGUMENTS
142
+ }, context);
143
+ }
144
+ const filters = [`u.username == "${formatUsername}"`];
145
+ if (!(0, import_isEmpty.default)(formatEmail)) {
146
+ filters.push(`u.email == "${formatEmail}"`);
147
+ }
148
+ if (!(0, import_isEmpty.default)(formatPhone)) {
149
+ filters.push(`u.phone == ${formatPhone}`);
150
+ }
151
+ const checkQuery = `FOR u IN users
152
+ FILTER ${filters.join(" || ")}
153
+ RETURN u`;
154
+ const existingUsers = await database.query(checkQuery).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
155
+ action,
156
+ args: { username },
157
+ category: eventCategory,
158
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
159
+ }, error, context).then(() => null));
160
+ if (existingUsers.length) {
161
+ return (0, import_utils2.logException)({
162
+ action,
163
+ args: { username },
164
+ category: eventCategory,
165
+ value: import_utils2.ErrorTypes.EXISTING_USERNAME
166
+ }, context);
167
+ }
168
+ const verifiedEmailCode = Math.floor(1e5 + Math.random() * 9e5);
169
+ const verifiedPhoneCode = Math.floor(1e5 + Math.random() * 9e5);
170
+ const insert = {
171
+ _key: (0, import_utils.createHash)(username, null),
172
+ added: Date.now(),
173
+ email: formatEmail,
174
+ modified: Date.now(),
175
+ password: encryptedPassword,
176
+ phone: formatPhone,
177
+ salt,
178
+ username: formatUsername,
179
+ userAccess: 1,
180
+ verifiedEmail: false,
181
+ verifiedEmailCode,
182
+ verifiedPhone: false,
183
+ verifiedPhoneCode
184
+ };
185
+ const insertQuery = import_arangojs.aql`INSERT ${insert} IN users RETURN NEW`;
186
+ return await database.query(insertQuery).then((cursor) => cursor.next() || {}).catch((error) => (0, import_utils2.logError)({
187
+ action,
188
+ args: { username },
189
+ category: eventCategory,
190
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
191
+ }, error, context).then(() => null));
192
+ };
193
+ const updateUser = async (context, user) => {
194
+ const action = "updateUser";
195
+ const { database } = context;
196
+ const _a = user, { _key, _id, id, tags = [], userId } = _a, updated = __objRest(_a, ["_key", "_id", "id", "tags", "userId"]);
197
+ let userDocId;
198
+ if (_id || id) {
199
+ userDocId = _id || id;
200
+ } else if (_key || userId) {
201
+ userDocId = `users/${_key || userId}`;
202
+ }
203
+ const userQuery = import_arangojs.aql`LET u = DOCUMENT(${userDocId})
204
+ UPDATE u WITH ${updated} IN users
205
+ RETURN NEW`;
206
+ const updatedUser = await database.query(userQuery).then((cursor) => cursor.next()).catch((error) => {
207
+ console.log(error);
208
+ throw error;
209
+ });
210
+ const tagCollection = database.collection("isTagged");
211
+ await Promise.all(tags.map(({ id: tagDocId, name }) => {
212
+ const edge = {
213
+ _from: tagDocId,
214
+ _key: (0, import_utils.createHash)(`isTagged-${tagDocId}-${userDocId}`),
215
+ _to: userDocId,
216
+ added: Date.now(),
217
+ name
218
+ };
219
+ const tagQuery = import_arangojs.aql`FOR it IN isTagged
220
+ FILTER it._from == ${tagDocId} && it._to == ${userDocId} && it.name == ${name}
221
+ LIMIT 1
222
+ RETURN it`;
223
+ return database.query(tagQuery).then((cursor) => cursor.next()).then((tagEdge) => {
224
+ if (!!tagEdge) {
225
+ return tagEdge;
226
+ }
227
+ return tagCollection.save(edge, { returnNew: true }).then(() => edge);
228
+ }).catch((error) => (0, import_utils2.logError)({
229
+ action,
230
+ category: eventCategory,
231
+ label: "db_error"
232
+ }, error, context).then(() => null));
233
+ }));
234
+ return updatedUser;
235
+ };
236
+ const confirmCode = async (context, args) => {
237
+ const action = "confirmEmail";
238
+ const { database, session: { userId: sessionId } } = context;
239
+ const { code, type } = args;
240
+ const userDocId = `users/${sessionId}`;
241
+ const aqlQuery = import_arangojs.aql`LET u = DOCUMENT(${userDocId}) RETURN u`;
242
+ try {
243
+ return database.query(aqlQuery).then((cursor) => cursor.next() || {}).then(({ verifiedEmailCode, verifiedPhoneCode }) => {
244
+ switch (type) {
245
+ case "email":
246
+ return code === verifiedEmailCode;
247
+ case "phone":
248
+ return code === verifiedPhoneCode;
249
+ default:
250
+ return false;
251
+ }
252
+ }).catch((error) => (0, import_utils2.logError)({
253
+ action,
254
+ args: { code, type, userId: sessionId },
255
+ category: eventCategory,
256
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
257
+ }, error, context));
258
+ } catch (error) {
259
+ return false;
260
+ }
261
+ };
262
+ const deleteUser = (context, args) => {
263
+ const action = "deleteUser";
264
+ const { database, session: { userId: sessionId, userAccess: sessionAccess } } = context;
265
+ const { userId } = args;
266
+ const isAdmin = sessionAccess > 2;
267
+ if (!isAdmin && sessionId !== userId) {
268
+ (0, import_utils2.logException)({
269
+ action,
270
+ args,
271
+ category: eventCategory,
272
+ label: "unauthorized",
273
+ value: "invalid_session"
274
+ }, context);
275
+ return null;
276
+ }
277
+ const aqlQuery = import_arangojs.aql`FOR u IN users
278
+ FILTER u._key == ${userId}
279
+ LIMIT 1
280
+ REMOVE u IN users
281
+ RETURN OLD`;
282
+ const stripeClient = new import_stripe.default(import_config.Config.get("stripe.token"), { apiVersion, typescript: true });
283
+ return database.query(aqlQuery).then((cursor) => cursor.next()).then((user = {}) => stripeClient.customers.del(user.stripeCustomerId).then(() => stripeClient.accounts.del(user.stripeAccountId)).then(() => user)).catch((error) => {
284
+ throw error;
285
+ });
286
+ };
287
+ const deactivateUser = (context, userId) => {
288
+ const action = "delete";
289
+ const { database, session: { userId: sessionId, userAccess: sessionAccess } } = context;
290
+ const isAdmin = sessionAccess > 2;
291
+ if (!isAdmin && sessionId !== userId) {
292
+ (0, import_utils2.logException)({
293
+ action,
294
+ category: eventCategory,
295
+ label: "unauthorized",
296
+ value: "invalid_session"
297
+ }, context);
298
+ return null;
299
+ }
300
+ const updated = {
301
+ userAccess: 0
302
+ };
303
+ const aqlQuery = import_arangojs.aql`UPDATE ${userId} WITH ${updated} IN users LIMIT 1 RETURN NEW`;
304
+ return database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => {
305
+ throw error;
306
+ });
307
+ };
308
+ const getDisplayName = (user = {}) => {
309
+ const { first, last, name = "", username = "" } = user;
310
+ const fullname = [first, last].join(" ").trim();
311
+ if (!(0, import_isEmpty.default)(name)) {
312
+ return name;
313
+ } else if (fullname !== "") {
314
+ return fullname;
315
+ } else if (!(0, import_isEmpty.default)(username)) {
316
+ return username;
317
+ }
318
+ return "Unknown";
319
+ };
320
+ const getSessionUser = (context) => {
321
+ const action = "getSessionUser";
322
+ const { database, fields, session: { userId: sessionId, username } } = context;
323
+ const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
324
+ const aqlQuery = `LET u = DOCUMENT("users/${sessionId}")
325
+ ${selectQueries.join("\n")}
326
+ RETURN MERGE(u, {${selectObjects.join(", ")}})`;
327
+ return database.query(aqlQuery).then((cursor) => cursor.next()).then((user = {}) => user).catch((error) => (0, import_utils2.logError)({
328
+ action,
329
+ args: { username, userId: sessionId },
330
+ category: eventCategory,
331
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
332
+ }, error, context));
333
+ };
334
+ const getUser = (context, args) => {
335
+ const action = "getUser";
336
+ const { userId } = args;
337
+ const { database, fields } = context;
338
+ const formatUserId = (0, import_utils.parseId)(userId);
339
+ const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
340
+ const aqlQuery = `LET u = DOCUMENT("users/${formatUserId}")
341
+ ${selectQueries.join("\n")}
342
+ FILTER u.userAccess > 0
343
+ RETURN MERGE(u, {${selectObjects.join(", ")}})`;
344
+ return database.query(aqlQuery).then((cursor) => cursor.next()).then((user = {}) => user).catch((error) => (0, import_utils2.logError)({
345
+ action,
346
+ args,
347
+ category: eventCategory,
348
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
349
+ }, error, context).then(() => {
350
+ }));
351
+ };
352
+ const getUsers = (context, options) => {
353
+ const action = "getUserList";
354
+ const { database, fields } = context;
355
+ const { limit, username } = parseUserOptions(options);
356
+ const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
357
+ const filterBy = ["u.userAccess > 0"];
358
+ if (!(0, import_isEmpty.default)(username)) {
359
+ filterBy.push(`CONTAINS(u.username, "${(0, import_utils.parseUsername)(username)}")`);
360
+ }
361
+ const aqlQuery = `FOR u IN users
362
+ FILTER ${filterBy.join(" && ")}
363
+ ${selectQueries.join("\n")}
364
+ ${limit.aql}
365
+ SORT u.username
366
+ RETURN MERGE(u, {${selectObjects.join(", ")}})`;
367
+ return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
368
+ action,
369
+ category: eventCategory,
370
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
371
+ }, error, context).then(() => []));
372
+ };
373
+ const getUsersByReactions = (context, { reactions = [], username }, options) => {
374
+ const action = "getUsersByReactions";
375
+ const { database, fields, session: { userId: sessionId } } = context;
376
+ const formatReactions = reactions.map((reactionName) => (0, import_utils.parseChar)(reactionName, 32).toLowerCase());
377
+ const { limit } = parseUserOptions(options);
378
+ const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
379
+ const formatSessionId = `users/${sessionId}`;
380
+ const formatUsername = (0, import_utils.parseUsername)(username);
381
+ const filterBy = [`POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`];
382
+ if (!(0, import_isEmpty.default)(username)) {
383
+ filterBy.push(`CONTAINS(u.username, "${formatUsername}")`);
384
+ }
385
+ const aqlQuery = `FOR u, r IN OUTBOUND "${formatSessionId}" hasReactions
386
+ OPTIONS {vertexCollections: "users"}
387
+ ${selectQueries.join("\n")}
388
+ ${filterBy.length ? `FILTER ${filterBy.join(" && ")}` : ""}
389
+ ${limit.aql}
390
+ RETURN MERGE(u, {${selectObjects.join(", ")}})`;
391
+ console.log({ aqlQuery });
392
+ return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
393
+ action,
394
+ category: eventCategory,
395
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
396
+ }, error, context).then(() => []));
397
+ };
398
+ const getUsersByTags = (context, { tags, username }, options) => {
399
+ const action = "getUsersByTags";
400
+ const { database, fields, session: { userId: sessionId } } = context;
401
+ const formatTags = tags.reduce((list, tagName) => {
402
+ if (!(0, import_isEmpty.default)(tagName)) {
403
+ list.push((0, import_utils.parseChar)(tagName, 32).toLowerCase());
404
+ }
405
+ return list;
406
+ }, []);
407
+ const { limit } = parseUserOptions(options);
408
+ const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
409
+ const formatUsername = (0, import_utils.parseUsername)(username);
410
+ const filterBy = [`u._key != "${sessionId}"`];
411
+ if (!(0, import_isEmpty.default)(username)) {
412
+ filterBy.push(`CONTAINS(u.username, "${formatUsername}")`);
413
+ }
414
+ const aqlQuery = `FOR t IN tags
415
+ FILTER POSITION(${JSON.stringify(formatTags)}, LOWER(t.name))
416
+ FOR u, it IN OUTBOUND t isTagged
417
+ OPTIONS {bfs: true, uniqueVertices: "global", vertexCollections: "users"}
418
+ ${selectQueries.join("\n")}
419
+ ${filterBy.length ? `FILTER ${filterBy.join(" && ")}` : ""}
420
+ ${limit.aql}
421
+ RETURN DISTINCT MERGE(u, {${selectObjects.join(", ")}})`;
422
+ console.log("getUsersByTags", aqlQuery);
423
+ return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
424
+ action,
425
+ category: eventCategory,
426
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
427
+ }, error, context).then(() => []));
428
+ };
429
+ const getUsersByLatest = (context, { username }, options) => {
430
+ const action = "getUsersByLatest";
431
+ const { database, fields, session: { userId } } = context;
432
+ const { limit } = parseUserOptions(options);
433
+ const filter = ["u._id != session._id"];
434
+ const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
435
+ if (!(0, import_isEmpty.default)(username)) {
436
+ filter.push(`CONTAINS(u.username, "${(0, import_utils.parseUsername)(username)}")`);
437
+ }
438
+ const aqlQuery = `FOR u IN users
439
+ LET session = DOCUMENT("users/${userId}")
440
+ FILTER ${filter.join(" && ")}
441
+ ${selectQueries.join("\n")}
442
+ LET distance = DISTANCE(u.latitude || 0, u.longitude || 0, session.latitude || 0, session.longitude || 0)
443
+ ${limit.aql}
444
+ SORT distance ASC, u.added DESC
445
+ RETURN MERGE(u, {${selectObjects.join(", ")}})`;
446
+ return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => (0, import_utils2.logError)({
447
+ action,
448
+ category: eventCategory,
449
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
450
+ }, error, context).then(() => []));
451
+ };
452
+ const refreshSession = async (context, { expires, token }) => {
453
+ const { error } = (0, import_session.getSession)(token);
454
+ if (error) {
455
+ return { errors: [error] };
456
+ }
457
+ const { exp, userId, username, userAccess } = (0, import_session.getSession)(token);
458
+ const now = Math.ceil(import_luxon.DateTime.local().toSeconds());
459
+ if (exp > now) {
460
+ return createToken(userId, username, userAccess, expires);
461
+ }
462
+ return { errors: ["session_expired"] };
463
+ };
464
+ const signIn = async (context, args) => {
465
+ const action = "signIn";
466
+ const { expires, password, username } = args;
467
+ const formatUsername = (0, import_utils.parseUsername)(username);
468
+ const formatPassword = (0, import_utils.parsePassword)(password);
469
+ const formatExpires = (0, import_utils.parseNum)(expires) || 15;
470
+ const { database } = context;
471
+ if ((0, import_isEmpty.default)(formatUsername) || (0, import_isEmpty.default)(formatPassword)) {
472
+ return (0, import_utils2.logException)({
473
+ action,
474
+ args: { username },
475
+ category: eventCategory,
476
+ value: import_utils2.ErrorTypes.INVALID_ARGUMENTS
477
+ }, context);
478
+ }
479
+ const checkQuery = import_arangojs.aql`FOR u IN users
480
+ FILTER u.username == ${formatUsername}
481
+ LIMIT 1
482
+ RETURN u`;
483
+ const checkUser = await database.query(checkQuery).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
484
+ action,
485
+ args: { username: formatUsername },
486
+ category: eventCategory,
487
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
488
+ }, error, context).then(() => null));
489
+ if (!checkUser) {
490
+ return (0, import_utils2.logException)({
491
+ action,
492
+ args: { username },
493
+ category: eventCategory,
494
+ value: import_utils2.ErrorTypes.INVALID_AUTHENTICATION
495
+ }, context);
496
+ }
497
+ const { _key: userId, salt, userAccess } = checkUser;
498
+ const authPassword = (0, import_utils.createPassword)(formatPassword, salt);
499
+ if (checkUser.password !== authPassword) {
500
+ return (0, import_utils2.logException)({
501
+ action,
502
+ args: { username },
503
+ category: eventCategory,
504
+ value: import_utils2.ErrorTypes.INVALID_AUTHENTICATION
505
+ }, context);
506
+ }
507
+ return createToken(userId, username, userAccess, formatExpires);
508
+ };
509
+ const signOut = async (context, args) => {
510
+ const action = "signOut";
511
+ const { database, session: { userId: sessionId, username } } = context;
512
+ const userDocId = `users/${sessionId}`;
513
+ const update = {
514
+ lastOnline: Date.now(),
515
+ sessionId: null
516
+ };
517
+ const sessionQuery = import_arangojs.aql`LET u = DOCUMENT(${userDocId})
518
+ UPDATE u WITH ${update} IN users
519
+ LIMIT 1
520
+ RETURN NEW`;
521
+ try {
522
+ await database.query(sessionQuery).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
523
+ action,
524
+ args: { username, userId: sessionId },
525
+ category: eventCategory,
526
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
527
+ }, error, context));
528
+ } catch (error) {
529
+ return false;
530
+ }
531
+ return true;
532
+ };
533
+ const getActiveUserCount = (context) => {
534
+ const action = "getActiveUserCount";
535
+ const { database } = context;
536
+ const countQuery = import_arangojs.aql`LET docs = (
537
+ FOR u IN users
538
+ FILTER u.active == true
539
+ RETURN u
540
+ )
541
+ RETURN LENGTH(docs)`;
542
+ return database.query(countQuery).then((cursor) => cursor.next()).catch((error) => (0, import_utils2.logError)({
543
+ action,
544
+ category: eventCategory,
545
+ label: import_utils2.ErrorTypes.DATABASE_ERROR
546
+ }, error, context).then(() => 0));
547
+ };
548
+ // Annotate the CommonJS export names for ESM import in node:
549
+ 0 && (module.exports = {
550
+ UserAccess,
551
+ addUser,
552
+ confirmCode,
553
+ createToken,
554
+ deactivateUser,
555
+ deleteUser,
556
+ getActiveUserCount,
557
+ getDisplayName,
558
+ getSessionUser,
559
+ getUser,
560
+ getUserOptional,
561
+ getUsers,
562
+ getUsersByLatest,
563
+ getUsersByReactions,
564
+ getUsersByTags,
565
+ parseUserOptions,
566
+ refreshSession,
567
+ signIn,
568
+ signOut,
569
+ updateUser
570
+ });
571
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/users.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,\n  createPassword,\n  parseChar,\n  parseEmail,\n  parseId,\n  parseNum,\n  parsePassword,\n  parsePhone,\n  parseUsername\n} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport isEmpty from 'lodash/isEmpty';\nimport {DateTime} from 'luxon';\nimport Stripe from 'stripe';\n\nimport {Config} from '../config';\nimport {ApiContext} from '../types/auth';\nimport {User} from '../types/users';\nimport {ErrorTypes, getLimit, logError, logException, selectReactionCountByType} from '../utils';\nimport {getSession, SessionError, SessionToken, setSession} from '../utils/session';\n\nconst eventCategory: string = 'users';\nconst apiVersion: any = '2020-03-02';\n\nexport interface UserOptions {\n  readonly from?: number;\n  readonly to?: number;\n  readonly username?: string;\n}\n\nexport enum UserAccess {\n  DEACTIVATED = 0,\n  ACTIVE = 1,\n  PREMIUM = 2,\n  CONTENT_ADMIN = 3,\n  ADMIN = 4\n}\n\nexport const createToken = (\n  userId: string,\n  username: string,\n  userAccess: number,\n  expires: number = 15\n): SessionToken => {\n  const now: DateTime = DateTime.local();\n  const sessionExpires: DateTime = now.plus({minutes: expires});\n  const iat: number = Math.floor(now.toSeconds());\n  const exp: number = Math.floor(sessionExpires.toSeconds());\n\n  const token = setSession({\n    exp,\n    iat,\n    username,\n    userAccess,\n    userId\n  });\n\n  return {\n    expires: sessionExpires.toMillis(),\n    issued: now.toMillis(),\n    token\n  };\n};\n\nexport const getUserOptional = (fields: string[] = []) =>\n  fields.reduce((selects: any, field: string) => {\n    if(field.includes('Count')) {\n      return selectReactionCountByType('users', 'u', field, selects);\n    }\n\n    return selects;\n  }, {objects: [], queries: []});\n\nexport const parseUserOptions = (options: UserOptions = {}) => {\n  const {\n    from = 0,\n    to = 30\n  } = options;\n\n  return {\n    ...options,\n    limit: getLimit(from, to)\n  };\n};\n\nexport const addUser = async (context: ApiContext, args: any): Promise<User> => {\n  const action: string = 'addUser';\n  const {database} = context;\n  const {user} = args;\n  const {email, password, phone, username} = user;\n  const salt: string = createHash(`${username}${password}`, null);\n  const encryptedPassword = createPassword(password, salt);\n  const formatUsername: string = parseUsername(username);\n  const formatEmail: string = parseEmail(email);\n  const formatPhone: string = parsePhone(phone);\n\n  if(isEmpty(formatUsername) || isEmpty(password) || (!isEmpty(formatPhone) && !isEmpty(formatEmail))) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const filters: string[] = [`u.username == \"${formatUsername}\"`];\n\n  if(!isEmpty(formatEmail)) {\n    filters.push(`u.email == \"${formatEmail}\"`);\n  }\n\n  if(!isEmpty(formatPhone)) {\n    filters.push(`u.phone == ${formatPhone}`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    RETURN u`;\n  const existingUsers = await database.query(checkQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error) => logError({\n      action,\n      args: {username},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n\n  if(existingUsers.length) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.EXISTING_USERNAME\n    }, context);\n  }\n\n  const verifiedEmailCode: number = Math.floor(100000 + (Math.random() * 900000));\n  const verifiedPhoneCode: number = Math.floor(100000 + (Math.random() * 900000));\n\n  // Add new user properties\n  const insert: User = {\n    _key: createHash(username, null),\n    added: Date.now(),\n    email: formatEmail,\n    modified: Date.now(),\n    password: encryptedPassword,\n    phone: formatPhone,\n    salt,\n    username: formatUsername,\n    userAccess: 1,\n    verifiedEmail: false,\n    verifiedEmailCode,\n    verifiedPhone: false,\n    verifiedPhoneCode\n  };\n\n  // Add new user in ArangoDB\n  const insertQuery: AqlQuery = aql`INSERT ${insert} IN users RETURN NEW`;\n  return await database.query(insertQuery)\n    .then((cursor: ArrayCursor) => cursor.next() || {})\n    .catch((error) => logError({\n      action,\n      args: {username},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n};\n\nexport const updateUser = async (context: ApiContext, user: User): Promise<any> => {\n  const action: string = 'updateUser';\n  const {database} = context;\n  const {_key, _id, id, tags = [], userId, ...updated} = user;\n  let userDocId: string;\n\n  if(_id || id) {\n    userDocId = _id || id;\n  } else if(_key || userId) {\n    userDocId = `users/${_key || userId}`;\n  }\n\n  const userQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId})\n    UPDATE u WITH ${updated} IN users\n    RETURN NEW`;\n\n  const updatedUser = await database.query(userQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      console.log(error);\n      throw error;\n    });\n\n  const tagCollection: EdgeCollection = database.collection('isTagged');\n  await Promise.all(tags.map(({id: tagDocId, name}) => {\n    const edge = {\n      _from: tagDocId,\n      _key: createHash(`isTagged-${tagDocId}-${userDocId}`),\n      _to: userDocId,\n      added: Date.now(),\n      name\n    };\n    const tagQuery: AqlQuery = aql`FOR it IN isTagged\n      FILTER it._from == ${tagDocId} && it._to == ${userDocId} && it.name == ${name}\n      LIMIT 1\n      RETURN it`;\n\n    return database.query(tagQuery)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .then((tagEdge) => {\n        if(!!tagEdge) {\n          return tagEdge;\n        }\n\n        return tagCollection.save(edge, {returnNew: true}).then(() => edge);\n      })\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        label: 'db_error'\n      }, error, context).then(() => null));\n  }));\n\n  return updatedUser;\n};\n\nexport const confirmCode = async (context: ApiContext, args): Promise<boolean> => {\n  const action: string = 'confirmEmail';\n  const {database, session: {userId: sessionId}} = context;\n  const {code, type} = args;\n  const userDocId: string = `users/${sessionId}`;\n\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId}) RETURN u`;\n\n  try {\n    return database.query(aqlQuery)\n      .then((cursor) => cursor.next() || {})\n      .then(({verifiedEmailCode, verifiedPhoneCode}: User) => {\n        switch(type) {\n          case 'email':\n            return code === verifiedEmailCode;\n          case 'phone':\n            return code === verifiedPhoneCode;\n          default:\n            return false;\n        }\n      })\n      .catch((error) => logError({\n        action,\n        args: {code, type, userId: sessionId},\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context)\n      );\n  } catch(error) {\n    return false;\n  }\n};\n\nexport const deleteUser = (context: ApiContext, args: any): Promise<any> => {\n  const action: string = 'deleteUser';\n  const {database, session: {userId: sessionId, userAccess: sessionAccess}} = context;\n  const {userId} = args;\n  const isAdmin: boolean = sessionAccess > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      args,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const aqlQuery: AqlQuery = aql`FOR u IN users\n    FILTER u._key == ${userId}\n    LIMIT 1\n    REMOVE u IN users\n    RETURN OLD`;\n\n  // Stripe\n  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user: User = {}) => stripeClient.customers.del(user.stripeCustomerId)\n      .then(() => stripeClient.accounts.del(user.stripeAccountId))\n      .then(() => user))\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const deactivateUser = (context: ApiContext, userId: string): Promise<User> => {\n  const action: string = 'delete';\n  const {database, session: {userId: sessionId, userAccess: sessionAccess}} = context;\n  const isAdmin: boolean = sessionAccess > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const updated: User = {\n    userAccess: 0\n  };\n  const aqlQuery: AqlQuery = aql`UPDATE ${userId} WITH ${updated} IN users LIMIT 1 RETURN NEW`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getDisplayName = (user: User = {}): string => {\n  const {first, last, name = '', username = ''} = user;\n  const fullname: string = ([first, last]).join(' ').trim();\n\n  if(!isEmpty(name)) {\n    return name;\n  } else if(fullname !== '') {\n    return fullname;\n  } else if(!isEmpty(username)) {\n    return username;\n  }\n\n  return 'Unknown';\n};\n\nexport const getSessionUser = (context: ApiContext): Promise<User> => {\n  const action: string = 'getSessionUser';\n  const {database, fields, session: {userId: sessionId, username}} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const aqlQuery: string = `LET u = DOCUMENT(\"users/${sessionId}\")\n  ${selectQueries.join('\\n')}\n  RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      args: {username, userId: sessionId},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUser = (context: ApiContext, args: any): Promise<User> => {\n  const action: string = 'getUser';\n  const {userId} = args;\n  const {database, fields} = context;\n  const formatUserId: string = parseId(userId);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  // Get data from database\n  const aqlQuery: string = `LET u = DOCUMENT(\"users/${formatUserId}\")\n    ${selectQueries.join('\\n')}\n    FILTER u.userAccess > 0\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      args,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => { }));\n};\n\nexport const getUsers = (context: ApiContext, options?: UserOptions): Promise<User[]> => {\n  const action: string = 'getUserList';\n  const {database, fields} = context;\n  const {limit, username} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const filterBy: string[] = ['u.userAccess > 0'];\n\n  if(!isEmpty(username)) {\n    filterBy.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u IN users\n    FILTER ${filterBy.join(' && ')}\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT u.username\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByReactions = (\n  context: ApiContext,\n  {reactions = [], username}: any,\n  options?: UserOptions\n): Promise<User[]> => {\n  const action: string = 'getUsersByReactions';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatReactions: string[] =  reactions.map((reactionName: string) => parseChar(reactionName, 32).toLowerCase());\n  const {limit} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [`POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`];\n\n  if(!isEmpty(username)) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u, r IN OUTBOUND \"${formatSessionId}\" hasReactions\n    OPTIONS {vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    ${filterBy.length ? `FILTER ${filterBy.join(' && ')}` : ''}\n    ${limit.aql}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  console.log({aqlQuery});\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByTags = (\n  context: ApiContext,\n  {tags, username}: any,\n  options?: UserOptions\n): Promise<User[]> => {\n  const action: string = 'getUsersByTags';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatTags: string[] =  tags.reduce((list: string[], tagName: string) => {\n    if(!isEmpty(tagName)) {\n      list.push(parseChar(tagName, 32).toLowerCase());\n    }\n\n    return list;\n  }, []);\n  const {limit} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [`u._key != \"${sessionId}\"`];\n\n  if(!isEmpty(username)) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR t IN tags\n    FILTER POSITION(${JSON.stringify(formatTags)}, LOWER(t.name))\n    FOR u, it IN OUTBOUND t isTagged\n    OPTIONS {bfs: true, uniqueVertices: \"global\", vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    ${filterBy.length ? `FILTER ${filterBy.join(' && ')}` : ''}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(u, {${selectObjects.join(', ')}})`;\n\n  console.log('getUsersByTags', aqlQuery);\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByLatest = (context: ApiContext, {username}, options?: UserOptions): Promise<User[]> => {\n  const action: string = 'getUsersByLatest';\n  const {database, fields, session: {userId}} = context;\n  const {limit} = parseUserOptions(options);\n  const filter = ['u._id != session._id'];\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  if(!isEmpty(username)) {\n    filter.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u IN users\n    LET session = DOCUMENT(\"users/${userId}\")\n    FILTER ${filter.join(' && ')}\n    ${selectQueries.join('\\n')}\n    LET distance = DISTANCE(u.latitude || 0, u.longitude || 0, session.latitude || 0, session.longitude || 0)\n    ${limit.aql}\n    SORT distance ASC, u.added DESC\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const refreshSession = async (context: ApiContext, {expires, token}): Promise<SessionToken | SessionError> => {\n  const {error} = getSession(token);\n\n  if(error) {\n    return {errors: [error]};\n  }\n\n  const {exp, userId, username, userAccess} = getSession(token);\n  const now: number = Math.ceil(DateTime.local().toSeconds());\n\n  if(exp > now) {\n    return createToken(userId, username, userAccess, expires);\n  }\n\n  return {errors: ['session_expired']};\n};\n\nexport const signIn = async (context: ApiContext, args): Promise<SessionToken> => {\n  const action: string = 'signIn';\n  const {expires, password, username} = args;\n  const formatUsername: string = parseUsername(username);\n  const formatPassword: string = parsePassword(password);\n  const formatExpires: number = parseNum(expires) || 15;\n  const {database} = context;\n\n  if(isEmpty(formatUsername) || isEmpty(formatPassword)) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const checkQuery: AqlQuery = aql`FOR u IN users\n    FILTER u.username == ${formatUsername}\n    LIMIT 1\n    RETURN u`;\n  const checkUser: User = await database.query(checkQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      args: {username: formatUsername},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n\n  if(!checkUser) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  const {_key: userId, salt, userAccess} = checkUser;\n  const authPassword: string = createPassword(formatPassword, salt);\n\n  if(checkUser.password !== authPassword) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  return createToken(userId, username, userAccess, formatExpires);\n};\n\nexport const signOut = async (context: ApiContext, args): Promise<boolean> => {\n  const action: string = 'signOut';\n  const {database, session: {userId: sessionId, username}} = context;\n  const userDocId: string = `users/${sessionId}`;\n\n  const update = {\n    lastOnline: Date.now(),\n    sessionId: null\n  };\n  const sessionQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId})\n    UPDATE u WITH ${update} IN users\n    LIMIT 1\n    RETURN NEW`;\n\n  try {\n    await database.query(sessionQuery)\n      .then((cursor) => cursor.next())\n      .catch((error) => logError({\n        action,\n        args: {username, userId: sessionId},\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context)\n      );\n  } catch(error) {\n    return false;\n  }\n\n  return true;\n};\n\nexport const getActiveUserCount = (context: ApiContext) => {\n  const action: string = 'getActiveUserCount';\n  const {database} = context;\n  const countQuery: AqlQuery = aql`LET docs = (\n    FOR u IN users\n    FILTER u.active == true\n    RETURN u\n  )\n  RETURN LENGTH(docs)`;\n\n  return database.query(countQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => 0)\n    );\n};\n"],
  "mappings": "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",
  "names": []
}

@@ -1,6 +1,6 @@
1
1
  import { ApiContext } from '../types/auth';
2
2
  export declare const addConnection: (connectionId: string, token: string) => Promise<any>;
3
- export declare const getConnectionBySub: (sub: string) => Promise<any>;
3
+ export declare const getConnectionByUser: (userId: string) => Promise<any>;
4
4
  export declare const getConnectionById: (connectionId: string) => Promise<any>;
5
5
  export declare const getConnectionByConvo: (context: ApiContext, convoId: string) => Promise<any>;
6
6
  export declare const removeConnection: (connectionId: string) => Promise<any>;