@nlabs/reaktor 0.5.4 → 0.8.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +7 -7
- package/lib/actions/conversations.js +68 -69
- package/lib/actions/dynamodb.js +22 -33
- package/lib/actions/email.js +32 -30
- package/lib/actions/files.js +39 -58
- package/lib/actions/groups.js +33 -28
- package/lib/actions/images.js +71 -82
- package/lib/actions/index.js +54 -37
- package/lib/actions/ios.js +21 -40
- package/lib/actions/locations.js +25 -22
- package/lib/actions/messages.js +31 -45
- package/lib/actions/notifications.js +13 -20
- package/lib/actions/payments.js +40 -52
- package/lib/actions/posts.js +68 -48
- package/lib/actions/reactions.js +29 -52
- package/lib/actions/s3.js +21 -18
- package/lib/actions/search.js +26 -40
- package/lib/actions/sms.js +24 -21
- package/lib/actions/statistics.js +14 -21
- package/lib/actions/subscription.js +27 -24
- package/lib/actions/tags.js +49 -56
- package/lib/actions/users.js +92 -106
- package/lib/actions/websockets.js +28 -41
- package/lib/config.js +21 -18
- package/lib/index.js +24 -22
- package/lib/lambdas/actions/websockets.js +21 -45
- package/lib/lambdas/authorizer.js +16 -24
- package/lib/lambdas/connection.js +18 -43
- package/lib/lambdas/utils/message.js +20 -17
- package/lib/lambdas/utils/websocket.js +33 -25
- package/lib/templates/email/layout.js +16 -9
- package/lib/templates/email/passwordForgot.js +16 -9
- package/lib/templates/email/passwordRecovery.js +16 -9
- package/lib/templates/email/verifyEmail.js +16 -9
- package/lib/templates/email/welcome.js +16 -9
- package/lib/templates/sms/passwordForgot.js +16 -9
- package/lib/templates/sms/passwordRecovery.js +16 -9
- package/lib/templates/sms/verifyEmail.js +16 -9
- package/lib/templates/sms/verifyPhone.js +16 -9
- package/lib/templates/sms/welcome.js +16 -9
- package/lib/types/apps.js +14 -5
- package/lib/types/arangodb.js +14 -5
- package/lib/types/auth.js +14 -5
- package/lib/types/connections.js +14 -5
- package/lib/types/conversations.js +14 -5
- package/lib/types/email.js +14 -5
- package/lib/types/files.js +14 -5
- package/lib/types/google.js +14 -5
- package/lib/types/groups.js +14 -5
- package/lib/types/images.js +14 -5
- package/lib/types/index.js +50 -35
- package/lib/types/locations.js +14 -5
- package/lib/types/messages.js +14 -5
- package/lib/types/notifications.js +14 -5
- package/lib/types/payments.js +14 -5
- package/lib/types/posts.js +14 -5
- package/lib/types/statistics.js +14 -5
- package/lib/types/tags.js +14 -5
- package/lib/types/users.js +14 -5
- package/lib/types/websocket.js +14 -5
- package/lib/utils/analytics.js +24 -44
- package/lib/utils/arangodb.js +15 -22
- package/lib/utils/auth.js +22 -36
- package/lib/utils/graphql.js +20 -17
- package/lib/utils/index.js +26 -23
- package/lib/utils/objects.js +22 -19
- package/lib/utils/session.js +20 -17
- package/package.json +31 -33
- package/.eslintrc +0 -18
- package/lib/actions/conversations.d.ts +0 -14
- package/lib/actions/dynamodb.d.ts +0 -8
- package/lib/actions/email.d.ts +0 -7
- package/lib/actions/files.d.ts +0 -16
- package/lib/actions/groups.d.ts +0 -14
- package/lib/actions/images.d.ts +0 -22
- package/lib/actions/index.d.ts +0 -20
- package/lib/actions/ios.d.ts +0 -7
- package/lib/actions/locations.d.ts +0 -3
- package/lib/actions/messages.d.ts +0 -13
- package/lib/actions/notifications.d.ts +0 -5
- package/lib/actions/payments.d.ts +0 -10
- package/lib/actions/posts.d.ts +0 -19
- package/lib/actions/reactions.d.ts +0 -30
- package/lib/actions/s3.d.ts +0 -7
- package/lib/actions/search.d.ts +0 -3
- package/lib/actions/sms.d.ts +0 -3
- package/lib/actions/statistics.d.ts +0 -2
- package/lib/actions/subscription.d.ts +0 -7
- package/lib/actions/tags.d.ts +0 -26
- package/lib/actions/users.d.ts +0 -44
- package/lib/actions/websockets.d.ts +0 -6
- package/lib/config.d.ts +0 -20
- package/lib/index.d.ts +0 -5
- package/lib/lambdas/actions/websockets.d.ts +0 -6
- package/lib/lambdas/authorizer.d.ts +0 -20
- package/lib/lambdas/connection.d.ts +0 -12
- package/lib/lambdas/utils/message.d.ts +0 -1
- package/lib/lambdas/utils/websocket.d.ts +0 -7
- package/lib/templates/email/layout.d.ts +0 -2
- package/lib/templates/email/passwordForgot.d.ts +0 -2
- package/lib/templates/email/passwordRecovery.d.ts +0 -2
- package/lib/templates/email/verifyEmail.d.ts +0 -2
- package/lib/templates/email/welcome.d.ts +0 -2
- package/lib/templates/sms/passwordForgot.d.ts +0 -2
- package/lib/templates/sms/passwordRecovery.d.ts +0 -2
- package/lib/templates/sms/verifyEmail.d.ts +0 -2
- package/lib/templates/sms/verifyPhone.d.ts +0 -2
- package/lib/templates/sms/welcome.d.ts +0 -2
- package/lib/types/apps.d.ts +0 -44
- package/lib/types/arangodb.d.ts +0 -17
- package/lib/types/auth.d.ts +0 -7
- package/lib/types/connections.d.ts +0 -8
- package/lib/types/conversations.d.ts +0 -11
- package/lib/types/email.d.ts +0 -12
- package/lib/types/files.d.ts +0 -26
- package/lib/types/google.d.ts +0 -27
- package/lib/types/groups.d.ts +0 -22
- package/lib/types/images.d.ts +0 -42
- package/lib/types/index.d.ts +0 -18
- package/lib/types/locations.d.ts +0 -20
- package/lib/types/messages.d.ts +0 -27
- package/lib/types/notifications.d.ts +0 -19
- package/lib/types/payments.d.ts +0 -115
- package/lib/types/posts.d.ts +0 -47
- package/lib/types/statistics.d.ts +0 -3
- package/lib/types/tags.d.ts +0 -15
- package/lib/types/users.d.ts +0 -80
- package/lib/types/websocket.d.ts +0 -14
- package/lib/utils/analytics.d.ts +0 -14
- package/lib/utils/arangodb.d.ts +0 -9
- package/lib/utils/auth.d.ts +0 -5
- package/lib/utils/graphql.d.ts +0 -1
- package/lib/utils/index.d.ts +0 -6
- package/lib/utils/objects.d.ts +0 -3
- package/lib/utils/session.d.ts +0 -18
package/lib/actions/users.js
CHANGED
|
@@ -1,59 +1,32 @@
|
|
|
1
1
|
var __create = Object.create;
|
|
2
2
|
var __defProp = Object.defineProperty;
|
|
3
|
-
var __defProps = Object.defineProperties;
|
|
4
3
|
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
5
|
-
var __getOwnPropDescs = Object.getOwnPropertyDescriptors;
|
|
6
4
|
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
7
|
-
var __getOwnPropSymbols = Object.getOwnPropertySymbols;
|
|
8
5
|
var __getProtoOf = Object.getPrototypeOf;
|
|
9
6
|
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 __require = typeof require !== "undefined" ? require : (x) => {
|
|
26
|
-
throw new Error('Dynamic require of "' + x + '" is not supported');
|
|
27
|
-
};
|
|
28
|
-
var __objRest = (source, exclude) => {
|
|
29
|
-
var target = {};
|
|
30
|
-
for (var prop in source)
|
|
31
|
-
if (__hasOwnProp.call(source, prop) && exclude.indexOf(prop) < 0)
|
|
32
|
-
target[prop] = source[prop];
|
|
33
|
-
if (source != null && __getOwnPropSymbols)
|
|
34
|
-
for (var prop of __getOwnPropSymbols(source)) {
|
|
35
|
-
if (exclude.indexOf(prop) < 0 && __propIsEnum.call(source, prop))
|
|
36
|
-
target[prop] = source[prop];
|
|
37
|
-
}
|
|
38
|
-
return target;
|
|
39
|
-
};
|
|
40
7
|
var __export = (target, all) => {
|
|
41
|
-
__markAsModule(target);
|
|
42
8
|
for (var name in all)
|
|
43
9
|
__defProp(target, name, { get: all[name], enumerable: true });
|
|
44
10
|
};
|
|
45
|
-
var
|
|
46
|
-
if (
|
|
47
|
-
for (let key of __getOwnPropNames(
|
|
48
|
-
if (!__hasOwnProp.call(
|
|
49
|
-
__defProp(
|
|
11
|
+
var __copyProps = (to, from, except, desc) => {
|
|
12
|
+
if (from && typeof from === "object" || typeof from === "function") {
|
|
13
|
+
for (let key of __getOwnPropNames(from))
|
|
14
|
+
if (!__hasOwnProp.call(to, key) && key !== except)
|
|
15
|
+
__defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
|
|
50
16
|
}
|
|
51
|
-
return
|
|
52
|
-
};
|
|
53
|
-
var __toModule = (module2) => {
|
|
54
|
-
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);
|
|
17
|
+
return to;
|
|
55
18
|
};
|
|
56
|
-
|
|
19
|
+
var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
|
|
20
|
+
// If the importer is in node compatibility mode or this is not an ESM
|
|
21
|
+
// file that has been converted to a CommonJS file using a Babel-
|
|
22
|
+
// compatible transform (i.e. "__esModule" has not been set), then set
|
|
23
|
+
// "default" to the CommonJS "module.exports" for node compatibility.
|
|
24
|
+
isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
|
|
25
|
+
mod
|
|
26
|
+
));
|
|
27
|
+
var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
|
|
28
|
+
var users_exports = {};
|
|
29
|
+
__export(users_exports, {
|
|
57
30
|
UserAccess: () => UserAccess,
|
|
58
31
|
addUser: () => addUser,
|
|
59
32
|
confirmCode: () => confirmCode,
|
|
@@ -75,24 +48,24 @@ __export(exports, {
|
|
|
75
48
|
signOut: () => signOut,
|
|
76
49
|
updateUser: () => updateUser
|
|
77
50
|
});
|
|
78
|
-
|
|
79
|
-
var
|
|
80
|
-
var
|
|
81
|
-
var import_luxon =
|
|
82
|
-
var import_stripe =
|
|
83
|
-
var import_config =
|
|
84
|
-
var import_utils2 =
|
|
85
|
-
var import_session =
|
|
51
|
+
module.exports = __toCommonJS(users_exports);
|
|
52
|
+
var import_utils = require("@nlabs/utils");
|
|
53
|
+
var import_arangojs = require("arangojs");
|
|
54
|
+
var import_luxon = require("luxon");
|
|
55
|
+
var import_stripe = __toESM(require("stripe"));
|
|
56
|
+
var import_config = require("../config");
|
|
57
|
+
var import_utils2 = require("../utils");
|
|
58
|
+
var import_session = require("../utils/session");
|
|
86
59
|
const eventCategory = "users";
|
|
87
60
|
const apiVersion = "2020-03-02";
|
|
88
|
-
var UserAccess
|
|
89
|
-
(function(UserAccess2) {
|
|
61
|
+
var UserAccess = /* @__PURE__ */ ((UserAccess2) => {
|
|
90
62
|
UserAccess2[UserAccess2["DEACTIVATED"] = 0] = "DEACTIVATED";
|
|
91
63
|
UserAccess2[UserAccess2["ACTIVE"] = 1] = "ACTIVE";
|
|
92
64
|
UserAccess2[UserAccess2["PREMIUM"] = 2] = "PREMIUM";
|
|
93
65
|
UserAccess2[UserAccess2["CONTENT_ADMIN"] = 3] = "CONTENT_ADMIN";
|
|
94
66
|
UserAccess2[UserAccess2["ADMIN"] = 4] = "ADMIN";
|
|
95
|
-
|
|
67
|
+
return UserAccess2;
|
|
68
|
+
})(UserAccess || {});
|
|
96
69
|
const createToken = (userId, username, userAccess, expires = 15) => {
|
|
97
70
|
const now = import_luxon.DateTime.local();
|
|
98
71
|
const sessionExpires = now.plus({ minutes: expires });
|
|
@@ -122,9 +95,10 @@ const parseUserOptions = (options = {}) => {
|
|
|
122
95
|
from = 0,
|
|
123
96
|
to = 30
|
|
124
97
|
} = options;
|
|
125
|
-
return
|
|
98
|
+
return {
|
|
99
|
+
...options,
|
|
126
100
|
limit: (0, import_utils2.getLimit)(from, to)
|
|
127
|
-
}
|
|
101
|
+
};
|
|
128
102
|
};
|
|
129
103
|
const addUser = async (context, args) => {
|
|
130
104
|
const action = "addUser";
|
|
@@ -136,7 +110,7 @@ const addUser = async (context, args) => {
|
|
|
136
110
|
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
137
111
|
const formatEmail = (0, import_utils.parseEmail)(email);
|
|
138
112
|
const formatPhone = (0, import_utils.parsePhone)(phone);
|
|
139
|
-
if (
|
|
113
|
+
if (!formatUsername || !password || !formatPhone && !formatEmail) {
|
|
140
114
|
return (0, import_utils2.logException)({
|
|
141
115
|
action,
|
|
142
116
|
args: { username },
|
|
@@ -145,21 +119,23 @@ const addUser = async (context, args) => {
|
|
|
145
119
|
}, context);
|
|
146
120
|
}
|
|
147
121
|
const filters = [`u.username == "${formatUsername}"`];
|
|
148
|
-
if (
|
|
122
|
+
if (formatEmail) {
|
|
149
123
|
filters.push(`u.email == "${formatEmail}"`);
|
|
150
124
|
}
|
|
151
|
-
if (
|
|
125
|
+
if (formatPhone) {
|
|
152
126
|
filters.push(`u.phone == ${formatPhone}`);
|
|
153
127
|
}
|
|
154
128
|
const checkQuery = `FOR u IN users
|
|
155
129
|
FILTER ${filters.join(" || ")}
|
|
156
130
|
RETURN u`;
|
|
157
|
-
const existingUsers = await database.query(checkQuery).then((cursor) => cursor.all()).catch(
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
131
|
+
const existingUsers = await database.query(checkQuery).then((cursor) => cursor.all()).catch(
|
|
132
|
+
(error) => (0, import_utils2.logError)({
|
|
133
|
+
action,
|
|
134
|
+
args: { username },
|
|
135
|
+
category: eventCategory,
|
|
136
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
137
|
+
}, error, context).then(() => null)
|
|
138
|
+
);
|
|
163
139
|
if (existingUsers.length) {
|
|
164
140
|
return (0, import_utils2.logException)({
|
|
165
141
|
action,
|
|
@@ -186,17 +162,19 @@ const addUser = async (context, args) => {
|
|
|
186
162
|
verifiedPhoneCode
|
|
187
163
|
};
|
|
188
164
|
const insertQuery = import_arangojs.aql`INSERT ${insert} IN users RETURN NEW`;
|
|
189
|
-
return await database.query(insertQuery).then((cursor) => cursor.next() || {}).catch(
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
165
|
+
return await database.query(insertQuery).then((cursor) => cursor.next() || {}).catch(
|
|
166
|
+
(error) => (0, import_utils2.logError)({
|
|
167
|
+
action,
|
|
168
|
+
args: { username },
|
|
169
|
+
category: eventCategory,
|
|
170
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
171
|
+
}, error, context).then(() => null)
|
|
172
|
+
);
|
|
195
173
|
};
|
|
196
174
|
const updateUser = async (context, user) => {
|
|
197
175
|
const action = "updateUser";
|
|
198
176
|
const { database } = context;
|
|
199
|
-
const
|
|
177
|
+
const { _key, _id, id, tags = [], userId, ...updated } = user;
|
|
200
178
|
let userDocId;
|
|
201
179
|
if (_id || id) {
|
|
202
180
|
userDocId = _id || id;
|
|
@@ -252,12 +230,14 @@ const confirmCode = async (context, args) => {
|
|
|
252
230
|
default:
|
|
253
231
|
return false;
|
|
254
232
|
}
|
|
255
|
-
}).catch(
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
|
|
233
|
+
}).catch(
|
|
234
|
+
(error) => (0, import_utils2.logError)({
|
|
235
|
+
action,
|
|
236
|
+
args: { code, type, userId: sessionId },
|
|
237
|
+
category: eventCategory,
|
|
238
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
239
|
+
}, error, context)
|
|
240
|
+
);
|
|
261
241
|
} catch (error) {
|
|
262
242
|
return false;
|
|
263
243
|
}
|
|
@@ -311,11 +291,11 @@ const deactivateUser = (context, userId) => {
|
|
|
311
291
|
const getDisplayName = (user = {}) => {
|
|
312
292
|
const { first, last, name = "", username = "" } = user;
|
|
313
293
|
const fullname = [first, last].join(" ").trim();
|
|
314
|
-
if (
|
|
294
|
+
if (name) {
|
|
315
295
|
return name;
|
|
316
296
|
} else if (fullname !== "") {
|
|
317
297
|
return fullname;
|
|
318
|
-
} else if (
|
|
298
|
+
} else if (username) {
|
|
319
299
|
return username;
|
|
320
300
|
}
|
|
321
301
|
return "Unknown";
|
|
@@ -358,7 +338,7 @@ const getUsers = (context, options) => {
|
|
|
358
338
|
const { limit, username } = parseUserOptions(options);
|
|
359
339
|
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
360
340
|
const filterBy = ["u.userAccess > 0"];
|
|
361
|
-
if (
|
|
341
|
+
if (username) {
|
|
362
342
|
filterBy.push(`CONTAINS(u.username, "${(0, import_utils.parseUsername)(username)}")`);
|
|
363
343
|
}
|
|
364
344
|
const aqlQuery = `FOR u IN users
|
|
@@ -382,10 +362,10 @@ const getUsersByReactions = (context, { reactions = [], username }, options) =>
|
|
|
382
362
|
const formatSessionId = `users/${sessionId}`;
|
|
383
363
|
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
384
364
|
const filterBy = [`POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`];
|
|
385
|
-
if (
|
|
365
|
+
if (username) {
|
|
386
366
|
filterBy.push(`CONTAINS(u.username, "${formatUsername}")`);
|
|
387
367
|
}
|
|
388
|
-
const aqlQuery = `FOR u, r IN OUTBOUND "${formatSessionId}"
|
|
368
|
+
const aqlQuery = `FOR u, r IN OUTBOUND "${formatSessionId}" hasReaction
|
|
389
369
|
OPTIONS {vertexCollections: "users"}
|
|
390
370
|
${selectQueries.join("\n")}
|
|
391
371
|
${filterBy.length ? `FILTER ${filterBy.join(" && ")}` : ""}
|
|
@@ -402,7 +382,7 @@ const getUsersByTags = (context, { tags, username }, options) => {
|
|
|
402
382
|
const action = "getUsersByTags";
|
|
403
383
|
const { database, fields, session: { userId: sessionId } } = context;
|
|
404
384
|
const formatTags = tags.reduce((list, tagName) => {
|
|
405
|
-
if (
|
|
385
|
+
if (tagName) {
|
|
406
386
|
list.push((0, import_utils.parseChar)(tagName, 32).toLowerCase());
|
|
407
387
|
}
|
|
408
388
|
return list;
|
|
@@ -411,7 +391,7 @@ const getUsersByTags = (context, { tags, username }, options) => {
|
|
|
411
391
|
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
412
392
|
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
413
393
|
const filterBy = [`u._key != "${sessionId}"`];
|
|
414
|
-
if (
|
|
394
|
+
if (username) {
|
|
415
395
|
filterBy.push(`CONTAINS(u.username, "${formatUsername}")`);
|
|
416
396
|
}
|
|
417
397
|
const aqlQuery = `FOR t IN tags
|
|
@@ -435,7 +415,7 @@ const getUsersByLatest = (context, { username }, options) => {
|
|
|
435
415
|
const { limit } = parseUserOptions(options);
|
|
436
416
|
const filter = ["u._id != session._id"];
|
|
437
417
|
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
438
|
-
if (
|
|
418
|
+
if (username) {
|
|
439
419
|
filter.push(`CONTAINS(u.username, "${(0, import_utils.parseUsername)(username)}")`);
|
|
440
420
|
}
|
|
441
421
|
const aqlQuery = `FOR u IN users
|
|
@@ -471,7 +451,7 @@ const signIn = async (context, args) => {
|
|
|
471
451
|
const formatPassword = (0, import_utils.parsePassword)(password);
|
|
472
452
|
const formatExpires = (0, import_utils.parseNum)(expires) || 15;
|
|
473
453
|
const { database } = context;
|
|
474
|
-
if (
|
|
454
|
+
if (!formatUsername || !formatPassword) {
|
|
475
455
|
return (0, import_utils2.logException)({
|
|
476
456
|
action,
|
|
477
457
|
args: { username },
|
|
@@ -483,12 +463,14 @@ const signIn = async (context, args) => {
|
|
|
483
463
|
FILTER u.username == ${formatUsername}
|
|
484
464
|
LIMIT 1
|
|
485
465
|
RETURN u`;
|
|
486
|
-
const checkUser = await database.query(checkQuery).then((cursor) => cursor.next()).catch(
|
|
487
|
-
|
|
488
|
-
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
|
|
466
|
+
const checkUser = await database.query(checkQuery).then((cursor) => cursor.next()).catch(
|
|
467
|
+
(error) => (0, import_utils2.logError)({
|
|
468
|
+
action,
|
|
469
|
+
args: { username: formatUsername },
|
|
470
|
+
category: eventCategory,
|
|
471
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
472
|
+
}, error, context).then(() => null)
|
|
473
|
+
);
|
|
492
474
|
if (!checkUser) {
|
|
493
475
|
return (0, import_utils2.logException)({
|
|
494
476
|
action,
|
|
@@ -522,12 +504,14 @@ const signOut = async (context, args) => {
|
|
|
522
504
|
LIMIT 1
|
|
523
505
|
RETURN NEW`;
|
|
524
506
|
try {
|
|
525
|
-
await database.query(sessionQuery).then((cursor) => cursor.next()).catch(
|
|
526
|
-
|
|
527
|
-
|
|
528
|
-
|
|
529
|
-
|
|
530
|
-
|
|
507
|
+
await database.query(sessionQuery).then((cursor) => cursor.next()).catch(
|
|
508
|
+
(error) => (0, import_utils2.logError)({
|
|
509
|
+
action,
|
|
510
|
+
args: { username, userId: sessionId },
|
|
511
|
+
category: eventCategory,
|
|
512
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
513
|
+
}, error, context)
|
|
514
|
+
);
|
|
531
515
|
} catch (error) {
|
|
532
516
|
return false;
|
|
533
517
|
}
|
|
@@ -542,11 +526,13 @@ const getActiveUserCount = (context) => {
|
|
|
542
526
|
RETURN u
|
|
543
527
|
)
|
|
544
528
|
RETURN LENGTH(docs)`;
|
|
545
|
-
return database.query(countQuery).then((cursor) => cursor.next()).catch(
|
|
546
|
-
|
|
547
|
-
|
|
548
|
-
|
|
549
|
-
|
|
529
|
+
return database.query(countQuery).then((cursor) => cursor.next()).catch(
|
|
530
|
+
(error) => (0, import_utils2.logError)({
|
|
531
|
+
action,
|
|
532
|
+
category: eventCategory,
|
|
533
|
+
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
534
|
+
}, error, context).then(() => 0)
|
|
535
|
+
);
|
|
550
536
|
};
|
|
551
537
|
// Annotate the CommonJS export names for ESM import in node:
|
|
552
538
|
0 && (module.exports = {
|
|
@@ -571,4 +557,4 @@ const getActiveUserCount = (context) => {
|
|
|
571
557
|
signOut,
|
|
572
558
|
updateUser
|
|
573
559
|
});
|
|
574
|
-
//# 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": []
}

|
|
560
|
+
//# 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 {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(!formatUsername || !password || (!formatPhone && !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(formatEmail) {\n    filters.push(`u.email == \"${formatEmail}\"`);\n  }\n\n  if(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(name) {\n    return name;\n  } else if(fullname !== '') {\n    return fullname;\n  } else if(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(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(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}\" hasReaction\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(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(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(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(!formatUsername || !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": ";;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAIA,mBAUO;AACP,sBAAkB;AAIlB,mBAAuB;AACvB,oBAAmB;AAEnB,oBAAqB;AAGrB,IAAAA,gBAAsF;AACtF,qBAAiE;AAEjE,MAAM,gBAAwB;AAC9B,MAAM,aAAkB;AAQjB,IAAK,aAAL,kBAAKC,gBAAL;AACL,EAAAA,wBAAA,iBAAc,KAAd;AACA,EAAAA,wBAAA,YAAS,KAAT;AACA,EAAAA,wBAAA,aAAU,KAAV;AACA,EAAAA,wBAAA,mBAAgB,KAAhB;AACA,EAAAA,wBAAA,WAAQ,KAAR;AALU,SAAAA;AAAA,GAAA;AAQL,MAAM,cAAc,CACzB,QACA,UACA,YACA,UAAkB,OACD;AACjB,QAAM,MAAgB,sBAAS,MAAM;AACrC,QAAM,iBAA2B,IAAI,KAAK,EAAC,SAAS,QAAO,CAAC;AAC5D,QAAM,MAAc,KAAK,MAAM,IAAI,UAAU,CAAC;AAC9C,QAAM,MAAc,KAAK,MAAM,eAAe,UAAU,CAAC;AAEzD,QAAM,YAAQ,2BAAW;AAAA,IACvB;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,IACA;AAAA,EACF,CAAC;AAED,SAAO;AAAA,IACL,SAAS,eAAe,SAAS;AAAA,IACjC,QAAQ,IAAI,SAAS;AAAA,IACrB;AAAA,EACF;AACF;AAEO,MAAM,kBAAkB,CAAC,SAAmB,CAAC,MAClD,OAAO,OAAO,CAAC,SAAc,UAAkB;AAC7C,MAAG,MAAM,SAAS,OAAO,GAAG;AAC1B,eAAO,yCAA0B,SAAS,KAAK,OAAO,OAAO;AAAA,EAC/D;AAEA,SAAO;AACT,GAAG,EAAC,SAAS,CAAC,GAAG,SAAS,CAAC,EAAC,CAAC;AAExB,MAAM,mBAAmB,CAAC,UAAuB,CAAC,MAAM;AAC7D,QAAM;AAAA,IACJ,OAAO;AAAA,IACP,KAAK;AAAA,EACP,IAAI;AAEJ,SAAO;AAAA,IACL,GAAG;AAAA,IACH,WAAO,wBAAS,MAAM,EAAE;AAAA,EAC1B;AACF;AAEO,MAAM,UAAU,OAAO,SAAqB,SAA6B;AAC9E,QAAM,SAAiB;AACvB,QAAM,EAAC,SAAQ,IAAI;AACnB,QAAM,EAAC,KAAI,IAAI;AACf,QAAM,EAAC,OAAO,UAAU,OAAO,SAAQ,IAAI;AAC3C,QAAM,WAAe,yBAAW,GAAG,QAAQ,GAAG,QAAQ,IAAI,IAAI;AAC9D,QAAM,wBAAoB,6BAAe,UAAU,IAAI;AACvD,QAAM,qBAAyB,4BAAc,QAAQ;AACrD,QAAM,kBAAsB,yBAAW,KAAK;AAC5C,QAAM,kBAAsB,yBAAW,KAAK;AAE5C,MAAG,CAAC,kBAAkB,CAAC,YAAa,CAAC,eAAe,CAAC,aAAc;AACjE,eAAO,4BAAa;AAAA,MAClB;AAAA,MACA,MAAM,EAAC,SAAQ;AAAA,MACf,UAAU;AAAA,MACV,OAAO,yBAAW;AAAA,IACpB,GAAG,OAAO;AAAA,EACZ;AAEA,QAAM,UAAoB,CAAC,kBAAkB,cAAc,GAAG;AAE9D,MAAG,aAAa;AACd,YAAQ,KAAK,eAAe,WAAW,GAAG;AAAA,EAC5C;AAEA,MAAG,aAAa;AACd,YAAQ,KAAK,cAAc,WAAW,EAAE;AAAA,EAC1C;AAEA,QAAM,aAAqB;AAAA,aAChB,QAAQ,KAAK,MAAM,CAAC;AAAA;AAE/B,QAAM,gBAAgB,MAAM,SAAS,MAAM,UAAU,EAClD,KAAK,CAAC,WAAwB,OAAO,IAAI,CAAC,EAC1C;AAAA,IAAM,CAAC,cAAU,wBAAS;AAAA,MACzB;AAAA,MACA,MAAM,EAAC,SAAQ;AAAA,MACf,UAAU;AAAA,MACV,OAAO,yBAAW;AAAA,IACpB,GAAG,OAAO,OAAO,EACd,KAAK,MAAM,IAAI;AAAA,EAClB;AAEF,MAAG,cAAc,QAAQ;AACvB,eAAO,4BAAa;AAAA,MAClB;AAAA,MACA,MAAM,EAAC,SAAQ;AAAA,MACf,UAAU;AAAA,MACV,OAAO,yBAAW;AAAA,IACpB,GAAG,OAAO;AAAA,EACZ;AAEA,QAAM,oBAA4B,KAAK,MAAM,MAAU,KAAK,OAAO,IAAI,GAAO;AAC9E,QAAM,oBAA4B,KAAK,MAAM,MAAU,KAAK,OAAO,IAAI,GAAO;AAG9E,QAAM,SAAe;AAAA,IACnB,UAAM,yBAAW,UAAU,IAAI;AAAA,IAC/B,OAAO,KAAK,IAAI;AAAA,IAChB,OAAO;AAAA,IACP,UAAU,KAAK,IAAI;AAAA,IACnB,UAAU;AAAA,IACV,OAAO;AAAA,IACP;AAAA,IACA,UAAU;AAAA,IACV,YAAY;AAAA,IACZ,eAAe;AAAA,IACf;AAAA,IACA,eAAe;AAAA,IACf;AAAA,EACF;AAGA,QAAM,cAAwB,6BAAa,MAAM;AACjD,SAAO,MAAM,SAAS,MAAM,WAAW,EACpC,KAAK,CAAC,WAAwB,OAAO,KAAK,KAAK,CAAC,CAAC,EACjD;AAAA,IAAM,CAAC,cAAU,wBAAS;AAAA,MACzB;AAAA,MACA,MAAM,EAAC,SAAQ;AAAA,MACf,UAAU;AAAA,MACV,OAAO,yBAAW;AAAA,IACpB,GAAG,OAAO,OAAO,EACd,KAAK,MAAM,IAAI;AAAA,EAClB;AACJ;AAEO,MAAM,aAAa,OAAO,SAAqB,SAA6B;AACjF,QAAM,SAAiB;AACvB,QAAM,EAAC,SAAQ,IAAI;AACnB,QAAM,EAAC,MAAM,KAAK,IAAI,OAAO,CAAC,GAAG,QAAQ,GAAG,QAAO,IAAI;AACvD,MAAI;AAEJ,MAAG,OAAO,IAAI;AACZ,gBAAY,OAAO;AAAA,EACrB,WAAU,QAAQ,QAAQ;AACxB,gBAAY,SAAS,QAAQ,MAAM;AAAA,EACrC;AAEA,QAAM,YAAsB,uCAAuB,SAAS;AAAA,oBAC1C,OAAO;AAAA;AAGzB,QAAM,cAAc,MAAM,SAAS,MAAM,SAAS,EAC/C,KAAK,CAAC,WAAW,OAAO,KAAK,CAAC,EAC9B,MAAM,CAAC,UAAiB;AACvB,YAAQ,IAAI,KAAK;AACjB,UAAM;AAAA,EACR,CAAC;AAEH,QAAM,gBAAgC,SAAS,WAAW,UAAU;AACpE,QAAM,QAAQ,IAAI,KAAK,IAAI,CAAC,EAAC,IAAI,UAAU,KAAI,MAAM;AACnD,UAAM,OAAO;AAAA,MACX,OAAO;AAAA,MACP,UAAM,yBAAW,YAAY,QAAQ,IAAI,SAAS,EAAE;AAAA,MACpD,KAAK;AAAA,MACL,OAAO,KAAK,IAAI;AAAA,MAChB;AAAA,IACF;AACA,UAAM,WAAqB;AAAA,2BACJ,QAAQ,iBAAiB,SAAS,kBAAkB,IAAI;AAAA;AAAA;AAI/E,WAAO,SAAS,MAAM,QAAQ,EAC3B,KAAK,CAAC,WAAwB,OAAO,KAAK,CAAC,EAC3C,KAAK,CAAC,YAAY;AACjB,UAAG,CAAC,CAAC,SAAS;AACZ,eAAO;AAAA,MACT;AAEA,aAAO,cAAc,KAAK,MAAM,EAAC,WAAW,KAAI,CAAC,EAAE,KAAK,MAAM,IAAI;AAAA,IACpE,CAAC,EACA,MAAM,CAAC,cAAiB,wBAAS;AAAA,MAChC;AAAA,MACA,UAAU;AAAA,MACV,OAAO;AAAA,IACT,GAAG,OAAO,OAAO,EAAE,KAAK,MAAM,IAAI,CAAC;AAAA,EACvC,CAAC,CAAC;AAEF,SAAO;AACT;AAEO,MAAM,cAAc,OAAO,SAAqB,SAA2B;AAChF,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACjD,QAAM,EAAC,MAAM,KAAI,IAAI;AACrB,QAAM,YAAoB,SAAS,SAAS;AAE5C,QAAM,WAAqB,uCAAuB,SAAS;AAE3D,MAAI;AACF,WAAO,SAAS,MAAM,QAAQ,EAC3B,KAAK,CAAC,WAAW,OAAO,KAAK,KAAK,CAAC,CAAC,EACpC,KAAK,CAAC,EAAC,mBAAmB,kBAAiB,MAAY;AACtD,cAAO,MAAM;AAAA,QACX,KAAK;AACH,iBAAO,SAAS;AAAA,QAClB,KAAK;AACH,iBAAO,SAAS;AAAA,QAClB;AACE,iBAAO;AAAA,MACX;AAAA,IACF,CAAC,EACA;AAAA,MAAM,CAAC,cAAU,wBAAS;AAAA,QACzB;AAAA,QACA,MAAM,EAAC,MAAM,MAAM,QAAQ,UAAS;AAAA,QACpC,UAAU;AAAA,QACV,OAAO,yBAAW;AAAA,MACpB,GAAG,OAAO,OAAO;AAAA,IACjB;AAAA,EACJ,SAAQ,OAAO;AACb,WAAO;AAAA,EACT;AACF;AAEO,MAAM,aAAa,CAAC,SAAqB,SAA4B;AAC1E,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,WAAW,YAAY,cAAa,EAAC,IAAI;AAC5E,QAAM,EAAC,OAAM,IAAI;AACjB,QAAM,UAAmB,gBAAgB;AAEzC,MAAG,CAAC,WAAY,cAAc,QAAS;AACrC,oCAAa;AAAA,MACX;AAAA,MACA;AAAA,MACA,UAAU;AAAA,MACV,OAAO;AAAA,MACP,OAAO;AAAA,IACT,GAAG,OAAO;AACV,WAAO;AAAA,EACT;AAEA,QAAM,WAAqB;AAAA,uBACN,MAAM;AAAA;AAAA;AAAA;AAM3B,QAAM,eAAe,IAAI,cAAAC,QAAO,qBAAO,IAAI,cAAc,GAAG,EAAC,YAAY,YAAY,KAAI,CAAC;AAE1F,SAAO,SAAS,MAAM,QAAQ,EAC3B,KAAK,CAAC,WAAwB,OAAO,KAAK,CAAC,EAC3C,KAAK,CAAC,OAAa,CAAC,MAAM,aAAa,UAAU,IAAI,KAAK,gBAAgB,EACxE,KAAK,MAAM,aAAa,SAAS,IAAI,KAAK,eAAe,CAAC,EAC1D,KAAK,MAAM,IAAI,CAAC,EAClB,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA,EACR,CAAC;AACL;AAEO,MAAM,iBAAiB,CAAC,SAAqB,WAAkC;AACpF,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,WAAW,YAAY,cAAa,EAAC,IAAI;AAC5E,QAAM,UAAmB,gBAAgB;AAEzC,MAAG,CAAC,WAAY,cAAc,QAAS;AACrC,oCAAa;AAAA,MACX;AAAA,MACA,UAAU;AAAA,MACV,OAAO;AAAA,MACP,OAAO;AAAA,IACT,GAAG,OAAO;AACV,WAAO;AAAA,EACT;AAEA,QAAM,UAAgB;AAAA,IACpB,YAAY;AAAA,EACd;AACA,QAAM,WAAqB,6BAAa,MAAM,SAAS,OAAO;AAE9D,SAAO,SAAS,MAAM,QAAQ,EAC3B,KAAK,CAAC,WAAwB,OAAO,KAAK,CAAC,EAC3C,MAAM,CAAC,UAAiB;AACvB,UAAM;AAAA,EACR,CAAC;AACL;AAEO,MAAM,iBAAiB,CAAC,OAAa,CAAC,MAAc;AACzD,QAAM,EAAC,OAAO,MAAM,OAAO,IAAI,WAAW,GAAE,IAAI;AAChD,QAAM,WAAoB,CAAC,OAAO,IAAI,EAAG,KAAK,GAAG,EAAE,KAAK;AAExD,MAAG,MAAM;AACP,WAAO;AAAA,EACT,WAAU,aAAa,IAAI;AACzB,WAAO;AAAA,EACT,WAAU,UAAU;AAClB,WAAO;AAAA,EACT;AAEA,SAAO;AACT;AAEO,MAAM,iBAAiB,CAAC,YAAuC;AACpE,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,QAAQ,WAAW,SAAQ,EAAC,IAAI;AACnE,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,MAAM;AAE/E,QAAM,WAAmB,2BAA2B,SAAS;AAAA,IAC3D,cAAc,KAAK,IAAI,CAAC;AAAA,qBACP,cAAc,KAAK,IAAI,CAAC;AAE3C,SAAO,SAAS,MAAM,QAAQ,EAC3B,KAAK,CAAC,WAAwB,OAAO,KAAK,CAAC,EAC3C,KAAK,CAAC,OAAO,CAAC,MAAM,IAAI,EACxB,MAAM,CAAC,cAAiB,wBAAS;AAAA,IAChC;AAAA,IACA,MAAM,EAAC,UAAU,QAAQ,UAAS;AAAA,IAClC,UAAU;AAAA,IACV,OAAO,yBAAW;AAAA,EACpB,GAAG,OAAO,OAAO,CAAC;AACtB;AAEO,MAAM,UAAU,CAAC,SAAqB,SAA6B;AACxE,QAAM,SAAiB;AACvB,QAAM,EAAC,OAAM,IAAI;AACjB,QAAM,EAAC,UAAU,OAAM,IAAI;AAC3B,QAAM,mBAAuB,sBAAQ,MAAM;AAC3C,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,MAAM;AAG/E,QAAM,WAAmB,2BAA2B,YAAY;AAAA,MAC5D,cAAc,KAAK,IAAI,CAAC;AAAA;AAAA,uBAEP,cAAc,KAAK,IAAI,CAAC;AAE7C,SAAO,SAAS,MAAM,QAAQ,EAC3B,KAAK,CAAC,WAAwB,OAAO,KAAK,CAAC,EAC3C,KAAK,CAAC,OAAO,CAAC,MAAM,IAAI,EACxB,MAAM,CAAC,cAAiB,wBAAS;AAAA,IAChC;AAAA,IACA;AAAA,IACA,UAAU;AAAA,IACV,OAAO,yBAAW;AAAA,EACpB,GAAG,OAAO,OAAO,EACd,KAAK,MAAM;AAAA,EAAE,CAAC,CAAC;AACtB;AAEO,MAAM,WAAW,CAAC,SAAqB,YAA2C;AACvF,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,OAAM,IAAI;AAC3B,QAAM,EAAC,OAAO,SAAQ,IAAI,iBAAiB,OAAO;AAClD,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,MAAM;AAC/E,QAAM,WAAqB,CAAC,kBAAkB;AAE9C,MAAG,UAAU;AACX,aAAS,KAAK,6BAAyB,4BAAc,QAAQ,CAAC,IAAI;AAAA,EACpE;AAGA,QAAM,WAAmB;AAAA,aACd,SAAS,KAAK,MAAM,CAAC;AAAA,MAC5B,cAAc,KAAK,IAAI,CAAC;AAAA,MACxB,MAAM,GAAG;AAAA;AAAA,uBAEQ,cAAc,KAAK,IAAI,CAAC;AAE7C,SAAO,SAAS,MAAM,QAAQ,EAC3B,KAAK,CAAC,WAAwB,OAAO,IAAI,CAAC,EAC1C,MAAM,CAAC,cAAiB,wBAAS;AAAA,IAChC;AAAA,IACA,UAAU;AAAA,IACV,OAAO,yBAAW;AAAA,EACpB,GAAG,OAAO,OAAO,EAAE,KAAK,MAAM,CAAC,CAAC,CAAC;AACrC;AAEO,MAAM,sBAAsB,CACjC,SACA,EAAC,YAAY,CAAC,GAAG,SAAQ,GACzB,YACoB;AACpB,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACzD,QAAM,kBAA6B,UAAU,IAAI,CAAC,qBAAyB,wBAAU,cAAc,EAAE,EAAE,YAAY,CAAC;AACpH,QAAM,EAAC,MAAK,IAAI,iBAAiB,OAAO;AACxC,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,MAAM;AAE/E,QAAM,kBAA0B,SAAS,SAAS;AAClD,QAAM,qBAAyB,4BAAc,QAAQ;AACrD,QAAM,WAAqB,CAAC,YAAY,KAAK,UAAU,eAAe,CAAC,kBAAkB;AAEzF,MAAG,UAAU;AACX,aAAS,KAAK,yBAAyB,cAAc,IAAI;AAAA,EAC3D;AAGA,QAAM,WAAmB,yBAAyB,eAAe;AAAA;AAAA,MAE7D,cAAc,KAAK,IAAI,CAAC;AAAA,MACxB,SAAS,SAAS,UAAU,SAAS,KAAK,MAAM,CAAC,KAAK,EAAE;AAAA,MACxD,MAAM,GAAG;AAAA,uBACQ,cAAc,KAAK,IAAI,CAAC;AAE7C,UAAQ,IAAI,EAAC,SAAQ,CAAC;AACtB,SAAO,SAAS,MAAM,QAAQ,EAC3B,KAAK,CAAC,WAAwB,OAAO,IAAI,CAAC,EAC1C,MAAM,CAAC,cAAiB,wBAAS;AAAA,IAChC;AAAA,IACA,UAAU;AAAA,IACV,OAAO,yBAAW;AAAA,EACpB,GAAG,OAAO,OAAO,EAAE,KAAK,MAAM,CAAC,CAAC,CAAC;AACrC;AAEO,MAAM,iBAAiB,CAC5B,SACA,EAAC,MAAM,SAAQ,GACf,YACoB;AACpB,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,QAAQ,UAAS,EAAC,IAAI;AACzD,QAAM,aAAwB,KAAK,OAAO,CAAC,MAAgB,YAAoB;AAC7E,QAAG,SAAS;AACV,WAAK,SAAK,wBAAU,SAAS,EAAE,EAAE,YAAY,CAAC;AAAA,IAChD;AAEA,WAAO;AAAA,EACT,GAAG,CAAC,CAAC;AACL,QAAM,EAAC,MAAK,IAAI,iBAAiB,OAAO;AACxC,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,MAAM;AAE/E,QAAM,qBAAyB,4BAAc,QAAQ;AACrD,QAAM,WAAqB,CAAC,cAAc,SAAS,GAAG;AAEtD,MAAG,UAAU;AACX,aAAS,KAAK,yBAAyB,cAAc,IAAI;AAAA,EAC3D;AAGA,QAAM,WAAmB;AAAA,sBACL,KAAK,UAAU,UAAU,CAAC;AAAA;AAAA;AAAA,MAG1C,cAAc,KAAK,IAAI,CAAC;AAAA,MACxB,SAAS,SAAS,UAAU,SAAS,KAAK,MAAM,CAAC,KAAK,EAAE;AAAA,MACxD,MAAM,GAAG;AAAA,gCACiB,cAAc,KAAK,IAAI,CAAC;AAEtD,UAAQ,IAAI,kBAAkB,QAAQ;AACtC,SAAO,SAAS,MAAM,QAAQ,EAC3B,KAAK,CAAC,WAAwB,OAAO,IAAI,CAAC,EAC1C,MAAM,CAAC,cAAiB,wBAAS;AAAA,IAChC;AAAA,IACA,UAAU;AAAA,IACV,OAAO,yBAAW;AAAA,EACpB,GAAG,OAAO,OAAO,EAAE,KAAK,MAAM,CAAC,CAAC,CAAC;AACrC;AAEO,MAAM,mBAAmB,CAAC,SAAqB,EAAC,SAAQ,GAAG,YAA2C;AAC3G,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,QAAQ,SAAS,EAAC,OAAM,EAAC,IAAI;AAC9C,QAAM,EAAC,MAAK,IAAI,iBAAiB,OAAO;AACxC,QAAM,SAAS,CAAC,sBAAsB;AACtC,QAAM,EAAC,SAAS,eAAe,SAAS,cAAa,IAAI,gBAAgB,MAAM;AAE/E,MAAG,UAAU;AACX,WAAO,KAAK,6BAAyB,4BAAc,QAAQ,CAAC,IAAI;AAAA,EAClE;AAGA,QAAM,WAAmB;AAAA,oCACS,MAAM;AAAA,aAC7B,OAAO,KAAK,MAAM,CAAC;AAAA,MAC1B,cAAc,KAAK,IAAI,CAAC;AAAA;AAAA,MAExB,MAAM,GAAG;AAAA;AAAA,uBAEQ,cAAc,KAAK,IAAI,CAAC;AAE7C,SAAO,SAAS,MAAM,QAAQ,EAC3B,KAAK,CAAC,WAAwB,OAAO,IAAI,CAAC,EAC1C,MAAM,CAAC,cAAiB,wBAAS;AAAA,IAChC;AAAA,IACA,UAAU;AAAA,IACV,OAAO,yBAAW;AAAA,EACpB,GAAG,OAAO,OAAO,EAAE,KAAK,MAAM,CAAC,CAAC,CAAC;AACrC;AAEO,MAAM,iBAAiB,OAAO,SAAqB,EAAC,SAAS,MAAK,MAA4C;AACnH,QAAM,EAAC,MAAK,QAAI,2BAAW,KAAK;AAEhC,MAAG,OAAO;AACR,WAAO,EAAC,QAAQ,CAAC,KAAK,EAAC;AAAA,EACzB;AAEA,QAAM,EAAC,KAAK,QAAQ,UAAU,WAAU,QAAI,2BAAW,KAAK;AAC5D,QAAM,MAAc,KAAK,KAAK,sBAAS,MAAM,EAAE,UAAU,CAAC;AAE1D,MAAG,MAAM,KAAK;AACZ,WAAO,YAAY,QAAQ,UAAU,YAAY,OAAO;AAAA,EAC1D;AAEA,SAAO,EAAC,QAAQ,CAAC,iBAAiB,EAAC;AACrC;AAEO,MAAM,SAAS,OAAO,SAAqB,SAAgC;AAChF,QAAM,SAAiB;AACvB,QAAM,EAAC,SAAS,UAAU,SAAQ,IAAI;AACtC,QAAM,qBAAyB,4BAAc,QAAQ;AACrD,QAAM,qBAAyB,4BAAc,QAAQ;AACrD,QAAM,oBAAwB,uBAAS,OAAO,KAAK;AACnD,QAAM,EAAC,SAAQ,IAAI;AAEnB,MAAG,CAAC,kBAAkB,CAAC,gBAAgB;AACrC,eAAO,4BAAa;AAAA,MAClB;AAAA,MACA,MAAM,EAAC,SAAQ;AAAA,MACf,UAAU;AAAA,MACV,OAAO,yBAAW;AAAA,IACpB,GAAG,OAAO;AAAA,EACZ;AAEA,QAAM,aAAuB;AAAA,2BACJ,cAAc;AAAA;AAAA;AAGvC,QAAM,YAAkB,MAAM,SAAS,MAAM,UAAU,EACpD,KAAK,CAAC,WAAW,OAAO,KAAK,CAAC,EAC9B;AAAA,IAAM,CAAC,cAAU,wBAAS;AAAA,MACzB;AAAA,MACA,MAAM,EAAC,UAAU,eAAc;AAAA,MAC/B,UAAU;AAAA,MACV,OAAO,yBAAW;AAAA,IACpB,GAAG,OAAO,OAAO,EACd,KAAK,MAAM,IAAI;AAAA,EAClB;AAEF,MAAG,CAAC,WAAW;AACb,eAAO,4BAAa;AAAA,MAClB;AAAA,MACA,MAAM,EAAC,SAAQ;AAAA,MACf,UAAU;AAAA,MACV,OAAO,yBAAW;AAAA,IACpB,GAAG,OAAO;AAAA,EACZ;AAEA,QAAM,EAAC,MAAM,QAAQ,MAAM,WAAU,IAAI;AACzC,QAAM,mBAAuB,6BAAe,gBAAgB,IAAI;AAEhE,MAAG,UAAU,aAAa,cAAc;AACtC,eAAO,4BAAa;AAAA,MAClB;AAAA,MACA,MAAM,EAAC,SAAQ;AAAA,MACf,UAAU;AAAA,MACV,OAAO,yBAAW;AAAA,IACpB,GAAG,OAAO;AAAA,EACZ;AAEA,SAAO,YAAY,QAAQ,UAAU,YAAY,aAAa;AAChE;AAEO,MAAM,UAAU,OAAO,SAAqB,SAA2B;AAC5E,QAAM,SAAiB;AACvB,QAAM,EAAC,UAAU,SAAS,EAAC,QAAQ,WAAW,SAAQ,EAAC,IAAI;AAC3D,QAAM,YAAoB,SAAS,SAAS;AAE5C,QAAM,SAAS;AAAA,IACb,YAAY,KAAK,IAAI;AAAA,IACrB,WAAW;AAAA,EACb;AACA,QAAM,eAAyB,uCAAuB,SAAS;AAAA,oBAC7C,MAAM;AAAA;AAAA;AAIxB,MAAI;AACF,UAAM,SAAS,MAAM,YAAY,EAC9B,KAAK,CAAC,WAAW,OAAO,KAAK,CAAC,EAC9B;AAAA,MAAM,CAAC,cAAU,wBAAS;AAAA,QACzB;AAAA,QACA,MAAM,EAAC,UAAU,QAAQ,UAAS;AAAA,QAClC,UAAU;AAAA,QACV,OAAO,yBAAW;AAAA,MACpB,GAAG,OAAO,OAAO;AAAA,IACjB;AAAA,EACJ,SAAQ,OAAO;AACb,WAAO;AAAA,EACT;AAEA,SAAO;AACT;AAEO,MAAM,qBAAqB,CAAC,YAAwB;AACzD,QAAM,SAAiB;AACvB,QAAM,EAAC,SAAQ,IAAI;AACnB,QAAM,aAAuB;AAAA;AAAA;AAAA;AAAA;AAAA;AAO7B,SAAO,SAAS,MAAM,UAAU,EAC7B,KAAK,CAAC,WAAW,OAAO,KAAK,CAAC,EAC9B;AAAA,IAAM,CAAC,cAAU,wBAAS;AAAA,MACzB;AAAA,MACA,UAAU;AAAA,MACV,OAAO,yBAAW;AAAA,IACpB,GAAG,OAAO,OAAO,EACd,KAAK,MAAM,CAAC;AAAA,EACf;AACJ;",
  "names": ["import_utils", "UserAccess", "Stripe"]
}

|