@nlabs/reaktor 0.9.0 → 0.10.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.env +1 -0
- package/.env.example +1 -0
- package/DATABASE_I18N_GUIDE.md +434 -0
- package/README.md +9 -0
- package/TEST_UTILITIES_GUIDE.md +360 -0
- package/coverage/index.html +61 -46
- package/dist/actions/apps.js +242 -0
- package/dist/actions/connections.js +90 -0
- package/dist/actions/conversations.js +350 -0
- package/dist/actions/dynamodb.js +150 -0
- package/dist/actions/email.js +152 -0
- package/dist/actions/files.js +283 -0
- package/dist/actions/groups.js +292 -0
- package/dist/actions/images.js +735 -0
- package/dist/actions/index.js +66 -0
- package/dist/actions/ios.js +164 -0
- package/dist/actions/locations.js +122 -0
- package/dist/actions/messages.js +208 -0
- package/dist/actions/notifications.js +59 -0
- package/dist/actions/payments.js +497 -0
- package/dist/actions/personas.js +110 -0
- package/dist/actions/posts.js +595 -0
- package/dist/actions/reactions.js +322 -0
- package/dist/actions/s3.js +133 -0
- package/dist/actions/search.js +90 -0
- package/dist/actions/sms.js +108 -0
- package/dist/actions/statistics.js +62 -0
- package/dist/actions/subscription.js +220 -0
- package/dist/actions/tags.js +292 -0
- package/dist/actions/users.js +784 -0
- package/dist/actions/websockets.js +174 -0
- package/dist/adapters/arangoAdapter.js +46 -0
- package/dist/adapters/fileAdapter.js +76 -0
- package/dist/adapters/imageAdapter.js +40 -0
- package/dist/adapters/messageAdapter.js +49 -0
- package/dist/adapters/postAdapter.js +70 -0
- package/dist/adapters/reaktorAdapter.js +44 -0
- package/dist/adapters/tagAdapter.js +50 -0
- package/dist/adapters/userAdapter.js +115 -0
- package/dist/config.js +125 -0
- package/dist/index.js +66 -0
- package/dist/lambdas/actions/websockets.js +132 -0
- package/dist/lambdas/authorizer.js +67 -0
- package/dist/lambdas/connection.js +91 -0
- package/dist/lambdas/utils/message.js +42 -0
- package/dist/lambdas/utils/websocket.js +105 -0
- package/dist/mocks/conversation.js +35 -0
- package/dist/mocks/file.js +38 -0
- package/dist/mocks/group.js +47 -0
- package/dist/mocks/image.js +44 -0
- package/dist/mocks/nlabs.png +0 -0
- package/dist/mocks/post.js +55 -0
- package/dist/mocks/tag.js +37 -0
- package/dist/mocks/user.js +88 -0
- package/dist/mutations/index.js +26 -0
- package/dist/mutations/locations.js +44 -0
- package/dist/mutations/messages.js +86 -0
- package/dist/mutations/personas.js +100 -0
- package/dist/mutations/posts.js +53 -0
- package/dist/mutations/reactions.js +51 -0
- package/dist/mutations/statistics.js +39 -0
- package/dist/mutations/subscriptions.js +56 -0
- package/dist/mutations/tags.js +120 -0
- package/dist/mutations/users.js +116 -0
- package/dist/objectTypes/app.js +173 -0
- package/dist/objectTypes/bankAccount.js +76 -0
- package/dist/objectTypes/connection.js +48 -0
- package/dist/objectTypes/conversation.js +77 -0
- package/dist/objectTypes/creditCard.js +86 -0
- package/dist/objectTypes/document.js +46 -0
- package/dist/objectTypes/error.js +46 -0
- package/dist/objectTypes/external.js +74 -0
- package/dist/objectTypes/file.js +100 -0
- package/dist/objectTypes/filter.js +43 -0
- package/dist/objectTypes/group.js +123 -0
- package/dist/objectTypes/iapSubscription.js +40 -0
- package/dist/objectTypes/image.js +129 -0
- package/dist/objectTypes/index.js +68 -0
- package/dist/objectTypes/location.js +109 -0
- package/dist/objectTypes/message.js +96 -0
- package/dist/objectTypes/passcode.js +42 -0
- package/dist/objectTypes/persona.js +87 -0
- package/dist/objectTypes/plan.js +95 -0
- package/dist/objectTypes/post.js +125 -0
- package/dist/objectTypes/reaction.js +61 -0
- package/dist/objectTypes/relation.js +49 -0
- package/dist/objectTypes/search.js +72 -0
- package/dist/objectTypes/statistics.js +39 -0
- package/dist/objectTypes/subscription.js +117 -0
- package/dist/objectTypes/tag.js +65 -0
- package/dist/objectTypes/user.js +144 -0
- package/dist/queries/index.js +33 -0
- package/dist/queries/locations.js +45 -0
- package/dist/queries/messages.js +52 -0
- package/dist/queries/posts.js +154 -0
- package/dist/queries/reactions.js +56 -0
- package/dist/queries/statistics.js +39 -0
- package/dist/queries/subscriptions.js +44 -0
- package/dist/queries/tags.js +75 -0
- package/dist/queries/users.js +64 -0
- package/dist/templates/email/layout.js +302 -0
- package/dist/templates/email/passwordForgot.js +38 -0
- package/dist/templates/email/passwordRecovery.js +35 -0
- package/dist/templates/email/verifyEmail.js +38 -0
- package/dist/templates/email/welcome.js +38 -0
- package/dist/templates/sms/passwordForgot.js +24 -0
- package/dist/templates/sms/passwordRecovery.js +24 -0
- package/dist/templates/sms/verifyEmail.js +24 -0
- package/dist/templates/sms/verifyPhone.js +24 -0
- package/dist/templates/sms/welcome.js +24 -0
- package/dist/types/apps.js +32 -0
- package/{lib → dist}/types/arangodb.js +1 -1
- package/{lib → dist}/types/auth.js +1 -1
- package/{lib → dist}/types/connections.js +1 -1
- package/dist/types/conversations.js +16 -0
- package/{lib → dist}/types/email.js +1 -1
- package/dist/types/files.js +16 -0
- package/dist/types/google.js +16 -0
- package/{lib → dist}/types/groups.js +1 -1
- package/dist/types/images.js +16 -0
- package/dist/types/index.js +60 -0
- package/{lib → dist}/types/locations.js +1 -1
- package/{lib → dist}/types/messages.js +1 -1
- package/{lib → dist}/types/notifications.js +1 -1
- package/dist/types/payments.js +16 -0
- package/dist/types/personas.js +16 -0
- package/dist/types/posts.js +16 -0
- package/{lib → dist}/types/tags.js +1 -1
- package/dist/types/users.js +16 -0
- package/dist/types/websockets.js +16 -0
- package/dist/utils/adapterUtils.js +45 -0
- package/dist/utils/analyticsUtils.js +72 -0
- package/dist/utils/arangodbUtils.js +165 -0
- package/dist/utils/auth.js +57 -0
- package/dist/utils/index.js +30 -0
- package/{lib → dist}/utils/session.js +10 -7
- package/index.js +1 -1
- package/jest.config.js +17 -0
- package/jest.setup.js +36 -0
- package/lex.config.cjs +13 -0
- package/lib/actions/apps.js +17 -249
- package/lib/actions/connections.js +7 -0
- package/lib/actions/content.js +17 -0
- package/lib/actions/conversations.js +19 -325
- package/lib/actions/dynamodb.js +2 -150
- package/lib/actions/email.js +2 -152
- package/lib/actions/files.js +5 -287
- package/lib/actions/groups.js +23 -263
- package/lib/actions/images.js +31 -646
- package/lib/actions/index.js +2 -62
- package/lib/actions/ios.js +9 -162
- package/lib/actions/locations.js +7 -110
- package/lib/actions/messages.js +21 -193
- package/lib/actions/notifications.js +2 -59
- package/lib/actions/payments.js +11 -461
- package/lib/actions/posts.js +77 -515
- package/lib/actions/profiles.js +8 -0
- package/lib/actions/reactions.js +25 -312
- package/lib/actions/s3.js +2 -133
- package/lib/actions/search.js +5 -90
- package/lib/actions/sms.js +2 -106
- package/lib/actions/statistics.js +6 -60
- package/lib/actions/subscriptions.js +12 -0
- package/lib/actions/tags.js +19 -262
- package/lib/actions/users.js +67 -537
- package/lib/actions/websockets.js +16 -158
- package/lib/adapters/arangoAdapter.js +2 -46
- package/lib/adapters/contentAdapter.js +2 -0
- package/lib/adapters/fileAdapter.js +2 -76
- package/lib/adapters/imageAdapter.js +2 -0
- package/lib/adapters/messageAdapter.js +2 -0
- package/lib/adapters/postAdapter.js +2 -70
- package/lib/adapters/reaktorAdapter.js +2 -44
- package/lib/adapters/tagAdapter.js +2 -50
- package/lib/adapters/userAdapter.js +2 -110
- package/lib/config.js +2 -125
- package/lib/handlers/graphqlHandler.js +2 -0
- package/lib/index.js +2 -30
- package/lib/lambdas/actions/websockets.js +14 -112
- package/lib/lambdas/authorizer.js +2 -67
- package/lib/lambdas/connection.js +2 -92
- package/lib/lambdas/utils/message.js +2 -42
- package/lib/lambdas/utils/websocket.js +2 -105
- package/lib/mocks/conversation.js +2 -35
- package/lib/mocks/file.js +2 -38
- package/lib/mocks/group.js +2 -47
- package/lib/mocks/image.js +2 -43
- package/lib/mocks/post.js +2 -55
- package/lib/mocks/tag.js +2 -37
- package/lib/mocks/user.js +2 -88
- package/lib/mutations/index.js +2 -0
- package/lib/mutations/locations.integration.js +2 -0
- package/lib/mutations/locations.js +2 -0
- package/lib/mutations/messages.integration.js +2 -0
- package/lib/mutations/messages.js +2 -0
- package/lib/mutations/posts.integration.js +2 -0
- package/lib/mutations/posts.js +2 -0
- package/lib/mutations/profiles.integration.js +2 -0
- package/lib/mutations/profiles.js +2 -0
- package/lib/mutations/reactions.integration.js +2 -0
- package/lib/mutations/reactions.js +2 -0
- package/lib/mutations/statistics.integration.js +2 -0
- package/lib/mutations/statistics.js +2 -0
- package/lib/mutations/subscriptions.integration.js +2 -0
- package/lib/mutations/subscriptions.js +2 -0
- package/lib/mutations/tags.integration.js +2 -0
- package/lib/mutations/tags.js +2 -0
- package/lib/mutations/users.integration.js +2 -0
- package/lib/mutations/users.js +2 -0
- package/lib/objectTypes/app.js +2 -0
- package/lib/objectTypes/bankAccount.js +2 -0
- package/lib/objectTypes/connection.js +2 -0
- package/lib/objectTypes/conversation.js +2 -0
- package/lib/objectTypes/creditCard.js +2 -0
- package/lib/objectTypes/document.js +2 -0
- package/lib/objectTypes/error.js +2 -0
- package/lib/objectTypes/external.js +2 -0
- package/lib/objectTypes/file.js +2 -0
- package/lib/objectTypes/filter.js +2 -0
- package/lib/objectTypes/group.js +2 -0
- package/lib/objectTypes/iapSubscription.js +2 -0
- package/lib/objectTypes/image.js +2 -0
- package/lib/objectTypes/index.js +2 -0
- package/lib/objectTypes/location.js +2 -0
- package/lib/objectTypes/message.js +2 -0
- package/lib/objectTypes/passcode.js +2 -0
- package/lib/objectTypes/plan.js +2 -0
- package/lib/objectTypes/post.js +2 -0
- package/lib/objectTypes/profile.js +2 -0
- package/lib/objectTypes/reaction.js +2 -0
- package/lib/objectTypes/relation.js +2 -0
- package/lib/objectTypes/search.js +2 -0
- package/lib/objectTypes/statistics.js +2 -0
- package/lib/objectTypes/subscription.js +2 -0
- package/lib/objectTypes/tag.js +2 -0
- package/lib/objectTypes/user.js +2 -0
- package/lib/queries/index.js +2 -0
- package/lib/queries/locations.integration.js +2 -0
- package/lib/queries/locations.js +2 -0
- package/lib/queries/messages.integration.js +2 -0
- package/lib/queries/messages.js +2 -0
- package/lib/queries/posts.integration.js +2 -0
- package/lib/queries/posts.js +2 -0
- package/lib/queries/reactions.integration.js +2 -0
- package/lib/queries/reactions.js +2 -0
- package/lib/queries/statistics.js +2 -0
- package/lib/queries/subscriptions.js +2 -0
- package/lib/queries/tags.integration.js +2 -0
- package/lib/queries/tags.js +2 -0
- package/lib/queries/users.integration.js +2 -0
- package/lib/queries/users.js +2 -0
- package/lib/templates/email/layout.js +3 -25
- package/lib/templates/email/passwordForgot.js +3 -25
- package/lib/templates/email/passwordRecovery.js +3 -25
- package/lib/templates/email/verifyEmail.js +3 -25
- package/lib/templates/email/welcome.js +3 -25
- package/lib/templates/sms/passwordForgot.js +2 -24
- package/lib/templates/sms/passwordRecovery.js +2 -24
- package/lib/templates/sms/verifyEmail.js +2 -24
- package/lib/templates/sms/verifyPhone.js +2 -24
- package/lib/templates/sms/welcome.js +2 -24
- package/lib/types/apps.types.js +2 -0
- package/lib/types/arangodb.types.js +1 -0
- package/lib/types/auth.types.js +1 -0
- package/lib/types/connections.types.js +1 -0
- package/lib/types/content.types.js +1 -0
- package/lib/types/conversations.types.js +1 -0
- package/lib/types/email.types.js +1 -0
- package/lib/types/error.types.js +2 -0
- package/lib/types/files.types.js +1 -0
- package/lib/types/google.types.js +1 -0
- package/lib/types/groups.types.js +1 -0
- package/lib/types/images.types.js +1 -0
- package/lib/types/index.js +2 -56
- package/lib/types/locations.types.js +1 -0
- package/lib/types/messages.types.js +1 -0
- package/lib/types/notifications.types.js +1 -0
- package/lib/types/payments.types.js +1 -0
- package/lib/types/posts.types.js +1 -0
- package/lib/types/profiles.types.js +1 -0
- package/lib/types/statistics.types.js +1 -0
- package/lib/types/tags.types.js +1 -0
- package/lib/types/users.types.js +1 -0
- package/lib/types/websockets.types.js +1 -0
- package/lib/utils/adapterUtils.js +2 -45
- package/lib/utils/analyticsUtils.js +2 -72
- package/lib/utils/arangodbUtils.js +5 -142
- package/lib/utils/authUtils.js +2 -0
- package/lib/utils/contextUtils.js +2 -0
- package/lib/utils/dbI18n.example.js +6 -0
- package/lib/utils/dbI18n.js +2 -0
- package/lib/utils/googleTranslate.js +2 -0
- package/lib/utils/graphqlUtils.js +2 -0
- package/lib/utils/index.js +2 -30
- package/lib/utils/languageDetection.js +2 -0
- package/lib/utils/localeUtils.example.js +2 -0
- package/lib/utils/localeUtils.js +2 -0
- package/lib/utils/middlewareUtils.js +2 -0
- package/lib/utils/sessionUtils.js +2 -0
- package/lib/utils/stripeUtils.js +2 -0
- package/lib/utils/templateUtils.js +2 -0
- package/lib/utils/testUtils.js +2 -0
- package/lib/utils/translationQueue.example.js +2 -0
- package/lib/utils/translationQueue.js +2 -0
- package/package.json +41 -37
- package/.eslintrc +0 -10
- package/.prettierrc.js +0 -4
- package/lib/actions/apps.d.ts +0 -25
- package/lib/actions/conversations.d.ts +0 -14
- package/lib/actions/dynamodb.d.ts +0 -8
- package/lib/actions/email.d.ts +0 -5
- package/lib/actions/files.d.ts +0 -19
- package/lib/actions/groups.d.ts +0 -14
- package/lib/actions/images.d.ts +0 -26
- package/lib/actions/index.d.ts +0 -21
- package/lib/actions/ios.d.ts +0 -7
- package/lib/actions/locations.d.ts +0 -5
- 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 -3
- package/lib/actions/subscription.d.ts +0 -7
- package/lib/actions/subscription.js +0 -218
- package/lib/actions/tags.d.ts +0 -29
- package/lib/actions/users.d.ts +0 -47
- package/lib/actions/websockets.d.ts +0 -19
- package/lib/adapters/arangoAdapter.d.ts +0 -2
- package/lib/adapters/fileAdapter.d.ts +0 -3
- package/lib/adapters/postAdapter.d.ts +0 -2
- package/lib/adapters/reaktorAdapter.d.ts +0 -6
- package/lib/adapters/tagAdapter.d.ts +0 -2
- package/lib/adapters/userAdapter.d.ts +0 -2
- 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/mocks/conversation.d.ts +0 -8
- package/lib/mocks/file.d.ts +0 -11
- package/lib/mocks/group.d.ts +0 -17
- package/lib/mocks/image.d.ts +0 -3
- package/lib/mocks/post.d.ts +0 -38
- package/lib/mocks/tag.d.ts +0 -2
- package/lib/mocks/user.d.ts +0 -4
- 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 -46
- package/lib/types/apps.js +0 -32
- package/lib/types/arangodb.d.ts +0 -30
- package/lib/types/auth.d.ts +0 -7
- package/lib/types/connections.d.ts +0 -7
- package/lib/types/conversations.d.ts +0 -29
- package/lib/types/conversations.js +0 -16
- package/lib/types/email.d.ts +0 -13
- package/lib/types/error.d.ts +0 -20
- package/lib/types/files.d.ts +0 -26
- package/lib/types/files.js +0 -16
- package/lib/types/google.d.ts +0 -29
- package/lib/types/google.js +0 -16
- package/lib/types/groups.d.ts +0 -21
- package/lib/types/images.d.ts +0 -51
- package/lib/types/images.js +0 -16
- package/lib/types/index.d.ts +0 -18
- package/lib/types/locations.d.ts +0 -20
- package/lib/types/messages.d.ts +0 -16
- package/lib/types/notifications.d.ts +0 -17
- package/lib/types/payments.d.ts +0 -112
- package/lib/types/payments.js +0 -16
- package/lib/types/posts.d.ts +0 -31
- package/lib/types/posts.js +0 -16
- package/lib/types/statistics.d.ts +0 -3
- package/lib/types/tags.d.ts +0 -10
- package/lib/types/users.d.ts +0 -76
- package/lib/types/users.js +0 -16
- package/lib/types/websocket.d.ts +0 -13
- package/lib/types/websocket.js +0 -16
- package/lib/utils/adapterUtils.d.ts +0 -1
- package/lib/utils/analyticsUtils.d.ts +0 -21
- package/lib/utils/arangodbUtils.d.ts +0 -65
- package/lib/utils/auth.d.ts +0 -20
- package/lib/utils/auth.js +0 -50
- package/lib/utils/index.d.ts +0 -5
- package/lib/utils/session.d.ts +0 -16
- /package/{lib → dist}/types/error.js +0 -0
- /package/{lib → dist}/types/statistics.js +0 -0
package/lib/actions/users.js
CHANGED
|
@@ -1,554 +1,84 @@
|
|
|
1
|
-
var
|
|
2
|
-
|
|
3
|
-
var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
|
|
4
|
-
var __getOwnPropNames = Object.getOwnPropertyNames;
|
|
5
|
-
var __getProtoOf = Object.getPrototypeOf;
|
|
6
|
-
var __hasOwnProp = Object.prototype.hasOwnProperty;
|
|
7
|
-
var __export = (target, all) => {
|
|
8
|
-
for (var name in all)
|
|
9
|
-
__defProp(target, name, { get: all[name], enumerable: true });
|
|
10
|
-
};
|
|
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 });
|
|
16
|
-
}
|
|
17
|
-
return to;
|
|
18
|
-
};
|
|
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, {
|
|
30
|
-
UserAccess: () => UserAccess,
|
|
31
|
-
addUser: () => addUser,
|
|
32
|
-
confirmCode: () => confirmCode,
|
|
33
|
-
createToken: () => createToken,
|
|
34
|
-
deactivateUser: () => deactivateUser,
|
|
35
|
-
deleteUser: () => deleteUser,
|
|
36
|
-
getActiveUserCount: () => getActiveUserCount,
|
|
37
|
-
getDisplayName: () => getDisplayName,
|
|
38
|
-
getSessionUser: () => getSessionUser,
|
|
39
|
-
getUser: () => getUser,
|
|
40
|
-
getUserByToken: () => getUserByToken,
|
|
41
|
-
getUserOptional: () => getUserOptional,
|
|
42
|
-
getUsers: () => getUsers,
|
|
43
|
-
getUsersByLatest: () => getUsersByLatest,
|
|
44
|
-
getUsersByReactions: () => getUsersByReactions,
|
|
45
|
-
getUsersByTags: () => getUsersByTags,
|
|
46
|
-
parseUserOptions: () => parseUserOptions,
|
|
47
|
-
refreshSession: () => refreshSession,
|
|
48
|
-
signIn: () => signIn,
|
|
49
|
-
signOut: () => signOut,
|
|
50
|
-
updateUser: () => updateUser
|
|
51
|
-
});
|
|
52
|
-
module.exports = __toCommonJS(users_exports);
|
|
53
|
-
var import_utils = require("@nlabs/utils");
|
|
54
|
-
var import_arangojs = require("arangojs");
|
|
55
|
-
var import_luxon = require("luxon");
|
|
56
|
-
var import_stripe = __toESM(require("stripe"));
|
|
57
|
-
var import_config = require("../config");
|
|
58
|
-
var import_analyticsUtils = require("../utils/analyticsUtils");
|
|
59
|
-
var import_arangodbUtils = require("../utils/arangodbUtils");
|
|
60
|
-
var import_session = require("../utils/session");
|
|
61
|
-
var import_error = require("../types/error");
|
|
62
|
-
var import_userAdapter = require("../adapters/userAdapter");
|
|
63
|
-
const eventCategory = "users";
|
|
64
|
-
const STRIPE_API_VERSION = "2020-03-02";
|
|
65
|
-
var UserAccess = /* @__PURE__ */ ((UserAccess2) => {
|
|
66
|
-
UserAccess2[UserAccess2["DEACTIVATED"] = 0] = "DEACTIVATED";
|
|
67
|
-
UserAccess2[UserAccess2["ACTIVE"] = 1] = "ACTIVE";
|
|
68
|
-
UserAccess2[UserAccess2["PREMIUM"] = 2] = "PREMIUM";
|
|
69
|
-
UserAccess2[UserAccess2["CONTENT_ADMIN"] = 3] = "CONTENT_ADMIN";
|
|
70
|
-
UserAccess2[UserAccess2["ADMIN"] = 4] = "ADMIN";
|
|
71
|
-
return UserAccess2;
|
|
72
|
-
})(UserAccess || {});
|
|
73
|
-
const createToken = (userId, username, userAccess, expiresInMinutes = 15) => {
|
|
74
|
-
const now = import_luxon.DateTime.local();
|
|
75
|
-
const sessionExpires = now.plus({ minutes: expiresInMinutes });
|
|
76
|
-
const iat = Math.floor(now.toSeconds());
|
|
77
|
-
const exp = Math.floor(sessionExpires.toSeconds());
|
|
78
|
-
const token = (0, import_session.setSession)({
|
|
79
|
-
exp,
|
|
80
|
-
iat,
|
|
81
|
-
username,
|
|
82
|
-
userAccess,
|
|
83
|
-
userId
|
|
84
|
-
});
|
|
85
|
-
return {
|
|
86
|
-
expires: sessionExpires.toMillis(),
|
|
87
|
-
issued: now.toMillis(),
|
|
88
|
-
token
|
|
89
|
-
};
|
|
90
|
-
};
|
|
91
|
-
const getUserOptional = (fields = []) => fields.reduce((selects, field) => {
|
|
92
|
-
if (field.includes("Count")) {
|
|
93
|
-
return (0, import_arangodbUtils.selectReactionCountByType)("users", "u", field, selects);
|
|
94
|
-
}
|
|
95
|
-
return selects;
|
|
96
|
-
}, { objects: [], queries: [] });
|
|
97
|
-
const parseUserOptions = (options = {}) => {
|
|
98
|
-
const {
|
|
99
|
-
from = 0,
|
|
100
|
-
to = 30
|
|
101
|
-
} = options;
|
|
102
|
-
const limit = (0, import_arangodbUtils.getLimit)(from, to);
|
|
103
|
-
return {
|
|
104
|
-
...options,
|
|
105
|
-
limit
|
|
106
|
-
};
|
|
107
|
-
};
|
|
108
|
-
const addUser = async (context, user) => {
|
|
109
|
-
const action = "addUser";
|
|
110
|
-
const { database } = context;
|
|
111
|
-
const { email, password, phone, username } = (0, import_userAdapter.parseUser)(user);
|
|
112
|
-
const salt = (0, import_utils.createHash)(`${username}${password}`, null);
|
|
113
|
-
const encryptedPassword = (0, import_utils.createPassword)(password, salt);
|
|
114
|
-
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
115
|
-
const formatEmail = (0, import_utils.parseEmail)(email);
|
|
116
|
-
const formatPhone = (0, import_utils.parsePhone)(phone);
|
|
117
|
-
if (!formatUsername || !password || !formatPhone && !formatEmail) {
|
|
118
|
-
return (0, import_analyticsUtils.logException)({
|
|
119
|
-
action,
|
|
120
|
-
category: eventCategory,
|
|
121
|
-
params: { username },
|
|
122
|
-
value: import_error.ErrorTypes.INVALID_ARGUMENTS
|
|
123
|
-
}, context);
|
|
124
|
-
}
|
|
125
|
-
const filters = [`u.username == "${formatUsername}"`];
|
|
126
|
-
if (formatEmail) {
|
|
127
|
-
filters.push(`u.email == "${formatEmail}"`);
|
|
128
|
-
}
|
|
129
|
-
if (formatPhone) {
|
|
130
|
-
filters.push(`u.phone == ${formatPhone}`);
|
|
131
|
-
}
|
|
132
|
-
const checkQuery = `FOR u IN users
|
|
133
|
-
FILTER ${filters.join(" || ")}
|
|
1
|
+
import{createHash as v,createPassword as P,parseArangoId as M,parseChar as Q,parseEmail as H,parseNum as J,parsePassword as L,parsePhone as Y,parseUsername as S}from"@nlabs/utils";import{aql as U}from"arangojs";import{DateTime as X}from"luxon";import z from"stripe";import{parseUser as D}from"../adapters/userAdapter";import{Config as K}from"../config";import{sendEmail as Z}from"./email";import{sendSms as ee}from"./sms";import{ErrorTypes as d}from"../types/error.types";import{logError as I,logException as q}from"../utils/analyticsUtils";import{getDocId as j,getLimit as se,selectReactionCountByType as re,useDb as g}from"../utils/arangodbUtils";import{detectLanguage as te}from"../utils/languageDetection";import{getSession as _,setSession as ne}from"../utils/sessionUtils";const E="users",oe="2025-05-28.basil";var ae=(t=>(t[t.DEACTIVATED=0]="DEACTIVATED",t[t.ACTIVE=1]="ACTIVE",t[t.PREMIUM=2]="PREMIUM",t[t.CONTENT_ADMIN=3]="CONTENT_ADMIN",t[t.ADMIN=4]="ADMIN",t))(ae||{});const B=(e,o,s,i=15)=>{const n=X.local(),t=n.plus({minutes:i}),u=Math.floor(n.toSeconds()),r=Math.floor(t.toSeconds()),a=ne({exp:r,iat:u,userAccess:s,userId:e,username:o});return{expires:t.toMillis(),issued:n.toMillis(),token:a,userId:e,username:o}},$=(e=[])=>e.reduce((o,s)=>s.includes("Count")?re("users","u",s,o):o,{objects:[],queries:[]}),b=(e={})=>{const{from:o=0,to:s=30}=e,i=se(o,s);return{...e,limit:i}},Ue=async(e,o)=>{const s="addUser",{databaseName:i,languageContext:n}=e,{email:t,password:u,phone:r,username:a,userId:c,_key:l,_id:p,...m}=D(o),T=L(u);if(!!!T||!(!!a||!!r||!!t))return q({action:s,category:E,params:{username:a},value:d.INVALID_ARGUMENTS},e);const A=v(`${a||r||t}${T}`,null),N=P(T,A),h=[];a&&h.push(`u.username == "${a}"`),t&&h.push(`u.email == "${t}"`),r&&h.push(`u.phone == ${r}`);const w=`FOR u IN users
|
|
2
|
+
FILTER ${h.join(" || ")}
|
|
134
3
|
LIMIT 1
|
|
135
|
-
RETURN u`;
|
|
136
|
-
|
|
137
|
-
const
|
|
138
|
-
|
|
139
|
-
return (0, import_analyticsUtils.logException)({
|
|
140
|
-
action,
|
|
141
|
-
category: eventCategory,
|
|
142
|
-
params: { username },
|
|
143
|
-
value: import_error.ErrorTypes.EXISTING_ITEM
|
|
144
|
-
}, context);
|
|
145
|
-
}
|
|
146
|
-
} catch (error) {
|
|
147
|
-
return (0, import_analyticsUtils.logError)({
|
|
148
|
-
action,
|
|
149
|
-
category: eventCategory,
|
|
150
|
-
params: { username },
|
|
151
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
152
|
-
}, error, context);
|
|
153
|
-
}
|
|
154
|
-
const verifiedEmailCode = Math.floor(1e5 + Math.random() * 9e5);
|
|
155
|
-
const verifiedPhoneCode = Math.floor(1e5 + Math.random() * 9e5);
|
|
156
|
-
const insert = {
|
|
157
|
-
_key: (0, import_utils.createHash)(username, null),
|
|
158
|
-
added: Date.now(),
|
|
159
|
-
email: formatEmail,
|
|
160
|
-
modified: Date.now(),
|
|
161
|
-
password: encryptedPassword,
|
|
162
|
-
phone: formatPhone,
|
|
163
|
-
salt,
|
|
164
|
-
username: formatUsername,
|
|
165
|
-
userAccess: 1,
|
|
166
|
-
verifiedEmail: false,
|
|
167
|
-
verifiedEmailCode,
|
|
168
|
-
verifiedPhone: false,
|
|
169
|
-
verifiedPhoneCode
|
|
170
|
-
};
|
|
171
|
-
const insertQuery = import_arangojs.aql`INSERT ${insert} IN users RETURN NEW`;
|
|
172
|
-
return await database.query(insertQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
|
|
173
|
-
action,
|
|
174
|
-
category: eventCategory,
|
|
175
|
-
params: { username },
|
|
176
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
177
|
-
}, error, context));
|
|
178
|
-
};
|
|
179
|
-
const updateUser = async (context, user) => {
|
|
180
|
-
const action = "updateUser";
|
|
181
|
-
const { database, session } = context;
|
|
182
|
-
const { _key, _id, id, tags = [], userId, ...updated } = (0, import_userAdapter.parseUser)(user);
|
|
183
|
-
if (!(0, import_session.isAdminUser)(session) && session?.userId !== userId) {
|
|
184
|
-
return (0, import_analyticsUtils.logException)({
|
|
185
|
-
action,
|
|
186
|
-
category: eventCategory,
|
|
187
|
-
params: { session },
|
|
188
|
-
value: import_error.ErrorTypes.INVALID_SESSION
|
|
189
|
-
}, context);
|
|
190
|
-
}
|
|
191
|
-
const userQuery = import_arangojs.aql`LET u = DOCUMENT(${id})
|
|
192
|
-
UPDATE u WITH ${updated} IN users
|
|
193
|
-
RETURN NEW`;
|
|
194
|
-
try {
|
|
195
|
-
const updatedUser = await database.query(userQuery).then((cursor) => cursor.next());
|
|
196
|
-
const tagCollection = database.collection("isTagged");
|
|
197
|
-
await Promise.all(tags.map(({ id: tagDocId, name }) => {
|
|
198
|
-
const tagQuery = import_arangojs.aql`FOR it IN isTagged
|
|
199
|
-
FILTER it._from == ${tagDocId} && it._to == ${id} && it.name == ${name}
|
|
4
|
+
RETURN u`;try{if((await g(i).query(w).then(W=>W.all())).length)return q({action:s,category:E,params:{email:t,phone:r,username:a},value:d.EXISTING_ITEM},e)}catch(C){return I({action:s,category:E,params:{email:t,phone:r,username:a},value:d.DATABASE_ERROR},C,e)}const x=r?.replace(/\D/g,"").substring(0,3),O=te({...n,phoneCountryCode:x,userPreference:m.locale||"en"}),F=Math.floor(1e5+Math.random()*9e5),k=Math.floor(1e5+Math.random()*9e5),G={...m,_key:v(a,null),added:Date.now(),email:t,locale:O,modified:Date.now(),password:N,phone:r,salt:A,userAccess:1,username:a,verifiedEmail:!1,verifiedEmailCode:F,verifiedPhone:!1,verifiedSmsCode:k},V=U`INSERT ${G} IN users RETURN NEW`;return await g(i).query(V).then(C=>C.next()).catch(C=>I({action:s,category:E,params:{username:a},value:d.DATABASE_ERROR},C,e))},Ne=async(e,o)=>{const s="updateUser",{databaseName:i}=e,n=D(o),{_key:t,_id:u,tags:r=[],...a}=n,{id:c}=n,l=U`LET u = DOCUMENT(${c})
|
|
5
|
+
UPDATE u WITH ${a} IN users
|
|
6
|
+
RETURN NEW`;try{const p=g(i),m=await p.query(l).then(y=>y.next()),T=p.collection("isTagged");return await Promise.all(r.map(({id:y,name:f})=>{const R=U`FOR it IN isTagged
|
|
7
|
+
FILTER it._from == ${y} && it._to == ${c} && it.name == ${f}
|
|
200
8
|
LIMIT 1
|
|
201
|
-
RETURN it`;
|
|
202
|
-
|
|
203
|
-
if (!!tagEdge) {
|
|
204
|
-
return tagEdge;
|
|
205
|
-
}
|
|
206
|
-
const edge = {
|
|
207
|
-
_from: tagDocId,
|
|
208
|
-
_key: (0, import_utils.createHash)(`isTagged-${tagDocId}-${id}`),
|
|
209
|
-
_to: id,
|
|
210
|
-
added: Date.now(),
|
|
211
|
-
name
|
|
212
|
-
};
|
|
213
|
-
return tagCollection.save(edge, { returnNew: true }).then(() => edge);
|
|
214
|
-
});
|
|
215
|
-
}));
|
|
216
|
-
return updatedUser;
|
|
217
|
-
} catch (error) {
|
|
218
|
-
return (0, import_analyticsUtils.logError)({
|
|
219
|
-
action,
|
|
220
|
-
category: eventCategory,
|
|
221
|
-
params: { user },
|
|
222
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
223
|
-
}, error, context);
|
|
224
|
-
}
|
|
225
|
-
};
|
|
226
|
-
const confirmCode = async (context, { code, type }) => {
|
|
227
|
-
const { database, session: { userId: sessionId } } = context;
|
|
228
|
-
const userDocId = (0, import_arangodbUtils.getDocId)("users", { userId: sessionId });
|
|
229
|
-
const aqlQuery = import_arangojs.aql`LET u = DOCUMENT(${userDocId}) RETURN u`;
|
|
230
|
-
try {
|
|
231
|
-
return await database.query(aqlQuery).then((cursor) => cursor.next()).then(({ verifiedEmailCode, verifiedPhoneCode }) => {
|
|
232
|
-
switch (type) {
|
|
233
|
-
case "email":
|
|
234
|
-
return code === verifiedEmailCode;
|
|
235
|
-
case "phone":
|
|
236
|
-
return code === verifiedPhoneCode;
|
|
237
|
-
default:
|
|
238
|
-
return false;
|
|
239
|
-
}
|
|
240
|
-
});
|
|
241
|
-
} catch (error) {
|
|
242
|
-
return false;
|
|
243
|
-
}
|
|
244
|
-
};
|
|
245
|
-
const deleteUser = (context, user) => {
|
|
246
|
-
const action = "deleteUser";
|
|
247
|
-
const { database } = context;
|
|
248
|
-
const { userId } = (0, import_userAdapter.parseUser)(user);
|
|
249
|
-
const aqlQuery = import_arangojs.aql`FOR u IN users
|
|
250
|
-
FILTER u._key == ${userId}
|
|
9
|
+
RETURN it`;return p.query(R).then(A=>A.next()).then(A=>{if(A)return A;const N={_from:y,_key:v(`isTagged-${y}-${c}`),_to:c,added:Date.now(),name:f};return T.save(N,{returnNew:!0}).then(()=>N)})})),m}catch(p){return I({action:s,category:E,params:{user:o},value:d.DATABASE_ERROR},p,e)}},he=async(e,{email:o,phone:s,username:i})=>{const n="forgotPassword",{databaseName:t}=e,u=U`FOR u IN users
|
|
10
|
+
FILTER u.email == ${o} || u.phone == ${s} || u.username == ${i}
|
|
251
11
|
LIMIT 1
|
|
12
|
+
RETURN u`;try{return await g(t).query(u).then(async r=>{const a=r.next();if(a){const{email:c,phone:l,verifiedEmail:p,verifiedPhone:m}=a,T=1e3*60*15,y=Math.floor(1e5+Math.random()*9e5),f=j("users",a);let R;if(c&&p&&(Z({context:e,text:`Your code is ${y}`}),R={verifiedEmailCode:y,verifiedEmailExpires:T}),l&&m&&(ee({context:e,text:`Your code is ${y}`}),R={verifiedSmsCode:y,verifiedPhoneExpires:T}),R.verifiedEmailCode||R.verifiedSmsCode){const A=U`UPDATE ${f} WITH ${R} IN users`;return await g(t).query(A),!0}return!1}return!1})}catch(r){return I({action:n,category:E,params:{email:o,phone:s,username:i},value:d.DATABASE_ERROR},r,e),!1}},Oe=async(e,{code:o,password:s,type:i,username:n})=>{const t="resetPassword",{databaseName:u}=e,r=L(s),a=U`FOR u IN users
|
|
13
|
+
FILTER u.username == ${n}
|
|
14
|
+
LIMIT 1
|
|
15
|
+
RETURN u`;try{return await g(u).query(a).then(async c=>{const l=c.next();if(l){const{_id:p,salt:m,verifiedEmailCode:T,verifiedEmailExpires:y,verifiedSmsCode:f,verifiedPhoneExpires:R}=l,A=Date.now();let N;switch(i){case"email":o===T&&y>A&&(N={password:P(r,m)});break;case"phone":o===f&&R>A&&(N={password:P(r,m)});break;default:return!1}if(N){const h=U`UPDATE ${p} WITH ${N} IN users`;return await g(u).query(h),!0}}return!1})}catch(c){return I({action:t,category:E,params:{username:n},value:d.DATABASE_ERROR},c,e),!1}},$e=async(e,{code:o,type:s})=>{const i="confirmCode",{databaseName:n,session:{userId:t}}=e,u=j("users",{userId:t}),r=U`LET u = DOCUMENT(${u}) RETURN u`;try{return await g(n).query(r).then(a=>a.next()).then(({verifiedEmailCode:a,verifiedSmsCode:c})=>{switch(s){case"email":return o===a;case"phone":return o===c;default:return!1}})}catch(a){return I({action:i,category:E,params:{code:o,type:s},value:d.DATABASE_ERROR},a,e),!1}},Ce=(e,o)=>{const s="deleteUser",{databaseName:i}=e,{id:n}=D(o),t=U`LET u = DOCUMENT(${n})
|
|
252
16
|
REMOVE u IN users
|
|
253
|
-
RETURN OLD`;
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
260
|
-
}, error, context));
|
|
261
|
-
};
|
|
262
|
-
const deactivateUser = (context, user) => {
|
|
263
|
-
const action = "delete";
|
|
264
|
-
const { database } = context;
|
|
265
|
-
const { userId } = (0, import_userAdapter.parseUser)(user);
|
|
266
|
-
const updated = {
|
|
267
|
-
userAccess: 0
|
|
268
|
-
};
|
|
269
|
-
const aqlQuery = import_arangojs.aql`UPDATE ${userId} WITH ${updated} IN users LIMIT 1 RETURN NEW`;
|
|
270
|
-
return database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
|
|
271
|
-
action,
|
|
272
|
-
category: eventCategory,
|
|
273
|
-
params: { userId },
|
|
274
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
275
|
-
}, error, context));
|
|
276
|
-
};
|
|
277
|
-
const getDisplayName = (user) => {
|
|
278
|
-
const { first, last, name = "", username = "" } = user;
|
|
279
|
-
const fullname = [first, last].join(" ").trim();
|
|
280
|
-
if (name) {
|
|
281
|
-
return name;
|
|
282
|
-
} else if (fullname !== "") {
|
|
283
|
-
return fullname;
|
|
284
|
-
} else if (username) {
|
|
285
|
-
return username;
|
|
286
|
-
}
|
|
287
|
-
return "Unknown";
|
|
288
|
-
};
|
|
289
|
-
const getSessionUser = (context) => {
|
|
290
|
-
const action = "getSessionUser";
|
|
291
|
-
const { database, fields, session: { userId: sessionId, username } } = context;
|
|
292
|
-
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
293
|
-
const aqlQuery = `LET u = DOCUMENT("users/${sessionId}")
|
|
294
|
-
${selectQueries.join("\n")}
|
|
295
|
-
RETURN MERGE(u, {${selectObjects.join(", ")}})`;
|
|
296
|
-
return database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
|
|
297
|
-
action,
|
|
298
|
-
category: eventCategory,
|
|
299
|
-
params: { username, userId: sessionId },
|
|
300
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
301
|
-
}, error, context));
|
|
302
|
-
};
|
|
303
|
-
const getUser = (context, user) => {
|
|
304
|
-
const action = "getUser";
|
|
305
|
-
const { userId } = (0, import_userAdapter.parseUser)(user);
|
|
306
|
-
const { database, fields } = context;
|
|
307
|
-
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
308
|
-
const aqlQuery = `LET u = DOCUMENT("users/${userId}")
|
|
309
|
-
${selectQueries.join("\n")}
|
|
17
|
+
RETURN OLD`,u=new z(K.get("stripe.token"),{apiVersion:oe,typescript:!0});return g(i).query(t).then(r=>r.next()).then(r=>u.customers.del(r?.stripeCustomerId).then(()=>u.accounts.del(r?.stripeAccountId)).then(()=>r)).catch(r=>I({action:s,category:E,params:{id:n},value:d.DATABASE_ERROR},r,e))},Se=(e,o)=>{const s="delete",{databaseName:i}=e,{id:n}=D(o),u=U`UPDATE ${n} WITH ${{userAccess:0}} IN users LIMIT 1 RETURN NEW`;return g(i).query(u).then(r=>r.next()).catch(r=>I({action:s,category:E,params:{id:n},value:d.DATABASE_ERROR},r,e))},qe=e=>{const{first:o,last:s,name:i="",username:n=""}=e,t=[o,s].join(" ").trim();return i||(t!==""?t:n||"Unknown")},De=e=>{const o="getSessionUser",{databaseName:s,fields:i,session:{userId:n,username:t}}=e,{objects:u,queries:r}=$(i),c=`LET u = DOCUMENT("${M(`users/${n}`)}")
|
|
18
|
+
${r.join(`
|
|
19
|
+
`)}
|
|
20
|
+
RETURN MERGE(u, {${u.join(", ")}})`;return g(s).query(c).then(l=>l.next()).catch(l=>(I({action:o,category:E,params:{userId:n,username:t},value:d.DATABASE_ERROR},l,e),null))},be=(e,o)=>{const s="getUser",{email:i,id:n,phone:t,userId:u,username:r}=D(o),{databaseName:a,fields:c}=e,{objects:l,queries:p}=$(c);let m;return n?m=`LET u = DOCUMENT("${n}")
|
|
21
|
+
${p.join(`
|
|
22
|
+
`)}
|
|
310
23
|
FILTER u.userAccess > 0
|
|
311
|
-
RETURN MERGE(u, {${
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
}
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
const aqlQuery = `FOR u IN users
|
|
329
|
-
FILTER ${filterBy.join(" && ")}
|
|
330
|
-
${selectQueries.join("\n")}
|
|
331
|
-
${limit.aql}
|
|
24
|
+
RETURN MERGE(u, {${l.join(", ")}})`:r?m=`FOR u IN users
|
|
25
|
+
FILTER u.username == "${r}"
|
|
26
|
+
${p.join(`
|
|
27
|
+
`)}
|
|
28
|
+
RETURN MERGE(u, {${l.join(", ")}})`:i?m=`FOR u IN users
|
|
29
|
+
FILTER u.email == "${i}"
|
|
30
|
+
${p.join(`
|
|
31
|
+
`)}
|
|
32
|
+
RETURN MERGE(u, {${l.join(", ")}})`:t&&(m=`FOR u IN users
|
|
33
|
+
FILTER u.phone == "${t}"
|
|
34
|
+
${p.join(`
|
|
35
|
+
`)}
|
|
36
|
+
RETURN MERGE(u, {${l.join(", ")}})`),g(a).query(m).then(T=>T.next()).then(T=>T).catch(T=>I({action:s,category:E,params:{id:n,userId:u,username:r},value:d.DATABASE_ERROR},T,e))},we=(e,o)=>{const s="getUserList",{databaseName:i,fields:n}=e,{limit:t,username:u}=b(o),{objects:r,queries:a}=$(n),c=["u.userAccess > 0"];u&&c.push(`CONTAINS(u.username, "${S(u)}")`);const l=`FOR u IN users
|
|
37
|
+
FILTER ${c.join(" && ")}
|
|
38
|
+
${a.join(`
|
|
39
|
+
`)}
|
|
40
|
+
${t.aql}
|
|
332
41
|
SORT u.username
|
|
333
|
-
RETURN MERGE(u, {${
|
|
334
|
-
return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => (0, import_analyticsUtils.logError)({
|
|
335
|
-
action,
|
|
336
|
-
category: eventCategory,
|
|
337
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
338
|
-
}, error, context));
|
|
339
|
-
};
|
|
340
|
-
const getUsersByReactions = (context, { reactions = [], username }, options) => {
|
|
341
|
-
const action = "getUsersByReactions";
|
|
342
|
-
const { database, fields, session: { userId: sessionId } } = context;
|
|
343
|
-
const formatReactions = reactions.map((reactionName) => (0, import_utils.parseChar)(reactionName, 32).toLowerCase());
|
|
344
|
-
const { limit } = parseUserOptions(options);
|
|
345
|
-
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
346
|
-
const formatSessionId = `users/${sessionId}`;
|
|
347
|
-
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
348
|
-
const filterBy = [`POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`];
|
|
349
|
-
if (username) {
|
|
350
|
-
filterBy.push(`CONTAINS(u.username, "${formatUsername}")`);
|
|
351
|
-
}
|
|
352
|
-
const aqlQuery = `FOR u, r IN OUTBOUND "${formatSessionId}" hasReaction
|
|
42
|
+
RETURN MERGE(u, {${r.join(", ")}})`;return g(i).query(l).then(p=>p.all()).catch(p=>(I({action:s,category:E,value:d.DATABASE_ERROR},p,e),[]))},Pe=(e,{reactions:o=[],username:s},i)=>{const n="getUsersByReactions",{databaseName:t,fields:u,session:{userId:r}}=e,a=o.map(R=>Q(R,32).toLowerCase()),{limit:c}=b(i),{objects:l,queries:p}=$(u),m=`users/${r}`,T=S(s),y=["u.userAccess > 0",`POSITION(${JSON.stringify(a)}, LOWER(r.name))`];s&&y.push(`CONTAINS(u.username, "${T}")`);const f=`FOR u, r IN OUTBOUND "${m}" hasReaction
|
|
353
43
|
OPTIONS {vertexCollections: "users"}
|
|
354
|
-
${
|
|
355
|
-
|
|
356
|
-
${
|
|
357
|
-
|
|
358
|
-
|
|
359
|
-
|
|
360
|
-
category: eventCategory,
|
|
361
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
362
|
-
}, error, context));
|
|
363
|
-
};
|
|
364
|
-
const getUsersByTags = (context, { tags, username }, options) => {
|
|
365
|
-
const action = "getUsersByTags";
|
|
366
|
-
const { database, fields, session: { userId: sessionId } } = context;
|
|
367
|
-
const formatTags = tags?.reduce((list, tagName) => {
|
|
368
|
-
if (tagName) {
|
|
369
|
-
list.push((0, import_utils.parseChar)(tagName, 32).toLowerCase());
|
|
370
|
-
}
|
|
371
|
-
return list;
|
|
372
|
-
}, []);
|
|
373
|
-
const { limit } = parseUserOptions(options);
|
|
374
|
-
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
375
|
-
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
376
|
-
const filterBy = [`u._key != "${sessionId}"`];
|
|
377
|
-
if (username) {
|
|
378
|
-
filterBy.push(`CONTAINS(u.username, "${formatUsername}")`);
|
|
379
|
-
}
|
|
380
|
-
const aqlQuery = `FOR t IN tags
|
|
381
|
-
FILTER POSITION(${JSON.stringify(formatTags)}, LOWER(t.name))
|
|
44
|
+
${p.join(`
|
|
45
|
+
`)}
|
|
46
|
+
FILTER ${y.join(" && ")}
|
|
47
|
+
${c.aql}
|
|
48
|
+
RETURN MERGE(u, {${l.join(", ")}})`;return g(t).query(f).then(R=>R.all()).catch(R=>(I({action:n,category:E,value:d.DATABASE_ERROR},R,e),[]))},xe=(e,{tags:o,username:s},i)=>{const n="getUsersByTags",{databaseName:t,fields:u,session:{userId:r}}=e,a=o?.reduce((f,R)=>(R&&f.push(Q(R,32).toLowerCase()),f),[]),{limit:c}=b(i),{objects:l,queries:p}=$(u),m=S(s),T=[`u._key != "${r}"`,"u.userAccess > 0"];s&&T.push(`CONTAINS(u.username, "${m}")`);const y=`FOR t IN tags
|
|
49
|
+
FILTER POSITION(${JSON.stringify(a)}, LOWER(t.name))
|
|
382
50
|
FOR u, it IN OUTBOUND t isTagged
|
|
383
51
|
OPTIONS {bfs: true, uniqueVertices: "global", vertexCollections: "users"}
|
|
384
|
-
${
|
|
385
|
-
|
|
386
|
-
${
|
|
387
|
-
|
|
388
|
-
|
|
389
|
-
|
|
390
|
-
|
|
391
|
-
|
|
392
|
-
|
|
393
|
-
};
|
|
394
|
-
const getUsersByLatest = (context, { username }, options) => {
|
|
395
|
-
const action = "getUsersByLatest";
|
|
396
|
-
const { database, fields, session: { userId } } = context;
|
|
397
|
-
const { limit } = parseUserOptions(options);
|
|
398
|
-
const filter = ["u._id != session._id"];
|
|
399
|
-
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
400
|
-
if (username) {
|
|
401
|
-
filter.push(`CONTAINS(u.username, "${(0, import_utils.parseUsername)(username)}")`);
|
|
402
|
-
}
|
|
403
|
-
const aqlQuery = `FOR u IN users
|
|
404
|
-
LET session = DOCUMENT("users/${userId}")
|
|
405
|
-
FILTER ${filter.join(" && ")}
|
|
406
|
-
${selectQueries.join("\n")}
|
|
52
|
+
${p.join(`
|
|
53
|
+
`)}
|
|
54
|
+
FILTER ${T.join(" && ")}
|
|
55
|
+
${c.aql}
|
|
56
|
+
RETURN DISTINCT MERGE(u, {${l.join(", ")}})`;return g(t).query(y).then(f=>f.all()).catch(f=>(I({action:n,category:E,value:d.DATABASE_ERROR},f,e),[]))},ve=(e,{username:o},s)=>{const i="getUsersByLatest",{databaseName:n,fields:t,session:{userId:u}}=e,{limit:r}=b(s),a=["u._id != session._id","u.userAccess > 0"],{objects:c,queries:l}=$(t);o&&a.push(`CONTAINS(u.username, "${S(o)}")`);const p=`FOR u IN users
|
|
57
|
+
LET session = DOCUMENT("users/${u}")
|
|
58
|
+
FILTER ${a.join(" && ")}
|
|
59
|
+
${l.join(`
|
|
60
|
+
`)}
|
|
407
61
|
LET distance = DISTANCE(u.latitude || 0, u.longitude || 0, session.latitude || 0, session.longitude || 0)
|
|
408
|
-
${
|
|
62
|
+
${r.aql}
|
|
409
63
|
SORT distance ASC, u.added DESC
|
|
410
|
-
RETURN MERGE(u, {${
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
414
|
-
|
|
415
|
-
|
|
416
|
-
}
|
|
417
|
-
|
|
418
|
-
|
|
419
|
-
|
|
420
|
-
};
|
|
421
|
-
const signIn = async (context, args) => {
|
|
422
|
-
const action = "signIn";
|
|
423
|
-
const { database } = context;
|
|
424
|
-
const { email, expires, password, username } = args;
|
|
425
|
-
const formatEmail = (0, import_utils.parseEmail)(email);
|
|
426
|
-
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
427
|
-
const formatPassword = (0, import_utils.parsePassword)(password);
|
|
428
|
-
const formatExpires = (0, import_utils.parseNum)(expires) || 15;
|
|
429
|
-
if (!formatUsername && !formatEmail || !formatPassword) {
|
|
430
|
-
return (0, import_analyticsUtils.logException)({
|
|
431
|
-
action,
|
|
432
|
-
category: eventCategory,
|
|
433
|
-
params: { username },
|
|
434
|
-
value: import_error.ErrorTypes.INVALID_ARGUMENTS
|
|
435
|
-
}, context);
|
|
436
|
-
}
|
|
437
|
-
const filters = [];
|
|
438
|
-
if (formatEmail) {
|
|
439
|
-
filters.push(`u.email == "${formatEmail}"`);
|
|
440
|
-
} else if (formatUsername) {
|
|
441
|
-
filters.push(`u.username == "${formatUsername}"`);
|
|
442
|
-
}
|
|
443
|
-
const checkQuery = `FOR u IN users
|
|
444
|
-
FILTER ${filters.join(" || ")}
|
|
64
|
+
RETURN MERGE(u, {${c.join(", ")}})`;return g(n).query(p).then(m=>m.all()).catch(m=>(I({action:i,category:E,value:d.DATABASE_ERROR},m,e),[]))},Le=(e,{userId:o},s)=>{const i="getUsersByConnection",{databaseName:n,fields:t}=e,{limit:u,username:r}=b(s),{objects:a,queries:c}=$(t),l=M(`users/${o}`),p=["u.userAccess > 0"];r&&p.push(`CONTAINS(u.username, "${S(r)}")`);const m=`FOR cu IN users
|
|
65
|
+
LET session = DOCUMENT("${l}")
|
|
66
|
+
FOR u, connection IN OUTBOUND cu hasConnection
|
|
67
|
+
OPTIONS {bfs: true, uniqueVertices: "global", vertexCollections: "users"}
|
|
68
|
+
${c.join(`
|
|
69
|
+
`)}
|
|
70
|
+
FILTER ${p.join(" && ")}
|
|
71
|
+
${u.aql}
|
|
72
|
+
RETURN DISTINCT MERGE(u, {${a.join(", ")}})`;return g(n).query(m).then(T=>T.all()).catch(T=>(I({action:i,category:E,value:d.DATABASE_ERROR},T,e),[]))},je=({expires:e,token:o})=>{try{const{userId:s,username:i,userAccess:n}=_(o);return B(s,i,n,e)}catch(s){throw s}},Me=async(e,o)=>{const s="signIn",{databaseName:i}=e,{email:n,expires:t,password:u,phone:r,username:a}=o,c=H(n),l=S(a),p=L(u),m=Y(r),T=J(t)||15;if(!l&&!c&&!m||!p)return q({action:s,category:E,params:{username:a},value:d.INVALID_ARGUMENTS},e),null;const y=[];c&&y.push(`u.email == "${c}"`),m&&y.push(`u.phone == ${m}`),l&&y.push(`u.username == "${l}"`);const f=`FOR u IN users
|
|
73
|
+
FILTER ${y.join(" || ")}
|
|
445
74
|
LIMIT 1
|
|
446
|
-
RETURN u`;
|
|
447
|
-
|
|
448
|
-
try {
|
|
449
|
-
checkUser = await database.query(checkQuery).then((cursor) => cursor.next());
|
|
450
|
-
} catch (error) {
|
|
451
|
-
return (0, import_analyticsUtils.logError)({
|
|
452
|
-
action,
|
|
453
|
-
category: eventCategory,
|
|
454
|
-
params: { username: formatUsername },
|
|
455
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
456
|
-
}, error, context);
|
|
457
|
-
}
|
|
458
|
-
if (!checkUser) {
|
|
459
|
-
return (0, import_analyticsUtils.logException)({
|
|
460
|
-
action,
|
|
461
|
-
category: eventCategory,
|
|
462
|
-
params: { username },
|
|
463
|
-
value: import_error.ErrorTypes.INVALID_AUTHENTICATION
|
|
464
|
-
}, context);
|
|
465
|
-
}
|
|
466
|
-
const { _key: userId, password: validPassword, salt, userAccess } = checkUser;
|
|
467
|
-
const authPassword = (0, import_utils.createPassword)(formatPassword, salt);
|
|
468
|
-
if (validPassword !== authPassword) {
|
|
469
|
-
return (0, import_analyticsUtils.logException)({
|
|
470
|
-
action,
|
|
471
|
-
category: eventCategory,
|
|
472
|
-
params: { userAccess, userId, username },
|
|
473
|
-
value: import_error.ErrorTypes.INVALID_AUTHENTICATION
|
|
474
|
-
}, context);
|
|
475
|
-
}
|
|
476
|
-
return createToken(userId, username, userAccess, formatExpires);
|
|
477
|
-
};
|
|
478
|
-
const signOut = async (context) => {
|
|
479
|
-
const action = "signOut";
|
|
480
|
-
const { database, session: { userId: sessionId, username } } = context;
|
|
481
|
-
const userDocId = `users/${sessionId}`;
|
|
482
|
-
const update = {
|
|
483
|
-
lastOnline: Date.now(),
|
|
484
|
-
sessionId: null
|
|
485
|
-
};
|
|
486
|
-
const sessionQuery = import_arangojs.aql`LET u = DOCUMENT(${userDocId})
|
|
487
|
-
UPDATE u WITH ${update} IN users
|
|
75
|
+
RETURN u`;let R;try{R=await g(i).query(f).then(O=>O.next())}catch(O){return I({action:s,category:E,params:{username:l},value:d.DATABASE_ERROR},O,e),null}if(!R)return q({action:s,category:E,params:{username:a},value:d.INVALID_AUTHENTICATION},e),null;const{_key:A,password:N,salt:h,userAccess:w}=R,x=P(p,h);if(N!==x)return q({action:s,category:E,params:{userAccess:w,userId:A,username:a},value:d.INVALID_AUTHENTICATION},e),null;try{return B(A,a,w,T)}catch(O){return I({action:s,category:E,params:{userId:A,username:a},value:d.DATABASE_ERROR},O,e),null}},Qe=async e=>{const o="signOut",{databaseName:s,session:{userId:i,username:n}}=e,t=`users/${i}`,u={lastOnline:Date.now(),sessionId:null},r=U`LET u = DOCUMENT(${t})
|
|
76
|
+
UPDATE u WITH ${u} IN users
|
|
488
77
|
LIMIT 1
|
|
489
|
-
RETURN NEW`;
|
|
490
|
-
try {
|
|
491
|
-
await database.query(sessionQuery).then((cursor) => cursor.next());
|
|
492
|
-
} catch (error) {
|
|
493
|
-
await (0, import_analyticsUtils.logError)({
|
|
494
|
-
action,
|
|
495
|
-
category: eventCategory,
|
|
496
|
-
params: { username, userId: sessionId },
|
|
497
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
498
|
-
}, error, context);
|
|
499
|
-
}
|
|
500
|
-
return true;
|
|
501
|
-
};
|
|
502
|
-
const getActiveUserCount = (context) => {
|
|
503
|
-
const action = "getActiveUserCount";
|
|
504
|
-
const { database } = context;
|
|
505
|
-
const countQuery = import_arangojs.aql`LET docs = (
|
|
78
|
+
RETURN NEW`;try{await g(s).query(r).then(a=>a.next())}catch(a){await I({action:o,category:E,params:{userId:i,username:n},value:d.DATABASE_ERROR},a,e)}return!0},_e=e=>{const o="getActiveUserCount",{databaseName:s}=e,i=U`LET docs = (
|
|
506
79
|
FOR u IN users
|
|
507
80
|
FILTER u.active == true
|
|
508
81
|
RETURN u
|
|
509
82
|
)
|
|
510
|
-
RETURN LENGTH(docs)`;
|
|
511
|
-
return database.query(countQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
|
|
512
|
-
action,
|
|
513
|
-
category: eventCategory,
|
|
514
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
515
|
-
}, error, context));
|
|
516
|
-
};
|
|
517
|
-
const getUserByToken = (context, token) => {
|
|
518
|
-
const action = "getUserByToken";
|
|
519
|
-
const { database } = context;
|
|
520
|
-
const { userId } = (0, import_session.getSession)(token);
|
|
521
|
-
const userDocId = (0, import_arangodbUtils.getDocId)("users", { userId });
|
|
522
|
-
const aqlQuery = import_arangojs.aql`LET u = DOCUMENT("${userDocId}") RETURN u`;
|
|
523
|
-
return database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
|
|
524
|
-
action,
|
|
525
|
-
category: eventCategory,
|
|
526
|
-
params: { userId },
|
|
527
|
-
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
528
|
-
}, error, context));
|
|
529
|
-
};
|
|
530
|
-
// Annotate the CommonJS export names for ESM import in node:
|
|
531
|
-
0 && (module.exports = {
|
|
532
|
-
UserAccess,
|
|
533
|
-
addUser,
|
|
534
|
-
confirmCode,
|
|
535
|
-
createToken,
|
|
536
|
-
deactivateUser,
|
|
537
|
-
deleteUser,
|
|
538
|
-
getActiveUserCount,
|
|
539
|
-
getDisplayName,
|
|
540
|
-
getSessionUser,
|
|
541
|
-
getUser,
|
|
542
|
-
getUserByToken,
|
|
543
|
-
getUserOptional,
|
|
544
|
-
getUsers,
|
|
545
|
-
getUsersByLatest,
|
|
546
|
-
getUsersByReactions,
|
|
547
|
-
getUsersByTags,
|
|
548
|
-
parseUserOptions,
|
|
549
|
-
refreshSession,
|
|
550
|
-
signIn,
|
|
551
|
-
signOut,
|
|
552
|
-
updateUser
|
|
553
|
-
});
|
|
554
|
-
//# 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 type {ApiContext} from '../types/auth';\nimport type {UserType} from '../types/users';\nimport {logError, logException} from '../utils/analyticsUtils';\nimport {getDocId, getLimit, selectReactionCountByType} from '../utils/arangodbUtils';\nimport {getSession, isAdminUser, SessionToken, setSession} from '../utils/session';\nimport {ErrorTypes, SessionError} from '../types/error';\nimport {parseUser} from '../adapters/userAdapter';\n\nconst eventCategory: string = 'users';\nconst STRIPE_API_VERSION: 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  expiresInMinutes: number = 15\n): SessionToken => {\n  const now: DateTime = DateTime.local();\n  const sessionExpires: DateTime = now.plus({minutes: expiresInMinutes});\n  const iat: number = Math.floor(now.toSeconds());\n  const exp: number = Math.floor(sessionExpires.toSeconds());\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  const limit = getLimit(from, to);\n\n  return {\n    ...options,\n    limit\n  };\n};\n\nexport const addUser = async (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'addUser';\n  const {database} = context;\n  const {email, password, phone, username} = parseUser(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      category: eventCategory,\n      params: {username},\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    LIMIT 1\n    RETURN u`;\n\n  try {\n    const existingUsers = await database.query(checkQuery).then((cursor: ArrayCursor) => cursor.all());\n\n    if(existingUsers.length) {\n      return logException({\n        action,\n        category: eventCategory,\n        params: {username},\n        value: ErrorTypes.EXISTING_ITEM\n      }, context);\n    }\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, 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  const insert: UserType = {\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  const insertQuery: AqlQuery = aql`INSERT ${insert} IN users RETURN NEW`;\n\n  return await database.query(insertQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const updateUser = async (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'updateUser';\n  const {database, session} = context;\n  const {_key, _id, id, tags = [], userId, ...updated} = parseUser(user);\n\n  if(!isAdminUser(session) && (session?.userId !== userId)) {\n    return logException({\n      action,\n      category: eventCategory,\n      params: {session},\n      value: ErrorTypes.INVALID_SESSION\n    }, context);\n  }\n\n  const userQuery: AqlQuery = aql`LET u = DOCUMENT(${id})\n    UPDATE u WITH ${updated} IN users\n    RETURN NEW`;\n\n  try {\n    const updatedUser = await database.query(userQuery).then((cursor) => cursor.next());\n    const tagCollection: EdgeCollection = database.collection('isTagged');\n\n    await Promise.all(tags.map(({id: tagDocId, name}) => {\n      const tagQuery: AqlQuery = aql`FOR it IN isTagged\n        FILTER it._from == ${tagDocId} && it._to == ${id} && 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          const edge = {\n            _from: tagDocId,\n            _key: createHash(`isTagged-${tagDocId}-${id}`),\n            _to: id,\n            added: Date.now(),\n            name\n          };\n\n          return tagCollection.save(edge, {returnNew: true}).then(() => edge);\n        });\n    }));\n\n    return updatedUser;\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      params: {user},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n};\n\nexport const confirmCode = async (context: ApiContext, {code, type}): Promise<boolean> => {\n  const {database, session: {userId: sessionId}} = context;\n  const userDocId = getDocId('users', {userId: sessionId});\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId}) RETURN u`;\n\n  try {\n    return await database.query(aqlQuery)\n      .then((cursor) => cursor.next())\n      .then(({verifiedEmailCode, verifiedPhoneCode}: UserType) => {\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) {\n    return false;\n  }\n};\n\nexport const deleteUser = (context: ApiContext, user: UserType): Promise<any> => {\n  const action = 'deleteUser';\n  const {database} = context;\n  const {userId} = parseUser(user);\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  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion: STRIPE_API_VERSION, typescript: true});\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((deletedUser) => stripeClient.customers.del(deletedUser?.stripeCustomerId)\n      .then(() => stripeClient.accounts.del(deletedUser?.stripeAccountId))\n      .then(() => deletedUser))\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {userId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error,context));\n};\n\nexport const deactivateUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'delete';\n  const {database} = context;\n  const {userId} = parseUser(user);\n  const updated: UserType = {\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) => logError({\n      action,\n      category: eventCategory,\n      params: {userId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error,context));\n};\n\nexport const getDisplayName = (user: UserType): string => {\n  const {first, last, name = '', username = ''} = user;\n  const fullname = ([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<UserType> => {\n  const action = '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    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {username, userId: sessionId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'getUser';\n  const {userId} = parseUser(user);\n  const {database, fields} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const aqlQuery: string = `LET u = DOCUMENT(\"users/${userId}\")\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      category: eventCategory,\n      params: {userId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUsers = (context: ApiContext, options?: UserOptions): Promise<UserType[]> => {\n  const action = '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  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      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUsersByReactions = (\n  context: ApiContext,\n  {reactions = [], username}: any,\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = '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  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  const aqlQuery: string = `FOR u, r IN OUTBOUND \"${formatSessionId}\" hasReaction\n    OPTIONS {vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\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      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUsersByTags = (\n  context: ApiContext,\n  {tags, username}: any,\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = '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  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  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    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT 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      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUsersByLatest = (context: ApiContext, {username}: any, options?: UserOptions): Promise<UserType[]> => {\n  const action = '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      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const refreshSession = async (context: ApiContext, {expires, token}): Promise<SessionToken | SessionError> => {\n  const {userId, username, userAccess} = getSession(token);\n\n  return createToken(userId, username, userAccess, expires);\n};\n\nexport const signIn = async (context: ApiContext, args): Promise<SessionToken> => {\n  const action = 'signIn';\n  const {database} = context;\n  const {email, expires, password, username} = args;\n  const formatEmail: string = parseEmail(email);\n  const formatUsername: string = parseUsername(username);\n  const formatPassword: string = parsePassword(password);\n  const formatExpires: number = parseNum(expires) || 15;\n\n  if((!formatUsername && !formatEmail) || !formatPassword) {\n    return logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const filters: string[] = [];\n\n  if(formatEmail) {\n    filters.push(`u.email == \"${formatEmail}\"`);\n  } else if(formatUsername) {\n    filters.push(`u.username == \"${formatUsername}\"`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    LIMIT 1\n    RETURN u`;\n\n  let checkUser: UserType;\n\n  try {\n    checkUser = await database.query(checkQuery).then((cursor) => cursor.next());\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      params: {username: formatUsername},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n\n  if(!checkUser) {\n    return logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  const {_key: userId, password: validPassword, salt, userAccess} = checkUser;\n  const authPassword: string = createPassword(formatPassword, salt);\n\n  if(validPassword !== authPassword) {\n    return logException({\n      action,\n      category: eventCategory,\n      params: {userAccess, userId, username},\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  return createToken(userId, username, userAccess, formatExpires);\n};\n\nexport const signOut = async (context: ApiContext): Promise<boolean> => {\n  const action = '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).then((cursor) => cursor.next());\n  } catch(error) {\n    await logError({\n      action,\n      category: eventCategory,\n      params: {username, userId: sessionId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n\n  return true;\n};\n\nexport const getActiveUserCount = (context: ApiContext) => {\n  const action = '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      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUserByToken = (context: ApiContext, token: string): Promise<UserType> => {\n  const action = 'getUserByToken';\n  const {database} = context;\n  const {userId} = getSession(token);\n  const userDocId = getDocId('users', {userId});\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(\"${userDocId}\") RETURN u`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {userId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n"],
  "mappings": "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",
  "names": ["UserAccess", "Stripe", "user"]
}

|
|
83
|
+
RETURN LENGTH(docs)`;return g(s).query(i).then(n=>n.next()).catch(n=>I({action:o,category:E,value:d.DATABASE_ERROR},n,e))},Be=(e,o)=>{const s="getUserByToken",{databaseName:i}=e,{userId:n}=_(o),t=j("users",{userId:n}),u=U`LET u = DOCUMENT("${t}") RETURN u`;return g(i).query(u).then(r=>r.next()).catch(r=>I({action:s,category:E,params:{userId:n},value:d.DATABASE_ERROR},r,e))};export{ae as UserAccess,Ue as addUser,$e as confirmCode,B as createToken,Se as deactivateUser,Ce as deleteUser,he as forgotPassword,_e as getActiveUserCount,qe as getDisplayName,De as getSessionUser,be as getUser,Be as getUserByToken,$ as getUserOptional,we as getUsers,Le as getUsersByConnection,ve as getUsersByLatest,Pe as getUsersByReactions,xe as getUsersByTags,b as parseUserOptions,je as refreshSession,Oe as resetPassword,Me as signIn,Qe as signOut,Ne as updateUser};
|
|
84
|
+
//# 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  parseArangoId,\n  parseChar,\n  parseEmail,\n  parseNum,\n  parsePassword,\n  parsePhone,\n  parseUsername\n} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {DateTime} from 'luxon';\nimport Stripe from 'stripe';\n\nimport {parseUser} from '../adapters/userAdapter';\nimport {Config} from '../config';\nimport {sendEmail} from './email';\nimport {sendSms} from './sms';\nimport {ErrorTypes, SessionError} from '../types/error.types';\nimport {logError, logException} from '../utils/analyticsUtils';\nimport {getDocId, getLimit, selectReactionCountByType, useDb} from '../utils/arangodbUtils';\nimport {detectLanguage} from '../utils/languageDetection';\nimport {getSession, SessionToken, setSession} from '../utils/sessionUtils';\n\nimport type {ApiContext} from '../types/auth.types';\nimport type {UserInput, UserType} from '../types/users.types';\nimport type {EdgeCollection} from 'arangojs/collections';\n\n\nconst eventCategory = 'users';\nconst STRIPE_API_VERSION = '2025-05-28.basil';\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  expiresInMinutes: number = 15\n): SessionToken => {\n  const now: DateTime = DateTime.local();\n  const sessionExpires: DateTime = now.plus({minutes: expiresInMinutes});\n  const iat: number = Math.floor(now.toSeconds());\n  const exp: number = Math.floor(sessionExpires.toSeconds());\n  const token = setSession({\n    exp,\n    iat,\n    userAccess,\n    userId,\n    username\n  });\n\n  return {\n    expires: sessionExpires.toMillis(),\n    issued: now.toMillis(),\n    token,\n    userId,\n    username\n  };\n};\n\ninterface SelectAccumulator {\n  objects: string[];\n  queries: string[];\n}\n\nexport const getUserOptional = (fields: string[] = []): SelectAccumulator =>\n  fields.reduce((selects: SelectAccumulator, 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  const limit = getLimit(from, to);\n\n  return {\n    ...options,\n    limit\n  };\n};\n\nexport const addUser = async (context: ApiContext, user: UserInput): Promise<UserType> => {\n  const action = 'addUser';\n  const {databaseName, languageContext} = context;\n  const {email, password, phone, username, userId, _key, _id, ...insertUser} = parseUser(user);\n  const formatPassword: string = parsePassword(password);\n  const hasPassword = !!formatPassword;\n  const hasUsername = !!username || !!phone || !!email;\n\n  if(!hasPassword || !hasUsername) {\n    return logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const hashId = username || phone || email;\n  const salt: string = createHash(`${hashId}${formatPassword}`, null);\n  const encryptedPassword = createPassword(formatPassword, salt);\n  const filters: string[] = [];\n\n  if(username) {\n    filters.push(`u.username == \"${username}\"`);\n  }\n\n  if(email) {\n    filters.push(`u.email == \"${email}\"`);\n  }\n\n  if(phone) {\n    filters.push(`u.phone == ${phone}`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    const existingUsers = await useDb(databaseName).query(checkQuery).then((cursor) => cursor.all());\n\n    if(existingUsers.length) {\n      return logException({\n        action,\n        category: eventCategory,\n        params: {\n          email,\n          phone,\n          username\n        },\n        value: ErrorTypes.EXISTING_ITEM\n      }, context);\n    }\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      params: {email, phone, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n\n  const phoneCountryCode = phone?.replace(/\\D/g, '').substring(0, 3);\n  const locale = detectLanguage({\n    ...languageContext,\n    phoneCountryCode,\n    userPreference: insertUser.locale || 'en'\n  });\n  const verifiedEmailCode: number = Math.floor(100000 + (Math.random() * 900000));\n  const verifiedSmsCode: number = Math.floor(100000 + (Math.random() * 900000));\n\n  const insert: UserType = {\n    ...insertUser,\n    _key: createHash(username, null),\n    added: Date.now(),\n    email,\n    locale,\n    modified: Date.now(),\n    password: encryptedPassword,\n    phone,\n    salt,\n    userAccess: 1,\n    username,\n    verifiedEmail: false,\n    verifiedEmailCode,\n    verifiedPhone: false,\n    verifiedSmsCode\n  };\n\n  const insertQuery: AqlQuery = aql`INSERT ${insert} IN users RETURN NEW`;\n\n  return await useDb(databaseName).query(insertQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const updateUser = async (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'updateUser';\n  const {databaseName} = context;\n  const updatedUser = parseUser(user);\n  const {_key, _id, tags = [], ...update} = updatedUser;\n  const {id} = updatedUser;\n\n  const userQuery: AqlQuery = aql`LET u = DOCUMENT(${id})\n    UPDATE u WITH ${update} IN users\n    RETURN NEW`;\n  try {\n    const database = useDb(databaseName);\n    const updatedUser = await database.query(userQuery).then((cursor) => cursor.next());\n    const tagCollection: EdgeCollection = database.collection('isTagged');\n\n    await Promise.all(tags.map(({id: tagDocId, name}) => {\n      const tagQuery: AqlQuery = aql`FOR it IN isTagged\n        FILTER it._from == ${tagDocId} && it._to == ${id} && it.name == ${name}\n        LIMIT 1\n        RETURN it`;\n\n      return database.query(tagQuery)\n        .then((cursor) => cursor.next())\n        .then((tagEdge) => {\n          console.log({tagEdge});\n          if(!!tagEdge) {\n            return tagEdge;\n          }\n\n          const edge = {\n            _from: tagDocId,\n            _key: createHash(`isTagged-${tagDocId}-${id}`),\n            _to: id,\n            added: Date.now(),\n            name\n          };\n\n          return tagCollection.save(edge, {returnNew: true}).then(() => edge);\n        });\n    }));\n\n    return updatedUser;\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      params: {user},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n};\n\nexport const forgotPassword = async (context: ApiContext, {email, phone, username}): Promise<boolean> => {\n  const action = 'forgotPassword';\n  const {databaseName} = context;\n  const aqlQuery: AqlQuery = aql`FOR u IN users\n    FILTER u.email == ${email} || u.phone == ${phone} || u.username == ${username}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    return await useDb(databaseName).query(aqlQuery)\n      .then(async (cursor) => {\n        const user = cursor.next();\n\n        if(user) {\n          const {email, phone, verifiedEmail, verifiedPhone} = user as UserType;\n          const codeExpires = 1000 * 60 * 15; // 15 minutes\n          const code = Math.floor(100000 + (Math.random() * 900000));\n          const userDocId = getDocId('users', user);\n          let update;\n\n          if(email && verifiedEmail) {\n            sendEmail({\n              context,\n              text: `Your code is ${code}`\n            });\n            update = {verifiedEmailCode: code, verifiedEmailExpires: codeExpires};\n          }\n\n          if(phone && verifiedPhone) {\n            sendSms({\n              context,\n              text: `Your code is ${code}`\n            });\n            update = {verifiedSmsCode: code, verifiedPhoneExpires: codeExpires};\n          }\n\n          if(update.verifiedEmailCode || update.verifiedSmsCode) {\n            const updateQuery: AqlQuery = aql`UPDATE ${userDocId} WITH ${update} IN users`;\n\n            await useDb(databaseName).query(updateQuery);\n\n            return true;\n          }\n\n          return false;\n        }\n\n        return false;\n      });\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {email, phone, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\n  }\n};\n\nexport const resetPassword = async (\n  context: ApiContext,\n  {\n    code,\n    password,\n    type,\n    username\n  }: {\n    code: number,\n    password: string,\n    type: 'phone' | 'email',\n    username: string\n  }\n): Promise<boolean> => {\n  const action = 'resetPassword';\n  const {databaseName} = context;\n  const formatPassword: string = parsePassword(password);\n  const aqlQuery: AqlQuery = aql`FOR u IN users\n    FILTER u.username == ${username}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    return await useDb(databaseName).query(aqlQuery)\n      .then(async (cursor) => {\n        const user = cursor.next();\n\n        if(user) {\n          const {\n            _id: userDocId,\n            salt,\n            verifiedEmailCode,\n            verifiedEmailExpires,\n            verifiedSmsCode,\n            verifiedPhoneExpires\n          } = user as UserType;\n          const now = Date.now();\n          let update;\n\n          switch(type) {\n            case 'email':\n              if(code === verifiedEmailCode && verifiedEmailExpires > now) {\n                const password: string = createPassword(formatPassword, salt);\n                update = {password};\n              }\n              break;\n            case 'phone':\n              if(code === verifiedSmsCode && verifiedPhoneExpires > now) {\n                const password: string = createPassword(formatPassword, salt);\n                update = {password};\n              }\n              break;\n            default:\n              return false;\n          }\n\n          if(update) {\n            const updateQuery: AqlQuery = aql`UPDATE ${userDocId} WITH ${update} IN users`;\n\n            await useDb(databaseName).query(updateQuery);\n\n            return true;\n          }\n        }\n\n        return false;\n      });\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\n  }\n};\n\nexport const confirmCode = async (\n  context: ApiContext,\n  {\n    code,\n    type\n  }: {\n    code: number,\n    type: 'phone' | 'email'\n  }\n): Promise<boolean> => {\n  const action = 'confirmCode';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const userDocId = getDocId('users', {userId: sessionId});\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId}) RETURN u`;\n\n  try {\n    return await useDb(databaseName).query(aqlQuery)\n      .then((cursor) => cursor.next())\n      .then(({verifiedEmailCode, verifiedSmsCode}: UserType) => {\n        switch(type) {\n          case 'email':\n            return code === verifiedEmailCode;\n          case 'phone':\n            return code === verifiedSmsCode;\n          default:\n            return false;\n        }\n      });\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {code, type},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\n  }\n};\n\nexport const deleteUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'deleteUser';\n  const {databaseName} = context;\n  const {id} = parseUser(user);\n\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(${id})\n    REMOVE u IN users\n    RETURN OLD`;\n\n  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion: STRIPE_API_VERSION, typescript: true});\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .then((deletedUser) => stripeClient.customers.del(deletedUser?.stripeCustomerId)\n      .then(() => stripeClient.accounts.del(deletedUser?.stripeAccountId))\n      .then(() => deletedUser))\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {id},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error,context));\n};\n\nexport const deactivateUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'delete';\n  const {databaseName} = context;\n  const {id} = parseUser(user);\n  const updated: UserType = {\n    userAccess: 0\n  };\n  const aqlQuery: AqlQuery = aql`UPDATE ${id} WITH ${updated} IN users LIMIT 1 RETURN NEW`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {id},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error,context));\n};\n\nexport const getDisplayName = (user: UserType): string => {\n  const {first, last, name = '', username = ''} = user;\n  const fullname = ([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<UserType> => {\n  const action = 'getSessionUser';\n  console.log('getSessionUser', {action, context});\n  const {databaseName, fields, session: {userId: sessionId, username}} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const formatSessionId = parseArangoId(`users/${sessionId}`);\n\n  const aqlQuery: string = `LET u = DOCUMENT(\"${formatSessionId}\")\n  ${selectQueries.join('\\n')}\n  RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.next() as unknown as UserType)\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        params: {userId: sessionId, username},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return null;\n    });\n};\n\nexport const getUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'getUser';\n  const {email, id, phone, userId, username} = parseUser(user);\n  const {databaseName, fields} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  let aqlQuery: string;\n\n  if(id) {\n    aqlQuery = `LET u = DOCUMENT(\"${id}\")\n    ${selectQueries.join('\\n')}\n    FILTER u.userAccess > 0\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  } else if(username) {\n    aqlQuery = `FOR u IN users\n    FILTER u.username == \"${username}\"\n    ${selectQueries.join('\\n')}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  } else if(email) {\n    aqlQuery = `FOR u IN users\n    FILTER u.email == \"${email}\"\n    ${selectQueries.join('\\n')}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  } else if(phone) {\n    aqlQuery = `FOR u IN users\n    FILTER u.phone == \"${phone}\"\n    ${selectQueries.join('\\n')}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  }\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .then((user) => user)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {id, userId, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUsers = (context: ApiContext, options?: UserOptions): Promise<UserType[]> => {\n  const action = 'getUserList';\n  const {databaseName, 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  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 useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByReactions = (\n  context: ApiContext,\n  {reactions = [], username}: {reactions?: string[], username?: string},\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = 'getUsersByReactions';\n  const {databaseName, 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  const formatSessionId: string = `users/${sessionId}`;\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [\n    'u.userAccess > 0',\n    `POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`\n  ];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  const aqlQuery: string = `FOR u, r IN OUTBOUND \"${formatSessionId}\" hasReaction\n    OPTIONS {vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByTags = (\n  context: ApiContext,\n  {tags, username},\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = 'getUsersByTags';\n  const {databaseName, 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  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [\n    `u._key != \"${sessionId}\"`,\n    'u.userAccess > 0'\n  ];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\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    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByLatest = (context: ApiContext, {username}, options?: UserOptions): Promise<UserType[]> => {\n  const action = 'getUsersByLatest';\n  const {databaseName, fields, session: {userId}} = context;\n  const {limit} = parseUserOptions(options);\n  const filter = [\n    'u._id != session._id',\n    'u.userAccess > 0'\n  ];\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 useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByConnection = (\n  context: ApiContext,\n  {userId}: UserType,\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = 'getUsersByConnection';\n  const {databaseName, fields} = context;\n  const {limit, username} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const formatUserId: string = parseArangoId(`users/${userId}`);\n  const filterBy: string[] = [\n    'u.userAccess > 0'\n  ];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  const aqlQuery: string = `FOR cu IN users\n    LET session = DOCUMENT(\"${formatUserId}\")\n    FOR u, connection IN OUTBOUND cu hasConnection\n    OPTIONS {bfs: true, uniqueVertices: \"global\", vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const refreshSession = ({expires, token}): SessionToken | SessionError => {\n  try {\n    const {userId, username, userAccess} = getSession(token);\n    return createToken(userId, username, userAccess, expires);\n  } catch(error) {\n    throw error; // Re-throw the error from getSession\n  }\n};\n\nexport const signIn = async (context: ApiContext, args): Promise<SessionToken> => {\n  const action = 'signIn';\n  const {databaseName} = context;\n  const {email, expires, password, phone, username} = args;\n  const formatEmail: string = parseEmail(email);\n  const formatUsername: string = parseUsername(username);\n  const formatPassword: string = parsePassword(password);\n  const formatPhone: string = parsePhone(phone);\n  const formatExpires: number = parseNum(expires) || 15;\n\n  if((!formatUsername && !formatEmail && !formatPhone) || !formatPassword) {\n    logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n    return null;\n  }\n\n  const filters: string[] = [];\n\n  if(formatEmail) {\n    filters.push(`u.email == \"${formatEmail}\"`);\n  }\n\n  if(formatPhone) {\n    filters.push(`u.phone == ${formatPhone}`);\n  }\n\n  if(formatUsername) {\n    filters.push(`u.username == \"${formatUsername}\"`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    LIMIT 1\n    RETURN u`;\n\n  let checkUser: UserType;\n\n  try {\n    checkUser = await useDb(databaseName).query(checkQuery).then((cursor) => cursor.next());\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {username: formatUsername},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return null;\n  }\n\n  if(!checkUser) {\n    logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n\n    return null;\n  }\n\n  const {_key: userId, password: validPassword, salt, userAccess} = checkUser;\n  const authPassword: string = createPassword(formatPassword, salt);\n\n  if(validPassword !== authPassword) {\n    logException({\n      action,\n      category: eventCategory,\n      params: {userAccess, userId, username},\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n\n    return null;\n  }\n\n  try {\n    console.log({formatExpires, userAccess, userId, username});\n    const token = createToken(userId, username, userAccess, formatExpires);\n    console.log({token});\n\n    return token;\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {userId, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return null;\n  }\n};\n\nexport const signOut = async (context: ApiContext): Promise<boolean> => {\n  const action = 'signOut';\n  const {databaseName, 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 useDb(databaseName).query(sessionQuery).then((cursor) => cursor.next());\n  } catch(error) {\n    await logError({\n      action,\n      category: eventCategory,\n      params: {userId: sessionId, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n\n  return true;\n};\n\nexport const getActiveUserCount = (context: ApiContext): Promise<number> => {\n  const action = 'getActiveUserCount';\n  const {databaseName} = 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 useDb(databaseName).query(countQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUserByToken = (context: ApiContext, token: string): Promise<UserType> => {\n  const action = 'getUserByToken';\n  const {databaseName} = context;\n  const {userId} = getSession(token);\n  const userDocId = getDocId('users', {userId});\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(\"${userDocId}\") RETURN u`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {userId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};"],
  "mappings": "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",
  "names": ["createHash", "createPassword", "parseArangoId", "parseChar", "parseEmail", "parseNum", "parsePassword", "parsePhone", "parseUsername", "aql", "DateTime", "Stripe", "parseUser", "Config", "sendEmail", "sendSms", "ErrorTypes", "logError", "logException", "getDocId", "getLimit", "selectReactionCountByType", "useDb", "detectLanguage", "getSession", "setSession", "eventCategory", "STRIPE_API_VERSION", "UserAccess", "createToken", "userId", "username", "userAccess", "expiresInMinutes", "now", "sessionExpires", "iat", "exp", "token", "getUserOptional", "fields", "selects", "field", "parseUserOptions", "options", "from", "to", "limit", "addUser", "context", "user", "action", "databaseName", "languageContext", "email", "password", "phone", "_key", "_id", "insertUser", "formatPassword", "salt", "encryptedPassword", "filters", "checkQuery", "cursor", "error", "phoneCountryCode", "locale", "verifiedEmailCode", "verifiedSmsCode", "insert", "insertQuery", "updateUser", "updatedUser", "tags", "update", "id", "userQuery", "database", "tagCollection", "tagDocId", "name", "tagQuery", "tagEdge", "edge", "forgotPassword", "aqlQuery", "verifiedEmail", "verifiedPhone", "codeExpires", "code", "userDocId", "updateQuery", "resetPassword", "type", "verifiedEmailExpires", "verifiedPhoneExpires", "confirmCode", "sessionId", "deleteUser", "stripeClient", "deletedUser", "deactivateUser", "getDisplayName", "first", "last", "fullname", "getSessionUser", "selectObjects", "selectQueries", "getUser", "getUsers", "filterBy", "getUsersByReactions", "reactions", "formatReactions", "reactionName", "formatSessionId", "formatUsername", "getUsersByTags", "formatTags", "list", "tagName", "getUsersByLatest", "filter", "getUsersByConnection", "formatUserId", "refreshSession", "expires", "signIn", "args", "formatEmail", "formatPhone", "formatExpires", "checkUser", "validPassword", "authPassword", "signOut", "sessionQuery", "getActiveUserCount", "countQuery", "getUserByToken"]
}

|