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

|
|
775
|
+
RETURN LENGTH(docs)`;
|
|
776
|
+
return useDb(databaseName).query(countQuery).then((cursor)=>cursor.next()).catch((error)=>{
|
|
777
|
+
throw logError({
|
|
778
|
+
action,
|
|
779
|
+
category: eventCategory,
|
|
780
|
+
value: ErrorTypes.DATABASE_ERROR
|
|
781
|
+
}, error, context);
|
|
782
|
+
});
|
|
783
|
+
};
|
|
784
|
+
export const getUserByToken = (context, token)=>{
|
|
785
|
+
const action = 'getUserByToken';
|
|
786
|
+
const { databaseName } = context;
|
|
787
|
+
const { userId } = getSession(token);
|
|
788
|
+
const userDocId = parseArangoId(`users/${userId}`);
|
|
789
|
+
const aqlQuery = aql`LET u = DOCUMENT("${userDocId}") RETURN u`;
|
|
790
|
+
return useDb(databaseName).query(aqlQuery).then((cursor)=>cursor.next()).catch((error)=>{
|
|
791
|
+
throw logError({
|
|
792
|
+
action,
|
|
793
|
+
category: eventCategory,
|
|
794
|
+
params: {
|
|
795
|
+
userId
|
|
796
|
+
},
|
|
797
|
+
value: ErrorTypes.DATABASE_ERROR
|
|
798
|
+
}, error, context);
|
|
799
|
+
});
|
|
800
|
+
};
|
|
801
|
+
|
|
802
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/nitrog7/Development/reaktor/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 {parseNum} from '@nlabs/utils/parsers/numbers';\nimport {\n  createHash,\n  createPassword,\n  parseArangoId,\n  parseChar,\n  parseEmail,\n  parsePassword,\n  parsePhone,\n  parseUsername\n} from '@nlabs/utils/parsers/strings';\nimport {aql} from 'arangojs';\nimport {DateTime} from 'luxon';\nimport Stripe from 'stripe';\n\nimport {parseUser} from '../adapters/userAdapter.js';\nimport {Config} from '../config.js';\nimport {ErrorTypes, type SessionError} from '../types/error.types.js';\nimport {logError, logException} from '../utils/analyticsUtils.js';\nimport {getDocId, getLimit, selectReactionCountByType, useDb} from '../utils/arangodbUtils.js';\nimport {detectLanguage} from '../utils/languageDetection.js';\nimport {getSession, setSession, type SessionToken} from '../utils/sessionUtils.js';\nimport {sendEmail} from './email.js';\nimport {sendSms} from './sms.js';\n\nimport type {AqlQuery} from 'arangojs/aql';\nimport type {EdgeCollection} from 'arangojs/collections';\nimport type {ApiContext} from '../types/auth.types.js';\nimport type {UserInput, UserType} from '../types/users.types.js';\n\nconst eventCategory = 'users';\nconst STRIPE_API_VERSION = '2025-12-15.clover';\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 = 0,\n  expiresInMinutes: number = 15\n): SessionToken => {\n  const now: DateTime = DateTime.local();\n  const sessionExpires: DateTime = now.plus({minutes: expiresInMinutes});\n  const iat: number = Math.floor(now.toSeconds());\n  const exp: number = Math.floor(sessionExpires.toSeconds());\n  const token = setSession({\n    exp,\n    iat,\n    userAccess,\n    userId,\n    username\n  });\n\n  return {\n    expires: sessionExpires.toMillis(),\n    issued: now.toMillis(),\n    token,\n    userId,\n    username\n  };\n};\n\ninterface SelectAccumulator {\n  objects: string[];\n  queries: string[];\n}\n\nexport const getUserOptional = (fields: string[] = []): SelectAccumulator =>\n  fields.reduce((selects: SelectAccumulator, field: string) => {\n    if(field.includes('Count')) {\n      return selectReactionCountByType('users', 'u', field, selects);\n    }\n\n    return selects;\n  }, {objects: [], queries: []});\n\nexport const parseUserOptions = (options: UserOptions = {}) => {\n  const {\n    from = 0,\n    to = 30\n  } = options;\n  const limit = getLimit(from, to);\n\n  return {\n    ...options,\n    limit\n  };\n};\n\nexport const addUser = async (context: ApiContext, user: UserInput): Promise<UserType | null> => {\n  const action = 'addUser';\n  const {databaseName, languageContext} = context;\n  const {confirm: _confirm, ...newUser} = user;\n  const {email, password, phone, username, userId, _key, _id, ...insertUser} = parseUser(newUser);\n  const hasPassword = !!password;\n  const hasUsername = !!username || !!phone || !!email;\n\n  if(!hasPassword || !hasUsername) {\n    logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n\n    return null;\n  }\n\n  const hashId = username || phone || email;\n  const salt: string = createHash(`${hashId}${password}`, '');\n  const encryptedPassword = createPassword(password, salt);\n  const filters: string[] = [];\n\n  if(username) {\n    filters.push(`u.username == \"${username}\"`);\n  }\n\n  if(email) {\n    filters.push(`u.email == \"${email}\"`);\n  }\n\n  if(phone) {\n    filters.push(`u.phone == ${phone}`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    const existingUsers = await useDb(databaseName).query(checkQuery).then((cursor) => cursor.all());\n\n    if(existingUsers.length) {\n      throw logException({\n        action,\n        category: eventCategory,\n        params: {\n          email,\n          phone,\n          username\n        },\n        value: ErrorTypes.EXISTING_ITEM\n      }, context);\n    }\n  } catch(error) {\n    throw logError({\n      action,\n      category: eventCategory,\n      params: {email, phone, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n\n  const phoneCountryCode = phone?.replace(/\\D/g, '').substring(0, 3);\n  const locale = detectLanguage({\n    ...languageContext,\n    phoneCountryCode: phoneCountryCode || '',\n    userPreference: insertUser.locale || 'en'\n  });\n  const verifiedEmailCode: number = Math.floor(100000 + (Math.random() * 900000));\n  const verifiedSmsCode: number = Math.floor(100000 + (Math.random() * 900000));\n\n  const insert: UserType = {\n    ...insertUser,\n    _key: createHash(username),\n    added: Date.now(),\n    email,\n    locale,\n    modified: Date.now(),\n    password: encryptedPassword,\n    phone,\n    salt,\n    userAccess: 1,\n    username,\n    verifiedEmail: false,\n    verifiedEmailCode,\n    verifiedPhone: false,\n    verifiedSmsCode\n  };\n\n  const insertQuery: AqlQuery = aql`INSERT ${insert} IN users RETURN NEW`;\n\n  return await useDb(databaseName).query(insertQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => {\n      throw logError({\n        action,\n        category: eventCategory,\n        params: {username},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n    });\n};\n\nexport const updateUser = async (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'updateUser';\n  const {databaseName} = context;\n  const updatedUser = parseUser(user);\n  const {_key, _id, tags = [], ...update} = updatedUser;\n  const {id} = updatedUser;\n\n  const userQuery: AqlQuery = aql`LET u = DOCUMENT(${id})\n    UPDATE u WITH ${update} IN users\n    RETURN NEW`;\n  try {\n    const database = useDb(databaseName);\n    const updatedUser = await database.query(userQuery).then((cursor) => cursor.next());\n    const tagCollection: EdgeCollection = database.collection('isTagged');\n\n    await Promise.all(tags.map(({id: tagDocId, name}) => {\n      const tagQuery: AqlQuery = aql`FOR it IN isTagged\n        FILTER it._from == ${tagDocId} && it._to == ${id} && it.name == ${name}\n        LIMIT 1\n        RETURN it`;\n\n      return database.query(tagQuery)\n        .then((cursor) => cursor.next())\n        .then((tagEdge) => {\n          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    throw logError({\n      action,\n      category: eventCategory,\n      params: {user},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n};\n\nexport const forgotPassword = async (context: ApiContext, {email, phone, username}): Promise<boolean> => {\n  const action = 'forgotPassword';\n  const {databaseName} = context;\n  const aqlQuery: AqlQuery = aql`FOR u IN users\n    FILTER u.email == ${email} || u.phone == ${phone} || u.username == ${username}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    return await useDb(databaseName).query(aqlQuery)\n      .then(async (cursor) => {\n        const user = cursor.next();\n\n        if(user) {\n          const {email, phone, verifiedEmail, verifiedPhone} = user as UserType;\n          const codeExpires = 1000 * 60 * 15; // 15 minutes\n          const code = Math.floor(100000 + (Math.random() * 900000));\n          const userDocId = getDocId('users', user);\n          let update;\n\n          if(email && verifiedEmail) {\n            sendEmail({\n              context,\n              text: `Your code is ${code}`\n            });\n            update = {verifiedEmailCode: code, verifiedEmailExpires: codeExpires};\n          }\n\n          if(phone && verifiedPhone) {\n            sendSms({\n              context,\n              text: `Your code is ${code}`\n            });\n            update = {verifiedPhoneExpires: codeExpires, verifiedSmsCode: code};\n          }\n\n          if(update.verifiedEmailCode || update.verifiedSmsCode) {\n            const updateQuery: AqlQuery = aql`UPDATE ${userDocId} WITH ${update} IN users`;\n\n            await useDb(databaseName).query(updateQuery);\n\n            return true;\n          }\n\n          return false;\n        }\n\n        return false;\n      });\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {email, phone, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\n  }\n};\n\nexport const resetPassword = async (\n  context: ApiContext,\n  {\n    code,\n    password,\n    type,\n    username\n  }: {\n    code: number;\n    password: string;\n    type: 'phone' | 'email';\n    username: string;\n  }\n): Promise<boolean> => {\n  const action = 'resetPassword';\n  const {databaseName} = context;\n  const formatPassword: string = parsePassword(password);\n  const aqlQuery: AqlQuery = aql`FOR u IN users\n    FILTER u.username == ${username}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    return await useDb(databaseName).query(aqlQuery)\n      .then(async (cursor) => {\n        const user = cursor.next();\n\n        if(user) {\n          const {\n            _id: userDocId,\n            salt,\n            verifiedEmailCode,\n            verifiedEmailExpires = 0,\n            verifiedSmsCode,\n            verifiedPhoneExpires = 0\n          } = user as UserType;\n          const now = Date.now();\n          let update;\n\n          switch(type) {\n            case 'email':\n              if(code === verifiedEmailCode && verifiedEmailExpires > now) {\n                const password: string = createPassword(formatPassword, salt);\n                update = {password};\n              }\n              break;\n            case 'phone':\n              if(code === verifiedSmsCode && verifiedPhoneExpires > now) {\n                const password: string = createPassword(formatPassword, salt);\n                update = {password};\n              }\n              break;\n            default:\n              return false;\n          }\n\n          if(update) {\n            const updateQuery: AqlQuery = aql`UPDATE ${userDocId} WITH ${update} IN users`;\n\n            await useDb(databaseName).query(updateQuery);\n\n            return true;\n          }\n        }\n\n        return false;\n      });\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\n  }\n};\n\nexport const confirmCode = async (\n  context: ApiContext,\n  {\n    code,\n    type,\n    value\n  }: {\n    code: number;\n    type: 'sms' | 'email';\n    value: string;\n  }\n): Promise<boolean> => {\n  const action = 'confirmCode';\n  const {databaseName} = context;\n  const isSms = type === 'sms';\n  const formattedValue = isSms ? parsePhone(value) : parseEmail(value);\n  const filterByType = isSms\n    ? `FILTER u.phone == \"${formattedValue}\" && u.verifiedSmsCode == ${code}`\n    : `FILTER u.email == \"${formattedValue}\" && u.verifiedEmailCode == ${code}`;\n  const aqlQuery: string = `FOR u IN users\n    ${filterByType}\n    LIMIT 1\n    RETURN u`;\n\n  try {\n    return await useDb(databaseName).query(aqlQuery)\n      .then((cursor) => cursor.next())\n      .then((user) => {\n        if(user) {\n          const updatedUser = isSms\n            ? {verifiedPhone: true, verifiedSmsCode: 0}\n            : {verifiedEmail: true, verifiedEmailCode: 0};\n          const aqlQuery: AqlQuery = aql`UPDATE ${user._key} WITH ${updatedUser} IN users`;\n\n          return useDb(databaseName).query(aqlQuery)\n            .then(() => true)\n            .catch((error: Error) => {\n              logError({\n                action,\n                category: eventCategory,\n                params: {code, type, value: formattedValue},\n                value: ErrorTypes.DATABASE_ERROR\n              }, error, context);\n\n              return false;\n            });\n        }\n\n        return false;\n      });\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {code, type, value: formattedValue},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\n  }\n};\n\nexport const deleteUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'deleteUser';\n  const {databaseName} = context;\n  const {id} = parseUser(user);\n\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(${id})\n    REMOVE u IN users\n    RETURN OLD`;\n\n  const stripeClient = new Stripe(Config.get('stripe.token'), {apiVersion: STRIPE_API_VERSION, typescript: true});\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .then((deletedUser) => stripeClient.customers.del(deletedUser?.stripeCustomerId)\n      .then(() => stripeClient.accounts.del(deletedUser?.stripeAccountId))\n      .then(() => deletedUser))\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {id},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const deactivateUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'delete';\n  const {databaseName} = context;\n  const {id} = parseUser(user);\n  const updated: UserType = {\n    userAccess: 0\n  };\n  const aqlQuery: AqlQuery = aql`UPDATE ${id} WITH ${updated} IN users LIMIT 1 RETURN NEW`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      throw logError({\n      action,\n      category: eventCategory,\n      params: {id},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  });\n};\n\nexport const getDisplayName = (user: UserType): string => {\n  const {first, last, name = '', username = ''} = user;\n  const fullname = ([first, last]).join(' ').trim();\n\n  if(name) {\n    return name;\n  } else if(fullname !== '') {\n    return fullname;\n  } else if(username) {\n    return username;\n  }\n\n  return 'Unknown';\n};\n\nexport const getSessionUser = (context: ApiContext): Promise<UserType> => {\n  const action = 'getSessionUser';\n  const {databaseName, fields, session: {userId: sessionId, username} = {}} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const formatSessionId = parseArangoId(`users/${sessionId}`);\n\n  const aqlQuery: string = `LET u = DOCUMENT(\"${formatSessionId}\")\n  ${selectQueries.join('\\n')}\n  RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.next() as unknown as UserType)\n    .catch((error: Error) => {\n      throw logError({\n        action,\n        category: eventCategory,\n        params: {userId: sessionId, username},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n    });\n};\n\nexport const getUser = (context: ApiContext, user: UserType): Promise<UserType> => {\n  const action = 'getUser';\n  const {email, id, phone, userId, username} = parseUser(user);\n  const {databaseName, fields} = context;\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  let aqlQuery: string = '';\n\n  if(id) {\n    aqlQuery = `LET u = DOCUMENT(\"${id}\")\n    ${selectQueries.join('\\n')}\n    FILTER u.userAccess > 0\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  } else if(username) {\n    aqlQuery = `FOR u IN users\n    FILTER u.username == \"${username}\"\n    ${selectQueries.join('\\n')}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  } else if(email) {\n    aqlQuery = `FOR u IN users\n    FILTER u.email == \"${email}\"\n    ${selectQueries.join('\\n')}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  } else if(phone) {\n    aqlQuery = `FOR u IN users\n    FILTER u.phone == \"${phone}\"\n    ${selectQueries.join('\\n')}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n  }\n\n  return useDb(databaseName)\n    .query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .then((user) => user)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {id, userId, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getUsers = (context: ApiContext, options?: UserOptions): Promise<UserType[]> => {\n  const action = 'getUserList';\n  const {databaseName, fields} = context;\n  const {limit, username} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const filterBy: string[] = ['u.userAccess > 0'];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  const aqlQuery: string = `FOR u IN users\n    FILTER ${filterBy.join(' && ')}\n    ${selectQueries.join('\\n')}\n    ${limit.aql}\n    SORT u.username\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByReactions = (\n  context: ApiContext,\n  {reactions = [], username}: {reactions?: string[]; username?: string},\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = 'getUsersByReactions';\n  const {databaseName, fields, session: {userId: sessionId} = {}} = context;\n  const formatReactions: string[] = reactions.map((reactionName: string) => parseChar(reactionName, 32).toLowerCase());\n  const {limit} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [\n    'u.userAccess > 0',\n    `POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`\n  ];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  const aqlQuery: string = `FOR u, r IN OUTBOUND \"${formatSessionId}\" hasReaction\n    OPTIONS {vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByTags = (\n  context: ApiContext,\n  {tags, username},\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = 'getUsersByTags';\n  const {databaseName, fields, session: {userId: sessionId} = {}} = context;\n  const formatTags: string[] = tags?.reduce((list: string[], tagName: string) => {\n    if(tagName) {\n      list.push(parseChar(tagName, 32).toLowerCase());\n    }\n\n    return list;\n  }, []);\n  const {limit} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const formatUsername: string = parseUsername(username);\n  const filterBy: string[] = [\n    `u._key != \"${sessionId}\"`,\n    'u.userAccess > 0'\n  ];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${formatUsername}\")`);\n  }\n\n  const aqlQuery: string = `FOR t IN tags\n    FILTER POSITION(${JSON.stringify(formatTags)}, LOWER(t.name))\n    FOR u, it IN OUTBOUND t isTagged\n    OPTIONS {bfs: true, uniqueVertices: \"global\", vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByLatest = (context: ApiContext, {username}, options?: UserOptions): Promise<UserType[]> => {\n  const action = 'getUsersByLatest';\n  const {databaseName, fields, session: {userId} = {}} = context;\n  const {limit} = parseUserOptions(options);\n  const filter = [\n    'u._id != session._id',\n    'u.userAccess > 0'\n  ];\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n\n  if(username) {\n    filter.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  // Get data from database\n  const aqlQuery: string = `FOR u IN users\n    LET session = DOCUMENT(\"users/${userId}\")\n    FILTER ${filter.join(' && ')}\n    ${selectQueries.join('\\n')}\n    LET distance = DISTANCE(u.latitude || 0, u.longitude || 0, session.latitude || 0, session.longitude || 0)\n    ${limit.aql}\n    SORT distance ASC, u.added DESC\n    RETURN MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const getUsersByConnection = (\n  context: ApiContext,\n  {userId}: UserType,\n  options?: UserOptions\n): Promise<UserType[]> => {\n  const action = 'getUsersByConnection';\n  const {databaseName, fields} = context;\n  const {limit, username} = parseUserOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getUserOptional(fields);\n  const formatUserId: string = parseArangoId(`users/${userId}`);\n  const filterBy: string[] = [\n    'u.userAccess > 0'\n  ];\n\n  if(username) {\n    filterBy.push(`CONTAINS(u.username, \"${parseUsername(username)}\")`);\n  }\n\n  const aqlQuery: string = `FOR cu IN users\n    LET session = DOCUMENT(\"${formatUserId}\")\n    FOR u, connection IN OUTBOUND cu hasConnection\n    OPTIONS {bfs: true, uniqueVertices: \"global\", vertexCollections: \"users\"}\n    ${selectQueries.join('\\n')}\n    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN DISTINCT MERGE(u, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQuery)\n    .then((cursor) => cursor.all() as unknown as UserType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [] as UserType[];\n    });\n};\n\nexport const refreshSession = ({expires, token}): SessionToken | SessionError => {\n  try {\n    const {userId, username, userAccess} = getSession(token);\n    return createToken(userId, username, userAccess, expires);\n  } catch(error) {\n    throw error; // Re-throw the error from getSession\n  }\n};\n\nexport const signIn = async (context: ApiContext, args): Promise<SessionToken | null> => {\n  const action = 'signIn';\n  const {databaseName} = context;\n  const {email, expires, password, phone, username} = args;\n  const formatEmail: string = parseEmail(email);\n  const formatUsername: string = parseUsername(username);\n  const formatPassword: string = parsePassword(password);\n  const formatPhone: string = parsePhone(phone);\n  const formatExpires: number = parseNum(expires) || 15;\n\n  if((!formatUsername && !formatEmail && !formatPhone) || !formatPassword) {\n    throw logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_ARGUMENTS\n    }, context);\n  }\n\n  const filters: string[] = [];\n\n  if(formatEmail) {\n    filters.push(`u.email == \"${formatEmail}\"`);\n  }\n\n  if(formatPhone) {\n    filters.push(`u.phone == ${formatPhone}`);\n  }\n\n  if(formatUsername) {\n    filters.push(`u.username == \"${formatUsername}\"`);\n  }\n\n  const checkQuery: string = `FOR u IN users\n    FILTER ${filters.join(' || ')}\n    LIMIT 1\n    RETURN u`;\n\n  let checkUser: UserType;\n\n  try {\n    checkUser = await useDb(databaseName).query(checkQuery).then((cursor) => cursor.next());\n  } catch(error) {\n    throw logError({\n      action,\n      category: eventCategory,\n      params: {username: formatUsername},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n\n  if(!checkUser) {\n    throw logException({\n      action,\n      category: eventCategory,\n      params: {username},\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  const {_key: userId, password: validPassword, salt, userAccess} = checkUser as UserType;\n  const authPassword: string = createPassword(formatPassword, salt);\n\n  console.log({\n    createPassword,\n    authPassword,\n    validPassword,\n    formatPassword,\n    salt\n  });\n\n  if(validPassword !== authPassword) {\n    throw logException({\n      action,\n      category: eventCategory,\n      params: {userAccess, userId, username},\n      value: ErrorTypes.INVALID_AUTHENTICATION\n    }, context);\n  }\n\n  try {\n    const token = createToken(userId || '', username || '', userAccess, formatExpires);\n\n    return token;\n  } catch(error) {\n    throw logError({\n      action,\n      category: eventCategory,\n      params: {userId, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n};\n\nexport const signOut = async (context: ApiContext): Promise<boolean> => {\n  const action = 'signOut';\n  const {databaseName, session: {userId: sessionId, username} = {}} = context;\n  const userDocId: string = `users/${sessionId}`;\n\n  const update = {\n    lastOnline: Date.now(),\n    sessionId: null\n  };\n  const sessionQuery: AqlQuery = aql`LET u = DOCUMENT(${userDocId})\n    UPDATE u WITH ${update} IN users\n    LIMIT 1\n    RETURN NEW`;\n\n  try {\n    await useDb(databaseName).query(sessionQuery).then((cursor) => cursor.next());\n  } catch(error) {\n    logError({\n      action,\n      category: eventCategory,\n      params: {userId: sessionId, username},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n\n    return false;\n  }\n\n  return true;\n};\n\nexport const getActiveUserCount = (context: ApiContext): Promise<number> => {\n  const action = 'getActiveUserCount';\n  const {databaseName} = context;\n  const countQuery: AqlQuery = aql`LET docs = (\n    FOR u IN users\n    FILTER u.active == true\n    RETURN u\n  )\n  RETURN LENGTH(docs)`;\n\n  return useDb(databaseName).query(countQuery)\n    .then((cursor) => cursor.next())\n    .catch((error) => {\n      throw logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n    });\n};\n\nexport const getUserByToken = (context: ApiContext, token: string): Promise<UserType> => {\n  const action = 'getUserByToken';\n  const {databaseName} = context;\n  const {userId} = getSession(token);\n  const userDocId = parseArangoId(`users/${userId}`);\n  const aqlQuery: AqlQuery = aql`LET u = DOCUMENT(\"${userDocId}\") RETURN u`;\n\n  return useDb(databaseName)\n    .query(aqlQuery)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => {\n      throw logError({\n        action,\n        category: eventCategory,\n        params: {userId},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n    });\n};"],"names":["parseNum","createHash","createPassword","parseArangoId","parseChar","parseEmail","parsePassword","parsePhone","parseUsername","aql","DateTime","Stripe","parseUser","Config","ErrorTypes","logError","logException","getDocId","getLimit","selectReactionCountByType","useDb","detectLanguage","getSession","setSession","sendEmail","sendSms","eventCategory","STRIPE_API_VERSION","UserAccess","createToken","userId","username","userAccess","expiresInMinutes","now","local","sessionExpires","plus","minutes","iat","Math","floor","toSeconds","exp","token","expires","toMillis","issued","getUserOptional","fields","reduce","selects","field","includes","objects","queries","parseUserOptions","options","from","to","limit","addUser","context","user","action","databaseName","languageContext","confirm","_confirm","newUser","email","password","phone","_key","_id","insertUser","hasPassword","hasUsername","category","params","value","INVALID_ARGUMENTS","hashId","salt","encryptedPassword","filters","push","checkQuery","join","existingUsers","query","then","cursor","all","length","EXISTING_ITEM","error","DATABASE_ERROR","phoneCountryCode","replace","substring","locale","userPreference","verifiedEmailCode","random","verifiedSmsCode","insert","added","Date","modified","verifiedEmail","verifiedPhone","insertQuery","next","catch","updateUser","updatedUser","tags","update","id","userQuery","database","tagCollection","collection","Promise","map","tagDocId","name","tagQuery","tagEdge","edge","_from","_to","save","returnNew","forgotPassword","aqlQuery","codeExpires","code","userDocId","text","verifiedEmailExpires","verifiedPhoneExpires","updateQuery","resetPassword","type","formatPassword","confirmCode","isSms","formattedValue","filterByType","deleteUser","stripeClient","get","apiVersion","typescript","deletedUser","customers","del","stripeCustomerId","accounts","stripeAccountId","deactivateUser","updated","getDisplayName","first","last","fullname","trim","getSessionUser","session","sessionId","selectObjects","selectQueries","formatSessionId","getUser","getUsers","filterBy","getUsersByReactions","reactions","formatReactions","reactionName","toLowerCase","formatUsername","JSON","stringify","getUsersByTags","formatTags","list","tagName","getUsersByLatest","filter","getUsersByConnection","formatUserId","refreshSession","signIn","args","formatEmail","formatPhone","formatExpires","checkUser","INVALID_AUTHENTICATION","validPassword","authPassword","console","log","signOut","lastOnline","sessionQuery","getActiveUserCount","countQuery","getUserByToken"],"mappings":"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"}
|