@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,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FjdGlvbnMvdXNlcnMudHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbIi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDE5LVByZXNlbnQsIE5pdHJvZ2VuIExhYnMsIEluYy5cbiAqIENvcHlyaWdodHMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIGFjY29tcGFueWluZyBMSUNFTlNFIGZpbGUgZm9yIHRlcm1zLlxuICovXG5pbXBvcnQge1xuICBjcmVhdGVIYXNoLFxuICBjcmVhdGVQYXNzd29yZCxcbiAgcGFyc2VDaGFyLFxuICBwYXJzZUVtYWlsLFxuICBwYXJzZUlkLFxuICBwYXJzZU51bSxcbiAgcGFyc2VQYXNzd29yZCxcbiAgcGFyc2VQaG9uZSxcbiAgcGFyc2VVc2VybmFtZVxufSBmcm9tICdAbmxhYnMvdXRpbHMnO1xuaW1wb3J0IHthcWx9IGZyb20gJ2FyYW5nb2pzJztcbmltcG9ydCB7QXFsUXVlcnl9IGZyb20gJ2FyYW5nb2pzL2FxbCc7XG5pbXBvcnQge0VkZ2VDb2xsZWN0aW9ufSBmcm9tICdhcmFuZ29qcy9jb2xsZWN0aW9uJztcbmltcG9ydCB7QXJyYXlDdXJzb3J9IGZyb20gJ2FyYW5nb2pzL2N1cnNvcic7XG5pbXBvcnQgaXNFbXB0eSBmcm9tICdsb2Rhc2gvaXNFbXB0eSc7XG5pbXBvcnQge0RhdGVUaW1lfSBmcm9tICdsdXhvbic7XG5pbXBvcnQgU3RyaXBlIGZyb20gJ3N0cmlwZSc7XG5cbmltcG9ydCB7Q29uZmlnfSBmcm9tICcuLi9jb25maWcnO1xuaW1wb3J0IHtBcGlDb250ZXh0fSBmcm9tICcuLi90eXBlcy9hdXRoJztcbmltcG9ydCB7VXNlcn0gZnJvbSAnLi4vdHlwZXMvdXNlcnMnO1xuaW1wb3J0IHtFcnJvclR5cGVzLCBnZXRMaW1pdCwgbG9nRXJyb3IsIGxvZ0V4Y2VwdGlvbiwgc2VsZWN0UmVhY3Rpb25Db3VudEJ5VHlwZX0gZnJvbSAnLi4vdXRpbHMnO1xuaW1wb3J0IHtnZXRTZXNzaW9uLCBTZXNzaW9uRXJyb3IsIFNlc3Npb25Ub2tlbiwgc2V0U2Vzc2lvbn0gZnJvbSAnLi4vdXRpbHMvc2Vzc2lvbic7XG5cbmNvbnN0IGV2ZW50Q2F0ZWdvcnk6IHN0cmluZyA9ICd1c2Vycyc7XG5jb25zdCBhcGlWZXJzaW9uOiBhbnkgPSAnMjAyMC0wMy0wMic7XG5cbmV4cG9ydCBpbnRlcmZhY2UgVXNlck9wdGlvbnMge1xuICByZWFkb25seSBmcm9tPzogbnVtYmVyO1xuICByZWFkb25seSB0bz86IG51bWJlcjtcbiAgcmVhZG9ubHkgdXNlcm5hbWU/OiBzdHJpbmc7XG59XG5cbmV4cG9ydCBlbnVtIFVzZXJBY2Nlc3Mge1xuICBERUFDVElWQVRFRCA9IDAsXG4gIEFDVElWRSA9IDEsXG4gIFBSRU1JVU0gPSAyLFxuICBDT05URU5UX0FETUlOID0gMyxcbiAgQURNSU4gPSA0XG59XG5cbmV4cG9ydCBjb25zdCBjcmVhdGVUb2tlbiA9IChcbiAgdXNlcklkOiBzdHJpbmcsXG4gIHVzZXJuYW1lOiBzdHJpbmcsXG4gIHVzZXJBY2Nlc3M6IG51bWJlcixcbiAgZXhwaXJlczogbnVtYmVyID0gMTVcbik6IFNlc3Npb25Ub2tlbiA9PiB7XG4gIGNvbnN0IG5vdzogRGF0ZVRpbWUgPSBEYXRlVGltZS5sb2NhbCgpO1xuICBjb25zdCBzZXNzaW9uRXhwaXJlczogRGF0ZVRpbWUgPSBub3cucGx1cyh7bWludXRlczogZXhwaXJlc30pO1xuICBjb25zdCBpYXQ6IG51bWJlciA9IE1hdGguZmxvb3Iobm93LnRvU2Vjb25kcygpKTtcbiAgY29uc3QgZXhwOiBudW1iZXIgPSBNYXRoLmZsb29yKHNlc3Npb25FeHBpcmVzLnRvU2Vjb25kcygpKTtcblxuICBjb25zdCB0b2tlbiA9IHNldFNlc3Npb24oe1xuICAgIGV4cCxcbiAgICBpYXQsXG4gICAgdXNlcm5hbWUsXG4gICAgdXNlckFjY2VzcyxcbiAgICB1c2VySWRcbiAgfSk7XG5cbiAgcmV0dXJuIHtcbiAgICBleHBpcmVzOiBzZXNzaW9uRXhwaXJlcy50b01pbGxpcygpLFxuICAgIGlzc3VlZDogbm93LnRvTWlsbGlzKCksXG4gICAgdG9rZW5cbiAgfTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRVc2VyT3B0aW9uYWwgPSAoZmllbGRzOiBzdHJpbmdbXSA9IFtdKSA9PlxuICBmaWVsZHMucmVkdWNlKChzZWxlY3RzOiBhbnksIGZpZWxkOiBzdHJpbmcpID0+IHtcbiAgICBpZihmaWVsZC5pbmNsdWRlcygnQ291bnQnKSkge1xuICAgICAgcmV0dXJuIHNlbGVjdFJlYWN0aW9uQ291bnRCeVR5cGUoJ3VzZXJzJywgJ3UnLCBmaWVsZCwgc2VsZWN0cyk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHNlbGVjdHM7XG4gIH0sIHtvYmplY3RzOiBbXSwgcXVlcmllczogW119KTtcblxuZXhwb3J0IGNvbnN0IHBhcnNlVXNlck9wdGlvbnMgPSAob3B0aW9uczogVXNlck9wdGlvbnMgPSB7fSkgPT4ge1xuICBjb25zdCB7XG4gICAgZnJvbSA9IDAsXG4gICAgdG8gPSAzMFxuICB9ID0gb3B0aW9ucztcblxuICByZXR1cm4ge1xuICAgIC4uLm9wdGlvbnMsXG4gICAgbGltaXQ6IGdldExpbWl0KGZyb20sIHRvKVxuICB9O1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZFVzZXIgPSBhc3luYyAoY29udGV4dDogQXBpQ29udGV4dCwgYXJnczogYW55KTogUHJvbWlzZTxVc2VyPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2FkZFVzZXInO1xuICBjb25zdCB7ZGF0YWJhc2V9ID0gY29udGV4dDtcbiAgY29uc3Qge3VzZXJ9ID0gYXJncztcbiAgY29uc3Qge2VtYWlsLCBwYXNzd29yZCwgcGhvbmUsIHVzZXJuYW1lfSA9IHVzZXI7XG4gIGNvbnN0IHNhbHQ6IHN0cmluZyA9IGNyZWF0ZUhhc2goYCR7dXNlcm5hbWV9JHtwYXNzd29yZH1gLCBudWxsKTtcbiAgY29uc3QgZW5jcnlwdGVkUGFzc3dvcmQgPSBjcmVhdGVQYXNzd29yZChwYXNzd29yZCwgc2FsdCk7XG4gIGNvbnN0IGZvcm1hdFVzZXJuYW1lOiBzdHJpbmcgPSBwYXJzZVVzZXJuYW1lKHVzZXJuYW1lKTtcbiAgY29uc3QgZm9ybWF0RW1haWw6IHN0cmluZyA9IHBhcnNlRW1haWwoZW1haWwpO1xuICBjb25zdCBmb3JtYXRQaG9uZTogc3RyaW5nID0gcGFyc2VQaG9uZShwaG9uZSk7XG5cbiAgaWYoaXNFbXB0eShmb3JtYXRVc2VybmFtZSkgfHwgaXNFbXB0eShwYXNzd29yZCkgfHwgKCFpc0VtcHR5KGZvcm1hdFBob25lKSAmJiAhaXNFbXB0eShmb3JtYXRFbWFpbCkpKSB7XG4gICAgcmV0dXJuIGxvZ0V4Y2VwdGlvbih7XG4gICAgICBhY3Rpb24sXG4gICAgICBhcmdzOiB7dXNlcm5hbWV9LFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICB2YWx1ZTogRXJyb3JUeXBlcy5JTlZBTElEX0FSR1VNRU5UU1xuICAgIH0sIGNvbnRleHQpO1xuICB9XG5cbiAgY29uc3QgZmlsdGVyczogc3RyaW5nW10gPSBbYHUudXNlcm5hbWUgPT0gXCIke2Zvcm1hdFVzZXJuYW1lfVwiYF07XG5cbiAgaWYoIWlzRW1wdHkoZm9ybWF0RW1haWwpKSB7XG4gICAgZmlsdGVycy5wdXNoKGB1LmVtYWlsID09IFwiJHtmb3JtYXRFbWFpbH1cImApO1xuICB9XG5cbiAgaWYoIWlzRW1wdHkoZm9ybWF0UGhvbmUpKSB7XG4gICAgZmlsdGVycy5wdXNoKGB1LnBob25lID09ICR7Zm9ybWF0UGhvbmV9YCk7XG4gIH1cblxuICBjb25zdCBjaGVja1F1ZXJ5OiBzdHJpbmcgPSBgRk9SIHUgSU4gdXNlcnNcbiAgICBGSUxURVIgJHtmaWx0ZXJzLmpvaW4oJyB8fCAnKX1cbiAgICBSRVRVUk4gdWA7XG4gIGNvbnN0IGV4aXN0aW5nVXNlcnMgPSBhd2FpdCBkYXRhYmFzZS5xdWVyeShjaGVja1F1ZXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLmNhdGNoKChlcnJvcikgPT4gbG9nRXJyb3Ioe1xuICAgICAgYWN0aW9uLFxuICAgICAgYXJnczoge3VzZXJuYW1lfSxcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1JcbiAgICB9LCBlcnJvciwgY29udGV4dClcbiAgICAgIC50aGVuKCgpID0+IG51bGwpXG4gICAgKTtcblxuICBpZihleGlzdGluZ1VzZXJzLmxlbmd0aCkge1xuICAgIHJldHVybiBsb2dFeGNlcHRpb24oe1xuICAgICAgYWN0aW9uLFxuICAgICAgYXJnczoge3VzZXJuYW1lfSxcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuRVhJU1RJTkdfVVNFUk5BTUVcbiAgICB9LCBjb250ZXh0KTtcbiAgfVxuXG4gIGNvbnN0IHZlcmlmaWVkRW1haWxDb2RlOiBudW1iZXIgPSBNYXRoLmZsb29yKDEwMDAwMCArIChNYXRoLnJhbmRvbSgpICogOTAwMDAwKSk7XG4gIGNvbnN0IHZlcmlmaWVkUGhvbmVDb2RlOiBudW1iZXIgPSBNYXRoLmZsb29yKDEwMDAwMCArIChNYXRoLnJhbmRvbSgpICogOTAwMDAwKSk7XG5cbiAgLy8gQWRkIG5ldyB1c2VyIHByb3BlcnRpZXNcbiAgY29uc3QgaW5zZXJ0OiBVc2VyID0ge1xuICAgIF9rZXk6IGNyZWF0ZUhhc2godXNlcm5hbWUsIG51bGwpLFxuICAgIGFkZGVkOiBEYXRlLm5vdygpLFxuICAgIGVtYWlsOiBmb3JtYXRFbWFpbCxcbiAgICBtb2RpZmllZDogRGF0ZS5ub3coKSxcbiAgICBwYXNzd29yZDogZW5jcnlwdGVkUGFzc3dvcmQsXG4gICAgcGhvbmU6IGZvcm1hdFBob25lLFxuICAgIHNhbHQsXG4gICAgdXNlcm5hbWU6IGZvcm1hdFVzZXJuYW1lLFxuICAgIHVzZXJBY2Nlc3M6IDEsXG4gICAgdmVyaWZpZWRFbWFpbDogZmFsc2UsXG4gICAgdmVyaWZpZWRFbWFpbENvZGUsXG4gICAgdmVyaWZpZWRQaG9uZTogZmFsc2UsXG4gICAgdmVyaWZpZWRQaG9uZUNvZGVcbiAgfTtcblxuICAvLyBBZGQgbmV3IHVzZXIgaW4gQXJhbmdvREJcbiAgY29uc3QgaW5zZXJ0UXVlcnk6IEFxbFF1ZXJ5ID0gYXFsYElOU0VSVCAke2luc2VydH0gSU4gdXNlcnMgUkVUVVJOIE5FV2A7XG4gIHJldHVybiBhd2FpdCBkYXRhYmFzZS5xdWVyeShpbnNlcnRRdWVyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSB8fCB7fSlcbiAgICAuY2F0Y2goKGVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBhcmdzOiB7dXNlcm5hbWV9LFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUlxuICAgIH0sIGVycm9yLCBjb250ZXh0KVxuICAgICAgLnRoZW4oKCkgPT4gbnVsbClcbiAgICApO1xufTtcblxuZXhwb3J0IGNvbnN0IHVwZGF0ZVVzZXIgPSBhc3luYyAoY29udGV4dDogQXBpQ29udGV4dCwgdXNlcjogVXNlcik6IFByb21pc2U8YW55PiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ3VwZGF0ZVVzZXInO1xuICBjb25zdCB7ZGF0YWJhc2V9ID0gY29udGV4dDtcbiAgY29uc3Qge19rZXksIF9pZCwgaWQsIHRhZ3MgPSBbXSwgdXNlcklkLCAuLi51cGRhdGVkfSA9IHVzZXI7XG4gIGxldCB1c2VyRG9jSWQ6IHN0cmluZztcblxuICBpZihfaWQgfHwgaWQpIHtcbiAgICB1c2VyRG9jSWQgPSBfaWQgfHwgaWQ7XG4gIH0gZWxzZSBpZihfa2V5IHx8IHVzZXJJZCkge1xuICAgIHVzZXJEb2NJZCA9IGB1c2Vycy8ke19rZXkgfHwgdXNlcklkfWA7XG4gIH1cblxuICBjb25zdCB1c2VyUXVlcnk6IEFxbFF1ZXJ5ID0gYXFsYExFVCB1ID0gRE9DVU1FTlQoJHt1c2VyRG9jSWR9KVxuICAgIFVQREFURSB1IFdJVEggJHt1cGRhdGVkfSBJTiB1c2Vyc1xuICAgIFJFVFVSTiBORVdgO1xuXG4gIGNvbnN0IHVwZGF0ZWRVc2VyID0gYXdhaXQgZGF0YWJhc2UucXVlcnkodXNlclF1ZXJ5KVxuICAgIC50aGVuKChjdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgIGNvbnNvbGUubG9nKGVycm9yKTtcbiAgICAgIHRocm93IGVycm9yO1xuICAgIH0pO1xuXG4gIGNvbnN0IHRhZ0NvbGxlY3Rpb246IEVkZ2VDb2xsZWN0aW9uID0gZGF0YWJhc2UuY29sbGVjdGlvbignaXNUYWdnZWQnKTtcbiAgYXdhaXQgUHJvbWlzZS5hbGwodGFncy5tYXAoKHtpZDogdGFnRG9jSWQsIG5hbWV9KSA9PiB7XG4gICAgY29uc3QgZWRnZSA9IHtcbiAgICAgIF9mcm9tOiB0YWdEb2NJZCxcbiAgICAgIF9rZXk6IGNyZWF0ZUhhc2goYGlzVGFnZ2VkLSR7dGFnRG9jSWR9LSR7dXNlckRvY0lkfWApLFxuICAgICAgX3RvOiB1c2VyRG9jSWQsXG4gICAgICBhZGRlZDogRGF0ZS5ub3coKSxcbiAgICAgIG5hbWVcbiAgICB9O1xuICAgIGNvbnN0IHRhZ1F1ZXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgaXQgSU4gaXNUYWdnZWRcbiAgICAgIEZJTFRFUiBpdC5fZnJvbSA9PSAke3RhZ0RvY0lkfSAmJiBpdC5fdG8gPT0gJHt1c2VyRG9jSWR9ICYmIGl0Lm5hbWUgPT0gJHtuYW1lfVxuICAgICAgTElNSVQgMVxuICAgICAgUkVUVVJOIGl0YDtcblxuICAgIHJldHVybiBkYXRhYmFzZS5xdWVyeSh0YWdRdWVyeSlcbiAgICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgLnRoZW4oKHRhZ0VkZ2UpID0+IHtcbiAgICAgICAgaWYoISF0YWdFZGdlKSB7XG4gICAgICAgICAgcmV0dXJuIHRhZ0VkZ2U7XG4gICAgICAgIH1cblxuICAgICAgICByZXR1cm4gdGFnQ29sbGVjdGlvbi5zYXZlKGVkZ2UsIHtyZXR1cm5OZXc6IHRydWV9KS50aGVuKCgpID0+IGVkZ2UpO1xuICAgICAgfSlcbiAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIGxhYmVsOiAnZGJfZXJyb3InXG4gICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG4gIH0pKTtcblxuICByZXR1cm4gdXBkYXRlZFVzZXI7XG59O1xuXG5leHBvcnQgY29uc3QgY29uZmlybUNvZGUgPSBhc3luYyAoY29udGV4dDogQXBpQ29udGV4dCwgYXJncyk6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdjb25maXJtRW1haWwnO1xuICBjb25zdCB7ZGF0YWJhc2UsIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH19ID0gY29udGV4dDtcbiAgY29uc3Qge2NvZGUsIHR5cGV9ID0gYXJncztcbiAgY29uc3QgdXNlckRvY0lkOiBzdHJpbmcgPSBgdXNlcnMvJHtzZXNzaW9uSWR9YDtcblxuICBjb25zdCBhcWxRdWVyeTogQXFsUXVlcnkgPSBhcWxgTEVUIHUgPSBET0NVTUVOVCgke3VzZXJEb2NJZH0pIFJFVFVSTiB1YDtcblxuICB0cnkge1xuICAgIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRdWVyeSlcbiAgICAgIC50aGVuKChjdXJzb3IpID0+IGN1cnNvci5uZXh0KCkgfHwge30pXG4gICAgICAudGhlbigoe3ZlcmlmaWVkRW1haWxDb2RlLCB2ZXJpZmllZFBob25lQ29kZX06IFVzZXIpID0+IHtcbiAgICAgICAgc3dpdGNoKHR5cGUpIHtcbiAgICAgICAgICBjYXNlICdlbWFpbCc6XG4gICAgICAgICAgICByZXR1cm4gY29kZSA9PT0gdmVyaWZpZWRFbWFpbENvZGU7XG4gICAgICAgICAgY2FzZSAncGhvbmUnOlxuICAgICAgICAgICAgcmV0dXJuIGNvZGUgPT09IHZlcmlmaWVkUGhvbmVDb2RlO1xuICAgICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgICAgIH1cbiAgICAgIH0pXG4gICAgICAuY2F0Y2goKGVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgIGFjdGlvbixcbiAgICAgICAgYXJnczoge2NvZGUsIHR5cGUsIHVzZXJJZDogc2Vzc2lvbklkfSxcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIGxhYmVsOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgICB9LCBlcnJvciwgY29udGV4dClcbiAgICAgICk7XG4gIH0gY2F0Y2goZXJyb3IpIHtcbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cbn07XG5cbmV4cG9ydCBjb25zdCBkZWxldGVVc2VyID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGFyZ3M6IGFueSk6IFByb21pc2U8YW55PiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2RlbGV0ZVVzZXInO1xuICBjb25zdCB7ZGF0YWJhc2UsIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZCwgdXNlckFjY2Vzczogc2Vzc2lvbkFjY2Vzc319ID0gY29udGV4dDtcbiAgY29uc3Qge3VzZXJJZH0gPSBhcmdzO1xuICBjb25zdCBpc0FkbWluOiBib29sZWFuID0gc2Vzc2lvbkFjY2VzcyA+IDI7XG5cbiAgaWYoIWlzQWRtaW4gJiYgKHNlc3Npb25JZCAhPT0gdXNlcklkKSkge1xuICAgIGxvZ0V4Y2VwdGlvbih7XG4gICAgICBhY3Rpb24sXG4gICAgICBhcmdzLFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogJ3VuYXV0aG9yaXplZCcsXG4gICAgICB2YWx1ZTogJ2ludmFsaWRfc2Vzc2lvbidcbiAgICB9LCBjb250ZXh0KTtcbiAgICByZXR1cm4gbnVsbDtcbiAgfVxuXG4gIGNvbnN0IGFxbFF1ZXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgdSBJTiB1c2Vyc1xuICAgIEZJTFRFUiB1Ll9rZXkgPT0gJHt1c2VySWR9XG4gICAgTElNSVQgMVxuICAgIFJFTU9WRSB1IElOIHVzZXJzXG4gICAgUkVUVVJOIE9MRGA7XG5cbiAgLy8gU3RyaXBlXG4gIGNvbnN0IHN0cmlwZUNsaWVudCA9IG5ldyBTdHJpcGUoQ29uZmlnLmdldCgnc3RyaXBlLnRva2VuJyksIHthcGlWZXJzaW9uLCB0eXBlc2NyaXB0OiB0cnVlfSk7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFF1ZXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKCh1c2VyOiBVc2VyID0ge30pID0+IHN0cmlwZUNsaWVudC5jdXN0b21lcnMuZGVsKHVzZXIuc3RyaXBlQ3VzdG9tZXJJZClcbiAgICAgIC50aGVuKCgpID0+IHN0cmlwZUNsaWVudC5hY2NvdW50cy5kZWwodXNlci5zdHJpcGVBY2NvdW50SWQpKVxuICAgICAgLnRoZW4oKCkgPT4gdXNlcikpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgIHRocm93IGVycm9yO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGRlYWN0aXZhdGVVc2VyID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIHVzZXJJZDogc3RyaW5nKTogUHJvbWlzZTxVc2VyPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2RlbGV0ZSc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkLCB1c2VyQWNjZXNzOiBzZXNzaW9uQWNjZXNzfX0gPSBjb250ZXh0O1xuICBjb25zdCBpc0FkbWluOiBib29sZWFuID0gc2Vzc2lvbkFjY2VzcyA+IDI7XG5cbiAgaWYoIWlzQWRtaW4gJiYgKHNlc3Npb25JZCAhPT0gdXNlcklkKSkge1xuICAgIGxvZ0V4Y2VwdGlvbih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiAndW5hdXRob3JpemVkJyxcbiAgICAgIHZhbHVlOiAnaW52YWxpZF9zZXNzaW9uJ1xuICAgIH0sIGNvbnRleHQpO1xuICAgIHJldHVybiBudWxsO1xuICB9XG5cbiAgY29uc3QgdXBkYXRlZDogVXNlciA9IHtcbiAgICB1c2VyQWNjZXNzOiAwXG4gIH07XG4gIGNvbnN0IGFxbFF1ZXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUgJHt1c2VySWR9IFdJVEggJHt1cGRhdGVkfSBJTiB1c2VycyBMSU1JVCAxIFJFVFVSTiBORVdgO1xuXG4gIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRdWVyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgdGhyb3cgZXJyb3I7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0RGlzcGxheU5hbWUgPSAodXNlcjogVXNlciA9IHt9KTogc3RyaW5nID0+IHtcbiAgY29uc3Qge2ZpcnN0LCBsYXN0LCBuYW1lID0gJycsIHVzZXJuYW1lID0gJyd9ID0gdXNlcjtcbiAgY29uc3QgZnVsbG5hbWU6IHN0cmluZyA9IChbZmlyc3QsIGxhc3RdKS5qb2luKCcgJykudHJpbSgpO1xuXG4gIGlmKCFpc0VtcHR5KG5hbWUpKSB7XG4gICAgcmV0dXJuIG5hbWU7XG4gIH0gZWxzZSBpZihmdWxsbmFtZSAhPT0gJycpIHtcbiAgICByZXR1cm4gZnVsbG5hbWU7XG4gIH0gZWxzZSBpZighaXNFbXB0eSh1c2VybmFtZSkpIHtcbiAgICByZXR1cm4gdXNlcm5hbWU7XG4gIH1cblxuICByZXR1cm4gJ1Vua25vd24nO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldFNlc3Npb25Vc2VyID0gKGNvbnRleHQ6IEFwaUNvbnRleHQpOiBQcm9taXNlPFVzZXI+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0U2Vzc2lvblVzZXInO1xuICBjb25zdCB7ZGF0YWJhc2UsIGZpZWxkcywgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkLCB1c2VybmFtZX19ID0gY29udGV4dDtcbiAgY29uc3Qge29iamVjdHM6IHNlbGVjdE9iamVjdHMsIHF1ZXJpZXM6IHNlbGVjdFF1ZXJpZXN9ID0gZ2V0VXNlck9wdGlvbmFsKGZpZWxkcyk7XG5cbiAgY29uc3QgYXFsUXVlcnk6IHN0cmluZyA9IGBMRVQgdSA9IERPQ1VNRU5UKFwidXNlcnMvJHtzZXNzaW9uSWR9XCIpXG4gICR7c2VsZWN0UXVlcmllcy5qb2luKCdcXG4nKX1cbiAgUkVUVVJOIE1FUkdFKHUsIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFF1ZXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKCh1c2VyID0ge30pID0+IHVzZXIpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGFyZ3M6IHt1c2VybmFtZSwgdXNlcklkOiBzZXNzaW9uSWR9LFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUlxuICAgIH0sIGVycm9yLCBjb250ZXh0KSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0VXNlciA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBhcmdzOiBhbnkpOiBQcm9taXNlPFVzZXI+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0VXNlcic7XG4gIGNvbnN0IHt1c2VySWR9ID0gYXJncztcbiAgY29uc3Qge2RhdGFiYXNlLCBmaWVsZHN9ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0VXNlcklkOiBzdHJpbmcgPSBwYXJzZUlkKHVzZXJJZCk7XG4gIGNvbnN0IHtvYmplY3RzOiBzZWxlY3RPYmplY3RzLCBxdWVyaWVzOiBzZWxlY3RRdWVyaWVzfSA9IGdldFVzZXJPcHRpb25hbChmaWVsZHMpO1xuXG4gIC8vIEdldCBkYXRhIGZyb20gZGF0YWJhc2VcbiAgY29uc3QgYXFsUXVlcnk6IHN0cmluZyA9IGBMRVQgdSA9IERPQ1VNRU5UKFwidXNlcnMvJHtmb3JtYXRVc2VySWR9XCIpXG4gICAgJHtzZWxlY3RRdWVyaWVzLmpvaW4oJ1xcbicpfVxuICAgIEZJTFRFUiB1LnVzZXJBY2Nlc3MgPiAwXG4gICAgUkVUVVJOIE1FUkdFKHUsIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFF1ZXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKCh1c2VyID0ge30pID0+IHVzZXIpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGFyZ3MsXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpXG4gICAgICAudGhlbigoKSA9PiB7IH0pKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRVc2VycyA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBvcHRpb25zPzogVXNlck9wdGlvbnMpOiBQcm9taXNlPFVzZXJbXT4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRVc2VyTGlzdCc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgZmllbGRzfSA9IGNvbnRleHQ7XG4gIGNvbnN0IHtsaW1pdCwgdXNlcm5hbWV9ID0gcGFyc2VVc2VyT3B0aW9ucyhvcHRpb25zKTtcbiAgY29uc3Qge29iamVjdHM6IHNlbGVjdE9iamVjdHMsIHF1ZXJpZXM6IHNlbGVjdFF1ZXJpZXN9ID0gZ2V0VXNlck9wdGlvbmFsKGZpZWxkcyk7XG4gIGNvbnN0IGZpbHRlckJ5OiBzdHJpbmdbXSA9IFsndS51c2VyQWNjZXNzID4gMCddO1xuXG4gIGlmKCFpc0VtcHR5KHVzZXJuYW1lKSkge1xuICAgIGZpbHRlckJ5LnB1c2goYENPTlRBSU5TKHUudXNlcm5hbWUsIFwiJHtwYXJzZVVzZXJuYW1lKHVzZXJuYW1lKX1cIilgKTtcbiAgfVxuXG4gIC8vIEdldCBkYXRhIGZyb20gZGF0YWJhc2VcbiAgY29uc3QgYXFsUXVlcnk6IHN0cmluZyA9IGBGT1IgdSBJTiB1c2Vyc1xuICAgIEZJTFRFUiAke2ZpbHRlckJ5LmpvaW4oJyAmJiAnKX1cbiAgICAke3NlbGVjdFF1ZXJpZXMuam9pbignXFxuJyl9XG4gICAgJHtsaW1pdC5hcWx9XG4gICAgU09SVCB1LnVzZXJuYW1lXG4gICAgUkVUVVJOIE1FUkdFKHUsIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFF1ZXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1JcbiAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBbXSkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGdldFVzZXJzQnlSZWFjdGlvbnMgPSAoXG4gIGNvbnRleHQ6IEFwaUNvbnRleHQsXG4gIHtyZWFjdGlvbnMgPSBbXSwgdXNlcm5hbWV9OiBhbnksXG4gIG9wdGlvbnM/OiBVc2VyT3B0aW9uc1xuKTogUHJvbWlzZTxVc2VyW10+ID0+IHtcbiAgY29uc3QgYWN0aW9uOiBzdHJpbmcgPSAnZ2V0VXNlcnNCeVJlYWN0aW9ucyc7XG4gIGNvbnN0IHtkYXRhYmFzZSwgZmllbGRzLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdFJlYWN0aW9uczogc3RyaW5nW10gPSAgcmVhY3Rpb25zLm1hcCgocmVhY3Rpb25OYW1lOiBzdHJpbmcpID0+IHBhcnNlQ2hhcihyZWFjdGlvbk5hbWUsIDMyKS50b0xvd2VyQ2FzZSgpKTtcbiAgY29uc3Qge2xpbWl0fSA9IHBhcnNlVXNlck9wdGlvbnMob3B0aW9ucyk7XG4gIGNvbnN0IHtvYmplY3RzOiBzZWxlY3RPYmplY3RzLCBxdWVyaWVzOiBzZWxlY3RRdWVyaWVzfSA9IGdldFVzZXJPcHRpb25hbChmaWVsZHMpO1xuXG4gIGNvbnN0IGZvcm1hdFNlc3Npb25JZDogc3RyaW5nID0gYHVzZXJzLyR7c2Vzc2lvbklkfWA7XG4gIGNvbnN0IGZvcm1hdFVzZXJuYW1lOiBzdHJpbmcgPSBwYXJzZVVzZXJuYW1lKHVzZXJuYW1lKTtcbiAgY29uc3QgZmlsdGVyQnk6IHN0cmluZ1tdID0gW2BQT1NJVElPTigke0pTT04uc3RyaW5naWZ5KGZvcm1hdFJlYWN0aW9ucyl9LCBMT1dFUihyLm5hbWUpKWBdO1xuXG4gIGlmKCFpc0VtcHR5KHVzZXJuYW1lKSkge1xuICAgIGZpbHRlckJ5LnB1c2goYENPTlRBSU5TKHUudXNlcm5hbWUsIFwiJHtmb3JtYXRVc2VybmFtZX1cIilgKTtcbiAgfVxuXG4gIC8vIEdldCBkYXRhIGZyb20gZGF0YWJhc2VcbiAgY29uc3QgYXFsUXVlcnk6IHN0cmluZyA9IGBGT1IgdSwgciBJTiBPVVRCT1VORCBcIiR7Zm9ybWF0U2Vzc2lvbklkfVwiIGhhc1JlYWN0aW9uc1xuICAgIE9QVElPTlMge3ZlcnRleENvbGxlY3Rpb25zOiBcInVzZXJzXCJ9XG4gICAgJHtzZWxlY3RRdWVyaWVzLmpvaW4oJ1xcbicpfVxuICAgICR7ZmlsdGVyQnkubGVuZ3RoID8gYEZJTFRFUiAke2ZpbHRlckJ5LmpvaW4oJyAmJiAnKX1gIDogJyd9XG4gICAgJHtsaW1pdC5hcWx9XG4gICAgUkVUVVJOIE1FUkdFKHUsIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG5cbiAgY29uc29sZS5sb2coe2FxbFF1ZXJ5fSk7XG4gIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRdWVyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gW10pKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRVc2Vyc0J5VGFncyA9IChcbiAgY29udGV4dDogQXBpQ29udGV4dCxcbiAge3RhZ3MsIHVzZXJuYW1lfTogYW55LFxuICBvcHRpb25zPzogVXNlck9wdGlvbnNcbik6IFByb21pc2U8VXNlcltdPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldFVzZXJzQnlUYWdzJztcbiAgY29uc3Qge2RhdGFiYXNlLCBmaWVsZHMsIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH19ID0gY29udGV4dDtcbiAgY29uc3QgZm9ybWF0VGFnczogc3RyaW5nW10gPSAgdGFncy5yZWR1Y2UoKGxpc3Q6IHN0cmluZ1tdLCB0YWdOYW1lOiBzdHJpbmcpID0+IHtcbiAgICBpZighaXNFbXB0eSh0YWdOYW1lKSkge1xuICAgICAgbGlzdC5wdXNoKHBhcnNlQ2hhcih0YWdOYW1lLCAzMikudG9Mb3dlckNhc2UoKSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGxpc3Q7XG4gIH0sIFtdKTtcbiAgY29uc3Qge2xpbWl0fSA9IHBhcnNlVXNlck9wdGlvbnMob3B0aW9ucyk7XG4gIGNvbnN0IHtvYmplY3RzOiBzZWxlY3RPYmplY3RzLCBxdWVyaWVzOiBzZWxlY3RRdWVyaWVzfSA9IGdldFVzZXJPcHRpb25hbChmaWVsZHMpO1xuXG4gIGNvbnN0IGZvcm1hdFVzZXJuYW1lOiBzdHJpbmcgPSBwYXJzZVVzZXJuYW1lKHVzZXJuYW1lKTtcbiAgY29uc3QgZmlsdGVyQnk6IHN0cmluZ1tdID0gW2B1Ll9rZXkgIT0gXCIke3Nlc3Npb25JZH1cImBdO1xuXG4gIGlmKCFpc0VtcHR5KHVzZXJuYW1lKSkge1xuICAgIGZpbHRlckJ5LnB1c2goYENPTlRBSU5TKHUudXNlcm5hbWUsIFwiJHtmb3JtYXRVc2VybmFtZX1cIilgKTtcbiAgfVxuXG4gIC8vIEdldCBkYXRhIGZyb20gZGF0YWJhc2VcbiAgY29uc3QgYXFsUXVlcnk6IHN0cmluZyA9IGBGT1IgdCBJTiB0YWdzXG4gICAgRklMVEVSIFBPU0lUSU9OKCR7SlNPTi5zdHJpbmdpZnkoZm9ybWF0VGFncyl9LCBMT1dFUih0Lm5hbWUpKVxuICAgIEZPUiB1LCBpdCBJTiBPVVRCT1VORCB0IGlzVGFnZ2VkXG4gICAgT1BUSU9OUyB7YmZzOiB0cnVlLCB1bmlxdWVWZXJ0aWNlczogXCJnbG9iYWxcIiwgdmVydGV4Q29sbGVjdGlvbnM6IFwidXNlcnNcIn1cbiAgICAke3NlbGVjdFF1ZXJpZXMuam9pbignXFxuJyl9XG4gICAgJHtmaWx0ZXJCeS5sZW5ndGggPyBgRklMVEVSICR7ZmlsdGVyQnkuam9pbignICYmICcpfWAgOiAnJ31cbiAgICAke2xpbWl0LmFxbH1cbiAgICBSRVRVUk4gRElTVElOQ1QgTUVSR0UodSwgeyR7c2VsZWN0T2JqZWN0cy5qb2luKCcsICcpfX0pYDtcblxuICBjb25zb2xlLmxvZygnZ2V0VXNlcnNCeVRhZ3MnLCBhcWxRdWVyeSk7XG4gIHJldHVybiBkYXRhYmFzZS5xdWVyeShhcWxRdWVyeSlcbiAgICAudGhlbigoY3Vyc29yOiBBcnJheUN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gW10pKTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRVc2Vyc0J5TGF0ZXN0ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIHt1c2VybmFtZX0sIG9wdGlvbnM/OiBVc2VyT3B0aW9ucyk6IFByb21pc2U8VXNlcltdPiA9PiB7XG4gIGNvbnN0IGFjdGlvbjogc3RyaW5nID0gJ2dldFVzZXJzQnlMYXRlc3QnO1xuICBjb25zdCB7ZGF0YWJhc2UsIGZpZWxkcywgc2Vzc2lvbjoge3VzZXJJZH19ID0gY29udGV4dDtcbiAgY29uc3Qge2xpbWl0fSA9IHBhcnNlVXNlck9wdGlvbnMob3B0aW9ucyk7XG4gIGNvbnN0IGZpbHRlciA9IFsndS5faWQgIT0gc2Vzc2lvbi5faWQnXTtcbiAgY29uc3Qge29iamVjdHM6IHNlbGVjdE9iamVjdHMsIHF1ZXJpZXM6IHNlbGVjdFF1ZXJpZXN9ID0gZ2V0VXNlck9wdGlvbmFsKGZpZWxkcyk7XG5cbiAgaWYoIWlzRW1wdHkodXNlcm5hbWUpKSB7XG4gICAgZmlsdGVyLnB1c2goYENPTlRBSU5TKHUudXNlcm5hbWUsIFwiJHtwYXJzZVVzZXJuYW1lKHVzZXJuYW1lKX1cIilgKTtcbiAgfVxuXG4gIC8vIEdldCBkYXRhIGZyb20gZGF0YWJhc2VcbiAgY29uc3QgYXFsUXVlcnk6IHN0cmluZyA9IGBGT1IgdSBJTiB1c2Vyc1xuICAgIExFVCBzZXNzaW9uID0gRE9DVU1FTlQoXCJ1c2Vycy8ke3VzZXJJZH1cIilcbiAgICBGSUxURVIgJHtmaWx0ZXIuam9pbignICYmICcpfVxuICAgICR7c2VsZWN0UXVlcmllcy5qb2luKCdcXG4nKX1cbiAgICBMRVQgZGlzdGFuY2UgPSBESVNUQU5DRSh1LmxhdGl0dWRlIHx8IDAsIHUubG9uZ2l0dWRlIHx8IDAsIHNlc3Npb24ubGF0aXR1ZGUgfHwgMCwgc2Vzc2lvbi5sb25naXR1ZGUgfHwgMClcbiAgICAke2xpbWl0LmFxbH1cbiAgICBTT1JUIGRpc3RhbmNlIEFTQywgdS5hZGRlZCBERVNDXG4gICAgUkVUVVJOIE1FUkdFKHUsIHske3NlbGVjdE9iamVjdHMuam9pbignLCAnKX19KWA7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGFxbFF1ZXJ5KVxuICAgIC50aGVuKChjdXJzb3I6IEFycmF5Q3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1JcbiAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBbXSkpO1xufTtcblxuZXhwb3J0IGNvbnN0IHJlZnJlc2hTZXNzaW9uID0gYXN5bmMgKGNvbnRleHQ6IEFwaUNvbnRleHQsIHtleHBpcmVzLCB0b2tlbn0pOiBQcm9taXNlPFNlc3Npb25Ub2tlbiB8IFNlc3Npb25FcnJvcj4gPT4ge1xuICBjb25zdCB7ZXJyb3J9ID0gZ2V0U2Vzc2lvbih0b2tlbik7XG5cbiAgaWYoZXJyb3IpIHtcbiAgICByZXR1cm4ge2Vycm9yczogW2Vycm9yXX07XG4gIH1cblxuICBjb25zdCB7ZXhwLCB1c2VySWQsIHVzZXJuYW1lLCB1c2VyQWNjZXNzfSA9IGdldFNlc3Npb24odG9rZW4pO1xuICBjb25zdCBub3c6IG51bWJlciA9IE1hdGguY2VpbChEYXRlVGltZS5sb2NhbCgpLnRvU2Vjb25kcygpKTtcblxuICBpZihleHAgPiBub3cpIHtcbiAgICByZXR1cm4gY3JlYXRlVG9rZW4odXNlcklkLCB1c2VybmFtZSwgdXNlckFjY2VzcywgZXhwaXJlcyk7XG4gIH1cblxuICByZXR1cm4ge2Vycm9yczogWydzZXNzaW9uX2V4cGlyZWQnXX07XG59O1xuXG5leHBvcnQgY29uc3Qgc2lnbkluID0gYXN5bmMgKGNvbnRleHQ6IEFwaUNvbnRleHQsIGFyZ3MpOiBQcm9taXNlPFNlc3Npb25Ub2tlbj4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdzaWduSW4nO1xuICBjb25zdCB7ZXhwaXJlcywgcGFzc3dvcmQsIHVzZXJuYW1lfSA9IGFyZ3M7XG4gIGNvbnN0IGZvcm1hdFVzZXJuYW1lOiBzdHJpbmcgPSBwYXJzZVVzZXJuYW1lKHVzZXJuYW1lKTtcbiAgY29uc3QgZm9ybWF0UGFzc3dvcmQ6IHN0cmluZyA9IHBhcnNlUGFzc3dvcmQocGFzc3dvcmQpO1xuICBjb25zdCBmb3JtYXRFeHBpcmVzOiBudW1iZXIgPSBwYXJzZU51bShleHBpcmVzKSB8fCAxNTtcbiAgY29uc3Qge2RhdGFiYXNlfSA9IGNvbnRleHQ7XG5cbiAgaWYoaXNFbXB0eShmb3JtYXRVc2VybmFtZSkgfHwgaXNFbXB0eShmb3JtYXRQYXNzd29yZCkpIHtcbiAgICByZXR1cm4gbG9nRXhjZXB0aW9uKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGFyZ3M6IHt1c2VybmFtZX0sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIHZhbHVlOiBFcnJvclR5cGVzLklOVkFMSURfQVJHVU1FTlRTXG4gICAgfSwgY29udGV4dCk7XG4gIH1cblxuICBjb25zdCBjaGVja1F1ZXJ5OiBBcWxRdWVyeSA9IGFxbGBGT1IgdSBJTiB1c2Vyc1xuICAgIEZJTFRFUiB1LnVzZXJuYW1lID09ICR7Zm9ybWF0VXNlcm5hbWV9XG4gICAgTElNSVQgMVxuICAgIFJFVFVSTiB1YDtcbiAgY29uc3QgY2hlY2tVc2VyOiBVc2VyID0gYXdhaXQgZGF0YWJhc2UucXVlcnkoY2hlY2tRdWVyeSlcbiAgICAudGhlbigoY3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC5jYXRjaCgoZXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGFyZ3M6IHt1c2VybmFtZTogZm9ybWF0VXNlcm5hbWV9LFxuICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICBsYWJlbDogRXJyb3JUeXBlcy5EQVRBQkFTRV9FUlJPUlxuICAgIH0sIGVycm9yLCBjb250ZXh0KVxuICAgICAgLnRoZW4oKCkgPT4gbnVsbClcbiAgICApO1xuXG4gIGlmKCFjaGVja1VzZXIpIHtcbiAgICByZXR1cm4gbG9nRXhjZXB0aW9uKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGFyZ3M6IHt1c2VybmFtZX0sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIHZhbHVlOiBFcnJvclR5cGVzLklOVkFMSURfQVVUSEVOVElDQVRJT05cbiAgICB9LCBjb250ZXh0KTtcbiAgfVxuXG4gIGNvbnN0IHtfa2V5OiB1c2VySWQsIHNhbHQsIHVzZXJBY2Nlc3N9ID0gY2hlY2tVc2VyO1xuICBjb25zdCBhdXRoUGFzc3dvcmQ6IHN0cmluZyA9IGNyZWF0ZVBhc3N3b3JkKGZvcm1hdFBhc3N3b3JkLCBzYWx0KTtcblxuICBpZihjaGVja1VzZXIucGFzc3dvcmQgIT09IGF1dGhQYXNzd29yZCkge1xuICAgIHJldHVybiBsb2dFeGNlcHRpb24oe1xuICAgICAgYWN0aW9uLFxuICAgICAgYXJnczoge3VzZXJuYW1lfSxcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgdmFsdWU6IEVycm9yVHlwZXMuSU5WQUxJRF9BVVRIRU5USUNBVElPTlxuICAgIH0sIGNvbnRleHQpO1xuICB9XG5cbiAgcmV0dXJuIGNyZWF0ZVRva2VuKHVzZXJJZCwgdXNlcm5hbWUsIHVzZXJBY2Nlc3MsIGZvcm1hdEV4cGlyZXMpO1xufTtcblxuZXhwb3J0IGNvbnN0IHNpZ25PdXQgPSBhc3luYyAoY29udGV4dDogQXBpQ29udGV4dCwgYXJncyk6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdzaWduT3V0JztcbiAgY29uc3Qge2RhdGFiYXNlLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWQsIHVzZXJuYW1lfX0gPSBjb250ZXh0O1xuICBjb25zdCB1c2VyRG9jSWQ6IHN0cmluZyA9IGB1c2Vycy8ke3Nlc3Npb25JZH1gO1xuXG4gIGNvbnN0IHVwZGF0ZSA9IHtcbiAgICBsYXN0T25saW5lOiBEYXRlLm5vdygpLFxuICAgIHNlc3Npb25JZDogbnVsbFxuICB9O1xuICBjb25zdCBzZXNzaW9uUXVlcnk6IEFxbFF1ZXJ5ID0gYXFsYExFVCB1ID0gRE9DVU1FTlQoJHt1c2VyRG9jSWR9KVxuICAgIFVQREFURSB1IFdJVEggJHt1cGRhdGV9IElOIHVzZXJzXG4gICAgTElNSVQgMVxuICAgIFJFVFVSTiBORVdgO1xuXG4gIHRyeSB7XG4gICAgYXdhaXQgZGF0YWJhc2UucXVlcnkoc2Vzc2lvblF1ZXJ5KVxuICAgICAgLnRoZW4oKGN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgIC5jYXRjaCgoZXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgYWN0aW9uLFxuICAgICAgICBhcmdzOiB7dXNlcm5hbWUsIHVzZXJJZDogc2Vzc2lvbklkfSxcbiAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgIGxhYmVsOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgICB9LCBlcnJvciwgY29udGV4dClcbiAgICAgICk7XG4gIH0gY2F0Y2goZXJyb3IpIHtcbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cblxuICByZXR1cm4gdHJ1ZTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRBY3RpdmVVc2VyQ291bnQgPSAoY29udGV4dDogQXBpQ29udGV4dCkgPT4ge1xuICBjb25zdCBhY3Rpb246IHN0cmluZyA9ICdnZXRBY3RpdmVVc2VyQ291bnQnO1xuICBjb25zdCB7ZGF0YWJhc2V9ID0gY29udGV4dDtcbiAgY29uc3QgY291bnRRdWVyeTogQXFsUXVlcnkgPSBhcWxgTEVUIGRvY3MgPSAoXG4gICAgRk9SIHUgSU4gdXNlcnNcbiAgICBGSUxURVIgdS5hY3RpdmUgPT0gdHJ1ZVxuICAgIFJFVFVSTiB1XG4gIClcbiAgUkVUVVJOIExFTkdUSChkb2NzKWA7XG5cbiAgcmV0dXJuIGRhdGFiYXNlLnF1ZXJ5KGNvdW50UXVlcnkpXG4gICAgLnRoZW4oKGN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAuY2F0Y2goKGVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpXG4gICAgICAudGhlbigoKSA9PiAwKVxuICAgICk7XG59O1xuIl0sCiAgIm1hcHBpbmdzIjogIjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBSUEsbUJBVU87QUFDUCxzQkFBa0I7QUFJbEIscUJBQW9CO0FBQ3BCLG1CQUF1QjtBQUN2QixvQkFBbUI7QUFFbkIsb0JBQXFCO0FBR3JCLG9CQUFzRjtBQUN0RixxQkFBaUU7QUFFakUsTUFBTSxnQkFBd0I7QUFDOUIsTUFBTSxhQUFrQjtBQVFqQixJQUFLO0FBQUwsVUFBSyxhQUFMO0FBQ0wsMkNBQWMsS0FBZDtBQUNBLHNDQUFTLEtBQVQ7QUFDQSx1Q0FBVSxLQUFWO0FBQ0EsNkNBQWdCLEtBQWhCO0FBQ0EscUNBQVEsS0FBUjtBQUFBLEdBTFU7QUFRTCxNQUFNLGNBQWMsQ0FDekIsUUFDQSxVQUNBLFlBQ0EsVUFBa0IsT0FDRDtBQUNqQixRQUFNLE1BQWdCLHNCQUFTO0FBQy9CLFFBQU0saUJBQTJCLElBQUksS0FBSyxF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
@@ -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>;