@nlabs/reaktor 0.8.1 → 0.10.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.prettierrc.js +4 -0
- package/README.md +10 -1
- package/coverage/actions/groups.ts.html +1039 -0
- package/coverage/actions/images.ts.html +2500 -0
- package/coverage/actions/index.html +131 -0
- package/coverage/actions/tags.ts.html +1000 -0
- package/coverage/adapters/arangoAdapter.ts.html +151 -0
- package/coverage/adapters/index.html +146 -0
- package/coverage/adapters/reaktorAdapter.ts.html +127 -0
- package/coverage/adapters/tagAdapter.ts.html +160 -0
- package/coverage/base.css +224 -0
- package/coverage/block-navigation.js +87 -0
- package/coverage/clover.xml +6 -0
- package/coverage/coverage-final.json +1 -0
- package/coverage/favicon.png +0 -0
- package/coverage/index.html +221 -0
- package/coverage/lcov-report/base.css +224 -0
- package/coverage/lcov-report/block-navigation.js +87 -0
- package/coverage/lcov-report/favicon.png +0 -0
- package/coverage/lcov-report/index.html +101 -0
- package/coverage/lcov-report/prettify.css +1 -0
- package/coverage/lcov-report/prettify.js +2 -0
- package/coverage/lcov-report/sort-arrow-sprite.png +0 -0
- package/coverage/lcov-report/sorter.js +196 -0
- package/coverage/lcov.info +0 -0
- package/coverage/mocks/file.ts.html +118 -0
- package/coverage/mocks/group.ts.html +145 -0
- package/coverage/mocks/image.ts.html +136 -0
- package/coverage/mocks/index.html +146 -0
- package/coverage/mocks/post.ts.html +169 -0
- package/coverage/mocks/tag.ts.html +121 -0
- package/coverage/mocks/user.ts.html +268 -0
- package/coverage/prettify.css +1 -0
- package/coverage/prettify.js +2 -0
- package/coverage/sort-arrow-sprite.png +0 -0
- package/coverage/sorter.js +196 -0
- package/coverage/types/error.ts.html +145 -0
- package/coverage/types/index.html +116 -0
- package/coverage/utils/adapterUtils.ts.html +151 -0
- package/coverage/utils/analyticsUtils.ts.html +292 -0
- package/coverage/utils/arangodbUtils.ts.html +463 -0
- package/coverage/utils/index.html +146 -0
- 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/dist/types/arangodb.js +16 -0
- package/{lib → dist}/types/auth.js +1 -1
- package/{lib → dist}/types/connections.js +1 -1
- package/{lib → dist}/types/conversations.js +1 -1
- package/{lib → dist}/types/email.js +1 -1
- package/dist/types/error.js +44 -0
- package/{lib → dist}/types/files.js +1 -1
- 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/dist/types/messages.js +16 -0
- 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/statistics.js +1 -1
- 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/dist/utils/session.js +60 -0
- package/jest.setup.js +0 -0
- package/jpg:- +0 -0
- package/lex.config.cjs +13 -0
- package/lib/actions/apps.d.ts +25 -0
- package/lib/actions/apps.js +242 -0
- package/lib/actions/connections.d.ts +4 -0
- package/lib/actions/connections.js +90 -0
- package/lib/actions/conversations.d.ts +12 -12
- package/lib/actions/conversations.js +147 -131
- package/lib/actions/dynamodb.d.ts +8 -8
- package/lib/actions/dynamodb.js +35 -32
- package/lib/actions/email.d.ts +3 -5
- package/lib/actions/email.js +33 -63
- package/lib/actions/files.d.ts +17 -14
- package/lib/actions/files.js +184 -202
- package/lib/actions/groups.d.ts +4 -4
- package/lib/actions/groups.js +47 -45
- package/lib/actions/images.d.ts +17 -13
- package/lib/actions/images.js +325 -264
- package/lib/actions/index.d.ts +3 -0
- package/lib/actions/index.js +7 -1
- package/lib/actions/ios.js +11 -10
- package/lib/actions/locations.d.ts +5 -2
- package/lib/actions/locations.js +41 -37
- package/lib/actions/messages.d.ts +4 -3
- package/lib/actions/messages.js +35 -32
- package/lib/actions/notifications.d.ts +2 -2
- package/lib/actions/notifications.js +1 -1
- package/lib/actions/payments.d.ts +2 -2
- package/lib/actions/payments.js +87 -83
- package/lib/actions/personas.d.ts +3 -0
- package/lib/actions/personas.js +110 -0
- package/lib/actions/posts.d.ts +11 -10
- package/lib/actions/posts.js +186 -152
- package/lib/actions/reactions.d.ts +5 -5
- package/lib/actions/reactions.js +30 -28
- package/lib/actions/s3.d.ts +7 -7
- package/lib/actions/s3.js +37 -32
- package/lib/actions/search.d.ts +3 -3
- package/lib/actions/search.js +13 -11
- package/lib/actions/sms.d.ts +9 -3
- package/lib/actions/sms.js +60 -34
- package/lib/actions/statistics.d.ts +3 -2
- package/lib/actions/statistics.js +21 -18
- package/lib/actions/subscription.d.ts +2 -2
- package/lib/actions/subscription.js +32 -39
- package/lib/actions/tags.d.ts +23 -20
- package/lib/actions/tags.js +161 -205
- package/lib/actions/users.d.ts +50 -22
- package/lib/actions/users.js +441 -217
- package/lib/actions/websockets.d.ts +19 -5
- package/lib/actions/websockets.js +89 -61
- package/lib/adapters/arangoAdapter.d.ts +2 -0
- package/lib/adapters/arangoAdapter.js +46 -0
- package/lib/adapters/fileAdapter.d.ts +3 -0
- package/lib/adapters/fileAdapter.js +76 -0
- package/lib/adapters/imageAdapter.d.ts +2 -0
- package/lib/adapters/imageAdapter.js +40 -0
- package/lib/adapters/messageAdapter.d.ts +2 -0
- package/lib/adapters/messageAdapter.js +49 -0
- package/lib/adapters/postAdapter.d.ts +2 -0
- package/lib/adapters/postAdapter.js +70 -0
- package/lib/adapters/reaktorAdapter.d.ts +6 -0
- package/lib/adapters/reaktorAdapter.js +44 -0
- package/lib/adapters/tagAdapter.d.ts +2 -0
- package/lib/adapters/tagAdapter.js +50 -0
- package/lib/adapters/userAdapter.d.ts +2 -0
- package/lib/adapters/userAdapter.js +115 -0
- package/lib/config.js +16 -17
- package/lib/index.d.ts +7 -0
- package/lib/index.js +44 -8
- package/lib/lambdas/actions/websockets.d.ts +7 -6
- package/lib/lambdas/actions/websockets.js +15 -11
- package/lib/lambdas/authorizer.js +4 -4
- package/lib/lambdas/connection.js +20 -20
- package/lib/lambdas/utils/message.js +1 -1
- package/lib/lambdas/utils/websocket.js +8 -7
- package/lib/mocks/conversation.d.ts +8 -0
- package/lib/mocks/conversation.js +35 -0
- package/lib/mocks/file.d.ts +11 -0
- package/lib/mocks/file.js +38 -0
- package/lib/mocks/group.d.ts +17 -0
- package/lib/mocks/group.js +47 -0
- package/lib/mocks/image.d.ts +3 -0
- package/lib/mocks/image.js +44 -0
- package/lib/mocks/nlabs.png +0 -0
- package/lib/mocks/post.d.ts +38 -0
- package/lib/mocks/post.js +55 -0
- package/lib/mocks/tag.d.ts +2 -0
- package/lib/mocks/tag.js +37 -0
- package/lib/mocks/user.d.ts +4 -0
- package/lib/mocks/user.js +88 -0
- package/lib/mutations/index.d.ts +3 -0
- package/lib/mutations/index.js +26 -0
- package/lib/mutations/locations.d.ts +2 -0
- package/lib/mutations/locations.js +44 -0
- package/lib/mutations/messages.d.ts +2 -0
- package/lib/mutations/messages.js +86 -0
- package/lib/mutations/personas.d.ts +2 -0
- package/lib/mutations/personas.js +100 -0
- package/lib/mutations/posts.d.ts +2 -0
- package/lib/mutations/posts.js +53 -0
- package/lib/mutations/reactions.d.ts +2 -0
- package/lib/mutations/reactions.js +51 -0
- package/lib/mutations/statistics.d.ts +2 -0
- package/lib/mutations/statistics.js +39 -0
- package/lib/mutations/subscriptions.d.ts +2 -0
- package/lib/mutations/subscriptions.js +56 -0
- package/lib/mutations/tags.d.ts +2 -0
- package/lib/mutations/tags.js +120 -0
- package/lib/mutations/users.d.ts +1 -0
- package/lib/mutations/users.js +116 -0
- package/lib/objectTypes/app.d.ts +3 -0
- package/lib/objectTypes/app.js +173 -0
- package/lib/objectTypes/bankAccount.d.ts +1 -0
- package/lib/objectTypes/bankAccount.js +76 -0
- package/lib/objectTypes/connection.d.ts +1 -0
- package/lib/objectTypes/connection.js +48 -0
- package/lib/objectTypes/conversation.d.ts +2 -0
- package/lib/objectTypes/conversation.js +77 -0
- package/lib/objectTypes/creditCard.d.ts +1 -0
- package/lib/objectTypes/creditCard.js +86 -0
- package/lib/objectTypes/document.d.ts +1 -0
- package/lib/objectTypes/document.js +46 -0
- package/lib/objectTypes/error.d.ts +1 -0
- package/lib/objectTypes/error.js +46 -0
- package/lib/objectTypes/external.d.ts +1 -0
- package/lib/objectTypes/external.js +74 -0
- package/lib/objectTypes/file.d.ts +2 -0
- package/lib/objectTypes/file.js +100 -0
- package/lib/objectTypes/filter.d.ts +1 -0
- package/lib/objectTypes/filter.js +43 -0
- package/lib/objectTypes/group.d.ts +3 -0
- package/lib/objectTypes/group.js +123 -0
- package/lib/objectTypes/iapSubscription.d.ts +1 -0
- package/lib/objectTypes/iapSubscription.js +40 -0
- package/lib/objectTypes/image.d.ts +2 -0
- package/lib/objectTypes/image.js +129 -0
- package/lib/objectTypes/index.d.ts +24 -0
- package/lib/objectTypes/index.js +68 -0
- package/lib/objectTypes/location.d.ts +2 -0
- package/lib/objectTypes/location.js +109 -0
- package/lib/objectTypes/message.d.ts +2 -0
- package/lib/objectTypes/message.js +96 -0
- package/lib/objectTypes/passcode.d.ts +1 -0
- package/lib/objectTypes/passcode.js +42 -0
- package/lib/objectTypes/persona.d.ts +3 -0
- package/lib/objectTypes/persona.js +87 -0
- package/lib/objectTypes/plan.d.ts +2 -0
- package/lib/objectTypes/plan.js +95 -0
- package/lib/objectTypes/post.d.ts +2 -0
- package/lib/objectTypes/post.js +125 -0
- package/lib/objectTypes/reaction.d.ts +2 -0
- package/lib/objectTypes/reaction.js +61 -0
- package/lib/objectTypes/relation.d.ts +1 -0
- package/lib/objectTypes/relation.js +49 -0
- package/lib/objectTypes/search.d.ts +1 -0
- package/lib/objectTypes/search.js +72 -0
- package/lib/objectTypes/statistics.d.ts +1 -0
- package/lib/objectTypes/statistics.js +39 -0
- package/lib/objectTypes/subscription.d.ts +2 -0
- package/lib/objectTypes/subscription.js +117 -0
- package/lib/objectTypes/tag.d.ts +2 -0
- package/lib/objectTypes/tag.js +65 -0
- package/lib/objectTypes/user.d.ts +4 -0
- package/lib/objectTypes/user.js +144 -0
- package/lib/queries/index.d.ts +3 -0
- package/lib/queries/index.js +33 -0
- package/lib/queries/locations.d.ts +2 -0
- package/lib/queries/locations.js +45 -0
- package/lib/queries/messages.d.ts +2 -0
- package/lib/queries/messages.js +52 -0
- package/lib/queries/posts.d.ts +2 -0
- package/lib/queries/posts.js +154 -0
- package/lib/queries/reactions.d.ts +2 -0
- package/lib/queries/reactions.js +56 -0
- package/lib/queries/statistics.d.ts +2 -0
- package/lib/queries/statistics.js +39 -0
- package/lib/queries/subscriptions.d.ts +2 -0
- package/lib/queries/subscriptions.js +44 -0
- package/lib/queries/tags.d.ts +2 -0
- package/lib/queries/tags.js +75 -0
- package/lib/queries/users.d.ts +1 -0
- package/lib/queries/users.js +64 -0
- package/lib/types/{apps.d.ts → apps.types.d.ts} +19 -17
- package/lib/types/apps.types.js +32 -0
- package/lib/types/arangodb.types.d.ts +34 -0
- package/lib/types/arangodb.types.js +16 -0
- package/lib/types/auth.types.d.ts +9 -0
- package/lib/types/auth.types.js +16 -0
- package/lib/types/connections.types.d.ts +5 -0
- package/lib/types/connections.types.js +16 -0
- package/lib/types/conversations.types.d.ts +27 -0
- package/lib/types/conversations.types.js +16 -0
- package/lib/types/email.types.d.ts +13 -0
- package/lib/types/email.types.js +16 -0
- package/lib/types/error.types.d.ts +20 -0
- package/lib/types/error.types.js +44 -0
- package/lib/types/{files.d.ts → files.types.d.ts} +9 -12
- package/lib/types/files.types.js +16 -0
- package/lib/types/google.types.d.ts +29 -0
- package/lib/types/google.types.js +16 -0
- package/lib/types/{groups.d.ts → groups.types.d.ts} +6 -10
- package/lib/types/groups.types.js +16 -0
- package/lib/types/images.types.d.ts +52 -0
- package/lib/types/images.types.js +16 -0
- package/lib/types/index.d.ts +20 -18
- package/lib/types/index.js +41 -37
- package/lib/types/{locations.d.ts → locations.types.d.ts} +4 -6
- package/lib/types/locations.types.js +16 -0
- package/lib/types/messages.types.d.ts +16 -0
- package/lib/types/messages.types.js +16 -0
- package/lib/types/notifications.types.d.ts +19 -0
- package/lib/types/notifications.types.js +16 -0
- package/lib/types/{payments.d.ts → payments.types.d.ts} +13 -19
- package/lib/types/payments.types.js +16 -0
- package/lib/types/personas.types.d.ts +32 -0
- package/lib/types/personas.types.js +16 -0
- package/lib/types/posts.types.d.ts +28 -0
- package/lib/types/posts.types.js +16 -0
- package/lib/types/statistics.types.d.ts +3 -0
- package/lib/types/statistics.types.js +16 -0
- package/lib/types/tags.types.d.ts +15 -0
- package/lib/types/tags.types.js +16 -0
- package/lib/types/{users.d.ts → users.types.d.ts} +21 -22
- package/lib/types/users.types.js +16 -0
- package/lib/types/{websocket.d.ts → websockets.types.d.ts} +7 -3
- package/lib/types/websockets.types.js +16 -0
- package/lib/utils/adapterUtils.d.ts +1 -0
- package/lib/utils/adapterUtils.js +45 -0
- package/lib/utils/analyticsUtils.d.ts +21 -0
- package/lib/utils/analyticsUtils.js +72 -0
- package/lib/utils/arangodbUtils.d.ts +66 -0
- package/lib/utils/arangodbUtils.js +165 -0
- package/lib/utils/auth.d.ts +19 -3
- package/lib/utils/auth.js +20 -30
- package/lib/utils/index.d.ts +3 -4
- package/lib/utils/index.js +7 -9
- package/lib/utils/session.d.ts +10 -10
- package/lib/utils/session.js +15 -3
- package/lib/utils/stripeUtils.d.ts +3 -0
- package/lib/utils/{graphql.js → stripeUtils.js} +12 -15
- package/package.json +38 -30
- package/lib/types/apps.js +0 -16
- package/lib/types/arangodb.d.ts +0 -17
- package/lib/types/arangodb.js +0 -16
- package/lib/types/auth.d.ts +0 -7
- package/lib/types/connections.d.ts +0 -8
- package/lib/types/conversations.d.ts +0 -16
- package/lib/types/email.d.ts +0 -12
- package/lib/types/google.d.ts +0 -27
- package/lib/types/google.js +0 -16
- package/lib/types/images.d.ts +0 -42
- package/lib/types/images.js +0 -16
- package/lib/types/messages.d.ts +0 -27
- package/lib/types/messages.js +0 -16
- package/lib/types/notifications.d.ts +0 -19
- package/lib/types/payments.js +0 -16
- package/lib/types/posts.d.ts +0 -47
- package/lib/types/posts.js +0 -16
- package/lib/types/statistics.d.ts +0 -3
- package/lib/types/tags.d.ts +0 -15
- package/lib/types/users.js +0 -16
- package/lib/types/websocket.js +0 -16
- package/lib/utils/analytics.d.ts +0 -14
- package/lib/utils/analytics.js +0 -88
- package/lib/utils/arangodb.d.ts +0 -9
- package/lib/utils/arangodb.js +0 -118
- package/lib/utils/graphql.d.ts +0 -1
- package/lib/utils/objects.d.ts +0 -3
- package/lib/utils/objects.js +0 -59
package/lib/actions/users.js
CHANGED
|
@@ -33,17 +33,21 @@ __export(users_exports, {
|
|
|
33
33
|
createToken: () => createToken,
|
|
34
34
|
deactivateUser: () => deactivateUser,
|
|
35
35
|
deleteUser: () => deleteUser,
|
|
36
|
+
forgotPassword: () => forgotPassword,
|
|
36
37
|
getActiveUserCount: () => getActiveUserCount,
|
|
37
38
|
getDisplayName: () => getDisplayName,
|
|
38
39
|
getSessionUser: () => getSessionUser,
|
|
39
40
|
getUser: () => getUser,
|
|
41
|
+
getUserByToken: () => getUserByToken,
|
|
40
42
|
getUserOptional: () => getUserOptional,
|
|
41
43
|
getUsers: () => getUsers,
|
|
44
|
+
getUsersByConnection: () => getUsersByConnection,
|
|
42
45
|
getUsersByLatest: () => getUsersByLatest,
|
|
43
46
|
getUsersByReactions: () => getUsersByReactions,
|
|
44
47
|
getUsersByTags: () => getUsersByTags,
|
|
45
48
|
parseUserOptions: () => parseUserOptions,
|
|
46
49
|
refreshSession: () => refreshSession,
|
|
50
|
+
resetPassword: () => resetPassword,
|
|
47
51
|
signIn: () => signIn,
|
|
48
52
|
signOut: () => signOut,
|
|
49
53
|
updateUser: () => updateUser
|
|
@@ -52,12 +56,17 @@ module.exports = __toCommonJS(users_exports);
|
|
|
52
56
|
var import_utils = require("@nlabs/utils");
|
|
53
57
|
var import_arangojs = require("arangojs");
|
|
54
58
|
var import_luxon = require("luxon");
|
|
55
|
-
var import_stripe = __toESM(require("stripe"));
|
|
59
|
+
var import_stripe = __toESM(require("stripe"), 1);
|
|
60
|
+
var import_userAdapter = require("../adapters/userAdapter");
|
|
56
61
|
var import_config = require("../config");
|
|
57
|
-
var
|
|
62
|
+
var import_email = require("./email");
|
|
63
|
+
var import_sms = require("./sms");
|
|
64
|
+
var import_error = require("../types/error.types");
|
|
65
|
+
var import_analyticsUtils = require("../utils/analyticsUtils");
|
|
66
|
+
var import_arangodbUtils = require("../utils/arangodbUtils");
|
|
58
67
|
var import_session = require("../utils/session");
|
|
59
68
|
const eventCategory = "users";
|
|
60
|
-
const
|
|
69
|
+
const STRIPE_API_VERSION = "2025-05-28.basil";
|
|
61
70
|
var UserAccess = /* @__PURE__ */ ((UserAccess2) => {
|
|
62
71
|
UserAccess2[UserAccess2["DEACTIVATED"] = 0] = "DEACTIVATED";
|
|
63
72
|
UserAccess2[UserAccess2["ACTIVE"] = 1] = "ACTIVE";
|
|
@@ -66,27 +75,29 @@ var UserAccess = /* @__PURE__ */ ((UserAccess2) => {
|
|
|
66
75
|
UserAccess2[UserAccess2["ADMIN"] = 4] = "ADMIN";
|
|
67
76
|
return UserAccess2;
|
|
68
77
|
})(UserAccess || {});
|
|
69
|
-
const createToken = (userId, username, userAccess,
|
|
78
|
+
const createToken = (userId, username, userAccess, expiresInMinutes = 15) => {
|
|
70
79
|
const now = import_luxon.DateTime.local();
|
|
71
|
-
const sessionExpires = now.plus({ minutes:
|
|
80
|
+
const sessionExpires = now.plus({ minutes: expiresInMinutes });
|
|
72
81
|
const iat = Math.floor(now.toSeconds());
|
|
73
82
|
const exp = Math.floor(sessionExpires.toSeconds());
|
|
74
83
|
const token = (0, import_session.setSession)({
|
|
75
84
|
exp,
|
|
76
85
|
iat,
|
|
77
|
-
username,
|
|
78
86
|
userAccess,
|
|
79
|
-
userId
|
|
87
|
+
userId,
|
|
88
|
+
username
|
|
80
89
|
});
|
|
81
90
|
return {
|
|
82
91
|
expires: sessionExpires.toMillis(),
|
|
83
92
|
issued: now.toMillis(),
|
|
84
|
-
token
|
|
93
|
+
token,
|
|
94
|
+
userId,
|
|
95
|
+
username
|
|
85
96
|
};
|
|
86
97
|
};
|
|
87
98
|
const getUserOptional = (fields = []) => fields.reduce((selects, field) => {
|
|
88
99
|
if (field.includes("Count")) {
|
|
89
|
-
return (0,
|
|
100
|
+
return (0, import_arangodbUtils.selectReactionCountByType)("users", "u", field, selects);
|
|
90
101
|
}
|
|
91
102
|
return selects;
|
|
92
103
|
}, { objects: [], queries: [] });
|
|
@@ -95,30 +106,37 @@ const parseUserOptions = (options = {}) => {
|
|
|
95
106
|
from = 0,
|
|
96
107
|
to = 30
|
|
97
108
|
} = options;
|
|
109
|
+
const limit = (0, import_arangodbUtils.getLimit)(from, to);
|
|
98
110
|
return {
|
|
99
111
|
...options,
|
|
100
|
-
limit
|
|
112
|
+
limit
|
|
101
113
|
};
|
|
102
114
|
};
|
|
103
|
-
const addUser = async (context,
|
|
115
|
+
const addUser = async (context, user) => {
|
|
104
116
|
const action = "addUser";
|
|
105
117
|
const { database } = context;
|
|
106
|
-
const {
|
|
107
|
-
const { email, password, phone, username } = user;
|
|
108
|
-
const salt = (0, import_utils.createHash)(`${username}${password}`, null);
|
|
109
|
-
const encryptedPassword = (0, import_utils.createPassword)(password, salt);
|
|
118
|
+
const { email, password, phone, username } = (0, import_userAdapter.parseUser)(user);
|
|
110
119
|
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
111
120
|
const formatEmail = (0, import_utils.parseEmail)(email);
|
|
112
121
|
const formatPhone = (0, import_utils.parsePhone)(phone);
|
|
113
|
-
|
|
114
|
-
|
|
122
|
+
const formatPassword = (0, import_utils.parsePassword)(password);
|
|
123
|
+
const hasPassword = !!formatPassword;
|
|
124
|
+
const hasUsername = !!formatUsername || !!formatPhone || !!formatEmail;
|
|
125
|
+
if (!hasPassword || !hasUsername) {
|
|
126
|
+
return (0, import_analyticsUtils.logException)({
|
|
115
127
|
action,
|
|
116
|
-
args: { username },
|
|
117
128
|
category: eventCategory,
|
|
118
|
-
|
|
129
|
+
params: { username },
|
|
130
|
+
value: import_error.ErrorTypes.INVALID_ARGUMENTS
|
|
119
131
|
}, context);
|
|
120
132
|
}
|
|
121
|
-
const
|
|
133
|
+
const hashId = formatUsername || formatPhone || formatEmail;
|
|
134
|
+
const salt = (0, import_utils.createHash)(`${hashId}${formatPassword}`, null);
|
|
135
|
+
const encryptedPassword = (0, import_utils.createPassword)(formatPassword, salt);
|
|
136
|
+
const filters = [];
|
|
137
|
+
if (formatUsername) {
|
|
138
|
+
filters.push(`u.username == "${formatUsername}"`);
|
|
139
|
+
}
|
|
122
140
|
if (formatEmail) {
|
|
123
141
|
filters.push(`u.email == "${formatEmail}"`);
|
|
124
142
|
}
|
|
@@ -127,101 +145,222 @@ const addUser = async (context, args) => {
|
|
|
127
145
|
}
|
|
128
146
|
const checkQuery = `FOR u IN users
|
|
129
147
|
FILTER ${filters.join(" || ")}
|
|
148
|
+
LIMIT 1
|
|
130
149
|
RETURN u`;
|
|
131
|
-
|
|
132
|
-
(
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
150
|
+
try {
|
|
151
|
+
const existingUsers = await database.query(checkQuery).then((cursor) => cursor.all());
|
|
152
|
+
if (existingUsers.length) {
|
|
153
|
+
return (0, import_analyticsUtils.logException)({
|
|
154
|
+
action,
|
|
155
|
+
category: eventCategory,
|
|
156
|
+
params: {
|
|
157
|
+
email: formatEmail,
|
|
158
|
+
phone: formatPhone,
|
|
159
|
+
username: formatUsername
|
|
160
|
+
},
|
|
161
|
+
value: import_error.ErrorTypes.EXISTING_ITEM
|
|
162
|
+
}, context);
|
|
163
|
+
}
|
|
164
|
+
} catch (error) {
|
|
165
|
+
return (0, import_analyticsUtils.logError)({
|
|
141
166
|
action,
|
|
142
|
-
args: { username },
|
|
143
167
|
category: eventCategory,
|
|
144
|
-
|
|
145
|
-
|
|
168
|
+
params: { username },
|
|
169
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
170
|
+
}, error, context);
|
|
146
171
|
}
|
|
147
172
|
const verifiedEmailCode = Math.floor(1e5 + Math.random() * 9e5);
|
|
148
173
|
const verifiedPhoneCode = Math.floor(1e5 + Math.random() * 9e5);
|
|
149
174
|
const insert = {
|
|
150
|
-
_key: (0, import_utils.createHash)(
|
|
175
|
+
_key: (0, import_utils.createHash)(formatUsername, null),
|
|
151
176
|
added: Date.now(),
|
|
152
177
|
email: formatEmail,
|
|
153
178
|
modified: Date.now(),
|
|
154
179
|
password: encryptedPassword,
|
|
155
180
|
phone: formatPhone,
|
|
156
181
|
salt,
|
|
157
|
-
username: formatUsername,
|
|
158
182
|
userAccess: 1,
|
|
183
|
+
username: formatUsername,
|
|
159
184
|
verifiedEmail: false,
|
|
160
185
|
verifiedEmailCode,
|
|
161
186
|
verifiedPhone: false,
|
|
162
187
|
verifiedPhoneCode
|
|
163
188
|
};
|
|
164
189
|
const insertQuery = import_arangojs.aql`INSERT ${insert} IN users RETURN NEW`;
|
|
165
|
-
return await database.query(insertQuery).then((cursor) => cursor.next()
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
}, error, context).then(() => null)
|
|
172
|
-
);
|
|
190
|
+
return await database.query(insertQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
|
|
191
|
+
action,
|
|
192
|
+
category: eventCategory,
|
|
193
|
+
params: { username },
|
|
194
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
195
|
+
}, error, context));
|
|
173
196
|
};
|
|
174
197
|
const updateUser = async (context, user) => {
|
|
175
198
|
const action = "updateUser";
|
|
176
|
-
const { database } = context;
|
|
177
|
-
const { _key, _id, id, tags = [], userId, ...updated } = user;
|
|
178
|
-
|
|
179
|
-
|
|
180
|
-
|
|
181
|
-
|
|
182
|
-
|
|
199
|
+
const { database, session } = context;
|
|
200
|
+
const { _key, _id, id, tags = [], userId, ...updated } = (0, import_userAdapter.parseUser)(user);
|
|
201
|
+
if (!(0, import_session.isAdminUser)(session) && session?.userId !== userId) {
|
|
202
|
+
return (0, import_analyticsUtils.logException)({
|
|
203
|
+
action,
|
|
204
|
+
category: eventCategory,
|
|
205
|
+
params: { session },
|
|
206
|
+
value: import_error.ErrorTypes.INVALID_SESSION
|
|
207
|
+
}, context);
|
|
183
208
|
}
|
|
184
|
-
const userQuery = import_arangojs.aql`LET u = DOCUMENT(${
|
|
209
|
+
const userQuery = import_arangojs.aql`LET u = DOCUMENT(${id})
|
|
185
210
|
UPDATE u WITH ${updated} IN users
|
|
186
211
|
RETURN NEW`;
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
return
|
|
212
|
+
try {
|
|
213
|
+
const updatedUser = await database.query(userQuery).then((cursor) => cursor.next());
|
|
214
|
+
const tagCollection = database.collection("isTagged");
|
|
215
|
+
await Promise.all(tags.map(({ id: tagDocId, name }) => {
|
|
216
|
+
const tagQuery = import_arangojs.aql`FOR it IN isTagged
|
|
217
|
+
FILTER it._from == ${tagDocId} && it._to == ${id} && it.name == ${name}
|
|
218
|
+
LIMIT 1
|
|
219
|
+
RETURN it`;
|
|
220
|
+
return database.query(tagQuery).then((cursor) => cursor.next()).then((tagEdge) => {
|
|
221
|
+
if (!!tagEdge) {
|
|
222
|
+
return tagEdge;
|
|
223
|
+
}
|
|
224
|
+
const edge = {
|
|
225
|
+
_from: tagDocId,
|
|
226
|
+
_key: (0, import_utils.createHash)(`isTagged-${tagDocId}-${id}`),
|
|
227
|
+
_to: id,
|
|
228
|
+
added: Date.now(),
|
|
229
|
+
name
|
|
230
|
+
};
|
|
231
|
+
return tagCollection.save(edge, { returnNew: true }).then(() => edge);
|
|
232
|
+
});
|
|
233
|
+
}));
|
|
234
|
+
return updatedUser;
|
|
235
|
+
} catch (error) {
|
|
236
|
+
return (0, import_analyticsUtils.logError)({
|
|
237
|
+
action,
|
|
238
|
+
category: eventCategory,
|
|
239
|
+
params: { user },
|
|
240
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
241
|
+
}, error, context);
|
|
242
|
+
}
|
|
243
|
+
};
|
|
244
|
+
const forgotPassword = async (context, { email, phone, username }) => {
|
|
245
|
+
const action = "forgotPassword";
|
|
246
|
+
const { app, database } = context;
|
|
247
|
+
const aqlQuery = import_arangojs.aql`FOR u IN users
|
|
248
|
+
FILTER u.email == ${email} || u.phone == ${phone} || u.username == ${username}
|
|
249
|
+
LIMIT 1
|
|
250
|
+
RETURN u`;
|
|
251
|
+
try {
|
|
252
|
+
return await database.query(aqlQuery).then(async (cursor) => {
|
|
253
|
+
const user = cursor.next();
|
|
254
|
+
if (user) {
|
|
255
|
+
const { email: email2, phone: phone2, verifiedEmail, verifiedPhone } = user;
|
|
256
|
+
const codeExpires = 1e3 * 60 * 15;
|
|
257
|
+
const code = Math.floor(1e5 + Math.random() * 9e5);
|
|
258
|
+
const userDocId = (0, import_arangodbUtils.getDocId)("users", user);
|
|
259
|
+
let update;
|
|
260
|
+
if (email2 && verifiedEmail) {
|
|
261
|
+
(0, import_email.sendEmail)({
|
|
262
|
+
app,
|
|
263
|
+
text: `Your code is ${code}`
|
|
264
|
+
});
|
|
265
|
+
update = { verifiedEmailCode: code, verifiedEmailExpires: codeExpires };
|
|
266
|
+
}
|
|
267
|
+
if (phone2 && verifiedPhone) {
|
|
268
|
+
(0, import_sms.sendSms)({
|
|
269
|
+
app,
|
|
270
|
+
text: `Your code is ${code}`
|
|
271
|
+
});
|
|
272
|
+
update = { verifiedPhoneCode: code, verifiedPhoneExpires: codeExpires };
|
|
273
|
+
}
|
|
274
|
+
if (update.verifiedEmailCode || update.verifiedPhoneCode) {
|
|
275
|
+
const updateQuery = import_arangojs.aql`UPDATE ${userDocId} WITH ${update} IN users`;
|
|
276
|
+
await database.query(updateQuery);
|
|
277
|
+
return true;
|
|
278
|
+
}
|
|
279
|
+
return false;
|
|
280
|
+
}
|
|
281
|
+
return false;
|
|
282
|
+
});
|
|
283
|
+
} catch (error) {
|
|
284
|
+
(0, import_analyticsUtils.logError)({
|
|
285
|
+
action,
|
|
286
|
+
category: eventCategory,
|
|
287
|
+
params: { email, phone, username },
|
|
288
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
289
|
+
}, error, context);
|
|
290
|
+
return false;
|
|
291
|
+
}
|
|
292
|
+
};
|
|
293
|
+
const resetPassword = async (context, {
|
|
294
|
+
code,
|
|
295
|
+
password,
|
|
296
|
+
type,
|
|
297
|
+
username
|
|
298
|
+
}) => {
|
|
299
|
+
const action = "resetPassword";
|
|
300
|
+
const { database } = context;
|
|
301
|
+
const formatPassword = (0, import_utils.parsePassword)(password);
|
|
302
|
+
const aqlQuery = import_arangojs.aql`FOR u IN users
|
|
303
|
+
FILTER u.username == ${username}
|
|
304
|
+
LIMIT 1
|
|
305
|
+
RETURN u`;
|
|
306
|
+
try {
|
|
307
|
+
return await database.query(aqlQuery).then(async (cursor) => {
|
|
308
|
+
const user = cursor.next();
|
|
309
|
+
if (user) {
|
|
310
|
+
const {
|
|
311
|
+
_id: userDocId,
|
|
312
|
+
salt,
|
|
313
|
+
verifiedEmailCode,
|
|
314
|
+
verifiedEmailExpires,
|
|
315
|
+
verifiedPhoneCode,
|
|
316
|
+
verifiedPhoneExpires
|
|
317
|
+
} = user;
|
|
318
|
+
const now = Date.now();
|
|
319
|
+
let update;
|
|
320
|
+
switch (type) {
|
|
321
|
+
case "email":
|
|
322
|
+
if (code === verifiedEmailCode && verifiedEmailExpires > now) {
|
|
323
|
+
const password2 = (0, import_utils.createPassword)(formatPassword, salt);
|
|
324
|
+
update = { password: password2 };
|
|
325
|
+
}
|
|
326
|
+
break;
|
|
327
|
+
case "phone":
|
|
328
|
+
if (code === verifiedPhoneCode && verifiedPhoneExpires > now) {
|
|
329
|
+
const password2 = (0, import_utils.createPassword)(formatPassword, salt);
|
|
330
|
+
update = { password: password2 };
|
|
331
|
+
}
|
|
332
|
+
break;
|
|
333
|
+
default:
|
|
334
|
+
return false;
|
|
335
|
+
}
|
|
336
|
+
if (update) {
|
|
337
|
+
const updateQuery = import_arangojs.aql`UPDATE ${userDocId} WITH ${update} IN users`;
|
|
338
|
+
await database.query(updateQuery);
|
|
339
|
+
return true;
|
|
340
|
+
}
|
|
207
341
|
}
|
|
208
|
-
return
|
|
209
|
-
})
|
|
342
|
+
return false;
|
|
343
|
+
});
|
|
344
|
+
} catch (error) {
|
|
345
|
+
(0, import_analyticsUtils.logError)({
|
|
210
346
|
action,
|
|
211
347
|
category: eventCategory,
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
348
|
+
params: { username },
|
|
349
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
350
|
+
}, error, context);
|
|
351
|
+
return false;
|
|
352
|
+
}
|
|
216
353
|
};
|
|
217
|
-
const confirmCode = async (context,
|
|
218
|
-
|
|
354
|
+
const confirmCode = async (context, {
|
|
355
|
+
code,
|
|
356
|
+
type
|
|
357
|
+
}) => {
|
|
358
|
+
const action = "confirmCode";
|
|
219
359
|
const { database, session: { userId: sessionId } } = context;
|
|
220
|
-
const
|
|
221
|
-
const userDocId = `users/${sessionId}`;
|
|
360
|
+
const userDocId = (0, import_arangodbUtils.getDocId)("users", { userId: sessionId });
|
|
222
361
|
const aqlQuery = import_arangojs.aql`LET u = DOCUMENT(${userDocId}) RETURN u`;
|
|
223
362
|
try {
|
|
224
|
-
return database.query(aqlQuery).then((cursor) => cursor.next()
|
|
363
|
+
return await database.query(aqlQuery).then((cursor) => cursor.next()).then(({ verifiedEmailCode, verifiedPhoneCode }) => {
|
|
225
364
|
switch (type) {
|
|
226
365
|
case "email":
|
|
227
366
|
return code === verifiedEmailCode;
|
|
@@ -230,65 +369,50 @@ const confirmCode = async (context, args) => {
|
|
|
230
369
|
default:
|
|
231
370
|
return false;
|
|
232
371
|
}
|
|
233
|
-
})
|
|
234
|
-
(error) => (0, import_utils2.logError)({
|
|
235
|
-
action,
|
|
236
|
-
args: { code, type, userId: sessionId },
|
|
237
|
-
category: eventCategory,
|
|
238
|
-
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
239
|
-
}, error, context)
|
|
240
|
-
);
|
|
372
|
+
});
|
|
241
373
|
} catch (error) {
|
|
374
|
+
(0, import_analyticsUtils.logError)({
|
|
375
|
+
action,
|
|
376
|
+
category: eventCategory,
|
|
377
|
+
params: { code, type },
|
|
378
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
379
|
+
}, error, context);
|
|
242
380
|
return false;
|
|
243
381
|
}
|
|
244
382
|
};
|
|
245
|
-
const deleteUser = (context,
|
|
383
|
+
const deleteUser = (context, user) => {
|
|
246
384
|
const action = "deleteUser";
|
|
247
|
-
const { database
|
|
248
|
-
const { userId } =
|
|
249
|
-
const isAdmin = sessionAccess > 2;
|
|
250
|
-
if (!isAdmin && sessionId !== userId) {
|
|
251
|
-
(0, import_utils2.logException)({
|
|
252
|
-
action,
|
|
253
|
-
args,
|
|
254
|
-
category: eventCategory,
|
|
255
|
-
label: "unauthorized",
|
|
256
|
-
value: "invalid_session"
|
|
257
|
-
}, context);
|
|
258
|
-
return null;
|
|
259
|
-
}
|
|
385
|
+
const { database } = context;
|
|
386
|
+
const { userId } = (0, import_userAdapter.parseUser)(user);
|
|
260
387
|
const aqlQuery = import_arangojs.aql`FOR u IN users
|
|
261
388
|
FILTER u._key == ${userId}
|
|
262
389
|
LIMIT 1
|
|
263
390
|
REMOVE u IN users
|
|
264
391
|
RETURN OLD`;
|
|
265
|
-
const stripeClient = new import_stripe.default(import_config.Config.get("stripe.token"), { apiVersion, typescript: true });
|
|
266
|
-
return database.query(aqlQuery).then((cursor) => cursor.next()).then((
|
|
267
|
-
|
|
268
|
-
|
|
392
|
+
const stripeClient = new import_stripe.default(import_config.Config.get("stripe.token"), { apiVersion: STRIPE_API_VERSION, typescript: true });
|
|
393
|
+
return database.query(aqlQuery).then((cursor) => cursor.next()).then((deletedUser) => stripeClient.customers.del(deletedUser?.stripeCustomerId).then(() => stripeClient.accounts.del(deletedUser?.stripeAccountId)).then(() => deletedUser)).catch((error) => (0, import_analyticsUtils.logError)({
|
|
394
|
+
action,
|
|
395
|
+
category: eventCategory,
|
|
396
|
+
params: { userId },
|
|
397
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
398
|
+
}, error, context));
|
|
269
399
|
};
|
|
270
|
-
const deactivateUser = (context,
|
|
400
|
+
const deactivateUser = (context, user) => {
|
|
271
401
|
const action = "delete";
|
|
272
|
-
const { database
|
|
273
|
-
const
|
|
274
|
-
if (!isAdmin && sessionId !== userId) {
|
|
275
|
-
(0, import_utils2.logException)({
|
|
276
|
-
action,
|
|
277
|
-
category: eventCategory,
|
|
278
|
-
label: "unauthorized",
|
|
279
|
-
value: "invalid_session"
|
|
280
|
-
}, context);
|
|
281
|
-
return null;
|
|
282
|
-
}
|
|
402
|
+
const { database } = context;
|
|
403
|
+
const { userId } = (0, import_userAdapter.parseUser)(user);
|
|
283
404
|
const updated = {
|
|
284
405
|
userAccess: 0
|
|
285
406
|
};
|
|
286
407
|
const aqlQuery = import_arangojs.aql`UPDATE ${userId} WITH ${updated} IN users LIMIT 1 RETURN NEW`;
|
|
287
|
-
return database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => {
|
|
288
|
-
|
|
289
|
-
|
|
408
|
+
return database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
|
|
409
|
+
action,
|
|
410
|
+
category: eventCategory,
|
|
411
|
+
params: { userId },
|
|
412
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
413
|
+
}, error, context));
|
|
290
414
|
};
|
|
291
|
-
const getDisplayName = (user
|
|
415
|
+
const getDisplayName = (user) => {
|
|
292
416
|
const { first, last, name = "", username = "" } = user;
|
|
293
417
|
const fullname = [first, last].join(" ").trim();
|
|
294
418
|
if (name) {
|
|
@@ -302,35 +426,47 @@ const getDisplayName = (user = {}) => {
|
|
|
302
426
|
};
|
|
303
427
|
const getSessionUser = (context) => {
|
|
304
428
|
const action = "getSessionUser";
|
|
429
|
+
console.log("getSessionUser", { action, context });
|
|
305
430
|
const { database, fields, session: { userId: sessionId, username } } = context;
|
|
306
431
|
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
307
|
-
const
|
|
432
|
+
const formatSessionId = (0, import_utils.parseArangoId)(`users/${sessionId}`);
|
|
433
|
+
const aqlQuery = `LET u = DOCUMENT("${formatSessionId}")
|
|
308
434
|
${selectQueries.join("\n")}
|
|
309
435
|
RETURN MERGE(u, {${selectObjects.join(", ")}})`;
|
|
310
|
-
return database.query(aqlQuery).then((cursor) => cursor.next()).
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
|
|
436
|
+
return database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => {
|
|
437
|
+
(0, import_analyticsUtils.logError)({
|
|
438
|
+
action,
|
|
439
|
+
category: eventCategory,
|
|
440
|
+
params: { userId: sessionId, username },
|
|
441
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
442
|
+
}, error, context);
|
|
443
|
+
return null;
|
|
444
|
+
});
|
|
316
445
|
};
|
|
317
|
-
const getUser = (context,
|
|
446
|
+
const getUser = (context, user) => {
|
|
318
447
|
const action = "getUser";
|
|
319
|
-
const { userId } =
|
|
448
|
+
const { id, userId, username } = (0, import_userAdapter.parseUser)(user);
|
|
320
449
|
const { database, fields } = context;
|
|
321
|
-
const formatUserId = (0, import_utils.parseId)(userId);
|
|
322
450
|
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
323
|
-
|
|
451
|
+
let aqlQuery;
|
|
452
|
+
console.log({ id, userId, username });
|
|
453
|
+
if (id) {
|
|
454
|
+
aqlQuery = `LET u = DOCUMENT("${id}")
|
|
324
455
|
${selectQueries.join("\n")}
|
|
325
456
|
FILTER u.userAccess > 0
|
|
326
457
|
RETURN MERGE(u, {${selectObjects.join(", ")}})`;
|
|
327
|
-
|
|
458
|
+
} else if (username) {
|
|
459
|
+
aqlQuery = `FOR u IN users
|
|
460
|
+
FILTER u.username == "${username}"
|
|
461
|
+
${selectQueries.join("\n")}
|
|
462
|
+
RETURN MERGE(u, {${selectObjects.join(", ")}})`;
|
|
463
|
+
}
|
|
464
|
+
return database.query(aqlQuery).then((cursor) => cursor.next()).then((user2) => user2).catch((error) => (0, import_analyticsUtils.logError)({
|
|
328
465
|
action,
|
|
329
|
-
args,
|
|
330
466
|
category: eventCategory,
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
}));
|
|
467
|
+
params: { id, userId, username },
|
|
468
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
469
|
+
}, error, context));
|
|
334
470
|
};
|
|
335
471
|
const getUsers = (context, options) => {
|
|
336
472
|
const action = "getUserList";
|
|
@@ -347,11 +483,14 @@ const getUsers = (context, options) => {
|
|
|
347
483
|
${limit.aql}
|
|
348
484
|
SORT u.username
|
|
349
485
|
RETURN MERGE(u, {${selectObjects.join(", ")}})`;
|
|
350
|
-
return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) =>
|
|
351
|
-
|
|
352
|
-
|
|
353
|
-
|
|
354
|
-
|
|
486
|
+
return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => {
|
|
487
|
+
(0, import_analyticsUtils.logError)({
|
|
488
|
+
action,
|
|
489
|
+
category: eventCategory,
|
|
490
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
491
|
+
}, error, context);
|
|
492
|
+
return [];
|
|
493
|
+
});
|
|
355
494
|
};
|
|
356
495
|
const getUsersByReactions = (context, { reactions = [], username }, options) => {
|
|
357
496
|
const action = "getUsersByReactions";
|
|
@@ -361,27 +500,32 @@ const getUsersByReactions = (context, { reactions = [], username }, options) =>
|
|
|
361
500
|
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
362
501
|
const formatSessionId = `users/${sessionId}`;
|
|
363
502
|
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
364
|
-
const filterBy = [
|
|
503
|
+
const filterBy = [
|
|
504
|
+
"u.userAccess > 0",
|
|
505
|
+
`POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`
|
|
506
|
+
];
|
|
365
507
|
if (username) {
|
|
366
508
|
filterBy.push(`CONTAINS(u.username, "${formatUsername}")`);
|
|
367
509
|
}
|
|
368
510
|
const aqlQuery = `FOR u, r IN OUTBOUND "${formatSessionId}" hasReaction
|
|
369
511
|
OPTIONS {vertexCollections: "users"}
|
|
370
512
|
${selectQueries.join("\n")}
|
|
371
|
-
|
|
513
|
+
FILTER ${filterBy.join(" && ")}
|
|
372
514
|
${limit.aql}
|
|
373
515
|
RETURN MERGE(u, {${selectObjects.join(", ")}})`;
|
|
374
|
-
|
|
375
|
-
|
|
376
|
-
|
|
377
|
-
|
|
378
|
-
|
|
379
|
-
|
|
516
|
+
return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => {
|
|
517
|
+
(0, import_analyticsUtils.logError)({
|
|
518
|
+
action,
|
|
519
|
+
category: eventCategory,
|
|
520
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
521
|
+
}, error, context);
|
|
522
|
+
return [];
|
|
523
|
+
});
|
|
380
524
|
};
|
|
381
525
|
const getUsersByTags = (context, { tags, username }, options) => {
|
|
382
526
|
const action = "getUsersByTags";
|
|
383
527
|
const { database, fields, session: { userId: sessionId } } = context;
|
|
384
|
-
const formatTags = tags
|
|
528
|
+
const formatTags = tags?.reduce((list, tagName) => {
|
|
385
529
|
if (tagName) {
|
|
386
530
|
list.push((0, import_utils.parseChar)(tagName, 32).toLowerCase());
|
|
387
531
|
}
|
|
@@ -390,7 +534,10 @@ const getUsersByTags = (context, { tags, username }, options) => {
|
|
|
390
534
|
const { limit } = parseUserOptions(options);
|
|
391
535
|
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
392
536
|
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
393
|
-
const filterBy = [
|
|
537
|
+
const filterBy = [
|
|
538
|
+
`u._key != "${sessionId}"`,
|
|
539
|
+
"u.userAccess > 0"
|
|
540
|
+
];
|
|
394
541
|
if (username) {
|
|
395
542
|
filterBy.push(`CONTAINS(u.username, "${formatUsername}")`);
|
|
396
543
|
}
|
|
@@ -399,21 +546,26 @@ const getUsersByTags = (context, { tags, username }, options) => {
|
|
|
399
546
|
FOR u, it IN OUTBOUND t isTagged
|
|
400
547
|
OPTIONS {bfs: true, uniqueVertices: "global", vertexCollections: "users"}
|
|
401
548
|
${selectQueries.join("\n")}
|
|
402
|
-
|
|
549
|
+
FILTER ${filterBy.join(" && ")}
|
|
403
550
|
${limit.aql}
|
|
404
551
|
RETURN DISTINCT MERGE(u, {${selectObjects.join(", ")}})`;
|
|
405
|
-
|
|
406
|
-
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
552
|
+
return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => {
|
|
553
|
+
(0, import_analyticsUtils.logError)({
|
|
554
|
+
action,
|
|
555
|
+
category: eventCategory,
|
|
556
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
557
|
+
}, error, context);
|
|
558
|
+
return [];
|
|
559
|
+
});
|
|
411
560
|
};
|
|
412
561
|
const getUsersByLatest = (context, { username }, options) => {
|
|
413
562
|
const action = "getUsersByLatest";
|
|
414
563
|
const { database, fields, session: { userId } } = context;
|
|
415
564
|
const { limit } = parseUserOptions(options);
|
|
416
|
-
const filter = [
|
|
565
|
+
const filter = [
|
|
566
|
+
"u._id != session._id",
|
|
567
|
+
"u.userAccess > 0"
|
|
568
|
+
];
|
|
417
569
|
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
418
570
|
if (username) {
|
|
419
571
|
filter.push(`CONTAINS(u.username, "${(0, import_utils.parseUsername)(username)}")`);
|
|
@@ -426,72 +578,131 @@ const getUsersByLatest = (context, { username }, options) => {
|
|
|
426
578
|
${limit.aql}
|
|
427
579
|
SORT distance ASC, u.added DESC
|
|
428
580
|
RETURN MERGE(u, {${selectObjects.join(", ")}})`;
|
|
429
|
-
return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) =>
|
|
430
|
-
|
|
431
|
-
|
|
432
|
-
|
|
433
|
-
|
|
581
|
+
return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => {
|
|
582
|
+
(0, import_analyticsUtils.logError)({
|
|
583
|
+
action,
|
|
584
|
+
category: eventCategory,
|
|
585
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
586
|
+
}, error, context);
|
|
587
|
+
return [];
|
|
588
|
+
});
|
|
434
589
|
};
|
|
435
|
-
const
|
|
436
|
-
const
|
|
437
|
-
|
|
438
|
-
|
|
590
|
+
const getUsersByConnection = (context, { userId }, options) => {
|
|
591
|
+
const action = "getUsersByConnection";
|
|
592
|
+
const { database, fields } = context;
|
|
593
|
+
const { limit, username } = parseUserOptions(options);
|
|
594
|
+
const { objects: selectObjects, queries: selectQueries } = getUserOptional(fields);
|
|
595
|
+
const formatUserId = (0, import_utils.parseArangoId)(`users/${userId}`);
|
|
596
|
+
const filterBy = [
|
|
597
|
+
"u.userAccess > 0"
|
|
598
|
+
];
|
|
599
|
+
if (username) {
|
|
600
|
+
filterBy.push(`CONTAINS(u.username, "${(0, import_utils.parseUsername)(username)}")`);
|
|
439
601
|
}
|
|
440
|
-
const
|
|
441
|
-
|
|
442
|
-
|
|
602
|
+
const aqlQuery = `FOR cu IN users
|
|
603
|
+
LET session = DOCUMENT("${formatUserId}")
|
|
604
|
+
FOR u, connection IN OUTBOUND cu hasConnection
|
|
605
|
+
OPTIONS {bfs: true, uniqueVertices: "global", vertexCollections: "users"}
|
|
606
|
+
${selectQueries.join("\n")}
|
|
607
|
+
FILTER ${filterBy.join(" && ")}
|
|
608
|
+
${limit.aql}
|
|
609
|
+
RETURN DISTINCT MERGE(u, {${selectObjects.join(", ")}})`;
|
|
610
|
+
return database.query(aqlQuery).then((cursor) => cursor.all()).catch((error) => {
|
|
611
|
+
(0, import_analyticsUtils.logError)({
|
|
612
|
+
action,
|
|
613
|
+
category: eventCategory,
|
|
614
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
615
|
+
}, error, context);
|
|
616
|
+
return [];
|
|
617
|
+
});
|
|
618
|
+
};
|
|
619
|
+
const refreshSession = ({ expires, token }) => {
|
|
620
|
+
try {
|
|
621
|
+
const { userId, username, userAccess } = (0, import_session.getSession)(token);
|
|
443
622
|
return createToken(userId, username, userAccess, expires);
|
|
623
|
+
} catch (error) {
|
|
624
|
+
throw error;
|
|
444
625
|
}
|
|
445
|
-
return { errors: ["session_expired"] };
|
|
446
626
|
};
|
|
447
627
|
const signIn = async (context, args) => {
|
|
448
628
|
const action = "signIn";
|
|
449
|
-
const {
|
|
629
|
+
const { database } = context;
|
|
630
|
+
const { email, expires, password, phone, username } = args;
|
|
631
|
+
const formatEmail = (0, import_utils.parseEmail)(email);
|
|
450
632
|
const formatUsername = (0, import_utils.parseUsername)(username);
|
|
451
633
|
const formatPassword = (0, import_utils.parsePassword)(password);
|
|
634
|
+
const formatPhone = (0, import_utils.parsePhone)(phone);
|
|
452
635
|
const formatExpires = (0, import_utils.parseNum)(expires) || 15;
|
|
453
|
-
|
|
454
|
-
|
|
455
|
-
return (0, import_utils2.logException)({
|
|
636
|
+
if (!formatUsername && !formatEmail && !formatPhone || !formatPassword) {
|
|
637
|
+
(0, import_analyticsUtils.logException)({
|
|
456
638
|
action,
|
|
457
|
-
args: { username },
|
|
458
639
|
category: eventCategory,
|
|
459
|
-
|
|
640
|
+
params: { username },
|
|
641
|
+
value: import_error.ErrorTypes.INVALID_ARGUMENTS
|
|
460
642
|
}, context);
|
|
643
|
+
return null;
|
|
461
644
|
}
|
|
462
|
-
const
|
|
463
|
-
|
|
645
|
+
const filters = [];
|
|
646
|
+
if (formatEmail) {
|
|
647
|
+
filters.push(`u.email == "${formatEmail}"`);
|
|
648
|
+
}
|
|
649
|
+
if (formatPhone) {
|
|
650
|
+
filters.push(`u.phone == ${formatPhone}`);
|
|
651
|
+
}
|
|
652
|
+
if (formatUsername) {
|
|
653
|
+
filters.push(`u.username == "${formatUsername}"`);
|
|
654
|
+
}
|
|
655
|
+
const checkQuery = `FOR u IN users
|
|
656
|
+
FILTER ${filters.join(" || ")}
|
|
464
657
|
LIMIT 1
|
|
465
658
|
RETURN u`;
|
|
466
|
-
|
|
467
|
-
|
|
659
|
+
let checkUser;
|
|
660
|
+
try {
|
|
661
|
+
checkUser = await database.query(checkQuery).then((cursor) => cursor.next());
|
|
662
|
+
} catch (error) {
|
|
663
|
+
(0, import_analyticsUtils.logError)({
|
|
468
664
|
action,
|
|
469
|
-
args: { username: formatUsername },
|
|
470
665
|
category: eventCategory,
|
|
471
|
-
|
|
472
|
-
|
|
473
|
-
|
|
666
|
+
params: { username: formatUsername },
|
|
667
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
668
|
+
}, error, context);
|
|
669
|
+
return null;
|
|
670
|
+
}
|
|
474
671
|
if (!checkUser) {
|
|
475
|
-
|
|
672
|
+
(0, import_analyticsUtils.logException)({
|
|
476
673
|
action,
|
|
477
|
-
args: { username },
|
|
478
674
|
category: eventCategory,
|
|
479
|
-
|
|
675
|
+
params: { username },
|
|
676
|
+
value: import_error.ErrorTypes.INVALID_AUTHENTICATION
|
|
480
677
|
}, context);
|
|
678
|
+
return null;
|
|
481
679
|
}
|
|
482
|
-
const { _key: userId, salt, userAccess } = checkUser;
|
|
680
|
+
const { _key: userId, password: validPassword, salt, userAccess } = checkUser;
|
|
483
681
|
const authPassword = (0, import_utils.createPassword)(formatPassword, salt);
|
|
484
|
-
if (
|
|
485
|
-
|
|
682
|
+
if (validPassword !== authPassword) {
|
|
683
|
+
(0, import_analyticsUtils.logException)({
|
|
486
684
|
action,
|
|
487
|
-
args: { username },
|
|
488
685
|
category: eventCategory,
|
|
489
|
-
|
|
686
|
+
params: { userAccess, userId, username },
|
|
687
|
+
value: import_error.ErrorTypes.INVALID_AUTHENTICATION
|
|
490
688
|
}, context);
|
|
689
|
+
return null;
|
|
690
|
+
}
|
|
691
|
+
try {
|
|
692
|
+
console.log({ formatExpires, userAccess, userId, username });
|
|
693
|
+
const token = createToken(userId, username, userAccess, formatExpires);
|
|
694
|
+
console.log({ token });
|
|
695
|
+
return token;
|
|
696
|
+
} catch (error) {
|
|
697
|
+
(0, import_analyticsUtils.logError)({
|
|
698
|
+
action,
|
|
699
|
+
category: eventCategory,
|
|
700
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
701
|
+
}, error, context);
|
|
702
|
+
return null;
|
|
491
703
|
}
|
|
492
|
-
return createToken(userId, username, userAccess, formatExpires);
|
|
493
704
|
};
|
|
494
|
-
const signOut = async (context
|
|
705
|
+
const signOut = async (context) => {
|
|
495
706
|
const action = "signOut";
|
|
496
707
|
const { database, session: { userId: sessionId, username } } = context;
|
|
497
708
|
const userDocId = `users/${sessionId}`;
|
|
@@ -504,16 +715,14 @@ const signOut = async (context, args) => {
|
|
|
504
715
|
LIMIT 1
|
|
505
716
|
RETURN NEW`;
|
|
506
717
|
try {
|
|
507
|
-
await database.query(sessionQuery).then((cursor) => cursor.next())
|
|
508
|
-
(error) => (0, import_utils2.logError)({
|
|
509
|
-
action,
|
|
510
|
-
args: { username, userId: sessionId },
|
|
511
|
-
category: eventCategory,
|
|
512
|
-
label: import_utils2.ErrorTypes.DATABASE_ERROR
|
|
513
|
-
}, error, context)
|
|
514
|
-
);
|
|
718
|
+
await database.query(sessionQuery).then((cursor) => cursor.next());
|
|
515
719
|
} catch (error) {
|
|
516
|
-
|
|
720
|
+
await (0, import_analyticsUtils.logError)({
|
|
721
|
+
action,
|
|
722
|
+
category: eventCategory,
|
|
723
|
+
params: { userId: sessionId, username },
|
|
724
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
725
|
+
}, error, context);
|
|
517
726
|
}
|
|
518
727
|
return true;
|
|
519
728
|
};
|
|
@@ -526,13 +735,24 @@ const getActiveUserCount = (context) => {
|
|
|
526
735
|
RETURN u
|
|
527
736
|
)
|
|
528
737
|
RETURN LENGTH(docs)`;
|
|
529
|
-
return database.query(countQuery).then((cursor) => cursor.next()).catch(
|
|
530
|
-
|
|
531
|
-
|
|
532
|
-
|
|
533
|
-
|
|
534
|
-
|
|
535
|
-
|
|
738
|
+
return database.query(countQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
|
|
739
|
+
action,
|
|
740
|
+
category: eventCategory,
|
|
741
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
742
|
+
}, error, context));
|
|
743
|
+
};
|
|
744
|
+
const getUserByToken = (context, token) => {
|
|
745
|
+
const action = "getUserByToken";
|
|
746
|
+
const { database } = context;
|
|
747
|
+
const { userId } = (0, import_session.getSession)(token);
|
|
748
|
+
const userDocId = (0, import_arangodbUtils.getDocId)("users", { userId });
|
|
749
|
+
const aqlQuery = import_arangojs.aql`LET u = DOCUMENT("${userDocId}") RETURN u`;
|
|
750
|
+
return database.query(aqlQuery).then((cursor) => cursor.next()).catch((error) => (0, import_analyticsUtils.logError)({
|
|
751
|
+
action,
|
|
752
|
+
category: eventCategory,
|
|
753
|
+
params: { userId },
|
|
754
|
+
value: import_error.ErrorTypes.DATABASE_ERROR
|
|
755
|
+
}, error, context));
|
|
536
756
|
};
|
|
537
757
|
// Annotate the CommonJS export names for ESM import in node:
|
|
538
758
|
0 && (module.exports = {
|
|
@@ -542,19 +762,23 @@ const getActiveUserCount = (context) => {
|
|
|
542
762
|
createToken,
|
|
543
763
|
deactivateUser,
|
|
544
764
|
deleteUser,
|
|
765
|
+
forgotPassword,
|
|
545
766
|
getActiveUserCount,
|
|
546
767
|
getDisplayName,
|
|
547
768
|
getSessionUser,
|
|
548
769
|
getUser,
|
|
770
|
+
getUserByToken,
|
|
549
771
|
getUserOptional,
|
|
550
772
|
getUsers,
|
|
773
|
+
getUsersByConnection,
|
|
551
774
|
getUsersByLatest,
|
|
552
775
|
getUsersByReactions,
|
|
553
776
|
getUsersByTags,
|
|
554
777
|
parseUserOptions,
|
|
555
778
|
refreshSession,
|
|
779
|
+
resetPassword,
|
|
556
780
|
signIn,
|
|
557
781
|
signOut,
|
|
558
782
|
updateUser
|
|
559
783
|
});
|
|
560
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/users.ts"],
  "sourcesContent": ["/**\n * Copyright (c) 2019-Present, Nitrogen Labs, Inc.\n * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.\n */\nimport {\n  createHash,\n  createPassword,\n  parseChar,\n  parseEmail,\n  parseId,\n  parseNum,\n  parsePassword,\n  parsePhone,\n  parseUsername\n} from '@nlabs/utils';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport {EdgeCollection} from 'arangojs/collection';\nimport {ArrayCursor} from 'arangojs/cursor';\nimport {DateTime} from 'luxon';\nimport Stripe from 'stripe';\n\nimport {Config} from '../config';\nimport {ApiContext} from '../types/auth';\nimport {User} from '../types/users';\nimport {ErrorTypes, getLimit, logError, logException, selectReactionCountByType} from '../utils';\nimport {getSession, SessionError, SessionToken, setSession} from '../utils/session';\n\nconst eventCategory: string = 'users';\nconst apiVersion: any = '2020-03-02';\n\nexport interface UserOptions {\n  readonly from?: number;\n  readonly to?: number;\n  readonly username?: string;\n}\n\nexport enum UserAccess {\n  DEACTIVATED = 0,\n  ACTIVE = 1,\n  PREMIUM = 2,\n  CONTENT_ADMIN = 3,\n  ADMIN = 4\n}\n\nexport const createToken = (\n  userId: string,\n  username: string,\n  userAccess: number,\n  expires: number = 15\n): SessionToken => {\n  const now: DateTime = DateTime.local();\n  const sessionExpires: DateTime = now.plus({minutes: expires});\n  const iat: number = Math.floor(now.toSeconds());\n  const exp: number = Math.floor(sessionExpires.toSeconds());\n\n  const token = setSession({\n    exp,\n    iat,\n    username,\n    userAccess,\n    userId\n  });\n\n  return {\n    expires: sessionExpires.toMillis(),\n    issued: now.toMillis(),\n    token\n  };\n};\n\nexport const getUserOptional = (fields: string[] = []) =>\n  fields.reduce((selects: any, field: string) => {\n    if(field.includes('Count')) {\n      return selectReactionCountByType('users', 'u', field, selects);\n    }\n\n    return selects;\n  }, {objects: [], queries: []});\n\nexport const parseUserOptions = (options: UserOptions = {}) => {\n  const {\n    from = 0,\n    to = 30\n  } = options;\n\n  return {\n    ...options,\n    limit: getLimit(from, to)\n  };\n};\n\nexport const addUser = async (context: ApiContext, args: any): Promise<User> => {\n  const action: string = 'addUser';\n  const {database} = context;\n  const {user} = args;\n  const {email, password, phone, username} = user;\n  const salt: string = createHash(`${username}${password}`, null);\n  const encryptedPassword = createPassword(password, salt);\n  const formatUsername: string = parseUsername(username);\n  const formatEmail: string = parseEmail(email);\n  const formatPhone: string = parsePhone(phone);\n\n  if(!formatUsername || !password || (!formatPhone && !formatEmail)) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const filters: string[] = [`u.username == \"${formatUsername}\"`];\n\n  if(formatEmail) {\n    filters.push(`u.email == \"${formatEmail}\"`);\n  }\n\n  if(formatPhone) {\n    filters.push(`u.phone == ${formatPhone}`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    RETURN u`;\n  const existingUsers = await database.query(checkQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error) => logError({\n      action,\n      args: {username},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n\n  if(existingUsers.length) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.EXISTING_USERNAME\n    }, context);\n  }\n\n  const verifiedEmailCode: number = Math.floor(100000 + (Math.random() * 900000));\n  const verifiedPhoneCode: number = Math.floor(100000 + (Math.random() * 900000));\n\n  // Add new user properties\n  const insert: User = {\n    _key: createHash(username, null),\n    added: Date.now(),\n    email: formatEmail,\n    modified: Date.now(),\n    password: encryptedPassword,\n    phone: formatPhone,\n    salt,\n    username: formatUsername,\n    userAccess: 1,\n    verifiedEmail: false,\n    verifiedEmailCode,\n    verifiedPhone: false,\n    verifiedPhoneCode\n  };\n\n  // Add new user in ArangoDB\n  const insertQuery: AqlQuery = aql`INSERT ${insert} IN users RETURN NEW`;\n  return await database.query(insertQuery)\n    .then((cursor: ArrayCursor) => cursor.next() || {})\n    .catch((error) => logError({\n      action,\n      args: {username},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n};\n\nexport const updateUser = async (context: ApiContext, user: User): Promise<any> => {\n  const action: string = 'updateUser';\n  const {database} = context;\n  const {_key, _id, id, tags = [], userId, ...updated} = user;\n  let userDocId: string;\n\n  if(_id || id) {\n    userDocId = _id || id;\n  } else if(_key || userId) {\n    userDocId = `users/${_key || userId}`;\n  }\n\n  const userQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId})\n    UPDATE u WITH ${updated} IN users\n    RETURN NEW`;\n\n  const updatedUser = await database.query(userQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      console.log(error);\n      throw error;\n    });\n\n  const tagCollection: EdgeCollection = database.collection('isTagged');\n  await Promise.all(tags.map(({id: tagDocId, name}) => {\n    const edge = {\n      _from: tagDocId,\n      _key: createHash(`isTagged-${tagDocId}-${userDocId}`),\n      _to: userDocId,\n      added: Date.now(),\n      name\n    };\n    const tagQuery: AqlQuery = aql`FOR it IN isTagged\n      FILTER it._from == ${tagDocId} && it._to == ${userDocId} && it.name == ${name}\n      LIMIT 1\n      RETURN it`;\n\n    return database.query(tagQuery)\n      .then((cursor: ArrayCursor) => cursor.next())\n      .then((tagEdge) => {\n        if(!!tagEdge) {\n          return tagEdge;\n        }\n\n        return tagCollection.save(edge, {returnNew: true}).then(() => edge);\n      })\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        label: 'db_error'\n      }, error, context).then(() => null));\n  }));\n\n  return updatedUser;\n};\n\nexport const confirmCode = async (context: ApiContext, args): Promise<boolean> => {\n  const action: string = 'confirmEmail';\n  const {database, session: {userId: sessionId}} = context;\n  const {code, type} = args;\n  const userDocId: string = `users/${sessionId}`;\n\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId}) RETURN u`;\n\n  try {\n    return database.query(aqlQuery)\n      .then((cursor) => cursor.next() || {})\n      .then(({verifiedEmailCode, verifiedPhoneCode}: User) => {\n        switch(type) {\n          case 'email':\n            return code === verifiedEmailCode;\n          case 'phone':\n            return code === verifiedPhoneCode;\n          default:\n            return false;\n        }\n      })\n      .catch((error) => logError({\n        action,\n        args: {code, type, userId: sessionId},\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context)\n      );\n  } catch(error) {\n    return false;\n  }\n};\n\nexport const deleteUser = (context: ApiContext, args: any): Promise<any> => {\n  const action: string = 'deleteUser';\n  const {database, session: {userId: sessionId, userAccess: sessionAccess}} = context;\n  const {userId} = args;\n  const isAdmin: boolean = sessionAccess > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      args,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const aqlQuery: AqlQuery = aql`FOR u IN users\n    FILTER u._key == ${userId}\n    LIMIT 1\n    REMOVE u IN users\n    RETURN OLD`;\n\n  // Stripe\n  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion, typescript: true});\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user: User = {}) => stripeClient.customers.del(user.stripeCustomerId)\n      .then(() => stripeClient.accounts.del(user.stripeAccountId))\n      .then(() => user))\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const deactivateUser = (context: ApiContext, userId: string): Promise<User> => {\n  const action: string = 'delete';\n  const {database, session: {userId: sessionId, userAccess: sessionAccess}} = context;\n  const isAdmin: boolean = sessionAccess > 2;\n\n  if(!isAdmin && (sessionId !== userId)) {\n    logException({\n      action,\n      category: eventCategory,\n      label: 'unauthorized',\n      value: 'invalid_session'\n    }, context);\n    return null;\n  }\n\n  const updated: User = {\n    userAccess: 0\n  };\n  const aqlQuery: AqlQuery = aql`UPDATE ${userId} WITH ${updated} IN users LIMIT 1 RETURN NEW`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .catch((error: Error) => {\n      throw error;\n    });\n};\n\nexport const getDisplayName = (user: User = {}): string => {\n  const {first, last, name = '', username = ''} = user;\n  const fullname: string = ([first, last]).join(' ').trim();\n\n  if(name) {\n    return name;\n  } else if(fullname !== '') {\n    return fullname;\n  } else if(username) {\n    return username;\n  }\n\n  return 'Unknown';\n};\n\nexport const getSessionUser = (context: ApiContext): Promise<User> => {\n  const action: string = 'getSessionUser';\n  const {database, fields, session: {userId: sessionId, username}} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const aqlQuery: string = `LET u = DOCUMENT(\"users/${sessionId}\")\n  ${selectQueries.join('\\n')}\n  RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      args: {username, userId: sessionId},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUser = (context: ApiContext, args: any): Promise<User> => {\n  const action: string = 'getUser';\n  const {userId} = args;\n  const {database, fields} = context;\n  const formatUserId: string = parseId(userId);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  // Get data from database\n  const aqlQuery: string = `LET u = DOCUMENT(\"users/${formatUserId}\")\n    ${selectQueries.join('\\n')}\n    FILTER u.userAccess > 0\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.next())\n    .then((user = {}) => user)\n    .catch((error: Error) => logError({\n      action,\n      args,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => { }));\n};\n\nexport const getUsers = (context: ApiContext, options?: UserOptions): Promise<User[]> => {\n  const action: string = 'getUserList';\n  const {database, fields} = context;\n  const {limit, username} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const filterBy: string[] = ['u.userAccess > 0'];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u IN users\n    FILTER ${filterBy.join(' && ')}\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT u.username\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByReactions = (\n  context: ApiContext,\n  {reactions = [], username}: any,\n  options?: UserOptions\n): Promise<User[]> => {\n  const action: string = 'getUsersByReactions';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatReactions: string[] =  reactions.map((reactionName: string) => parseChar(reactionName, 32).toLowerCase());\n  const {limit} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [`POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u, r IN OUTBOUND \"${formatSessionId}\" hasReaction\n    OPTIONS {vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    ${filterBy.length ? `FILTER ${filterBy.join(' && ')}` : ''}\n    ${limit.aql}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  console.log({aqlQuery});\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByTags = (\n  context: ApiContext,\n  {tags, username}: any,\n  options?: UserOptions\n): Promise<User[]> => {\n  const action: string = 'getUsersByTags';\n  const {database, fields, session: {userId: sessionId}} = context;\n  const formatTags: string[] =  tags.reduce((list: string[], tagName: string) => {\n    if(tagName) {\n      list.push(parseChar(tagName, 32).toLowerCase());\n    }\n\n    return list;\n  }, []);\n  const {limit} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [`u._key != \"${sessionId}\"`];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR t IN tags\n    FILTER POSITION(${JSON.stringify(formatTags)}, LOWER(t.name))\n    FOR u, it IN OUTBOUND t isTagged\n    OPTIONS {bfs: true, uniqueVertices: \"global\", vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    ${filterBy.length ? `FILTER ${filterBy.join(' && ')}` : ''}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(u, {${selectObjects.join(', ')}})`;\n\n  console.log('getUsersByTags', aqlQuery);\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const getUsersByLatest = (context: ApiContext, {username}, options?: UserOptions): Promise<User[]> => {\n  const action: string = 'getUsersByLatest';\n  const {database, fields, session: {userId}} = context;\n  const {limit} = parseUserOptions(options);\n  const filter = ['u._id != session._id'];\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  if(username) {\n    filter.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u IN users\n    LET session = DOCUMENT(\"users/${userId}\")\n    FILTER ${filter.join(' && ')}\n    ${selectQueries.join('\\n')}\n    LET distance = DISTANCE(u.latitude || 0, u.longitude || 0, session.latitude || 0, session.longitude || 0)\n    ${limit.aql}\n    SORT distance ASC, u.added DESC\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return database.query(aqlQuery)\n    .then((cursor: ArrayCursor) => cursor.all())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => []));\n};\n\nexport const refreshSession = async (context: ApiContext, {expires, token}): Promise<SessionToken | SessionError> => {\n  const {error} = getSession(token);\n\n  if(error) {\n    return {errors: [error]};\n  }\n\n  const {exp, userId, username, userAccess} = getSession(token);\n  const now: number = Math.ceil(DateTime.local().toSeconds());\n\n  if(exp > now) {\n    return createToken(userId, username, userAccess, expires);\n  }\n\n  return {errors: ['session_expired']};\n};\n\nexport const signIn = async (context: ApiContext, args): Promise<SessionToken> => {\n  const action: string = 'signIn';\n  const {expires, password, username} = args;\n  const formatUsername: string = parseUsername(username);\n  const formatPassword: string = parsePassword(password);\n  const formatExpires: number = parseNum(expires) || 15;\n  const {database} = context;\n\n  if(!formatUsername || !formatPassword) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const checkQuery: AqlQuery = aql`FOR u IN users\n    FILTER u.username == ${formatUsername}\n    LIMIT 1\n    RETURN u`;\n  const checkUser: User = await database.query(checkQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      args: {username: formatUsername},\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => null)\n    );\n\n  if(!checkUser) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  const {_key: userId, salt, userAccess} = checkUser;\n  const authPassword: string = createPassword(formatPassword, salt);\n\n  if(checkUser.password !== authPassword) {\n    return logException({\n      action,\n      args: {username},\n      category: eventCategory,\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  return createToken(userId, username, userAccess, formatExpires);\n};\n\nexport const signOut = async (context: ApiContext, args): Promise<boolean> => {\n  const action: string = 'signOut';\n  const {database, session: {userId: sessionId, username}} = context;\n  const userDocId: string = `users/${sessionId}`;\n\n  const update = {\n    lastOnline: Date.now(),\n    sessionId: null\n  };\n  const sessionQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId})\n    UPDATE u WITH ${update} IN users\n    LIMIT 1\n    RETURN NEW`;\n\n  try {\n    await database.query(sessionQuery)\n      .then((cursor) => cursor.next())\n      .catch((error) => logError({\n        action,\n        args: {username, userId: sessionId},\n        category: eventCategory,\n        label: ErrorTypes.DATABASE_ERROR\n      }, error, context)\n      );\n  } catch(error) {\n    return false;\n  }\n\n  return true;\n};\n\nexport const getActiveUserCount = (context: ApiContext) => {\n  const action: string = 'getActiveUserCount';\n  const {database} = context;\n  const countQuery: AqlQuery = aql`LET docs = (\n    FOR u IN users\n    FILTER u.active == true\n    RETURN u\n  )\n  RETURN LENGTH(docs)`;\n\n  return database.query(countQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context)\n      .then(() => 0)\n    );\n};\n"],
  "mappings": "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",
  "names": ["import_utils", "UserAccess", "Stripe"]
}

|
|
784
|
+
//# 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} from '../utils/arangodbUtils';\nimport {getSession, isAdminUser, SessionToken, setSession} from '../utils/session';\n\nimport type {ApiContext} from '../types/auth.types';\nimport type {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: UserType): Promise<UserType> => {\n  const action = 'addUser';\n  const {database} = context;\n  const {email, password, phone, username} = parseUser(user);\n  const formatUsername: string = parseUsername(username);\n  const formatEmail: string = parseEmail(email);\n  const formatPhone: string = parsePhone(phone);\n  const formatPassword: string = parsePassword(password);\n  const hasPassword = !!formatPassword;\n  const hasUsername = !!formatUsername || !!formatPhone || !!formatEmail;\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 = formatUsername || formatPhone || formatEmail;\n  const salt: string = createHash(`${hashId}${formatPassword}`, null);\n  const encryptedPassword = createPassword(formatPassword, salt);\n  const filters: string[] = [];\n\n  if(formatUsername) {\n    filters.push(`u.username == \"${formatUsername}\"`);\n  }\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) => cursor.all());\n\n    if(existingUsers.length) {\n      return logException({\n        action,\n        category: eventCategory,\n        params: {\n          email: formatEmail,\n          phone: formatPhone,\n          username: formatUsername\n        },\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(formatUsername, null),\n    added: Date.now(),\n    email: formatEmail,\n    modified: Date.now(),\n    password: encryptedPassword,\n    phone: formatPhone,\n    salt,\n    userAccess: 1,\n    username: formatUsername,\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) => 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) => 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 forgotPassword = async (context: ApiContext, {email, phone, username}): Promise<boolean> => {\n  const action = 'forgotPassword';\n  const {app, database} = 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 database.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              app,\n              text: `Your code is ${code}`\n            });\n            update = {verifiedEmailCode: code, verifiedEmailExpires: codeExpires};\n          }\n\n          if(phone && verifiedPhone) {\n            sendSms({\n              app,\n              text: `Your code is ${code}`\n            });\n            update = {verifiedPhoneCode: code, verifiedPhoneExpires: codeExpires};\n          }\n\n          if(update.verifiedEmailCode || update.verifiedPhoneCode) {\n            const updateQuery: AqlQuery = aql`UPDATE ${userDocId} WITH ${update} IN users`;\n\n            await database.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 {database} = 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 database.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            verifiedPhoneCode,\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 === verifiedPhoneCode && 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 database.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 {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    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 {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) => 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) => 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  console.log('getSessionUser', {action, context});\n  const {database, 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 database.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 {id, userId, username} = parseUser(user);\n  const {database, fields} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  let aqlQuery: string;\n\n  console.log({id, userId, username});\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  }\n\n  return database.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 {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) => 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},\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[] = [\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 database.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 {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[] = [\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 database.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 {database, 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 database.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 {database, 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 database.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 {database} = 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 database.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      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 {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: {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 {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) => 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": ["UserAccess", "email", "phone", "password", "Stripe", "user"]
}

|