@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/payments.js
CHANGED
|
@@ -1,32 +1,449 @@
|
|
|
1
|
-
|
|
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, parseChar, parseId, parseString, parseVarChar } from '@nlabs/utils/parsers/strings';
|
|
6
|
+
import { aql } from 'arangojs';
|
|
7
|
+
import { ErrorTypes } from '../types/error.types.js';
|
|
8
|
+
import { logError, UserError } from '../utils/analyticsUtils.js';
|
|
9
|
+
import { useDb } from '../utils/arangodbUtils.js';
|
|
10
|
+
import { getStripeClient } from '../utils/stripeUtils.js';
|
|
11
|
+
import { getUser } from './users.js';
|
|
12
|
+
const eventCategory = 'payments';
|
|
13
|
+
export const addCustomerAccount = (context)=>{
|
|
14
|
+
const action = 'addCustomerAccount';
|
|
15
|
+
const { databaseName, session: { userId: sessionId, username } = {} } = context;
|
|
16
|
+
const stripeClient = getStripeClient();
|
|
17
|
+
return stripeClient.customers.create({
|
|
18
|
+
metadata: {
|
|
19
|
+
userId: sessionId,
|
|
20
|
+
username
|
|
21
|
+
}
|
|
22
|
+
}).then((customer)=>{
|
|
23
|
+
// Create session
|
|
24
|
+
const now = Date.now();
|
|
25
|
+
const update = {
|
|
26
|
+
modified: now,
|
|
27
|
+
stripeCustomerId: customer.id
|
|
28
|
+
};
|
|
29
|
+
const aqlQry = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;
|
|
30
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((updatedUser)=>!!updatedUser).catch((error)=>logError({
|
|
31
|
+
action,
|
|
32
|
+
category: eventCategory,
|
|
33
|
+
label: ErrorTypes.DATABASE_ERROR
|
|
34
|
+
}, error, context).then(()=>null));
|
|
35
|
+
});
|
|
36
|
+
};
|
|
37
|
+
export const addBankAccount = (context, bankAccount)=>{
|
|
38
|
+
const action = 'addPaymentAccountBank';
|
|
39
|
+
const { databaseName, session: { userId: sessionId } = {} } = context;
|
|
40
|
+
// Params
|
|
41
|
+
const { accountNumber, fullName, routing } = bankAccount;
|
|
42
|
+
const formatAccount = parseString(accountNumber, 32);
|
|
43
|
+
if (formatAccount === '') {
|
|
44
|
+
throw new UserError('required_account_number');
|
|
45
|
+
}
|
|
46
|
+
const formatFullName = parseVarChar(fullName, 128);
|
|
47
|
+
if (formatFullName === '') {
|
|
48
|
+
throw new UserError('required_full_name');
|
|
49
|
+
}
|
|
50
|
+
const formatRouting = parseString(routing, 32);
|
|
51
|
+
if (formatRouting === '') {
|
|
52
|
+
throw new UserError('required_routing_number');
|
|
53
|
+
}
|
|
54
|
+
return getUser(context, {
|
|
55
|
+
userId: sessionId
|
|
56
|
+
}).then((user)=>{
|
|
57
|
+
const { stripeAccountId } = user;
|
|
58
|
+
const stripeClient = getStripeClient();
|
|
59
|
+
// Create a token first, then use the token as the source
|
|
60
|
+
return stripeClient.tokens.create({
|
|
61
|
+
bank_account: {
|
|
62
|
+
account_holder_name: formatFullName,
|
|
63
|
+
account_holder_type: 'individual',
|
|
64
|
+
account_number: formatAccount,
|
|
65
|
+
country: 'US',
|
|
66
|
+
currency: 'USD',
|
|
67
|
+
routing_number: formatRouting
|
|
68
|
+
}
|
|
69
|
+
}).then((token)=>stripeClient.customers.createSource(stripeAccountId, {
|
|
70
|
+
source: token.id
|
|
71
|
+
})).then((account)=>{
|
|
72
|
+
// Use type assertion for card properties
|
|
73
|
+
// const cardSource = account as unknown as Card;
|
|
74
|
+
// const brand = cardSource.brand || '';
|
|
75
|
+
// const cvcCheck = cardSource.cvc_check || '';
|
|
76
|
+
// const last4 = cardSource.last4 || '';
|
|
77
|
+
// Create session
|
|
78
|
+
const now = Date.now();
|
|
79
|
+
const update = {
|
|
80
|
+
bankAccount,
|
|
81
|
+
bankFullName: formatFullName,
|
|
82
|
+
bankId: account.id,
|
|
83
|
+
bankRouting: formatRouting,
|
|
84
|
+
modified: now
|
|
85
|
+
};
|
|
86
|
+
const aqlQry = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;
|
|
87
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((updatedUser)=>updatedUser);
|
|
88
|
+
}).catch((error)=>{
|
|
89
|
+
const msg = error.message;
|
|
90
|
+
if (msg === 'A bank account with that routing number and account number ' + 'already exists for this customer.') {
|
|
91
|
+
return logError({
|
|
92
|
+
action,
|
|
93
|
+
category: eventCategory,
|
|
94
|
+
label: 'bank_account_exists'
|
|
95
|
+
}, error, context).then(()=>null);
|
|
96
|
+
}
|
|
97
|
+
return logError({
|
|
98
|
+
action,
|
|
99
|
+
category: eventCategory,
|
|
100
|
+
label: 'payment_error'
|
|
101
|
+
}, error, context).then(()=>null);
|
|
102
|
+
});
|
|
103
|
+
});
|
|
104
|
+
};
|
|
105
|
+
export const addCreditCard = (context, card)=>{
|
|
106
|
+
const action = 'addCreditCard';
|
|
107
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
108
|
+
return getUser(context, {
|
|
109
|
+
userId: sessionId
|
|
110
|
+
}).then((user)=>{
|
|
111
|
+
// User
|
|
112
|
+
const { stripeAccountId } = user;
|
|
113
|
+
// Card
|
|
114
|
+
const { accountNumber, city, country, cvc, expMonth, expYear, fullName, street1, street2, state, zip } = card;
|
|
115
|
+
const formatNumber = parseNum(accountNumber, 16);
|
|
116
|
+
if (!formatNumber) {
|
|
117
|
+
throw new UserError('required_credit_card_number');
|
|
118
|
+
}
|
|
119
|
+
const formatExpMonth = parseNum(expMonth, 2);
|
|
120
|
+
if (!formatExpMonth) {
|
|
121
|
+
throw new UserError('required_credit_card_exp_month');
|
|
122
|
+
}
|
|
123
|
+
const formatExpYear = parseNum(expYear, 2);
|
|
124
|
+
if (!formatExpYear) {
|
|
125
|
+
throw new UserError('required_credit_card_exp_year');
|
|
126
|
+
}
|
|
127
|
+
const formatCvc = parseNum(cvc, 3);
|
|
128
|
+
// Address
|
|
129
|
+
const paymentCard = {};
|
|
130
|
+
const formatCity = parseVarChar(city, 32);
|
|
131
|
+
if (formatCity) {
|
|
132
|
+
paymentCard.city = formatCity;
|
|
133
|
+
}
|
|
134
|
+
const formatCountry = parseChar(country, 2);
|
|
135
|
+
if (formatCountry) {
|
|
136
|
+
paymentCard.country = formatCountry;
|
|
137
|
+
}
|
|
138
|
+
const formatFullName = parseVarChar(fullName, 32);
|
|
139
|
+
if (formatFullName) {
|
|
140
|
+
paymentCard.fullName = formatFullName;
|
|
141
|
+
}
|
|
142
|
+
const formatStreet1 = parseVarChar(street1, 32);
|
|
143
|
+
if (formatStreet1) {
|
|
144
|
+
paymentCard.street1 = formatStreet1;
|
|
145
|
+
}
|
|
146
|
+
const formatStreet2 = parseVarChar(street2, 32);
|
|
147
|
+
if (formatStreet2) {
|
|
148
|
+
paymentCard.street2 = formatStreet2;
|
|
149
|
+
}
|
|
150
|
+
const formatState = parseChar(state, 2);
|
|
151
|
+
if (formatState) {
|
|
152
|
+
paymentCard.state = formatState;
|
|
153
|
+
}
|
|
154
|
+
const formatZip = parseVarChar(zip, 10);
|
|
155
|
+
if (formatZip) {
|
|
156
|
+
paymentCard.zip = formatZip;
|
|
157
|
+
}
|
|
158
|
+
const stripeClient = getStripeClient();
|
|
159
|
+
// Create a token first, then use the token as the source
|
|
160
|
+
return stripeClient.tokens.create({
|
|
161
|
+
card: {
|
|
162
|
+
address_city: formatCity,
|
|
163
|
+
address_country: formatCountry,
|
|
164
|
+
address_line1: formatStreet1,
|
|
165
|
+
address_line2: formatStreet2,
|
|
166
|
+
address_state: formatState,
|
|
167
|
+
address_zip: formatZip,
|
|
168
|
+
cvc: formatCvc.toString(),
|
|
169
|
+
exp_month: formatExpMonth.toString(),
|
|
170
|
+
exp_year: formatExpYear.toString(),
|
|
171
|
+
name: fullName,
|
|
172
|
+
number: formatNumber.toString()
|
|
173
|
+
}
|
|
174
|
+
}).then((token)=>stripeClient.customers.createSource(stripeAccountId, {
|
|
175
|
+
source: token.id
|
|
176
|
+
})).then((newSource)=>{
|
|
177
|
+
// Use type assertion for card properties
|
|
178
|
+
const cardSource = newSource;
|
|
179
|
+
const brand = cardSource.brand || '';
|
|
180
|
+
const cvcCheck = cardSource.cvc_check || '';
|
|
181
|
+
const last4 = cardSource.last4 || '';
|
|
182
|
+
// Create session
|
|
183
|
+
const now = Date.now();
|
|
184
|
+
const insert = {
|
|
185
|
+
...paymentCard,
|
|
186
|
+
_key: createHash(`user-payment-${sessionId}`),
|
|
187
|
+
accountNumber: last4,
|
|
188
|
+
added: now,
|
|
189
|
+
brand,
|
|
190
|
+
cvcCheck,
|
|
191
|
+
expMonth,
|
|
192
|
+
expYear,
|
|
193
|
+
modified: now,
|
|
194
|
+
userId: sessionId
|
|
195
|
+
};
|
|
196
|
+
const insertAqlQry = aql`INSERT ${insert} IN creditCards RETURN NEW`;
|
|
197
|
+
return useDb(databaseName).query(insertAqlQry).then((cursor)=>cursor.next()).then((newCard)=>{
|
|
198
|
+
if (newCard) {
|
|
199
|
+
// Add linked edge
|
|
200
|
+
const { _id: cardId, _key: cardKey } = card;
|
|
201
|
+
const edgeCollection = useDb(databaseName).collection('hasPayment');
|
|
202
|
+
const edgeId = createHash(`payment-${cardKey}`);
|
|
203
|
+
const edge = {
|
|
204
|
+
_from: `users/${sessionId}`,
|
|
205
|
+
_key: edgeId,
|
|
206
|
+
_to: cardId
|
|
207
|
+
};
|
|
208
|
+
return edgeCollection.save(edge, {
|
|
209
|
+
returnNew: true
|
|
210
|
+
}).then(()=>card);
|
|
211
|
+
}
|
|
212
|
+
return newCard;
|
|
213
|
+
}).catch((error)=>logError({
|
|
214
|
+
action,
|
|
215
|
+
category: eventCategory,
|
|
216
|
+
label: 'payment_error'
|
|
217
|
+
}, error, context).then(()=>null));
|
|
218
|
+
});
|
|
219
|
+
});
|
|
220
|
+
};
|
|
221
|
+
export const updateCreditCard = (context, card)=>{
|
|
222
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
223
|
+
const { city, country, expMonth, expYear, fullName, id, street1, state, zip } = card;
|
|
224
|
+
const formatId = parseId(id);
|
|
225
|
+
if (formatId) {
|
|
226
|
+
throw new UserError('required_credit_card_id');
|
|
227
|
+
}
|
|
228
|
+
const paymentCard = {};
|
|
229
|
+
const formatExpMonth = parseNum(expMonth, 2);
|
|
230
|
+
const formatExpYear = parseNum(expYear, 2);
|
|
231
|
+
const formatCity = parseVarChar(city, 32);
|
|
232
|
+
const formatCountry = parseChar(country, 2);
|
|
233
|
+
const formatFullName = parseVarChar(fullName, 32);
|
|
234
|
+
const formatStreet1 = parseString(street1, 32);
|
|
235
|
+
const formatState = parseChar(state, 2);
|
|
236
|
+
const formatZip = parseVarChar(zip, 10);
|
|
237
|
+
if (formatExpMonth) {
|
|
238
|
+
paymentCard.expMonth = formatExpMonth;
|
|
239
|
+
}
|
|
240
|
+
if (formatExpYear) {
|
|
241
|
+
paymentCard.expYear = formatExpYear;
|
|
242
|
+
}
|
|
243
|
+
if (formatCity) {
|
|
244
|
+
paymentCard.city = formatCity;
|
|
245
|
+
}
|
|
246
|
+
if (formatCountry) {
|
|
247
|
+
paymentCard.country = formatCountry;
|
|
248
|
+
}
|
|
249
|
+
if (formatFullName) {
|
|
250
|
+
paymentCard.fullName = formatFullName;
|
|
251
|
+
}
|
|
252
|
+
if (formatStreet1) {
|
|
253
|
+
paymentCard.street1 = formatStreet1;
|
|
254
|
+
}
|
|
255
|
+
if (formatState) {
|
|
256
|
+
paymentCard.state = formatState;
|
|
257
|
+
}
|
|
258
|
+
if (formatZip) {
|
|
259
|
+
paymentCard.zip = formatZip;
|
|
260
|
+
}
|
|
261
|
+
const update = paymentCard;
|
|
262
|
+
const aqlQry = aql`
|
|
2
263
|
LET updatedCard = FIRST(
|
|
3
264
|
FOR c IN creditCards
|
|
4
|
-
FILTER c._key == ${
|
|
5
|
-
UPDATE c WITH ${
|
|
265
|
+
FILTER c._key == ${formatId} && c.userId == ${sessionId}
|
|
266
|
+
UPDATE c WITH ${update} IN creditCards
|
|
6
267
|
LIMIT 1
|
|
7
268
|
RETURN NEW
|
|
8
269
|
)
|
|
9
270
|
LET user = FIRST(
|
|
10
271
|
FOR u IN users
|
|
11
|
-
FILTER u._key == ${
|
|
272
|
+
FILTER u._key == ${sessionId}
|
|
12
273
|
LIMIT 1
|
|
13
274
|
RETURN u
|
|
14
275
|
)
|
|
15
|
-
RETURN {user: user, card: updatedCard}`;
|
|
16
|
-
|
|
17
|
-
|
|
276
|
+
RETURN {user: user, card: updatedCard}`;
|
|
277
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((results = {
|
|
278
|
+
card: {},
|
|
279
|
+
user: {}
|
|
280
|
+
})=>{
|
|
281
|
+
const updatedCard = results.card;
|
|
282
|
+
const { user } = results;
|
|
283
|
+
if (!updatedCard) {
|
|
284
|
+
throw new UserError('not_found');
|
|
285
|
+
}
|
|
286
|
+
const { stripeCustomerId } = user;
|
|
287
|
+
const { stripeId } = card;
|
|
288
|
+
const stripeClient = getStripeClient();
|
|
289
|
+
const update = {
|
|
290
|
+
address_city: formatCity,
|
|
291
|
+
address_country: formatCountry,
|
|
292
|
+
address_line1: formatStreet1,
|
|
293
|
+
address_state: formatState,
|
|
294
|
+
address_zip: formatZip,
|
|
295
|
+
exp_month: formatExpMonth.toString(),
|
|
296
|
+
exp_year: formatExpYear.toString(),
|
|
297
|
+
name: formatFullName
|
|
298
|
+
};
|
|
299
|
+
return stripeClient.customers.updateSource(stripeCustomerId, stripeId, update).then(()=>card).catch((error)=>{
|
|
300
|
+
console.log('payments::updateCard::error', error);
|
|
301
|
+
throw new UserError('payment_error');
|
|
302
|
+
});
|
|
303
|
+
});
|
|
304
|
+
};
|
|
305
|
+
export const getCreditCards = (context)=>{
|
|
306
|
+
const action = 'getCreditCards';
|
|
307
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
308
|
+
const aqlQry = aql`FOR c IN creditCards
|
|
309
|
+
FILTER c.userId == ${sessionId}
|
|
310
|
+
RETURN c`;
|
|
311
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.all()).then((list = [])=>list).catch((error)=>logError({
|
|
312
|
+
action,
|
|
313
|
+
category: eventCategory,
|
|
314
|
+
label: ErrorTypes.DATABASE_ERROR
|
|
315
|
+
}, error, context).then(()=>null));
|
|
316
|
+
};
|
|
317
|
+
export const deleteCreditCard = (context, cardId)=>{
|
|
318
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
319
|
+
const formatCardId = parseId(cardId);
|
|
320
|
+
const aqlQry = aql`
|
|
18
321
|
LET card = FIRST(
|
|
19
322
|
FOR c IN creditCards
|
|
20
|
-
FILTER c._key == ${
|
|
323
|
+
FILTER c._key == ${formatCardId} && c.userId == ${sessionId}
|
|
21
324
|
LIMIT 1
|
|
22
325
|
REMOVE c IN creditCards
|
|
23
326
|
RETURN OLD
|
|
24
327
|
)
|
|
25
328
|
LET user = FIRST(
|
|
26
329
|
FOR u IN users
|
|
27
|
-
FILTER u._key == ${
|
|
330
|
+
FILTER u._key == ${sessionId}
|
|
28
331
|
LIMIT 1
|
|
29
332
|
RETURN u
|
|
30
333
|
)
|
|
31
|
-
RETURN {user: user, card: card}`;
|
|
32
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/payments.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 {createHash, parseChar, parseId, parseString, parseVarChar} from '@nlabs/utils/parsers/strings';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport stripe from 'stripe';\n\nimport {ApiContext} from '../types/auth.types.js';\nimport {ErrorTypes} from '../types/error.types.js';\nimport {PaymentBankAccount, PaymentCardType, PaymentCharge, PaymentTransfer} from '../types/payments.types.js';\nimport {UserType} from '../types/users.types.js';\nimport {logError, UserError} from '../utils/analyticsUtils.js';\nimport {useDb} from '../utils/arangodbUtils.js';\nimport {getStripeClient} from '../utils/stripeUtils.js';\nimport {getUser} from './users.js';\n\nimport type {EdgeCollection} from 'arangojs/collections';\n\nconst eventCategory = 'payments';\n\nexport const addCustomerAccount = (context: ApiContext): Promise<boolean> => {\n  const action = 'addCustomerAccount';\n  const {databaseName, session: {userId: sessionId, username} = {}} = context;\n\n  const stripeClient = getStripeClient();\n\n  return stripeClient.customers\n    .create({\n      metadata: {\n        userId: sessionId,\n        username\n      }\n    } as any)\n    .then((customer) => {\n      // Create session\n      const now: number = Date.now();\n      const update: UserType = {\n        modified: now,\n        stripeCustomerId: customer.id\n      };\n\n      const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n      return useDb(databaseName).query(aqlQry)\n        .then((cursor) => cursor.next())\n        .then((updatedUser: UserType) => !!updatedUser)\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          label: ErrorTypes.DATABASE_ERROR\n        }, error, context).then(() => null));\n    });\n};\n\nexport const addBankAccount = (context: ApiContext, bankAccount: PaymentBankAccount): Promise<boolean> => {\n  const action = 'addPaymentAccountBank';\n  const {databaseName, session: {userId: sessionId} = {}} = context;\n\n  // Params\n  const {\n    accountNumber,\n    fullName,\n    routing\n  } = bankAccount;\n\n  const formatAccount: string = parseString(accountNumber, 32);\n\n  if(formatAccount === '') {\n    throw new UserError('required_account_number');\n  }\n\n  const formatFullName: string = parseVarChar(fullName, 128);\n\n  if(formatFullName === '') {\n    throw new UserError('required_full_name');\n  }\n\n  const formatRouting: string = parseString(routing, 32);\n\n  if(formatRouting === '') {\n    throw new UserError('required_routing_number');\n  }\n\n  return getUser(context, {userId: sessionId})\n    .then((user: UserType) => {\n      const {stripeAccountId} = user;\n      const stripeClient = getStripeClient();\n\n      // Create a token first, then use the token as the source\n      return stripeClient.tokens.create({\n        bank_account: {\n          account_holder_name: formatFullName,\n          account_holder_type: 'individual',\n          account_number: formatAccount,\n          country: 'US',\n          currency: 'USD',\n          routing_number: formatRouting\n        }\n      })\n        .then((token) => stripeClient.customers.createSource(\n          stripeAccountId,\n          {source: token.id}\n        ))\n        .then((account) => {\n          // Use type assertion for card properties\n          // const cardSource = account as unknown as Card;\n          // const brand = cardSource.brand || '';\n          // const cvcCheck = cardSource.cvc_check || '';\n          // const last4 = cardSource.last4 || '';\n\n          // Create session\n          const now: number = Date.now();\n          const update = {\n            bankAccount,\n            bankFullName: formatFullName,\n            bankId: account.id,\n            bankRouting: formatRouting,\n            modified: now\n          };\n\n          const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n          return useDb(databaseName).query(aqlQry)\n            .then((cursor) => cursor.next())\n            .then((updatedUser: UserType) => updatedUser);\n        })\n        .catch((error: Error) => {\n          const msg = error.message;\n\n          if(msg === 'A bank account with that routing number and account number ' +\n            'already exists for this customer.') {\n            return logError({\n              action,\n              category: eventCategory,\n              label: 'bank_account_exists'\n            }, error, context).then(() => null);\n          }\n          return logError({\n            action,\n            category: eventCategory,\n            label: 'payment_error'\n          }, error, context).then(() => null);\n        });\n    });\n};\n\nexport const addCreditCard = (context: ApiContext, card: PaymentCardType): Promise<UserType> => {\n  const action = 'addCreditCard';\n  const {databaseName, session: {userId: sessionId}} = context;\n\n  return getUser(context, {userId: sessionId})\n    .then((user: UserType) => {\n      // User\n      const {stripeAccountId} = user;\n\n      // Card\n      const {\n        accountNumber,\n        city,\n        country,\n        cvc,\n        expMonth,\n        expYear,\n        fullName,\n        street1,\n        street2,\n        state,\n        zip\n      }: PaymentCardType = card;\n\n      const formatNumber: number = parseNum(accountNumber, 16);\n\n      if(!formatNumber) {\n        throw new UserError('required_credit_card_number');\n      }\n\n      const formatExpMonth: number = parseNum(expMonth, 2);\n\n      if(!formatExpMonth) {\n        throw new UserError('required_credit_card_exp_month');\n      }\n\n      const formatExpYear: number = parseNum(expYear, 2);\n\n      if(!formatExpYear) {\n        throw new UserError('required_credit_card_exp_year');\n      }\n\n      const formatCvc: number = parseNum(cvc, 3);\n\n      // Address\n      const paymentCard: PaymentCardType = {};\n      const formatCity: string = parseVarChar(city, 32);\n\n      if(formatCity) {\n        paymentCard.city = formatCity;\n      }\n\n      const formatCountry: string = parseChar(country, 2);\n\n      if(formatCountry) {\n        paymentCard.country = formatCountry;\n      }\n\n      const formatFullName: string = parseVarChar(fullName, 32);\n\n      if(formatFullName) {\n        paymentCard.fullName = formatFullName;\n      }\n\n      const formatStreet1: string = parseVarChar(street1, 32);\n\n      if(formatStreet1) {\n        paymentCard.street1 = formatStreet1;\n      }\n\n      const formatStreet2: string = parseVarChar(street2, 32);\n\n      if(formatStreet2) {\n        paymentCard.street2 = formatStreet2;\n      }\n\n      const formatState: string = parseChar(state, 2);\n\n      if(formatState) {\n        paymentCard.state = formatState;\n      }\n\n      const formatZip: string = parseVarChar(zip, 10);\n\n      if(formatZip) {\n        paymentCard.zip = formatZip;\n      }\n\n      const stripeClient = getStripeClient();\n\n      // Create a token first, then use the token as the source\n      return stripeClient.tokens.create({\n        card: {\n          address_city: formatCity,\n          address_country: formatCountry,\n          address_line1: formatStreet1,\n          address_line2: formatStreet2,\n          address_state: formatState,\n          address_zip: formatZip,\n          cvc: formatCvc.toString(),\n          exp_month: formatExpMonth.toString(),\n          exp_year: formatExpYear.toString(),\n          name: fullName,\n          number: formatNumber.toString()\n        }\n      })\n        .then((token) => stripeClient.customers.createSource(\n          stripeAccountId,\n          {source: token.id}\n        ))\n        .then((newSource) => {\n          // Use type assertion for card properties\n          const cardSource = newSource as unknown as Card;\n          const brand = cardSource.brand || '';\n          const cvcCheck = cardSource.cvc_check || '';\n          const last4 = cardSource.last4 || '';\n\n          // Create session\n          const now: number = Date.now();\n          const insert = {\n            ...paymentCard,\n            _key: createHash(`user-payment-${sessionId}`),\n            accountNumber: last4,\n            added: now,\n            brand,\n            cvcCheck,\n            expMonth,\n            expYear,\n            modified: now,\n            userId: sessionId\n          };\n          const insertAqlQry: AqlQuery = aql`INSERT ${insert} IN creditCards RETURN NEW`;\n\n          return useDb(databaseName).query(insertAqlQry)\n            .then((cursor) => cursor.next())\n            .then((newCard: PaymentCardType) => {\n              if(newCard) {\n                // Add linked edge\n                const {_id: cardId, _key: cardKey} = card;\n                const edgeCollection: EdgeCollection = useDb(databaseName).collection('hasPayment');\n                const edgeId = createHash(`payment-${cardKey}`);\n                const edge = {\n                  _from: `users/${sessionId}`,\n                  _key: edgeId,\n                  _to: cardId\n                };\n\n                return edgeCollection.save(edge, {returnNew: true}).then(() => card);\n              }\n\n              return newCard;\n            })\n            .catch((error: Error) => logError({\n              action,\n              category: eventCategory,\n              label: 'payment_error'\n            }, error, context).then(() => null));\n        });\n    });\n};\n\nexport const updateCreditCard = (context: ApiContext, card: PaymentCardType): Promise<PaymentCardType> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n\n  const {\n    city,\n    country,\n    expMonth,\n    expYear,\n    fullName,\n    id,\n    street1,\n    state,\n    zip\n  }: PaymentCardType = card;\n\n  const formatId: string = parseId(id);\n\n  if(formatId) {\n    throw new UserError('required_credit_card_id');\n  }\n\n  const paymentCard: PaymentCardType = {};\n  const formatExpMonth: number = parseNum(expMonth, 2);\n  const formatExpYear: number = parseNum(expYear, 2);\n  const formatCity: string = parseVarChar(city, 32);\n  const formatCountry: string = parseChar(country, 2);\n  const formatFullName: string = parseVarChar(fullName, 32);\n  const formatStreet1: string = parseString(street1, 32);\n  const formatState: string = parseChar(state, 2);\n  const formatZip: string = parseVarChar(zip, 10);\n\n  if(formatExpMonth) {\n    paymentCard.expMonth = formatExpMonth;\n  }\n\n  if(formatExpYear) {\n    paymentCard.expYear = formatExpYear;\n  }\n\n  if(formatCity) {\n    paymentCard.city = formatCity;\n  }\n\n  if(formatCountry) {\n    paymentCard.country = formatCountry;\n  }\n\n  if(formatFullName) {\n    paymentCard.fullName = formatFullName;\n  }\n\n  if(formatStreet1) {\n    paymentCard.street1 = formatStreet1;\n  }\n\n  if(formatState) {\n    paymentCard.state = formatState;\n  }\n\n  if(formatZip) {\n    paymentCard.zip = formatZip;\n  }\n\n  const update = paymentCard;\n  const aqlQry: AqlQuery = aql`\n      LET updatedCard = FIRST(\n        FOR c IN creditCards\n        FILTER c._key == ${formatId} && c.userId == ${sessionId}\n        UPDATE c WITH ${update} IN creditCards\n        LIMIT 1\n        RETURN NEW\n      )\n      LET user = FIRST(\n        FOR u IN users\n        FILTER u._key == ${sessionId}\n        LIMIT 1\n        RETURN u\n      )\n      RETURN {user: user, card: updatedCard}`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((results = {card: {}, user: {}}) => {\n      const updatedCard: PaymentCardType = results.card;\n      const {user} = results;\n\n      if(!updatedCard) {\n        throw new UserError('not_found');\n      }\n\n      const {stripeCustomerId} = user;\n      const {stripeId} = card;\n      const stripeClient = getStripeClient();\n      const update: stripe.CustomerUpdateSourceParams = {\n        address_city: formatCity,\n        address_country: formatCountry,\n        address_line1: formatStreet1,\n        address_state: formatState,\n        address_zip: formatZip,\n        exp_month: formatExpMonth.toString(),\n        exp_year: formatExpYear.toString(),\n        name: formatFullName\n      };\n\n      return stripeClient.customers\n        .updateSource(stripeCustomerId, stripeId, update)\n        .then(() => card)\n        .catch((error: Error) => {\n          console.log('payments::updateCard::error', error);\n          throw new UserError('payment_error');\n        });\n    });\n};\n\nexport const getCreditCards = (context: ApiContext): Promise<PaymentCardType[]> => {\n  const action = 'getCreditCards';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const aqlQry: AqlQuery = aql`FOR c IN creditCards\n    FILTER c.userId == ${sessionId}\n    RETURN c`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all())\n    .then((list: PaymentCardType[] = []) => list)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => null));\n};\n\nexport const deleteCreditCard = (context: ApiContext, cardId: string): Promise<boolean> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n  const formatCardId: string = parseId(cardId);\n  const aqlQry: AqlQuery = aql`\n    LET card = FIRST(\n      FOR c IN creditCards\n      FILTER c._key == ${formatCardId} && c.userId == ${sessionId}\n      LIMIT 1\n      REMOVE c IN creditCards\n      RETURN OLD\n    )\n    LET user = FIRST(\n      FOR u IN users\n      FILTER u._key == ${sessionId}\n      LIMIT 1\n      RETURN u\n    )\n    RETURN {user: user, card: card}`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((result = {card: {}, user: {}}) => {\n      if(!result) {\n        return false;\n      }\n\n      const {card, user} = result;\n      const {_key: cardKey} = card;\n\n      // Remove linked edges\n      const edgeCollection = useDb(databaseName).collection('hasPayment');\n\n      return edgeCollection.outEdges(cardKey, {})\n        .then(async (response) => {\n          // Extract edges from the response\n          const edges = Array.isArray(response) ? response : [];\n\n          if(edges.length) {\n            await Promise.all(\n              edges.map((edge) => {\n                const {_key: edgeKey} = edge;\n                const removeAqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN hasPayment`;\n                return useDb(databaseName).query(removeAqlQry);\n              }))\n              .then(() => {\n                // Stripe\n                const stripeClient = getStripeClient();\n\n                return stripeClient.customers\n                  .deleteSource(user.stripeCustomerId, card.stripeId)\n                  .then(() => true)\n                  .catch((error: Error) => {\n                    console.log('payments::deleteCard::error', error);\n                    throw new UserError('payment_error');\n                  });\n              });\n\n            return true;\n          }\n\n          return false;\n        });\n    });\n};\n\nexport const deleteBankAccount = (context: ApiContext, bankId: string): Promise<boolean> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n\n  // Clean db\n  const update: UserType = {\n    bankAccount: '',\n    bankFullName: '',\n    bankId: '',\n    bankRouting: '',\n    modified: Date.now()\n  };\n  const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((user: UserType) => {\n      const {stripeAccountId} = user;\n      const stripeClient = getStripeClient();\n\n      return stripeClient.customers\n        .deleteSource(stripeAccountId, bankId)\n        .then(() => true)\n        .catch(() => Promise.resolve(false));\n    });\n};\n\nexport const createPaymentTransfer = (context: ApiContext, transfer: PaymentTransfer): Promise<PaymentTransfer> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {amount, currency} = transfer;\n  const formatAmount: number = parseNum(amount);\n  const formatCurrency: string = parseChar(currency, 3, 'USD').toUpperCase();\n\n  return getUser(context, {userId: sessionId})\n    .then((user: UserType) => {\n      const {stripeAccountId} = user;\n      const stripeClient = getStripeClient();\n\n      return stripeClient.transfers\n        .create({\n          amount: formatAmount,\n          currency: formatCurrency,\n          destination: stripeAccountId\n        })\n        .then((stripeTransfer) => {\n          console.log(stripeTransfer);\n          const now: number = Date.now();\n          const insert: PaymentTransfer = {\n            added: now,\n            amount: formatAmount,\n            currency: formatCurrency,\n            modified: now,\n            userId: sessionId\n          };\n          const aqlQry: AqlQuery = aql`INSERT ${insert} IN transfers RETURN NEW`;\n\n          return useDb(databaseName).query(aqlQry)\n            .then((cursor) => cursor.next())\n            .then((newTransfer: PaymentTransfer) => newTransfer);\n        });\n    });\n};\n\nexport const createPaymentHold = (context: ApiContext, payment: PaymentCharge): Promise<PaymentCharge> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {amount, capture, cardId, currency, description} = payment;\n  const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();\n  const stripeClient = getStripeClient();\n\n  return stripeClient.charges\n    .create({\n      amount,\n      capture,\n      currency: formatCurrency,\n      description,\n      source: cardId\n    })\n    .then((stripeCharge) => {\n      const now: number = Date.now();\n      const insert: PaymentCharge = {\n        added: now,\n        amount,\n        capture,\n        cardId,\n        chargeFailCode: stripeCharge.failure_code,\n        chargeFailMsg: stripeCharge.failure_message,\n        chargeId: stripeCharge.id,\n        chargeStatus: stripeCharge.status,\n        currency: formatCurrency,\n        description,\n        modified: now,\n        userId: sessionId\n      };\n      const aqlQry: AqlQuery = aql`INSERT ${insert} IN payments RETURN NEW`;\n\n      return useDb(databaseName).query(aqlQry)\n        .then((cursor) => cursor.next())\n        .then((newPayment: PaymentCharge) => newPayment);\n    })\n    .catch((error: Error) => {\n      console.log('payments::createHold::error', error);\n      throw new UserError('payment_error');\n    });\n};\n\ninterface Card {\n  id: string;\n  brand?: string;\n  cvc_check?: string;\n  last4?: string;\n}\n"],
  "mappings": "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",
  "names": ["parseNum", "createHash", "parseChar", "parseId", "parseString", "parseVarChar", "aql", "ErrorTypes", "logError", "UserError", "useDb", "getStripeClient", "getUser", "eventCategory", "addCustomerAccount", "context", "action", "databaseName", "sessionId", "username", "customer", "update", "aqlQry", "cursor", "updatedUser", "error", "addBankAccount", "bankAccount", "accountNumber", "fullName", "routing", "formatAccount", "formatFullName", "formatRouting", "user", "stripeAccountId", "stripeClient", "token", "account", "now", "addCreditCard", "card", "city", "country", "cvc", "expMonth", "expYear", "street1", "street2", "state", "zip", "formatNumber", "formatExpMonth", "formatExpYear", "formatCvc", "paymentCard", "formatCity", "formatCountry", "formatStreet1", "formatStreet2", "formatState", "formatZip", "newSource", "cardSource", "brand", "cvcCheck", "last4", "insert", "insertAqlQry", "newCard", "cardId", "cardKey", "edgeCollection", "edgeId", "edge", "updateCreditCard", "id", "formatId", "results", "updatedCard", "stripeCustomerId", "stripeId", "getCreditCards", "list", "deleteCreditCard", "formatCardId", "result", "response", "edges", "edgeKey", "removeAqlQry", "deleteBankAccount", "bankId", "createPaymentTransfer", "transfer", "amount", "currency", "formatAmount", "formatCurrency", "stripeTransfer", "newTransfer", "createPaymentHold", "payment", "capture", "description", "stripeCharge", "newPayment"]
}

|
|
334
|
+
RETURN {user: user, card: card}`;
|
|
335
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((result = {
|
|
336
|
+
card: {},
|
|
337
|
+
user: {}
|
|
338
|
+
})=>{
|
|
339
|
+
if (!result) {
|
|
340
|
+
return false;
|
|
341
|
+
}
|
|
342
|
+
const { card, user } = result;
|
|
343
|
+
const { _key: cardKey } = card;
|
|
344
|
+
// Remove linked edges
|
|
345
|
+
const edgeCollection = useDb(databaseName).collection('hasPayment');
|
|
346
|
+
return edgeCollection.outEdges(cardKey, {}).then(async (response)=>{
|
|
347
|
+
// Extract edges from the response
|
|
348
|
+
const edges = Array.isArray(response) ? response : [];
|
|
349
|
+
if (edges.length) {
|
|
350
|
+
await Promise.all(edges.map((edge)=>{
|
|
351
|
+
const { _key: edgeKey } = edge;
|
|
352
|
+
const removeAqlQry = aql`REMOVE {_key:${edgeKey}} IN hasPayment`;
|
|
353
|
+
return useDb(databaseName).query(removeAqlQry);
|
|
354
|
+
})).then(()=>{
|
|
355
|
+
// Stripe
|
|
356
|
+
const stripeClient = getStripeClient();
|
|
357
|
+
return stripeClient.customers.deleteSource(user.stripeCustomerId, card.stripeId).then(()=>true).catch((error)=>{
|
|
358
|
+
console.log('payments::deleteCard::error', error);
|
|
359
|
+
throw new UserError('payment_error');
|
|
360
|
+
});
|
|
361
|
+
});
|
|
362
|
+
return true;
|
|
363
|
+
}
|
|
364
|
+
return false;
|
|
365
|
+
});
|
|
366
|
+
});
|
|
367
|
+
};
|
|
368
|
+
export const deleteBankAccount = (context, bankId)=>{
|
|
369
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
370
|
+
// Clean db
|
|
371
|
+
const update = {
|
|
372
|
+
bankAccount: '',
|
|
373
|
+
bankFullName: '',
|
|
374
|
+
bankId: '',
|
|
375
|
+
bankRouting: '',
|
|
376
|
+
modified: Date.now()
|
|
377
|
+
};
|
|
378
|
+
const aqlQry = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;
|
|
379
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((user)=>{
|
|
380
|
+
const { stripeAccountId } = user;
|
|
381
|
+
const stripeClient = getStripeClient();
|
|
382
|
+
return stripeClient.customers.deleteSource(stripeAccountId, bankId).then(()=>true).catch(()=>Promise.resolve(false));
|
|
383
|
+
});
|
|
384
|
+
};
|
|
385
|
+
export const createPaymentTransfer = (context, transfer)=>{
|
|
386
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
387
|
+
const { amount, currency } = transfer;
|
|
388
|
+
const formatAmount = parseNum(amount);
|
|
389
|
+
const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();
|
|
390
|
+
return getUser(context, {
|
|
391
|
+
userId: sessionId
|
|
392
|
+
}).then((user)=>{
|
|
393
|
+
const { stripeAccountId } = user;
|
|
394
|
+
const stripeClient = getStripeClient();
|
|
395
|
+
return stripeClient.transfers.create({
|
|
396
|
+
amount: formatAmount,
|
|
397
|
+
currency: formatCurrency,
|
|
398
|
+
destination: stripeAccountId
|
|
399
|
+
}).then((stripeTransfer)=>{
|
|
400
|
+
console.log(stripeTransfer);
|
|
401
|
+
const now = Date.now();
|
|
402
|
+
const insert = {
|
|
403
|
+
added: now,
|
|
404
|
+
amount: formatAmount,
|
|
405
|
+
currency: formatCurrency,
|
|
406
|
+
modified: now,
|
|
407
|
+
userId: sessionId
|
|
408
|
+
};
|
|
409
|
+
const aqlQry = aql`INSERT ${insert} IN transfers RETURN NEW`;
|
|
410
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((newTransfer)=>newTransfer);
|
|
411
|
+
});
|
|
412
|
+
});
|
|
413
|
+
};
|
|
414
|
+
export const createPaymentHold = (context, payment)=>{
|
|
415
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
416
|
+
const { amount, capture, cardId, currency, description } = payment;
|
|
417
|
+
const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();
|
|
418
|
+
const stripeClient = getStripeClient();
|
|
419
|
+
return stripeClient.charges.create({
|
|
420
|
+
amount,
|
|
421
|
+
capture,
|
|
422
|
+
currency: formatCurrency,
|
|
423
|
+
description,
|
|
424
|
+
source: cardId
|
|
425
|
+
}).then((stripeCharge)=>{
|
|
426
|
+
const now = Date.now();
|
|
427
|
+
const insert = {
|
|
428
|
+
added: now,
|
|
429
|
+
amount,
|
|
430
|
+
capture,
|
|
431
|
+
cardId,
|
|
432
|
+
chargeFailCode: stripeCharge.failure_code,
|
|
433
|
+
chargeFailMsg: stripeCharge.failure_message,
|
|
434
|
+
chargeId: stripeCharge.id,
|
|
435
|
+
chargeStatus: stripeCharge.status,
|
|
436
|
+
currency: formatCurrency,
|
|
437
|
+
description,
|
|
438
|
+
modified: now,
|
|
439
|
+
userId: sessionId
|
|
440
|
+
};
|
|
441
|
+
const aqlQry = aql`INSERT ${insert} IN payments RETURN NEW`;
|
|
442
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((newPayment)=>newPayment);
|
|
443
|
+
}).catch((error)=>{
|
|
444
|
+
console.log('payments::createHold::error', error);
|
|
445
|
+
throw new UserError('payment_error');
|
|
446
|
+
});
|
|
447
|
+
};
|
|
448
|
+
|
|
449
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/nitrog7/Development/reaktor/src/actions/payments.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 {createHash, parseChar, parseId, parseString, parseVarChar} from '@nlabs/utils/parsers/strings';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport stripe from 'stripe';\n\nimport {ApiContext} from '../types/auth.types.js';\nimport {ErrorTypes} from '../types/error.types.js';\nimport {PaymentBankAccount, PaymentCardType, PaymentCharge, PaymentTransfer} from '../types/payments.types.js';\nimport {UserType} from '../types/users.types.js';\nimport {logError, UserError} from '../utils/analyticsUtils.js';\nimport {useDb} from '../utils/arangodbUtils.js';\nimport {getStripeClient} from '../utils/stripeUtils.js';\nimport {getUser} from './users.js';\n\nimport type {EdgeCollection} from 'arangojs/collections';\n\nconst eventCategory = 'payments';\n\nexport const addCustomerAccount = (context: ApiContext): Promise<boolean> => {\n  const action = 'addCustomerAccount';\n  const {databaseName, session: {userId: sessionId, username} = {}} = context;\n\n  const stripeClient = getStripeClient();\n\n  return stripeClient.customers\n    .create({\n      metadata: {\n        userId: sessionId,\n        username\n      }\n    } as any)\n    .then((customer) => {\n      // Create session\n      const now: number = Date.now();\n      const update: UserType = {\n        modified: now,\n        stripeCustomerId: customer.id\n      };\n\n      const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n      return useDb(databaseName).query(aqlQry)\n        .then((cursor) => cursor.next())\n        .then((updatedUser: UserType) => !!updatedUser)\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          label: ErrorTypes.DATABASE_ERROR\n        }, error, context).then(() => null));\n    });\n};\n\nexport const addBankAccount = (context: ApiContext, bankAccount: PaymentBankAccount): Promise<boolean> => {\n  const action = 'addPaymentAccountBank';\n  const {databaseName, session: {userId: sessionId} = {}} = context;\n\n  // Params\n  const {\n    accountNumber,\n    fullName,\n    routing\n  } = bankAccount;\n\n  const formatAccount: string = parseString(accountNumber, 32);\n\n  if(formatAccount === '') {\n    throw new UserError('required_account_number');\n  }\n\n  const formatFullName: string = parseVarChar(fullName, 128);\n\n  if(formatFullName === '') {\n    throw new UserError('required_full_name');\n  }\n\n  const formatRouting: string = parseString(routing, 32);\n\n  if(formatRouting === '') {\n    throw new UserError('required_routing_number');\n  }\n\n  return getUser(context, {userId: sessionId})\n    .then((user: UserType) => {\n      const {stripeAccountId} = user;\n      const stripeClient = getStripeClient();\n\n      // Create a token first, then use the token as the source\n      return stripeClient.tokens.create({\n        bank_account: {\n          account_holder_name: formatFullName,\n          account_holder_type: 'individual',\n          account_number: formatAccount,\n          country: 'US',\n          currency: 'USD',\n          routing_number: formatRouting\n        }\n      })\n        .then((token) => stripeClient.customers.createSource(\n          stripeAccountId,\n          {source: token.id}\n        ))\n        .then((account) => {\n          // Use type assertion for card properties\n          // const cardSource = account as unknown as Card;\n          // const brand = cardSource.brand || '';\n          // const cvcCheck = cardSource.cvc_check || '';\n          // const last4 = cardSource.last4 || '';\n\n          // Create session\n          const now: number = Date.now();\n          const update = {\n            bankAccount,\n            bankFullName: formatFullName,\n            bankId: account.id,\n            bankRouting: formatRouting,\n            modified: now\n          };\n\n          const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n          return useDb(databaseName).query(aqlQry)\n            .then((cursor) => cursor.next())\n            .then((updatedUser: UserType) => updatedUser);\n        })\n        .catch((error: Error) => {\n          const msg = error.message;\n\n          if(msg === 'A bank account with that routing number and account number ' +\n            'already exists for this customer.') {\n            return logError({\n              action,\n              category: eventCategory,\n              label: 'bank_account_exists'\n            }, error, context).then(() => null);\n          }\n          return logError({\n            action,\n            category: eventCategory,\n            label: 'payment_error'\n          }, error, context).then(() => null);\n        });\n    });\n};\n\nexport const addCreditCard = (context: ApiContext, card: PaymentCardType): Promise<UserType> => {\n  const action = 'addCreditCard';\n  const {databaseName, session: {userId: sessionId}} = context;\n\n  return getUser(context, {userId: sessionId})\n    .then((user: UserType) => {\n      // User\n      const {stripeAccountId} = user;\n\n      // Card\n      const {\n        accountNumber,\n        city,\n        country,\n        cvc,\n        expMonth,\n        expYear,\n        fullName,\n        street1,\n        street2,\n        state,\n        zip\n      }: PaymentCardType = card;\n\n      const formatNumber: number = parseNum(accountNumber, 16);\n\n      if(!formatNumber) {\n        throw new UserError('required_credit_card_number');\n      }\n\n      const formatExpMonth: number = parseNum(expMonth, 2);\n\n      if(!formatExpMonth) {\n        throw new UserError('required_credit_card_exp_month');\n      }\n\n      const formatExpYear: number = parseNum(expYear, 2);\n\n      if(!formatExpYear) {\n        throw new UserError('required_credit_card_exp_year');\n      }\n\n      const formatCvc: number = parseNum(cvc, 3);\n\n      // Address\n      const paymentCard: PaymentCardType = {};\n      const formatCity: string = parseVarChar(city, 32);\n\n      if(formatCity) {\n        paymentCard.city = formatCity;\n      }\n\n      const formatCountry: string = parseChar(country, 2);\n\n      if(formatCountry) {\n        paymentCard.country = formatCountry;\n      }\n\n      const formatFullName: string = parseVarChar(fullName, 32);\n\n      if(formatFullName) {\n        paymentCard.fullName = formatFullName;\n      }\n\n      const formatStreet1: string = parseVarChar(street1, 32);\n\n      if(formatStreet1) {\n        paymentCard.street1 = formatStreet1;\n      }\n\n      const formatStreet2: string = parseVarChar(street2, 32);\n\n      if(formatStreet2) {\n        paymentCard.street2 = formatStreet2;\n      }\n\n      const formatState: string = parseChar(state, 2);\n\n      if(formatState) {\n        paymentCard.state = formatState;\n      }\n\n      const formatZip: string = parseVarChar(zip, 10);\n\n      if(formatZip) {\n        paymentCard.zip = formatZip;\n      }\n\n      const stripeClient = getStripeClient();\n\n      // Create a token first, then use the token as the source\n      return stripeClient.tokens.create({\n        card: {\n          address_city: formatCity,\n          address_country: formatCountry,\n          address_line1: formatStreet1,\n          address_line2: formatStreet2,\n          address_state: formatState,\n          address_zip: formatZip,\n          cvc: formatCvc.toString(),\n          exp_month: formatExpMonth.toString(),\n          exp_year: formatExpYear.toString(),\n          name: fullName,\n          number: formatNumber.toString()\n        }\n      })\n        .then((token) => stripeClient.customers.createSource(\n          stripeAccountId,\n          {source: token.id}\n        ))\n        .then((newSource) => {\n          // Use type assertion for card properties\n          const cardSource = newSource as unknown as Card;\n          const brand = cardSource.brand || '';\n          const cvcCheck = cardSource.cvc_check || '';\n          const last4 = cardSource.last4 || '';\n\n          // Create session\n          const now: number = Date.now();\n          const insert = {\n            ...paymentCard,\n            _key: createHash(`user-payment-${sessionId}`),\n            accountNumber: last4,\n            added: now,\n            brand,\n            cvcCheck,\n            expMonth,\n            expYear,\n            modified: now,\n            userId: sessionId\n          };\n          const insertAqlQry: AqlQuery = aql`INSERT ${insert} IN creditCards RETURN NEW`;\n\n          return useDb(databaseName).query(insertAqlQry)\n            .then((cursor) => cursor.next())\n            .then((newCard: PaymentCardType) => {\n              if(newCard) {\n                // Add linked edge\n                const {_id: cardId, _key: cardKey} = card;\n                const edgeCollection: EdgeCollection = useDb(databaseName).collection('hasPayment');\n                const edgeId = createHash(`payment-${cardKey}`);\n                const edge = {\n                  _from: `users/${sessionId}`,\n                  _key: edgeId,\n                  _to: cardId\n                };\n\n                return edgeCollection.save(edge, {returnNew: true}).then(() => card);\n              }\n\n              return newCard;\n            })\n            .catch((error: Error) => logError({\n              action,\n              category: eventCategory,\n              label: 'payment_error'\n            }, error, context).then(() => null));\n        });\n    });\n};\n\nexport const updateCreditCard = (context: ApiContext, card: PaymentCardType): Promise<PaymentCardType> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n\n  const {\n    city,\n    country,\n    expMonth,\n    expYear,\n    fullName,\n    id,\n    street1,\n    state,\n    zip\n  }: PaymentCardType = card;\n\n  const formatId: string = parseId(id);\n\n  if(formatId) {\n    throw new UserError('required_credit_card_id');\n  }\n\n  const paymentCard: PaymentCardType = {};\n  const formatExpMonth: number = parseNum(expMonth, 2);\n  const formatExpYear: number = parseNum(expYear, 2);\n  const formatCity: string = parseVarChar(city, 32);\n  const formatCountry: string = parseChar(country, 2);\n  const formatFullName: string = parseVarChar(fullName, 32);\n  const formatStreet1: string = parseString(street1, 32);\n  const formatState: string = parseChar(state, 2);\n  const formatZip: string = parseVarChar(zip, 10);\n\n  if(formatExpMonth) {\n    paymentCard.expMonth = formatExpMonth;\n  }\n\n  if(formatExpYear) {\n    paymentCard.expYear = formatExpYear;\n  }\n\n  if(formatCity) {\n    paymentCard.city = formatCity;\n  }\n\n  if(formatCountry) {\n    paymentCard.country = formatCountry;\n  }\n\n  if(formatFullName) {\n    paymentCard.fullName = formatFullName;\n  }\n\n  if(formatStreet1) {\n    paymentCard.street1 = formatStreet1;\n  }\n\n  if(formatState) {\n    paymentCard.state = formatState;\n  }\n\n  if(formatZip) {\n    paymentCard.zip = formatZip;\n  }\n\n  const update = paymentCard;\n  const aqlQry: AqlQuery = aql`\n      LET updatedCard = FIRST(\n        FOR c IN creditCards\n        FILTER c._key == ${formatId} && c.userId == ${sessionId}\n        UPDATE c WITH ${update} IN creditCards\n        LIMIT 1\n        RETURN NEW\n      )\n      LET user = FIRST(\n        FOR u IN users\n        FILTER u._key == ${sessionId}\n        LIMIT 1\n        RETURN u\n      )\n      RETURN {user: user, card: updatedCard}`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((results = {card: {}, user: {}}) => {\n      const updatedCard: PaymentCardType = results.card;\n      const {user} = results;\n\n      if(!updatedCard) {\n        throw new UserError('not_found');\n      }\n\n      const {stripeCustomerId} = user;\n      const {stripeId} = card;\n      const stripeClient = getStripeClient();\n      const update: stripe.CustomerUpdateSourceParams = {\n        address_city: formatCity,\n        address_country: formatCountry,\n        address_line1: formatStreet1,\n        address_state: formatState,\n        address_zip: formatZip,\n        exp_month: formatExpMonth.toString(),\n        exp_year: formatExpYear.toString(),\n        name: formatFullName\n      };\n\n      return stripeClient.customers\n        .updateSource(stripeCustomerId, stripeId, update)\n        .then(() => card)\n        .catch((error: Error) => {\n          console.log('payments::updateCard::error', error);\n          throw new UserError('payment_error');\n        });\n    });\n};\n\nexport const getCreditCards = (context: ApiContext): Promise<PaymentCardType[]> => {\n  const action = 'getCreditCards';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const aqlQry: AqlQuery = aql`FOR c IN creditCards\n    FILTER c.userId == ${sessionId}\n    RETURN c`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all())\n    .then((list: PaymentCardType[] = []) => list)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      label: ErrorTypes.DATABASE_ERROR\n    }, error, context).then(() => null));\n};\n\nexport const deleteCreditCard = (context: ApiContext, cardId: string): Promise<boolean> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n  const formatCardId: string = parseId(cardId);\n  const aqlQry: AqlQuery = aql`\n    LET card = FIRST(\n      FOR c IN creditCards\n      FILTER c._key == ${formatCardId} && c.userId == ${sessionId}\n      LIMIT 1\n      REMOVE c IN creditCards\n      RETURN OLD\n    )\n    LET user = FIRST(\n      FOR u IN users\n      FILTER u._key == ${sessionId}\n      LIMIT 1\n      RETURN u\n    )\n    RETURN {user: user, card: card}`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((result = {card: {}, user: {}}) => {\n      if(!result) {\n        return false;\n      }\n\n      const {card, user} = result;\n      const {_key: cardKey} = card;\n\n      // Remove linked edges\n      const edgeCollection = useDb(databaseName).collection('hasPayment');\n\n      return edgeCollection.outEdges(cardKey, {})\n        .then(async (response) => {\n          // Extract edges from the response\n          const edges = Array.isArray(response) ? response : [];\n\n          if(edges.length) {\n            await Promise.all(\n              edges.map((edge) => {\n                const {_key: edgeKey} = edge;\n                const removeAqlQry: AqlQuery = aql`REMOVE {_key:${edgeKey}} IN hasPayment`;\n                return useDb(databaseName).query(removeAqlQry);\n              }))\n              .then(() => {\n                // Stripe\n                const stripeClient = getStripeClient();\n\n                return stripeClient.customers\n                  .deleteSource(user.stripeCustomerId, card.stripeId)\n                  .then(() => true)\n                  .catch((error: Error) => {\n                    console.log('payments::deleteCard::error', error);\n                    throw new UserError('payment_error');\n                  });\n              });\n\n            return true;\n          }\n\n          return false;\n        });\n    });\n};\n\nexport const deleteBankAccount = (context: ApiContext, bankId: string): Promise<boolean> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n\n  // Clean db\n  const update: UserType = {\n    bankAccount: '',\n    bankFullName: '',\n    bankId: '',\n    bankRouting: '',\n    modified: Date.now()\n  };\n  const aqlQry: AqlQuery = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then((user: UserType) => {\n      const {stripeAccountId} = user;\n      const stripeClient = getStripeClient();\n\n      return stripeClient.customers\n        .deleteSource(stripeAccountId, bankId)\n        .then(() => true)\n        .catch(() => Promise.resolve(false));\n    });\n};\n\nexport const createPaymentTransfer = (context: ApiContext, transfer: PaymentTransfer): Promise<PaymentTransfer> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {amount, currency} = transfer;\n  const formatAmount: number = parseNum(amount);\n  const formatCurrency: string = parseChar(currency, 3, 'USD').toUpperCase();\n\n  return getUser(context, {userId: sessionId})\n    .then((user: UserType) => {\n      const {stripeAccountId} = user;\n      const stripeClient = getStripeClient();\n\n      return stripeClient.transfers\n        .create({\n          amount: formatAmount,\n          currency: formatCurrency,\n          destination: stripeAccountId\n        })\n        .then((stripeTransfer) => {\n          console.log(stripeTransfer);\n          const now: number = Date.now();\n          const insert: PaymentTransfer = {\n            added: now,\n            amount: formatAmount,\n            currency: formatCurrency,\n            modified: now,\n            userId: sessionId\n          };\n          const aqlQry: AqlQuery = aql`INSERT ${insert} IN transfers RETURN NEW`;\n\n          return useDb(databaseName).query(aqlQry)\n            .then((cursor) => cursor.next())\n            .then((newTransfer: PaymentTransfer) => newTransfer);\n        });\n    });\n};\n\nexport const createPaymentHold = (context: ApiContext, payment: PaymentCharge): Promise<PaymentCharge> => {\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {amount, capture, cardId, currency, description} = payment;\n  const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();\n  const stripeClient = getStripeClient();\n\n  return stripeClient.charges\n    .create({\n      amount,\n      capture,\n      currency: formatCurrency,\n      description,\n      source: cardId\n    })\n    .then((stripeCharge) => {\n      const now: number = Date.now();\n      const insert: PaymentCharge = {\n        added: now,\n        amount,\n        capture,\n        cardId,\n        chargeFailCode: stripeCharge.failure_code,\n        chargeFailMsg: stripeCharge.failure_message,\n        chargeId: stripeCharge.id,\n        chargeStatus: stripeCharge.status,\n        currency: formatCurrency,\n        description,\n        modified: now,\n        userId: sessionId\n      };\n      const aqlQry: AqlQuery = aql`INSERT ${insert} IN payments RETURN NEW`;\n\n      return useDb(databaseName).query(aqlQry)\n        .then((cursor) => cursor.next())\n        .then((newPayment: PaymentCharge) => newPayment);\n    })\n    .catch((error: Error) => {\n      console.log('payments::createHold::error', error);\n      throw new UserError('payment_error');\n    });\n};\n\ninterface Card {\n  id: string;\n  brand?: string;\n  cvc_check?: string;\n  last4?: string;\n}\n"],"names":["parseNum","createHash","parseChar","parseId","parseString","parseVarChar","aql","ErrorTypes","logError","UserError","useDb","getStripeClient","getUser","eventCategory","addCustomerAccount","context","action","databaseName","session","userId","sessionId","username","stripeClient","customers","create","metadata","then","customer","now","Date","update","modified","stripeCustomerId","id","aqlQry","query","cursor","next","updatedUser","catch","error","category","label","DATABASE_ERROR","addBankAccount","bankAccount","accountNumber","fullName","routing","formatAccount","formatFullName","formatRouting","user","stripeAccountId","tokens","bank_account","account_holder_name","account_holder_type","account_number","country","currency","routing_number","token","createSource","source","account","bankFullName","bankId","bankRouting","msg","message","addCreditCard","card","city","cvc","expMonth","expYear","street1","street2","state","zip","formatNumber","formatExpMonth","formatExpYear","formatCvc","paymentCard","formatCity","formatCountry","formatStreet1","formatStreet2","formatState","formatZip","address_city","address_country","address_line1","address_line2","address_state","address_zip","toString","exp_month","exp_year","name","number","newSource","cardSource","brand","cvcCheck","cvc_check","last4","insert","_key","added","insertAqlQry","newCard","_id","cardId","cardKey","edgeCollection","collection","edgeId","edge","_from","_to","save","returnNew","updateCreditCard","formatId","results","updatedCard","stripeId","updateSource","console","log","getCreditCards","all","list","deleteCreditCard","formatCardId","result","outEdges","response","edges","Array","isArray","length","Promise","map","edgeKey","removeAqlQry","deleteSource","deleteBankAccount","resolve","createPaymentTransfer","transfer","amount","formatAmount","formatCurrency","toUpperCase","transfers","destination","stripeTransfer","newTransfer","createPaymentHold","payment","capture","description","charges","stripeCharge","chargeFailCode","failure_code","chargeFailMsg","failure_message","chargeId","chargeStatus","status","newPayment"],"mappings":"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"}
|