@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/images.js
CHANGED
|
@@ -1,37 +1,176 @@
|
|
|
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 { get as httpGet } from '@nlabs/rip-hunter';
|
|
5
|
+
import { parseNum } from '@nlabs/utils/parsers/numbers';
|
|
6
|
+
import { createHash, parseArangoId, parseChar, parseId } from '@nlabs/utils/parsers/strings';
|
|
7
|
+
import { aql } from 'arangojs';
|
|
8
|
+
import gm from 'gm';
|
|
9
|
+
import { DateTime } from 'luxon';
|
|
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 { getGroupDetails, isGrouped } from './groups.js';
|
|
15
|
+
import { s3DeleteList, s3GetSignedUrl, s3Put } from './s3.js';
|
|
16
|
+
import { lowerCaseKeys } from '@nlabs/utils/parsers/objects';
|
|
17
|
+
const eventCategory = 'images';
|
|
18
|
+
gm.subClass({
|
|
19
|
+
imageMagick: '7+'
|
|
20
|
+
});
|
|
21
|
+
export const parseImageOptions = (options = {})=>{
|
|
22
|
+
const { from = 0, to = 30, type = 'default' } = options;
|
|
23
|
+
return {
|
|
24
|
+
limit: getLimit(from, to),
|
|
25
|
+
type: parseChar(type, 32)
|
|
26
|
+
};
|
|
27
|
+
};
|
|
28
|
+
export const getImageOptional = (fields = [])=>fields.reduce((selects, field)=>{
|
|
29
|
+
if (field.includes('Count')) {
|
|
30
|
+
return selectReactionCountByType('images', 'i', field, selects);
|
|
31
|
+
}
|
|
32
|
+
switch(field){
|
|
33
|
+
case 'reactions':
|
|
34
|
+
{
|
|
35
|
+
selects.queries.push(`LET reactions = (
|
|
2
36
|
FOR image, r IN INBOUND i._id reactions
|
|
3
37
|
COLLECT reactionName = r.value INTO reactionItems
|
|
4
38
|
RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}
|
|
5
|
-
)`)
|
|
39
|
+
)`);
|
|
40
|
+
selects.objects.push('reactions:reactions');
|
|
41
|
+
return selects;
|
|
42
|
+
}
|
|
43
|
+
case 'tags':
|
|
44
|
+
{
|
|
45
|
+
selects.queries.push(`LET tags = (
|
|
6
46
|
FOR t, pl IN INBOUND i._id isTagged
|
|
7
47
|
RETURN t
|
|
8
|
-
)`)
|
|
48
|
+
)`);
|
|
49
|
+
selects.objects.push('tags:tags');
|
|
50
|
+
return selects;
|
|
51
|
+
}
|
|
52
|
+
case 'users':
|
|
53
|
+
{
|
|
54
|
+
selects.queries.push(`LET users = FIRST(
|
|
9
55
|
FOR u IN users
|
|
10
56
|
FILTER i.userId == u._key
|
|
11
57
|
LIMIT 1
|
|
12
58
|
RETURN u
|
|
13
|
-
)`)
|
|
14
|
-
|
|
59
|
+
)`);
|
|
60
|
+
selects.objects.push('users:users');
|
|
61
|
+
return selects;
|
|
62
|
+
}
|
|
63
|
+
default:
|
|
64
|
+
{
|
|
65
|
+
return selects;
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
}, {
|
|
69
|
+
objects: [],
|
|
70
|
+
queries: []
|
|
71
|
+
});
|
|
72
|
+
export const getImagesByUser = (context, userId, from = 0, to = 30)=>{
|
|
73
|
+
const action = 'getImagesByUser';
|
|
74
|
+
const { databaseName } = context;
|
|
75
|
+
const formatUserId = parseId(userId);
|
|
76
|
+
const limit = getLimit(from, to);
|
|
77
|
+
const aqlQry = `FOR i IN images
|
|
78
|
+
FILTER i.userId == "${formatUserId}"
|
|
15
79
|
LET user = FIRST(
|
|
16
80
|
FOR u IN users
|
|
17
81
|
FILTER u._key == i.userId
|
|
18
82
|
LIMIT 1
|
|
19
83
|
RETURN u
|
|
20
84
|
)
|
|
21
|
-
${
|
|
85
|
+
${limit.aql}
|
|
22
86
|
SORT i.added
|
|
23
|
-
RETURN MERGE(i, {user:user})`;
|
|
24
|
-
|
|
25
|
-
|
|
87
|
+
RETURN MERGE(i, {user:user})`;
|
|
88
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.all()).catch((error)=>{
|
|
89
|
+
logError({
|
|
90
|
+
action,
|
|
91
|
+
category: eventCategory,
|
|
92
|
+
params: {
|
|
93
|
+
from,
|
|
94
|
+
to,
|
|
95
|
+
userId
|
|
96
|
+
},
|
|
97
|
+
value: ErrorTypes.DATABASE_ERROR
|
|
98
|
+
}, error, context);
|
|
99
|
+
return [];
|
|
100
|
+
});
|
|
101
|
+
};
|
|
102
|
+
export const getImageCountByItem = (context, itemId)=>{
|
|
103
|
+
const action = 'getImageCountByItem';
|
|
104
|
+
const { databaseName } = context;
|
|
105
|
+
const formatItemId = parseArangoId(itemId);
|
|
106
|
+
const aqlQry = aql`FOR i IN hasImage
|
|
107
|
+
FILTER i._to == ${formatItemId}
|
|
108
|
+
RETURN {count: COUNT(i)}`;
|
|
109
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then(({ count })=>count).catch((error)=>logError({
|
|
110
|
+
action,
|
|
111
|
+
category: eventCategory,
|
|
112
|
+
params: {
|
|
113
|
+
itemId
|
|
114
|
+
},
|
|
115
|
+
value: ErrorTypes.DATABASE_ERROR
|
|
116
|
+
}, error, context));
|
|
117
|
+
};
|
|
118
|
+
export const getImagesByItem = async (context, itemId, options = {})=>{
|
|
119
|
+
const action = 'getImagesByItem';
|
|
120
|
+
const { databaseName, fields = [] } = context;
|
|
121
|
+
const formatItemId = parseArangoId(itemId);
|
|
122
|
+
const { limit } = parseImageOptions(options);
|
|
123
|
+
const { objects: selectObjects, queries: selectQueries } = getImageOptional(fields);
|
|
124
|
+
const aqlQry = `FOR i, l IN 1..1 OUTBOUND "${formatItemId}" hasImage
|
|
26
125
|
FILTER NOT IS_NULL(i)
|
|
27
|
-
${
|
|
28
|
-
`)}
|
|
126
|
+
${selectQueries.join('\n')}
|
|
29
127
|
SORT i.added
|
|
30
|
-
${
|
|
31
|
-
RETURN MERGE(i, {${
|
|
128
|
+
${limit.aql}
|
|
129
|
+
RETURN MERGE(i, {${selectObjects.join(', ')}})`;
|
|
130
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.all()).catch((error)=>{
|
|
131
|
+
logError({
|
|
132
|
+
action,
|
|
133
|
+
category: eventCategory,
|
|
134
|
+
params: {
|
|
135
|
+
itemId,
|
|
136
|
+
options
|
|
137
|
+
},
|
|
138
|
+
value: ErrorTypes.DATABASE_ERROR
|
|
139
|
+
}, error, context);
|
|
140
|
+
return [];
|
|
141
|
+
});
|
|
142
|
+
};
|
|
143
|
+
export const getImagesByGroup = (context, params)=>{
|
|
144
|
+
const action = 'getImagesByGroup';
|
|
145
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
146
|
+
const { filters, groupId, from, to } = params;
|
|
147
|
+
const formatGroupId = parseId(groupId);
|
|
148
|
+
const limit = getLimit(from, to);
|
|
149
|
+
filters.map((filter)=>{
|
|
150
|
+
const { conditional, name, value } = filter;
|
|
151
|
+
let formatCond = conditional;
|
|
152
|
+
if (conditional !== '>=' && conditional !== '<=' && conditional !== '>' && conditional !== '<') {
|
|
153
|
+
formatCond = '==';
|
|
154
|
+
}
|
|
155
|
+
switch(name){
|
|
156
|
+
case 'added':
|
|
157
|
+
return `p.added ${formatCond} ${parseNum(value)}`;
|
|
158
|
+
default:
|
|
159
|
+
return '';
|
|
160
|
+
}
|
|
161
|
+
});
|
|
162
|
+
return getGroupDetails(context, formatGroupId).then((group)=>{
|
|
163
|
+
if (group.privacy === 'public') {
|
|
164
|
+
filters.push({
|
|
165
|
+
conditional: '==',
|
|
166
|
+
name: 'groupId',
|
|
167
|
+
value: formatGroupId
|
|
168
|
+
});
|
|
169
|
+
const filterStr = filters.join(' && ');
|
|
170
|
+
const aqlQry = `FOR i IN
|
|
32
171
|
FLATTEN(
|
|
33
172
|
FOR p IN posts
|
|
34
|
-
FILTER ${
|
|
173
|
+
FILTER ${filterStr}
|
|
35
174
|
LET images = (
|
|
36
175
|
FOR i, e IN INBOUND p._id hasImage
|
|
37
176
|
RETURN i
|
|
@@ -40,27 +179,545 @@ import{get as J}from"@nlabs/rip-hunter";import{parseNum as Z}from"@nlabs/utils/p
|
|
|
40
179
|
RETURN images
|
|
41
180
|
)
|
|
42
181
|
SORT i.added DESC
|
|
43
|
-
${
|
|
44
|
-
RETURN i`;
|
|
45
|
-
|
|
182
|
+
${limit.aql}
|
|
183
|
+
RETURN i`;
|
|
184
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.all()).catch((error)=>{
|
|
185
|
+
logError({
|
|
186
|
+
action,
|
|
187
|
+
category: eventCategory,
|
|
188
|
+
params: {
|
|
189
|
+
filters,
|
|
190
|
+
from,
|
|
191
|
+
groupId,
|
|
192
|
+
to
|
|
193
|
+
},
|
|
194
|
+
value: ErrorTypes.DATABASE_ERROR
|
|
195
|
+
}, error, context);
|
|
196
|
+
return [];
|
|
197
|
+
});
|
|
198
|
+
}
|
|
199
|
+
return isGrouped(context, sessionId, groupId).then((grouped)=>{
|
|
200
|
+
if (grouped.isValid) {
|
|
201
|
+
filters.push({
|
|
202
|
+
conditional: '==',
|
|
203
|
+
name: 'groupId',
|
|
204
|
+
value: grouped.groupId
|
|
205
|
+
});
|
|
206
|
+
const filterList = filters.join(' && ');
|
|
207
|
+
const aqlQry = `FOR p IN post
|
|
208
|
+
FILTER ${filterList}
|
|
46
209
|
FOR f IN p.files
|
|
47
210
|
FILTER f.type == "image/jpeg" || f.type == "image/png"
|
|
48
|
-
${
|
|
211
|
+
${limit.aql}
|
|
49
212
|
SORT p.added DESC
|
|
50
|
-
RETURN f`;
|
|
213
|
+
RETURN f`;
|
|
214
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.all()).catch((error)=>{
|
|
215
|
+
logError({
|
|
216
|
+
action,
|
|
217
|
+
category: eventCategory,
|
|
218
|
+
params: {
|
|
219
|
+
filters,
|
|
220
|
+
from,
|
|
221
|
+
groupId,
|
|
222
|
+
to
|
|
223
|
+
},
|
|
224
|
+
value: ErrorTypes.DATABASE_ERROR
|
|
225
|
+
}, error, context);
|
|
226
|
+
return [];
|
|
227
|
+
});
|
|
228
|
+
}
|
|
229
|
+
return [];
|
|
230
|
+
});
|
|
231
|
+
});
|
|
232
|
+
};
|
|
233
|
+
export const getImagesByReactions = (context, reactions = [], options)=>{
|
|
234
|
+
const action = 'getUsersByImage';
|
|
235
|
+
const { databaseName, fields = [], session: { userId: sessionId } } = context;
|
|
236
|
+
const { limit } = parseImageOptions(options);
|
|
237
|
+
const { objects: selectObjects, queries: selectQueries } = getImageOptional(fields);
|
|
238
|
+
const formatSessionId = `users/${sessionId}`;
|
|
239
|
+
const formatReactions = reactions.map((reactionName)=>parseChar(reactionName, 32));
|
|
240
|
+
const filterBy = [
|
|
241
|
+
'r.type == "images"',
|
|
242
|
+
`POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`
|
|
243
|
+
];
|
|
244
|
+
// Get data from database
|
|
245
|
+
const aqlQry = `FOR i, r IN OUTBOUND "${formatSessionId}" hasReaction
|
|
51
246
|
OPTIONS {vertexCollections: "images"}
|
|
52
|
-
${
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
${
|
|
56
|
-
|
|
57
|
-
|
|
247
|
+
${selectQueries.join('\n')}
|
|
248
|
+
FILTER ${filterBy.join(' && ')}
|
|
249
|
+
${limit.aql}
|
|
250
|
+
RETURN MERGE(i, {${selectObjects.join(', ')}})`;
|
|
251
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.all()).catch((error)=>{
|
|
252
|
+
logError({
|
|
253
|
+
action,
|
|
254
|
+
category: eventCategory,
|
|
255
|
+
params: {
|
|
256
|
+
options,
|
|
257
|
+
reactions
|
|
258
|
+
},
|
|
259
|
+
value: ErrorTypes.DATABASE_ERROR
|
|
260
|
+
}, error, context);
|
|
261
|
+
return [];
|
|
262
|
+
});
|
|
263
|
+
};
|
|
264
|
+
export const getImage = (context, imageId)=>{
|
|
265
|
+
const action = 'getItem';
|
|
266
|
+
const { databaseName } = context;
|
|
267
|
+
const formatImageId = parseId(imageId);
|
|
268
|
+
const aqlQry = aql`FOR i IN images
|
|
269
|
+
FILTER i._key==${formatImageId}
|
|
58
270
|
LIMIT 1
|
|
59
|
-
RETURN i`;
|
|
60
|
-
|
|
271
|
+
RETURN i`;
|
|
272
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).catch((error)=>logError({
|
|
273
|
+
action,
|
|
274
|
+
category: eventCategory,
|
|
275
|
+
params: {
|
|
276
|
+
imageId
|
|
277
|
+
},
|
|
278
|
+
value: ErrorTypes.DATABASE_ERROR
|
|
279
|
+
}, error, context));
|
|
280
|
+
};
|
|
281
|
+
export const getPathUserImages = (userId, imageId, imageType, dir = 'images')=>{
|
|
282
|
+
let filename = imageId;
|
|
283
|
+
switch(imageType){
|
|
284
|
+
case 'image/png':
|
|
285
|
+
filename = `${imageId}.png`;
|
|
286
|
+
break;
|
|
287
|
+
default:
|
|
288
|
+
filename = `${imageId}.jpg`;
|
|
289
|
+
break;
|
|
290
|
+
}
|
|
291
|
+
return `users/${userId}/${dir}/${filename}`;
|
|
292
|
+
};
|
|
293
|
+
export const getAppImageUrl = ({ bucket, directory = 'images', imageId, imageType = 'jpg', isThumb = false, type, typeId })=>{
|
|
294
|
+
if (!type) {
|
|
295
|
+
return '';
|
|
296
|
+
}
|
|
297
|
+
if (!imageId) {
|
|
298
|
+
if (imageType === 'profile') {
|
|
299
|
+
return `${Config.get('image.defaultUrl')}/users_bk.jpg`;
|
|
300
|
+
}
|
|
301
|
+
return `${Config.get('image.defaultUrl')}/users_wh.jpg`;
|
|
302
|
+
}
|
|
303
|
+
const bucketName = bucket || Config.get('image.bucket') || 'dev.reaktor.io';
|
|
304
|
+
const imageName = isThumb ? `${imageId}-th.${imageType}` : `${imageId}.${imageType === 'profile' || imageType === 'other' ? 'jpg' : imageType}`;
|
|
305
|
+
const imageUrl = Config.get('image.url') || `https://s3.amazonaws.com/${bucketName}`;
|
|
306
|
+
if (type === 'apps') {
|
|
307
|
+
return `${imageUrl}/${type}/${directory}/${imageName}`;
|
|
308
|
+
}
|
|
309
|
+
return `${imageUrl}/${type}/${typeId}/${directory}/${imageName}`;
|
|
310
|
+
};
|
|
311
|
+
export const getImageUrl = async ({ bucket, imageDir = 'images', imageId, imageType = 'jpg', isThumb = false, type, typeId, urlType = 'public' })=>{
|
|
312
|
+
if (!imageId) {
|
|
313
|
+
return '';
|
|
314
|
+
}
|
|
315
|
+
// Always use signed URLs for thumbnails
|
|
316
|
+
if (isThumb) {
|
|
317
|
+
urlType = 'signed';
|
|
318
|
+
}
|
|
319
|
+
const thumbSuffix = isThumb ? '-th' : '';
|
|
320
|
+
const imageKey = `${type}/${typeId}/${imageDir}/${imageId}${thumbSuffix}.${imageType}`;
|
|
321
|
+
switch(urlType){
|
|
322
|
+
case 'signed':
|
|
323
|
+
{
|
|
324
|
+
try {
|
|
325
|
+
const defaultBucket = Config.get('image.bucket') || 'dev.reaktor.io';
|
|
326
|
+
return await s3GetSignedUrl({
|
|
327
|
+
Bucket: bucket || defaultBucket,
|
|
328
|
+
Expires: 900,
|
|
329
|
+
Key: imageKey
|
|
330
|
+
});
|
|
331
|
+
} catch (error) {
|
|
332
|
+
throw error;
|
|
333
|
+
}
|
|
334
|
+
}
|
|
335
|
+
case 'public':
|
|
336
|
+
{
|
|
337
|
+
return `https://box.${Config.get('app.url')}/${imageKey}`;
|
|
338
|
+
}
|
|
339
|
+
case 'dev':
|
|
340
|
+
{
|
|
341
|
+
return `https://s3.amazonaws.com/dev.${Config.get('app.url')}/${imageKey}`;
|
|
342
|
+
}
|
|
343
|
+
default:
|
|
344
|
+
{
|
|
345
|
+
return '';
|
|
346
|
+
}
|
|
347
|
+
}
|
|
348
|
+
};
|
|
349
|
+
export const resizeSaveImage = async (context, imageId, buffer, fileType = 'image/jpeg', s3Options)=>{
|
|
350
|
+
const action = 'resizeSaveImage';
|
|
351
|
+
const { session: { userId: sessionId } } = context;
|
|
352
|
+
const imgW = Config.get('image.imgSize');
|
|
353
|
+
const imgQ = Config.get('image.imgQuality');
|
|
354
|
+
const format = fileType.split('/')[1];
|
|
355
|
+
const imageOptimizedBuffer = await new Promise((resolve, reject)=>{
|
|
356
|
+
gm(buffer, 'img').setFormat(format).quality(imgQ).autoOrient().resize(imgW, imgW, '>').stream((streamError, stdout)=>{
|
|
357
|
+
if (streamError) {
|
|
358
|
+
reject(logError({
|
|
359
|
+
action,
|
|
360
|
+
category: eventCategory,
|
|
361
|
+
params: {
|
|
362
|
+
fileType,
|
|
363
|
+
imageId
|
|
364
|
+
},
|
|
365
|
+
value: ErrorTypes.IMAGE_SAVE
|
|
366
|
+
}, streamError, context));
|
|
367
|
+
return;
|
|
368
|
+
}
|
|
369
|
+
let imageBuffer = Buffer.from('');
|
|
370
|
+
stdout.on('data', (data)=>{
|
|
371
|
+
imageBuffer = Buffer.concat([
|
|
372
|
+
imageBuffer,
|
|
373
|
+
data
|
|
374
|
+
]);
|
|
375
|
+
});
|
|
376
|
+
stdout.on('end', ()=>resolve(imageBuffer));
|
|
377
|
+
});
|
|
378
|
+
});
|
|
379
|
+
try {
|
|
380
|
+
const imageObj = {
|
|
381
|
+
ACL: 'authenticated-read',
|
|
382
|
+
Body: imageOptimizedBuffer,
|
|
383
|
+
Bucket: null,
|
|
384
|
+
ContentType: fileType,
|
|
385
|
+
...s3Options || {},
|
|
386
|
+
Key: getPathUserImages(sessionId, imageId, fileType, 'images')
|
|
387
|
+
};
|
|
388
|
+
await s3Put(imageObj);
|
|
389
|
+
} catch (error) {
|
|
390
|
+
return logError({
|
|
391
|
+
action,
|
|
392
|
+
category: eventCategory,
|
|
393
|
+
params: {
|
|
394
|
+
fileType,
|
|
395
|
+
imageId
|
|
396
|
+
},
|
|
397
|
+
value: ErrorTypes.IMAGE_SAVE
|
|
398
|
+
}, error, context);
|
|
399
|
+
}
|
|
400
|
+
const imageIdentity = await new Promise((resolve)=>{
|
|
401
|
+
gm(imageOptimizedBuffer, 'img').identify({
|
|
402
|
+
bufferStream: true
|
|
403
|
+
}, (error, imageData = {})=>{
|
|
404
|
+
if (error) {
|
|
405
|
+
return logError({
|
|
406
|
+
action,
|
|
407
|
+
category: eventCategory,
|
|
408
|
+
value: ErrorTypes.IMAGE_SAVE
|
|
409
|
+
}, error, context);
|
|
410
|
+
}
|
|
411
|
+
const formatValues = lowerCaseKeys(imageData);
|
|
412
|
+
const { make: cameraMake, model: cameraModel, datetimeoriginal: taken } = formatValues;
|
|
413
|
+
const stats = formatValues['channel statistics'];
|
|
414
|
+
let color;
|
|
415
|
+
if (stats) {
|
|
416
|
+
let { red, green, blue, mean } = stats;
|
|
417
|
+
if (red) {
|
|
418
|
+
mean = red['standard deviation'] || red.mean;
|
|
419
|
+
red = mean ? +mean.split(' ')[0].substring(0, 3) : 0;
|
|
420
|
+
} else {
|
|
421
|
+
red = 0;
|
|
422
|
+
}
|
|
423
|
+
if (green) {
|
|
424
|
+
mean = green['standard deviation'] || green.mean;
|
|
425
|
+
green = mean ? +mean.split(' ')[0].substring(0, 3) : 0;
|
|
426
|
+
} else {
|
|
427
|
+
green = 0;
|
|
428
|
+
}
|
|
429
|
+
if (blue) {
|
|
430
|
+
mean = blue['standard deviation'] || blue.mean;
|
|
431
|
+
blue = mean ? +mean.split(' ')[0].substring(0, 3) : 0;
|
|
432
|
+
} else {
|
|
433
|
+
blue = 0;
|
|
434
|
+
}
|
|
435
|
+
const rgb = blue | green << 8 | red << 16;
|
|
436
|
+
const rgbColor = rgb.toString(16);
|
|
437
|
+
color = rgbColor === '0' ? '000000' : rgbColor;
|
|
438
|
+
}
|
|
439
|
+
const { 'JPEG-Quality': quality, Orientation: orientationData, Resolution: resolutionData, size } = imageData;
|
|
440
|
+
let resolution;
|
|
441
|
+
if (resolutionData) {
|
|
442
|
+
const [resolutionNumber, resolutionUnit] = resolutionData.split(' ');
|
|
443
|
+
const resolutionValue = resolutionNumber.split('x')[0];
|
|
444
|
+
const resolutionUnitValue = resolutionUnit === 'pixels/inch' ? 'ppi' : '';
|
|
445
|
+
resolution = `${resolutionValue} ${resolutionUnitValue}`;
|
|
446
|
+
}
|
|
447
|
+
return resolve({
|
|
448
|
+
color,
|
|
449
|
+
fileSize: imageOptimizedBuffer.length,
|
|
450
|
+
fileType: format,
|
|
451
|
+
height: size?.height || 0,
|
|
452
|
+
imageId,
|
|
453
|
+
make: cameraMake,
|
|
454
|
+
model: cameraModel,
|
|
455
|
+
orientation: orientationData === 'Unknown' ? undefined : orientationData?.toLowerCase(),
|
|
456
|
+
quality: quality ? +quality : undefined,
|
|
457
|
+
resolution,
|
|
458
|
+
taken: taken ? DateTime.fromMillis(taken).millisecond : undefined,
|
|
459
|
+
width: size?.width || 0
|
|
460
|
+
});
|
|
461
|
+
});
|
|
462
|
+
});
|
|
463
|
+
const thumbOptimizedBuffer = await new Promise((resolve, reject)=>{
|
|
464
|
+
const thmW = Config.get('image.thmSize');
|
|
465
|
+
const thmQ = Config.get('image.thmQuality');
|
|
466
|
+
gm(imageOptimizedBuffer, 'img').setFormat(format).gravity('Center').resize(thmW, thmW, '^').extent(thmW, thmW).quality(thmQ).stream((streamError, thumbStdout)=>{
|
|
467
|
+
if (streamError) {
|
|
468
|
+
logError({
|
|
469
|
+
action,
|
|
470
|
+
category: eventCategory,
|
|
471
|
+
params: {
|
|
472
|
+
fileType,
|
|
473
|
+
imageId
|
|
474
|
+
},
|
|
475
|
+
value: ErrorTypes.IMAGE_SAVE
|
|
476
|
+
}, streamError, context);
|
|
477
|
+
return;
|
|
478
|
+
}
|
|
479
|
+
let thumbBuffer = Buffer.from('');
|
|
480
|
+
thumbStdout.on('data', (data)=>{
|
|
481
|
+
thumbBuffer = Buffer.concat([
|
|
482
|
+
thumbBuffer,
|
|
483
|
+
data
|
|
484
|
+
]);
|
|
485
|
+
});
|
|
486
|
+
thumbStdout.on('end', ()=>resolve(thumbBuffer));
|
|
487
|
+
});
|
|
488
|
+
});
|
|
489
|
+
try {
|
|
490
|
+
const thumbObj = {
|
|
491
|
+
ACL: 'authenticated-read',
|
|
492
|
+
Body: thumbOptimizedBuffer,
|
|
493
|
+
Bucket: null,
|
|
494
|
+
ContentType: fileType,
|
|
495
|
+
...s3Options || {},
|
|
496
|
+
Key: getPathUserImages(sessionId, imageId, fileType, 'thumbs')
|
|
497
|
+
};
|
|
498
|
+
await s3Put(thumbObj);
|
|
499
|
+
} catch (error) {
|
|
500
|
+
return logError({
|
|
501
|
+
action,
|
|
502
|
+
category: eventCategory,
|
|
503
|
+
value: ErrorTypes.IMAGE_SAVE
|
|
504
|
+
}, error, context);
|
|
505
|
+
}
|
|
506
|
+
return imageIdentity;
|
|
507
|
+
};
|
|
508
|
+
export const addImage = (context, image, s3Options)=>{
|
|
509
|
+
const action = 'addImage';
|
|
510
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
511
|
+
const { imageId, description, buffer, fileType } = image;
|
|
512
|
+
const now = Date.now();
|
|
513
|
+
return resizeSaveImage(context, imageId, buffer, fileType, s3Options).then((resizedImage)=>{
|
|
514
|
+
const insert = {
|
|
515
|
+
...resizedImage,
|
|
516
|
+
_key: imageId,
|
|
517
|
+
added: now,
|
|
518
|
+
description,
|
|
519
|
+
fileType,
|
|
520
|
+
modified: now,
|
|
521
|
+
userId: sessionId
|
|
522
|
+
};
|
|
523
|
+
const aqlQry = aql`INSERT ${insert} IN images RETURN NEW`;
|
|
524
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).catch((error)=>logError({
|
|
525
|
+
action,
|
|
526
|
+
category: eventCategory,
|
|
527
|
+
params: {
|
|
528
|
+
image,
|
|
529
|
+
s3Options
|
|
530
|
+
},
|
|
531
|
+
value: ErrorTypes.DATABASE_ERROR
|
|
532
|
+
}, error, context));
|
|
533
|
+
}).catch((error)=>logError({
|
|
534
|
+
action,
|
|
535
|
+
category: eventCategory,
|
|
536
|
+
value: ErrorTypes.IMAGE_RESIZE
|
|
537
|
+
}, error, context));
|
|
538
|
+
};
|
|
539
|
+
export const addImageEdge = async (context, imageEdge)=>{
|
|
540
|
+
const action = 'addImageEdge';
|
|
541
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
542
|
+
const { imageId, itemId, itemType } = imageEdge;
|
|
543
|
+
const now = Date.now();
|
|
544
|
+
// const edgeCollection: EdgeCollection = useDb(databaseName).collection('hasImage');
|
|
545
|
+
const edgeId = createHash(`hasImage-${imageId}-${itemId}-${sessionId}`);
|
|
546
|
+
const formatItemType = parseChar(itemType).toLowerCase();
|
|
547
|
+
const formatItemId = parseId(itemId);
|
|
548
|
+
const itemDocId = getDocId(formatItemType, {
|
|
549
|
+
_key: formatItemId
|
|
550
|
+
});
|
|
551
|
+
const formatImageId = parseId(imageId);
|
|
552
|
+
const fileDocId = `images/${formatImageId}`;
|
|
553
|
+
const edge = {
|
|
554
|
+
_from: itemDocId,
|
|
555
|
+
_key: edgeId,
|
|
556
|
+
_to: fileDocId,
|
|
557
|
+
added: now,
|
|
558
|
+
itemType
|
|
559
|
+
};
|
|
560
|
+
if (itemDocId) {
|
|
561
|
+
return useDb(databaseName).collection('hasImage').save(edge, {
|
|
562
|
+
returnNew: true
|
|
563
|
+
}).then((fileEdge)=>useDb(databaseName).collection('hasImage').document(fileEdge)).catch((error)=>logError({
|
|
564
|
+
action,
|
|
565
|
+
category: eventCategory,
|
|
566
|
+
params: {
|
|
567
|
+
edge,
|
|
568
|
+
fileDocId,
|
|
569
|
+
imageId,
|
|
570
|
+
itemDocId,
|
|
571
|
+
itemId,
|
|
572
|
+
itemType
|
|
573
|
+
},
|
|
574
|
+
value: ErrorTypes.DATABASE_ERROR
|
|
575
|
+
}, error, context));
|
|
576
|
+
}
|
|
577
|
+
return null;
|
|
578
|
+
};
|
|
579
|
+
export const updateImage = async (context, image, s3Options)=>{
|
|
580
|
+
const action = 'updateImage';
|
|
581
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
582
|
+
const { base64 = '', buffer: imageBuffer, description = '', imageId, itemId, itemType, fileType, url = '' } = image;
|
|
583
|
+
const now = Date.now();
|
|
584
|
+
const formatImageId = imageId || createHash(`image-${sessionId}`);
|
|
585
|
+
const formatItemId = parseId(itemId);
|
|
586
|
+
const formatItemType = parseChar(itemType, 16).toLowerCase();
|
|
587
|
+
const customParams = {
|
|
588
|
+
description,
|
|
589
|
+
imageId: formatImageId,
|
|
590
|
+
userId: sessionId
|
|
591
|
+
};
|
|
592
|
+
if (base64 || imageBuffer) {
|
|
593
|
+
let buffer = imageBuffer;
|
|
594
|
+
if (base64) {
|
|
595
|
+
const formatBase64 = base64.substr(base64.indexOf(',') + 1);
|
|
596
|
+
buffer = Buffer.from(formatBase64, 'base64');
|
|
597
|
+
}
|
|
598
|
+
let updatedFileType = fileType;
|
|
599
|
+
if (!fileType) {
|
|
600
|
+
const { fileTypeFromBuffer } = await import('file-type');
|
|
601
|
+
const { mime } = await fileTypeFromBuffer(buffer);
|
|
602
|
+
updatedFileType = mime;
|
|
603
|
+
}
|
|
604
|
+
const imgParams = {
|
|
605
|
+
...customParams,
|
|
606
|
+
buffer,
|
|
607
|
+
fileType: updatedFileType
|
|
608
|
+
};
|
|
609
|
+
return addImage(context, imgParams, s3Options).then((image)=>{
|
|
610
|
+
if (formatItemId && formatItemType) {
|
|
611
|
+
const imageEdge = {
|
|
612
|
+
imageId: formatImageId,
|
|
613
|
+
itemId: formatItemId,
|
|
614
|
+
itemType: formatItemType
|
|
615
|
+
};
|
|
616
|
+
return addImageEdge(context, imageEdge).then(()=>image);
|
|
617
|
+
}
|
|
618
|
+
return image;
|
|
619
|
+
}).catch((error)=>logError({
|
|
620
|
+
action,
|
|
621
|
+
category: eventCategory,
|
|
622
|
+
value: ErrorTypes.IMAGE_SAVE
|
|
623
|
+
}, error, context));
|
|
624
|
+
} else if (url !== '') {
|
|
625
|
+
let contentType;
|
|
626
|
+
return httpGet(url).then((res)=>{
|
|
627
|
+
if (res.status !== 200) {
|
|
628
|
+
return logException({
|
|
629
|
+
action,
|
|
630
|
+
category: eventCategory,
|
|
631
|
+
value: ErrorTypes.IMAGE_REQUEST
|
|
632
|
+
}, context);
|
|
633
|
+
}
|
|
634
|
+
contentType = res.headers.get('content-type');
|
|
635
|
+
return res;
|
|
636
|
+
}).then((res)=>res.buffer()).then((buffer)=>{
|
|
637
|
+
const imgParams = {
|
|
638
|
+
...customParams,
|
|
639
|
+
buffer,
|
|
640
|
+
fileType: contentType
|
|
641
|
+
};
|
|
642
|
+
return addImage(context, imgParams, s3Options).then((image)=>{
|
|
643
|
+
if (formatItemId && formatItemType) {
|
|
644
|
+
const imageEdge = {
|
|
645
|
+
imageId: formatImageId,
|
|
646
|
+
itemId: formatItemId,
|
|
647
|
+
itemType: formatItemType
|
|
648
|
+
};
|
|
649
|
+
return addImageEdge(context, imageEdge).then(()=>image);
|
|
650
|
+
}
|
|
651
|
+
return image;
|
|
652
|
+
});
|
|
653
|
+
}).catch((error)=>logError({
|
|
654
|
+
action,
|
|
655
|
+
category: eventCategory,
|
|
656
|
+
value: 'fetch_error'
|
|
657
|
+
}, error, context));
|
|
658
|
+
} else if (imageId !== '') {
|
|
659
|
+
const update = {
|
|
660
|
+
description,
|
|
661
|
+
modified: now
|
|
662
|
+
};
|
|
663
|
+
const aqlQry = aql`UPDATE {_key: ${formatImageId}} WITH ${update} IN images RETURN NEW`;
|
|
664
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).catch((error)=>logError({
|
|
665
|
+
action,
|
|
666
|
+
category: eventCategory,
|
|
667
|
+
params: {
|
|
668
|
+
aqlQry,
|
|
669
|
+
formatImageId,
|
|
670
|
+
update
|
|
671
|
+
},
|
|
672
|
+
value: ErrorTypes.DATABASE_ERROR
|
|
673
|
+
}, error, context));
|
|
674
|
+
}
|
|
675
|
+
return null;
|
|
676
|
+
};
|
|
677
|
+
export const deleteImage = async (context, imageId)=>{
|
|
678
|
+
const action = 'delete';
|
|
679
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
680
|
+
const formatImageId = parseId(imageId);
|
|
681
|
+
try {
|
|
682
|
+
const removeEdgeQuery = aql`FOR hi IN hasImage
|
|
683
|
+
FILTER hi._from == ${formatImageId}
|
|
61
684
|
REMOVE hi IN hasImage
|
|
62
|
-
RETURN OLD`;
|
|
63
|
-
|
|
685
|
+
RETURN OLD`;
|
|
686
|
+
await useDb(databaseName).query(removeEdgeQuery);
|
|
687
|
+
const aqlQuery = aql`FOR i IN images
|
|
688
|
+
FILTER i._key == ${formatImageId} && i.userId == ${sessionId}
|
|
64
689
|
REMOVE i IN images
|
|
65
|
-
RETURN OLD
|
|
66
|
-
//# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../src/actions/images.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 {DeleteObjectsCommandInput, PutObjectCommandInput} from '@aws-sdk/client-s3';\nimport {get as httpGet} from '@nlabs/rip-hunter';\nimport {parseNum} from '@nlabs/utils/parsers/numbers';\nimport {createHash, parseArangoId, parseChar, parseId} from '@nlabs/utils/parsers/strings';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport gm from 'gm';\nimport {DateTime} from 'luxon';\n\nimport {Config} from '../config.js';\nimport {ErrorTypes} from '../types/error.types.js';\nimport {logError, logException} from '../utils/analyticsUtils.js';\nimport {getDocId, getLimit, selectReactionCountByType, useDb} from '../utils/arangodbUtils.js';\nimport {getGroupDetails, isGrouped} from './groups.js';\nimport {s3DeleteList, s3GetSignedUrl, s3Put} from './s3.js';\n\nimport {lowerCaseKeys} from '@nlabs/utils/parsers/objects';\nimport type {QueryFilter} from '../types/apps.types.js';\nimport type {ArangoDbCollection, ArangoDbLimit} from '../types/arangodb.types.js';\nimport type {ApiContext} from '../types/auth.types.js';\nimport type {GroupType, GroupUser} from '../types/groups.types.js';\nimport type {ImageEdgeType, ImageIdentifyType, ImageOptions, ImageType, ImageUrlData} from '../types/images.types.js';\n\n// Define interface for HTTP response with buffer method\ninterface HttpResponse extends Response {\n  buffer(): Promise<Buffer>;\n}\n\nconst eventCategory: string = 'images';\ngm.subClass({imageMagick: '7+'});\n\nexport const parseImageOptions = (options: ImageOptions = {}) => {\n  const {\n    from = 0,\n    to = 30,\n    type = 'default'\n  } = options;\n\n  return {\n    limit: getLimit(from, to) as ArangoDbLimit,\n    type: parseChar(type, 32)\n  };\n};\n\nexport const getImageOptional = (fields: string[] = []) =>\n  fields.reduce((selects, field: string) => {\n    if(field.includes('Count')) {\n      return selectReactionCountByType('images', 'i', field, selects);\n    }\n\n    switch(field) {\n      case 'reactions': {\n        selects.queries.push(`LET reactions = (\n          FOR image, r IN INBOUND i._id reactions\n          COLLECT reactionName = r.value INTO reactionItems\n          RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n        )`);\n        selects.objects.push('reactions:reactions');\n        return selects;\n      }\n      case 'tags': {\n        selects.queries.push(`LET tags = (\n          FOR t, pl IN INBOUND i._id isTagged\n          RETURN t\n        )`);\n        selects.objects.push('tags:tags');\n        return selects;\n      }\n      case 'users': {\n        selects.queries.push(`LET users = FIRST(\n          FOR u IN users\n          FILTER i.userId == u._key\n          LIMIT 1\n          RETURN u\n        )`);\n        selects.objects.push('users:users');\n        return selects;\n      }\n      default: {\n        return selects;\n      }\n    }\n  }, {objects: [], queries: []});\n\nexport const getImagesByUser = (\n  context: ApiContext,\n  userId: string,\n  from: number = 0,\n  to: number = 30\n): Promise<ImageType[]> => {\n  const action: string = 'getImagesByUser';\n  const {databaseName} = context;\n  const formatUserId: string = parseId(userId);\n  const limit: ArangoDbLimit = getLimit(from, to);\n  const aqlQry: string = `FOR i IN images\n      FILTER i.userId == \"${formatUserId}\"\n      LET user = FIRST(\n        FOR u IN users\n        FILTER u._key == i.userId\n        LIMIT 1\n        RETURN u\n      )\n      ${limit.aql}\n      SORT i.added\n      RETURN MERGE(i, {user:user})`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as ImageType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        params: {from, to, userId},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [];\n    });\n};\n\nexport const getImageCountByItem = (context: ApiContext, itemId: string): Promise<number> => {\n  const action: string = 'getImageCountByItem';\n  const {databaseName} = context;\n  const formatItemId: string = parseArangoId(itemId);\n  const aqlQry: AqlQuery = aql`FOR i IN hasImage\n      FILTER i._to == ${formatItemId}\n      RETURN {count: COUNT(i)}`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then(({count}) => count)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {itemId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getImagesByItem = async (\n  context: ApiContext,\n  itemId: string,\n  options: ImageOptions = {}\n): Promise<ImageType[]> => {\n  const action: string = 'getImagesByItem';\n  const {databaseName, fields = []} = context;\n  const formatItemId: string = parseArangoId(itemId);\n  const {limit} = parseImageOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getImageOptional(fields);\n  const aqlQry: string = `FOR i, l IN 1..1 OUTBOUND \"${formatItemId}\" hasImage\n    FILTER NOT IS_NULL(i)\n    ${selectQueries.join('\\n')}\n    SORT i.added\n    ${limit.aql}\n    RETURN MERGE(i, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as ImageType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        params: {itemId, options},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n      return [];\n    });\n};\n\nexport const getImagesByGroup = (\n  context: ApiContext,\n  params: {filters: QueryFilter[], groupId: string, from: number, to: number}\n): Promise<ImageType[]> => {\n  const action: string = 'getImagesByGroup';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {filters, groupId, from, to} = params;\n  const formatGroupId: string = parseId(groupId);\n  const limit: ArangoDbLimit = getLimit(from, to);\n\n  filters\n    .map((filter: QueryFilter) => {\n      const {conditional, name, value} = filter;\n      let formatCond: string = conditional;\n\n      if(conditional !== '>=' && conditional !== '<=' && conditional !== '>' && conditional !== '<') {\n        formatCond = '==';\n      }\n\n      switch(name) {\n        case 'added':\n          return `p.added ${formatCond} ${parseNum(value)}`;\n        default:\n          return '';\n      }\n    });\n\n  return getGroupDetails(context, formatGroupId)\n    .then((group: GroupType) => {\n      if(group.privacy === 'public') {\n        filters.push({conditional: '==', name: 'groupId', value: formatGroupId});\n        const filterStr = filters.join(' && ');\n        const aqlQry: string = `FOR i IN\n          FLATTEN(\n            FOR p IN posts\n            FILTER ${filterStr}\n            LET images = (\n              FOR i, e IN INBOUND p._id hasImage\n              RETURN i\n            )\n            SORT p.added DESC\n            RETURN images\n          )\n          SORT i.added DESC\n          ${limit.aql}\n          RETURN i`;\n\n        return useDb(databaseName).query(aqlQry)\n          .then((cursor) => cursor.all() as unknown as ImageType[])\n          .catch((error: Error) => {\n            logError({\n              action,\n              category: eventCategory,\n              params: {filters, from, groupId, to},\n              value: ErrorTypes.DATABASE_ERROR\n            }, error, context);\n            return [];\n          });\n      }\n      return isGrouped(context, sessionId, groupId)\n        .then((grouped: GroupUser) => {\n          if(grouped.isValid) {\n            filters.push({conditional: '==', name: 'groupId', value: grouped.groupId});\n            const filterList: string = filters.join(' && ');\n            const aqlQry: string = `FOR p IN post\n                FILTER ${filterList}\n                FOR f IN p.files\n                FILTER f.type == \"image/jpeg\" || f.type == \"image/png\"\n                ${limit.aql}\n                SORT p.added DESC\n                RETURN f`;\n\n            return useDb(databaseName).query(aqlQry)\n              .then((cursor) => cursor.all() as unknown as ImageType[])\n              .catch((error: Error) => {\n                logError({\n                  action,\n                  category: eventCategory,\n                  params: {filters, from, groupId, to},\n                  value: ErrorTypes.DATABASE_ERROR\n                }, error, context);\n                return [];\n              });\n          }\n          return [];\n        });\n    });\n};\n\nexport const getImagesByReactions = (\n  context: ApiContext,\n  reactions: string[] = [],\n  options?: ImageOptions\n): Promise<ImageType[]> => {\n  const action: string = 'getUsersByImage';\n  const {databaseName, fields = [], session: {userId: sessionId}} = context;\n  const {limit} = parseImageOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getImageOptional(fields);\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatReactions: string[] = reactions.map((reactionName) => parseChar(reactionName, 32));\n  const filterBy: string[] = [\n    'r.type == \"images\"',\n    `POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`];\n\n  // Get data from database\n  const aqlQry: string = `FOR i, r IN OUTBOUND \"${formatSessionId}\" hasReaction\n    OPTIONS {vertexCollections: \"images\"}\n    ${selectQueries.join('\\n')}\n    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN MERGE(i, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as ImageType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        params: {options, reactions},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n      return [];\n    });\n};\n\nexport const getImage = (context: ApiContext, imageId: string): Promise<ImageType> => {\n  const action: string = 'getItem';\n  const {databaseName} = context;\n  const formatImageId: string = parseId(imageId);\n  const aqlQry: AqlQuery = aql`FOR i IN images\n    FILTER i._key==${formatImageId}\n    LIMIT 1\n    RETURN i`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {imageId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getPathUserImages = (userId: string, imageId: string, imageType: string, dir: string = 'images'): string => {\n  let filename: string = imageId;\n\n  switch(imageType) {\n    case 'image/png':\n      filename = `${imageId}.png`;\n      break;\n    default:\n      filename = `${imageId}.jpg`;\n      break;\n  }\n\n  return `users/${userId}/${dir}/${filename}`;\n};\n\nexport const getAppImageUrl = ({\n  bucket,\n  directory = 'images',\n  imageId,\n  imageType = 'jpg',\n  isThumb = false,\n  type,\n  typeId\n}: ImageUrlData): string => {\n  if(!type) {\n    return '';\n  }\n\n  if(!imageId) {\n    if(imageType === 'profile') {\n      return `${Config.get('image.defaultUrl')}/users_bk.jpg`;\n    }\n    return `${Config.get('image.defaultUrl')}/users_wh.jpg`;\n  }\n\n  const bucketName = bucket || Config.get('image.bucket') || 'dev.reaktor.io';\n  const imageName = isThumb ? `${imageId}-th.${imageType}` : `${imageId}.${imageType === 'profile' || imageType === 'other' ? 'jpg' : imageType}`;\n  const imageUrl = Config.get('image.url') || `https://s3.amazonaws.com/${bucketName}`;\n\n  if(type === 'apps') {\n    return `${imageUrl}/${type}/${directory}/${imageName}`;\n  }\n\n  return `${imageUrl}/${type}/${typeId}/${directory}/${imageName}`;\n};\n\nexport const getImageUrl = async ({\n  bucket,\n  imageDir = 'images',\n  imageId,\n  imageType = 'jpg',\n  isThumb = false,\n  type,\n  typeId,\n  urlType = 'public'\n}: ImageUrlData): Promise<string> => {\n  if(!imageId) {\n    return '';\n  }\n\n  // Always use signed URLs for thumbnails\n  if(isThumb) {\n    urlType = 'signed';\n  }\n\n  const thumbSuffix = isThumb ? '-th' : '';\n  const imageKey: string = `${type}/${typeId}/${imageDir}/${imageId}${thumbSuffix}.${imageType}`;\n\n  switch(urlType) {\n    case 'signed': {\n      try {\n        const defaultBucket = Config.get('image.bucket') || 'dev.reaktor.io';\n        return await s3GetSignedUrl({Bucket: bucket || defaultBucket, Expires: 900, Key: imageKey});\n      } catch(error) {\n        throw error;\n      }\n    }\n    case 'public': {\n      return `https://box.${Config.get('app.url')}/${imageKey}`;\n    }\n    case 'dev': {\n      return `https://s3.amazonaws.com/dev.${Config.get('app.url')}/${imageKey}`;\n    }\n    default: {\n      return '';\n    }\n  }\n};\n\nexport const resizeSaveImage = async (\n  context: ApiContext,\n  imageId: string,\n  buffer: Buffer,\n  fileType: string = 'image/jpeg', s3Options?: PutObjectCommandInput\n): Promise<ImageType> => {\n  const action: string = 'resizeSaveImage';\n  const {session: {userId: sessionId}} = context;\n  const imgW: number = Config.get('image.imgSize');\n  const imgQ: number = Config.get('image.imgQuality');\n  const format: string = (fileType.split('/'))[1];\n\n  const imageOptimizedBuffer: Buffer = await new Promise((resolve, reject) => {\n    gm(buffer, 'img')\n      .setFormat(format)\n      .quality(imgQ)\n      .autoOrient()\n      .resize(imgW, imgW, '>')\n      .stream((streamError: Error, stdout): void => {\n        if(streamError) {\n          reject(\n            logError({\n              action,\n              category: eventCategory,\n              params: {fileType, imageId},\n              value: ErrorTypes.IMAGE_SAVE\n            }, streamError, context)\n          );\n          return;\n        }\n\n        let imageBuffer: Buffer = Buffer.from('');\n\n        stdout.on('data', (data) => {\n          imageBuffer = Buffer.concat([imageBuffer, data]);\n        });\n\n        stdout.on('end', () => resolve(imageBuffer));\n      });\n  });\n\n  try {\n    const imageObj: PutObjectCommandInput = {\n      ACL: 'authenticated-read',\n      Body: imageOptimizedBuffer,\n      Bucket: null,\n      ContentType: fileType,\n      ...(s3Options || {}),\n      Key: getPathUserImages(sessionId, imageId, fileType, 'images')\n    };\n    await s3Put(imageObj);\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      params: {fileType, imageId},\n      value: ErrorTypes.IMAGE_SAVE\n    }, error, context);\n  }\n\n  const imageIdentity = await new Promise((resolve) => {\n    gm(imageOptimizedBuffer, 'img')\n      .identify({bufferStream: true}, (error: Error, imageData: ImageIdentifyType = {}) => {\n        if(error) {\n          return logError({\n            action,\n            category: eventCategory,\n            value: ErrorTypes.IMAGE_SAVE\n          }, error, context);\n        }\n\n        const formatValues = lowerCaseKeys(imageData);\n        const {make: cameraMake, model: cameraModel, datetimeoriginal: taken}: ImageIdentifyType = formatValues;\n        const stats = formatValues['channel statistics'];\n        let color: string;\n\n        if(stats) {\n          let {red, green, blue, mean} = stats;\n\n          if(red) {\n            mean = red['standard deviation'] || red.mean;\n            red = mean ? +((mean.split(' ')[0]).substring(0, 3)) : 0;\n          } else {\n            red = 0;\n          }\n\n          if(green) {\n            mean = green['standard deviation'] || green.mean;\n            green = mean ? +((mean.split(' ')[0]).substring(0, 3)) : 0;\n          } else {\n            green = 0;\n          }\n\n          if(blue) {\n            mean = blue['standard deviation'] || blue.mean;\n            blue = mean ? +((mean.split(' ')[0]).substring(0, 3)) : 0;\n          } else {\n            blue = 0;\n          }\n\n          const rgb = blue | (green << 8) | (red << 16);\n          const rgbColor = rgb.toString(16);\n          color = rgbColor === '0' ? '000000' : rgbColor;\n        }\n\n        const {\n          'JPEG-Quality': quality,\n          Orientation: orientationData,\n          Resolution: resolutionData,\n          size\n        } = imageData;\n        let resolution;\n\n        if(resolutionData) {\n          const [resolutionNumber, resolutionUnit] = resolutionData.split(' ');\n          const resolutionValue = resolutionNumber.split('x')[0];\n          const resolutionUnitValue = resolutionUnit === 'pixels/inch' ? 'ppi' : '';\n          resolution = `${resolutionValue} ${resolutionUnitValue}`;\n        }\n\n        return resolve({\n          color,\n          fileSize: imageOptimizedBuffer.length,\n          fileType: format,\n          height: size?.height || 0,\n          imageId,\n          make: cameraMake,\n          model: cameraModel,\n          orientation: orientationData === 'Unknown' ? undefined : orientationData?.toLowerCase(),\n          quality: quality ? +quality : undefined,\n          resolution,\n          taken: taken ? DateTime.fromMillis(taken).millisecond : undefined,\n          width: size?.width || 0\n        });\n      });\n  });\n\n  const thumbOptimizedBuffer: Buffer = await new Promise((resolve, reject) => {\n    const thmW = Config.get('image.thmSize');\n    const thmQ = Config.get('image.thmQuality');\n\n    gm(imageOptimizedBuffer, 'img')\n      .setFormat(format)\n      .gravity('Center')\n      .resize(thmW, thmW, '^')\n      .extent(thmW, thmW)\n      .quality(thmQ)\n      .stream((streamError: Error, thumbStdout): void => {\n        if(streamError) {\n          logError({\n            action,\n            category: eventCategory,\n            params: {fileType, imageId},\n            value: ErrorTypes.IMAGE_SAVE\n          }, streamError, context);\n          return;\n        }\n\n        let thumbBuffer: Buffer = Buffer.from('');\n\n        thumbStdout.on('data', (data) => {\n          thumbBuffer = Buffer.concat([thumbBuffer, data]);\n        });\n\n        thumbStdout.on('end', () => resolve(thumbBuffer));\n      });\n  });\n\n  try {\n    const thumbObj: PutObjectCommandInput = {\n      ACL: 'authenticated-read',\n      Body: thumbOptimizedBuffer,\n      Bucket: null,\n      ContentType: fileType,\n      ...(s3Options || {}),\n      Key: getPathUserImages(sessionId, imageId, fileType, 'thumbs')\n    };\n    await s3Put(thumbObj);\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.IMAGE_SAVE\n    }, error, context);\n  }\n\n  return imageIdentity;\n};\n\nexport const addImage = (\n  context: ApiContext,\n  image: ImageType,\n  s3Options?: PutObjectCommandInput\n): Promise<ImageType> => {\n  const action: string = 'addImage';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {imageId, description, buffer, fileType} = image;\n  const now: number = Date.now();\n\n  return resizeSaveImage(context, imageId, buffer, fileType, s3Options)\n    .then((resizedImage: ImageType) => {\n      const insert: ImageType = {\n        ...resizedImage,\n        _key: imageId,\n        added: now,\n        description,\n        fileType,\n        modified: now,\n        userId: sessionId\n      };\n\n      const aqlQry: AqlQuery = aql`INSERT ${insert} IN images RETURN NEW`;\n\n      return useDb(databaseName).query(aqlQry)\n        .then((cursor) => cursor.next())\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          params: {image, s3Options},\n          value: ErrorTypes.DATABASE_ERROR\n        }, error, context));\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.IMAGE_RESIZE\n    }, error, context));\n};\n\nexport const addImageEdge = async (context: ApiContext, imageEdge: ImageEdgeType): Promise<object> => {\n  const action: string = 'addImageEdge';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {imageId, itemId, itemType} = imageEdge;\n  const now: number = Date.now();\n  // const edgeCollection: EdgeCollection = useDb(databaseName).collection('hasImage');\n  const edgeId: string = createHash(`hasImage-${imageId}-${itemId}-${sessionId}`);\n  const formatItemType: ArangoDbCollection = parseChar(itemType).toLowerCase() as ArangoDbCollection;\n  const formatItemId: string = parseId(itemId);\n  const itemDocId: string = getDocId(formatItemType, {_key: formatItemId});\n  const formatImageId: string = parseId(imageId);\n  const fileDocId: string = `images/${formatImageId}`;\n\n  const edge: ImageEdgeType = {\n    _from: itemDocId,\n    _key: edgeId,\n    _to: fileDocId,\n    added: now,\n    itemType\n  };\n\n  if(itemDocId) {\n    return useDb(databaseName).collection('hasImage').save(edge, {returnNew: true})\n      .then((fileEdge) => useDb(databaseName).collection('hasImage').document(fileEdge))\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        params: {\n          edge,\n          fileDocId,\n          imageId,\n          itemDocId,\n          itemId,\n          itemType\n        },\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context));\n  }\n\n  return null;\n};\n\nexport const updateImage = async (\n  context: ApiContext,\n  image: ImageType,\n  s3Options?: PutObjectCommandInput\n): Promise<ImageType> => {\n  const action: string = 'updateImage';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {\n    base64 = '',\n    buffer: imageBuffer,\n    description = '',\n    imageId,\n    itemId,\n    itemType,\n    fileType,\n    url = ''\n  } = image;\n\n  const now: number = Date.now();\n  const formatImageId: string = imageId || createHash(`image-${sessionId}`);\n  const formatItemId: string = parseId(itemId);\n  const formatItemType: ArangoDbCollection = parseChar(itemType, 16).toLowerCase() as ArangoDbCollection;\n  const customParams: ImageType = {\n    description,\n    imageId: formatImageId,\n    userId: sessionId\n  };\n\n  if(base64 || imageBuffer) {\n    let buffer: Buffer = imageBuffer;\n\n    if(base64) {\n      const formatBase64: string = base64.substr(base64.indexOf(',') + 1);\n      buffer = Buffer.from(formatBase64, 'base64');\n    }\n\n    let updatedFileType = fileType;\n\n    if(!fileType) {\n      const {fileTypeFromBuffer} = await import('file-type');\n      const {mime} = await fileTypeFromBuffer(buffer);\n      updatedFileType = mime;\n    }\n\n    const imgParams: ImageType = {\n      ...customParams,\n      buffer,\n      fileType: updatedFileType\n    };\n\n    return addImage(context, imgParams, s3Options)\n      .then((image: ImageType) => {\n        if(formatItemId && formatItemType) {\n          const imageEdge: ImageEdgeType = {\n            imageId: formatImageId,\n            itemId: formatItemId,\n            itemType: formatItemType\n          };\n          return addImageEdge(context, imageEdge).then(() => image);\n        }\n\n        return image;\n      })\n      .catch((error) => logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.IMAGE_SAVE\n      }, error, context));\n  } else if(url !== '') {\n    let contentType;\n\n    return httpGet(url)\n      .then((res: HttpResponse) => {\n        if(res.status !== 200) {\n          return logException({\n            action,\n            category: eventCategory,\n            value: ErrorTypes.IMAGE_REQUEST\n          }, context);\n        }\n\n        contentType = res.headers.get('content-type');\n\n        return res;\n      })\n      .then((res) => (res as HttpResponse).buffer())\n      .then((buffer: Buffer) => {\n        const imgParams: ImageType = {\n          ...customParams,\n          buffer,\n          fileType: contentType\n        };\n\n        return addImage(context, imgParams, s3Options)\n          .then((image: ImageType) => {\n            if(formatItemId && formatItemType) {\n              const imageEdge: ImageEdgeType = {imageId: formatImageId, itemId: formatItemId, itemType: formatItemType};\n              return addImageEdge(context, imageEdge).then(() => image);\n            }\n\n            return image;\n          });\n      })\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        value: 'fetch_error'\n      }, error, context));\n  } else if(imageId !== '') {\n    const update = {\n      description,\n      modified: now\n    };\n    const aqlQry: AqlQuery = aql`UPDATE {_key: ${formatImageId}} WITH ${update} IN images RETURN NEW`;\n\n    return useDb(databaseName).query(aqlQry)\n      .then((cursor) => cursor.next())\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        params: {\n          aqlQry,\n          formatImageId,\n          update\n        },\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context));\n  }\n\n  return null;\n};\n\nexport const deleteImage = async (context: ApiContext, imageId: string): Promise<ImageType> => {\n  const action: string = 'delete';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const formatImageId = parseId(imageId);\n\n  try {\n    const removeEdgeQuery = aql`FOR hi IN hasImage\n    FILTER hi._from == ${formatImageId}\n    REMOVE hi IN hasImage\n    RETURN OLD`;\n\n    await useDb(databaseName).query(removeEdgeQuery);\n\n    const aqlQuery = aql`FOR i IN images\n    FILTER i._key == ${formatImageId} && i.userId == ${sessionId}\n    REMOVE i IN images\n    RETURN OLD`;\n\n    const image = await useDb(databaseName).query(aqlQuery).then((cursor) => cursor.next());\n\n    if(image) {\n      const {_key: imageKey} = image;\n      const params: DeleteObjectsCommandInput = {\n        Bucket: null,\n        Delete: {\n          Objects: [\n            {Key: `users/${sessionId}/images/${imageKey}.jpg`},\n            {Key: `users/${sessionId}/thumbs/${imageKey}.jpg`}\n          ],\n          Quiet: true\n        }\n      };\n\n      return s3DeleteList(params).then(() => image);\n    }\n\n    return null;\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      params: {imageId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n};\n"],
  "mappings": "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",
  "names": ["httpGet", "parseNum", "createHash", "parseArangoId", "parseChar", "parseId", "aql", "gm", "DateTime", "Config", "ErrorTypes", "logError", "logException", "getDocId", "getLimit", "selectReactionCountByType", "useDb", "getGroupDetails", "isGrouped", "s3DeleteList", "s3GetSignedUrl", "s3Put", "lowerCaseKeys", "eventCategory", "parseImageOptions", "options", "from", "to", "type", "getImageOptional", "fields", "selects", "field", "getImagesByUser", "context", "userId", "action", "databaseName", "formatUserId", "limit", "aqlQry", "cursor", "error", "getImageCountByItem", "itemId", "formatItemId", "count", "getImagesByItem", "selectObjects", "selectQueries", "getImagesByGroup", "params", "sessionId", "filters", "groupId", "formatGroupId", "filter", "conditional", "name", "value", "formatCond", "group", "grouped", "getImagesByReactions", "reactions", "formatSessionId", "formatReactions", "reactionName", "filterBy", "getImage", "imageId", "formatImageId", "getPathUserImages", "imageType", "dir", "filename", "getAppImageUrl", "bucket", "directory", "isThumb", "typeId", "bucketName", "imageName", "imageUrl", "getImageUrl", "imageDir", "urlType", "imageKey", "defaultBucket", "resizeSaveImage", "buffer", "fileType", "s3Options", "imgW", "imgQ", "format", "imageOptimizedBuffer", "resolve", "reject", "streamError", "stdout", "imageBuffer", "data", "imageObj", "imageIdentity", "imageData", "formatValues", "cameraMake", "cameraModel", "taken", "stats", "color", "red", "green", "blue", "mean", "rgbColor", "quality", "orientationData", "resolutionData", "size", "resolution", "resolutionNumber", "resolutionUnit", "thumbOptimizedBuffer", "thmW", "thmQ", "thumbStdout", "thumbBuffer", "thumbObj", "addImage", "image", "description", "now", "resizedImage", "insert", "addImageEdge", "imageEdge", "itemType", "edgeId", "formatItemType", "itemDocId", "fileDocId", "edge", "fileEdge", "updateImage", "base64", "url", "customParams", "formatBase64", "updatedFileType", "fileTypeFromBuffer", "mime", "imgParams", "contentType", "res", "update", "deleteImage", "removeEdgeQuery", "aqlQuery"]
}

|
|
690
|
+
RETURN OLD`;
|
|
691
|
+
const image = await useDb(databaseName).query(aqlQuery).then((cursor)=>cursor.next());
|
|
692
|
+
if (image) {
|
|
693
|
+
const { _key: imageKey } = image;
|
|
694
|
+
const params = {
|
|
695
|
+
Bucket: null,
|
|
696
|
+
Delete: {
|
|
697
|
+
Objects: [
|
|
698
|
+
{
|
|
699
|
+
Key: `users/${sessionId}/images/${imageKey}.jpg`
|
|
700
|
+
},
|
|
701
|
+
{
|
|
702
|
+
Key: `users/${sessionId}/thumbs/${imageKey}.jpg`
|
|
703
|
+
}
|
|
704
|
+
],
|
|
705
|
+
Quiet: true
|
|
706
|
+
}
|
|
707
|
+
};
|
|
708
|
+
return s3DeleteList(params).then(()=>image);
|
|
709
|
+
}
|
|
710
|
+
return null;
|
|
711
|
+
} catch (error) {
|
|
712
|
+
return logError({
|
|
713
|
+
action,
|
|
714
|
+
category: eventCategory,
|
|
715
|
+
params: {
|
|
716
|
+
imageId
|
|
717
|
+
},
|
|
718
|
+
value: ErrorTypes.DATABASE_ERROR
|
|
719
|
+
}, error, context);
|
|
720
|
+
}
|
|
721
|
+
};
|
|
722
|
+
|
|
723
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["/Users/nitrog7/Development/reaktor/src/actions/images.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 {DeleteObjectsCommandInput, PutObjectCommandInput} from '@aws-sdk/client-s3';\nimport {get as httpGet} from '@nlabs/rip-hunter';\nimport {parseNum} from '@nlabs/utils/parsers/numbers';\nimport {createHash, parseArangoId, parseChar, parseId} from '@nlabs/utils/parsers/strings';\nimport {aql} from 'arangojs';\nimport {AqlQuery} from 'arangojs/aql';\nimport gm from 'gm';\nimport {DateTime} from 'luxon';\n\nimport {Config} from '../config.js';\nimport {ErrorTypes} from '../types/error.types.js';\nimport {logError, logException} from '../utils/analyticsUtils.js';\nimport {getDocId, getLimit, selectReactionCountByType, useDb} from '../utils/arangodbUtils.js';\nimport {getGroupDetails, isGrouped} from './groups.js';\nimport {s3DeleteList, s3GetSignedUrl, s3Put} from './s3.js';\n\nimport {lowerCaseKeys} from '@nlabs/utils/parsers/objects';\nimport type {QueryFilter} from '../types/apps.types.js';\nimport type {ArangoDbCollection, ArangoDbLimit} from '../types/arangodb.types.js';\nimport type {ApiContext} from '../types/auth.types.js';\nimport type {GroupType, GroupUser} from '../types/groups.types.js';\nimport type {ImageEdgeType, ImageIdentifyType, ImageOptions, ImageType, ImageUrlData} from '../types/images.types.js';\n\n// Define interface for HTTP response with buffer method\ninterface HttpResponse extends Response {\n  buffer(): Promise<Buffer>;\n}\n\nconst eventCategory: string = 'images';\ngm.subClass({imageMagick: '7+'});\n\nexport const parseImageOptions = (options: ImageOptions = {}) => {\n  const {\n    from = 0,\n    to = 30,\n    type = 'default'\n  } = options;\n\n  return {\n    limit: getLimit(from, to) as ArangoDbLimit,\n    type: parseChar(type, 32)\n  };\n};\n\nexport const getImageOptional = (fields: string[] = []) =>\n  fields.reduce((selects, field: string) => {\n    if(field.includes('Count')) {\n      return selectReactionCountByType('images', 'i', field, selects);\n    }\n\n    switch(field) {\n      case 'reactions': {\n        selects.queries.push(`LET reactions = (\n          FOR image, r IN INBOUND i._id reactions\n          COLLECT reactionName = r.value INTO reactionItems\n          RETURN {name: reactionName, count: LENGTH(reactionItems[*].r.value)}\n        )`);\n        selects.objects.push('reactions:reactions');\n        return selects;\n      }\n      case 'tags': {\n        selects.queries.push(`LET tags = (\n          FOR t, pl IN INBOUND i._id isTagged\n          RETURN t\n        )`);\n        selects.objects.push('tags:tags');\n        return selects;\n      }\n      case 'users': {\n        selects.queries.push(`LET users = FIRST(\n          FOR u IN users\n          FILTER i.userId == u._key\n          LIMIT 1\n          RETURN u\n        )`);\n        selects.objects.push('users:users');\n        return selects;\n      }\n      default: {\n        return selects;\n      }\n    }\n  }, {objects: [], queries: []});\n\nexport const getImagesByUser = (\n  context: ApiContext,\n  userId: string,\n  from: number = 0,\n  to: number = 30\n): Promise<ImageType[]> => {\n  const action: string = 'getImagesByUser';\n  const {databaseName} = context;\n  const formatUserId: string = parseId(userId);\n  const limit: ArangoDbLimit = getLimit(from, to);\n  const aqlQry: string = `FOR i IN images\n      FILTER i.userId == \"${formatUserId}\"\n      LET user = FIRST(\n        FOR u IN users\n        FILTER u._key == i.userId\n        LIMIT 1\n        RETURN u\n      )\n      ${limit.aql}\n      SORT i.added\n      RETURN MERGE(i, {user:user})`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as ImageType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        params: {from, to, userId},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n\n      return [];\n    });\n};\n\nexport const getImageCountByItem = (context: ApiContext, itemId: string): Promise<number> => {\n  const action: string = 'getImageCountByItem';\n  const {databaseName} = context;\n  const formatItemId: string = parseArangoId(itemId);\n  const aqlQry: AqlQuery = aql`FOR i IN hasImage\n      FILTER i._to == ${formatItemId}\n      RETURN {count: COUNT(i)}`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .then(({count}) => count)\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {itemId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getImagesByItem = async (\n  context: ApiContext,\n  itemId: string,\n  options: ImageOptions = {}\n): Promise<ImageType[]> => {\n  const action: string = 'getImagesByItem';\n  const {databaseName, fields = []} = context;\n  const formatItemId: string = parseArangoId(itemId);\n  const {limit} = parseImageOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getImageOptional(fields);\n  const aqlQry: string = `FOR i, l IN 1..1 OUTBOUND \"${formatItemId}\" hasImage\n    FILTER NOT IS_NULL(i)\n    ${selectQueries.join('\\n')}\n    SORT i.added\n    ${limit.aql}\n    RETURN MERGE(i, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as ImageType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        params: {itemId, options},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n      return [];\n    });\n};\n\nexport const getImagesByGroup = (\n  context: ApiContext,\n  params: {filters: QueryFilter[], groupId: string, from: number, to: number}\n): Promise<ImageType[]> => {\n  const action: string = 'getImagesByGroup';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {filters, groupId, from, to} = params;\n  const formatGroupId: string = parseId(groupId);\n  const limit: ArangoDbLimit = getLimit(from, to);\n\n  filters\n    .map((filter: QueryFilter) => {\n      const {conditional, name, value} = filter;\n      let formatCond: string = conditional;\n\n      if(conditional !== '>=' && conditional !== '<=' && conditional !== '>' && conditional !== '<') {\n        formatCond = '==';\n      }\n\n      switch(name) {\n        case 'added':\n          return `p.added ${formatCond} ${parseNum(value)}`;\n        default:\n          return '';\n      }\n    });\n\n  return getGroupDetails(context, formatGroupId)\n    .then((group: GroupType) => {\n      if(group.privacy === 'public') {\n        filters.push({conditional: '==', name: 'groupId', value: formatGroupId});\n        const filterStr = filters.join(' && ');\n        const aqlQry: string = `FOR i IN\n          FLATTEN(\n            FOR p IN posts\n            FILTER ${filterStr}\n            LET images = (\n              FOR i, e IN INBOUND p._id hasImage\n              RETURN i\n            )\n            SORT p.added DESC\n            RETURN images\n          )\n          SORT i.added DESC\n          ${limit.aql}\n          RETURN i`;\n\n        return useDb(databaseName).query(aqlQry)\n          .then((cursor) => cursor.all() as unknown as ImageType[])\n          .catch((error: Error) => {\n            logError({\n              action,\n              category: eventCategory,\n              params: {filters, from, groupId, to},\n              value: ErrorTypes.DATABASE_ERROR\n            }, error, context);\n            return [];\n          });\n      }\n      return isGrouped(context, sessionId, groupId)\n        .then((grouped: GroupUser) => {\n          if(grouped.isValid) {\n            filters.push({conditional: '==', name: 'groupId', value: grouped.groupId});\n            const filterList: string = filters.join(' && ');\n            const aqlQry: string = `FOR p IN post\n                FILTER ${filterList}\n                FOR f IN p.files\n                FILTER f.type == \"image/jpeg\" || f.type == \"image/png\"\n                ${limit.aql}\n                SORT p.added DESC\n                RETURN f`;\n\n            return useDb(databaseName).query(aqlQry)\n              .then((cursor) => cursor.all() as unknown as ImageType[])\n              .catch((error: Error) => {\n                logError({\n                  action,\n                  category: eventCategory,\n                  params: {filters, from, groupId, to},\n                  value: ErrorTypes.DATABASE_ERROR\n                }, error, context);\n                return [];\n              });\n          }\n          return [];\n        });\n    });\n};\n\nexport const getImagesByReactions = (\n  context: ApiContext,\n  reactions: string[] = [],\n  options?: ImageOptions\n): Promise<ImageType[]> => {\n  const action: string = 'getUsersByImage';\n  const {databaseName, fields = [], session: {userId: sessionId}} = context;\n  const {limit} = parseImageOptions(options);\n  const {objects: selectObjects, queries: selectQueries} = getImageOptional(fields);\n  const formatSessionId: string = `users/${sessionId}`;\n  const formatReactions: string[] = reactions.map((reactionName) => parseChar(reactionName, 32));\n  const filterBy: string[] = [\n    'r.type == \"images\"',\n    `POSITION(${JSON.stringify(formatReactions)}, LOWER(r.name))`];\n\n  // Get data from database\n  const aqlQry: string = `FOR i, r IN OUTBOUND \"${formatSessionId}\" hasReaction\n    OPTIONS {vertexCollections: \"images\"}\n    ${selectQueries.join('\\n')}\n    FILTER ${filterBy.join(' && ')}\n    ${limit.aql}\n    RETURN MERGE(i, {${selectObjects.join(', ')}})`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.all() as unknown as ImageType[])\n    .catch((error: Error) => {\n      logError({\n        action,\n        category: eventCategory,\n        params: {options, reactions},\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context);\n      return [];\n    });\n};\n\nexport const getImage = (context: ApiContext, imageId: string): Promise<ImageType> => {\n  const action: string = 'getItem';\n  const {databaseName} = context;\n  const formatImageId: string = parseId(imageId);\n  const aqlQry: AqlQuery = aql`FOR i IN images\n    FILTER i._key==${formatImageId}\n    LIMIT 1\n    RETURN i`;\n\n  return useDb(databaseName).query(aqlQry)\n    .then((cursor) => cursor.next())\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      params: {imageId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context));\n};\n\nexport const getPathUserImages = (userId: string, imageId: string, imageType: string, dir: string = 'images'): string => {\n  let filename: string = imageId;\n\n  switch(imageType) {\n    case 'image/png':\n      filename = `${imageId}.png`;\n      break;\n    default:\n      filename = `${imageId}.jpg`;\n      break;\n  }\n\n  return `users/${userId}/${dir}/${filename}`;\n};\n\nexport const getAppImageUrl = ({\n  bucket,\n  directory = 'images',\n  imageId,\n  imageType = 'jpg',\n  isThumb = false,\n  type,\n  typeId\n}: ImageUrlData): string => {\n  if(!type) {\n    return '';\n  }\n\n  if(!imageId) {\n    if(imageType === 'profile') {\n      return `${Config.get('image.defaultUrl')}/users_bk.jpg`;\n    }\n    return `${Config.get('image.defaultUrl')}/users_wh.jpg`;\n  }\n\n  const bucketName = bucket || Config.get('image.bucket') || 'dev.reaktor.io';\n  const imageName = isThumb ? `${imageId}-th.${imageType}` : `${imageId}.${imageType === 'profile' || imageType === 'other' ? 'jpg' : imageType}`;\n  const imageUrl = Config.get('image.url') || `https://s3.amazonaws.com/${bucketName}`;\n\n  if(type === 'apps') {\n    return `${imageUrl}/${type}/${directory}/${imageName}`;\n  }\n\n  return `${imageUrl}/${type}/${typeId}/${directory}/${imageName}`;\n};\n\nexport const getImageUrl = async ({\n  bucket,\n  imageDir = 'images',\n  imageId,\n  imageType = 'jpg',\n  isThumb = false,\n  type,\n  typeId,\n  urlType = 'public'\n}: ImageUrlData): Promise<string> => {\n  if(!imageId) {\n    return '';\n  }\n\n  // Always use signed URLs for thumbnails\n  if(isThumb) {\n    urlType = 'signed';\n  }\n\n  const thumbSuffix = isThumb ? '-th' : '';\n  const imageKey: string = `${type}/${typeId}/${imageDir}/${imageId}${thumbSuffix}.${imageType}`;\n\n  switch(urlType) {\n    case 'signed': {\n      try {\n        const defaultBucket = Config.get('image.bucket') || 'dev.reaktor.io';\n        return await s3GetSignedUrl({Bucket: bucket || defaultBucket, Expires: 900, Key: imageKey});\n      } catch(error) {\n        throw error;\n      }\n    }\n    case 'public': {\n      return `https://box.${Config.get('app.url')}/${imageKey}`;\n    }\n    case 'dev': {\n      return `https://s3.amazonaws.com/dev.${Config.get('app.url')}/${imageKey}`;\n    }\n    default: {\n      return '';\n    }\n  }\n};\n\nexport const resizeSaveImage = async (\n  context: ApiContext,\n  imageId: string,\n  buffer: Buffer,\n  fileType: string = 'image/jpeg', s3Options?: PutObjectCommandInput\n): Promise<ImageType> => {\n  const action: string = 'resizeSaveImage';\n  const {session: {userId: sessionId}} = context;\n  const imgW: number = Config.get('image.imgSize');\n  const imgQ: number = Config.get('image.imgQuality');\n  const format: string = (fileType.split('/'))[1];\n\n  const imageOptimizedBuffer: Buffer = await new Promise((resolve, reject) => {\n    gm(buffer, 'img')\n      .setFormat(format)\n      .quality(imgQ)\n      .autoOrient()\n      .resize(imgW, imgW, '>')\n      .stream((streamError: Error, stdout): void => {\n        if(streamError) {\n          reject(\n            logError({\n              action,\n              category: eventCategory,\n              params: {fileType, imageId},\n              value: ErrorTypes.IMAGE_SAVE\n            }, streamError, context)\n          );\n          return;\n        }\n\n        let imageBuffer: Buffer = Buffer.from('');\n\n        stdout.on('data', (data) => {\n          imageBuffer = Buffer.concat([imageBuffer, data]);\n        });\n\n        stdout.on('end', () => resolve(imageBuffer));\n      });\n  });\n\n  try {\n    const imageObj: PutObjectCommandInput = {\n      ACL: 'authenticated-read',\n      Body: imageOptimizedBuffer,\n      Bucket: null,\n      ContentType: fileType,\n      ...(s3Options || {}),\n      Key: getPathUserImages(sessionId, imageId, fileType, 'images')\n    };\n    await s3Put(imageObj);\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      params: {fileType, imageId},\n      value: ErrorTypes.IMAGE_SAVE\n    }, error, context);\n  }\n\n  const imageIdentity = await new Promise((resolve) => {\n    gm(imageOptimizedBuffer, 'img')\n      .identify({bufferStream: true}, (error: Error, imageData: ImageIdentifyType = {}) => {\n        if(error) {\n          return logError({\n            action,\n            category: eventCategory,\n            value: ErrorTypes.IMAGE_SAVE\n          }, error, context);\n        }\n\n        const formatValues = lowerCaseKeys(imageData);\n        const {make: cameraMake, model: cameraModel, datetimeoriginal: taken}: ImageIdentifyType = formatValues;\n        const stats = formatValues['channel statistics'];\n        let color: string;\n\n        if(stats) {\n          let {red, green, blue, mean} = stats;\n\n          if(red) {\n            mean = red['standard deviation'] || red.mean;\n            red = mean ? +((mean.split(' ')[0]).substring(0, 3)) : 0;\n          } else {\n            red = 0;\n          }\n\n          if(green) {\n            mean = green['standard deviation'] || green.mean;\n            green = mean ? +((mean.split(' ')[0]).substring(0, 3)) : 0;\n          } else {\n            green = 0;\n          }\n\n          if(blue) {\n            mean = blue['standard deviation'] || blue.mean;\n            blue = mean ? +((mean.split(' ')[0]).substring(0, 3)) : 0;\n          } else {\n            blue = 0;\n          }\n\n          const rgb = blue | (green << 8) | (red << 16);\n          const rgbColor = rgb.toString(16);\n          color = rgbColor === '0' ? '000000' : rgbColor;\n        }\n\n        const {\n          'JPEG-Quality': quality,\n          Orientation: orientationData,\n          Resolution: resolutionData,\n          size\n        } = imageData;\n        let resolution;\n\n        if(resolutionData) {\n          const [resolutionNumber, resolutionUnit] = resolutionData.split(' ');\n          const resolutionValue = resolutionNumber.split('x')[0];\n          const resolutionUnitValue = resolutionUnit === 'pixels/inch' ? 'ppi' : '';\n          resolution = `${resolutionValue} ${resolutionUnitValue}`;\n        }\n\n        return resolve({\n          color,\n          fileSize: imageOptimizedBuffer.length,\n          fileType: format,\n          height: size?.height || 0,\n          imageId,\n          make: cameraMake,\n          model: cameraModel,\n          orientation: orientationData === 'Unknown' ? undefined : orientationData?.toLowerCase(),\n          quality: quality ? +quality : undefined,\n          resolution,\n          taken: taken ? DateTime.fromMillis(taken).millisecond : undefined,\n          width: size?.width || 0\n        });\n      });\n  });\n\n  const thumbOptimizedBuffer: Buffer = await new Promise((resolve, reject) => {\n    const thmW = Config.get('image.thmSize');\n    const thmQ = Config.get('image.thmQuality');\n\n    gm(imageOptimizedBuffer, 'img')\n      .setFormat(format)\n      .gravity('Center')\n      .resize(thmW, thmW, '^')\n      .extent(thmW, thmW)\n      .quality(thmQ)\n      .stream((streamError: Error, thumbStdout): void => {\n        if(streamError) {\n          logError({\n            action,\n            category: eventCategory,\n            params: {fileType, imageId},\n            value: ErrorTypes.IMAGE_SAVE\n          }, streamError, context);\n          return;\n        }\n\n        let thumbBuffer: Buffer = Buffer.from('');\n\n        thumbStdout.on('data', (data) => {\n          thumbBuffer = Buffer.concat([thumbBuffer, data]);\n        });\n\n        thumbStdout.on('end', () => resolve(thumbBuffer));\n      });\n  });\n\n  try {\n    const thumbObj: PutObjectCommandInput = {\n      ACL: 'authenticated-read',\n      Body: thumbOptimizedBuffer,\n      Bucket: null,\n      ContentType: fileType,\n      ...(s3Options || {}),\n      Key: getPathUserImages(sessionId, imageId, fileType, 'thumbs')\n    };\n    await s3Put(thumbObj);\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.IMAGE_SAVE\n    }, error, context);\n  }\n\n  return imageIdentity;\n};\n\nexport const addImage = (\n  context: ApiContext,\n  image: ImageType,\n  s3Options?: PutObjectCommandInput\n): Promise<ImageType> => {\n  const action: string = 'addImage';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {imageId, description, buffer, fileType} = image;\n  const now: number = Date.now();\n\n  return resizeSaveImage(context, imageId, buffer, fileType, s3Options)\n    .then((resizedImage: ImageType) => {\n      const insert: ImageType = {\n        ...resizedImage,\n        _key: imageId,\n        added: now,\n        description,\n        fileType,\n        modified: now,\n        userId: sessionId\n      };\n\n      const aqlQry: AqlQuery = aql`INSERT ${insert} IN images RETURN NEW`;\n\n      return useDb(databaseName).query(aqlQry)\n        .then((cursor) => cursor.next())\n        .catch((error: Error) => logError({\n          action,\n          category: eventCategory,\n          params: {image, s3Options},\n          value: ErrorTypes.DATABASE_ERROR\n        }, error, context));\n    })\n    .catch((error: Error) => logError({\n      action,\n      category: eventCategory,\n      value: ErrorTypes.IMAGE_RESIZE\n    }, error, context));\n};\n\nexport const addImageEdge = async (context: ApiContext, imageEdge: ImageEdgeType): Promise<object> => {\n  const action: string = 'addImageEdge';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {imageId, itemId, itemType} = imageEdge;\n  const now: number = Date.now();\n  // const edgeCollection: EdgeCollection = useDb(databaseName).collection('hasImage');\n  const edgeId: string = createHash(`hasImage-${imageId}-${itemId}-${sessionId}`);\n  const formatItemType: ArangoDbCollection = parseChar(itemType).toLowerCase() as ArangoDbCollection;\n  const formatItemId: string = parseId(itemId);\n  const itemDocId: string = getDocId(formatItemType, {_key: formatItemId});\n  const formatImageId: string = parseId(imageId);\n  const fileDocId: string = `images/${formatImageId}`;\n\n  const edge: ImageEdgeType = {\n    _from: itemDocId,\n    _key: edgeId,\n    _to: fileDocId,\n    added: now,\n    itemType\n  };\n\n  if(itemDocId) {\n    return useDb(databaseName).collection('hasImage').save(edge, {returnNew: true})\n      .then((fileEdge) => useDb(databaseName).collection('hasImage').document(fileEdge))\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        params: {\n          edge,\n          fileDocId,\n          imageId,\n          itemDocId,\n          itemId,\n          itemType\n        },\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context));\n  }\n\n  return null;\n};\n\nexport const updateImage = async (\n  context: ApiContext,\n  image: ImageType,\n  s3Options?: PutObjectCommandInput\n): Promise<ImageType> => {\n  const action: string = 'updateImage';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const {\n    base64 = '',\n    buffer: imageBuffer,\n    description = '',\n    imageId,\n    itemId,\n    itemType,\n    fileType,\n    url = ''\n  } = image;\n\n  const now: number = Date.now();\n  const formatImageId: string = imageId || createHash(`image-${sessionId}`);\n  const formatItemId: string = parseId(itemId);\n  const formatItemType: ArangoDbCollection = parseChar(itemType, 16).toLowerCase() as ArangoDbCollection;\n  const customParams: ImageType = {\n    description,\n    imageId: formatImageId,\n    userId: sessionId\n  };\n\n  if(base64 || imageBuffer) {\n    let buffer: Buffer = imageBuffer;\n\n    if(base64) {\n      const formatBase64: string = base64.substr(base64.indexOf(',') + 1);\n      buffer = Buffer.from(formatBase64, 'base64');\n    }\n\n    let updatedFileType = fileType;\n\n    if(!fileType) {\n      const {fileTypeFromBuffer} = await import('file-type');\n      const {mime} = await fileTypeFromBuffer(buffer);\n      updatedFileType = mime;\n    }\n\n    const imgParams: ImageType = {\n      ...customParams,\n      buffer,\n      fileType: updatedFileType\n    };\n\n    return addImage(context, imgParams, s3Options)\n      .then((image: ImageType) => {\n        if(formatItemId && formatItemType) {\n          const imageEdge: ImageEdgeType = {\n            imageId: formatImageId,\n            itemId: formatItemId,\n            itemType: formatItemType\n          };\n          return addImageEdge(context, imageEdge).then(() => image);\n        }\n\n        return image;\n      })\n      .catch((error) => logError({\n        action,\n        category: eventCategory,\n        value: ErrorTypes.IMAGE_SAVE\n      }, error, context));\n  } else if(url !== '') {\n    let contentType;\n\n    return httpGet(url)\n      .then((res: HttpResponse) => {\n        if(res.status !== 200) {\n          return logException({\n            action,\n            category: eventCategory,\n            value: ErrorTypes.IMAGE_REQUEST\n          }, context);\n        }\n\n        contentType = res.headers.get('content-type');\n\n        return res;\n      })\n      .then((res) => (res as HttpResponse).buffer())\n      .then((buffer: Buffer) => {\n        const imgParams: ImageType = {\n          ...customParams,\n          buffer,\n          fileType: contentType\n        };\n\n        return addImage(context, imgParams, s3Options)\n          .then((image: ImageType) => {\n            if(formatItemId && formatItemType) {\n              const imageEdge: ImageEdgeType = {imageId: formatImageId, itemId: formatItemId, itemType: formatItemType};\n              return addImageEdge(context, imageEdge).then(() => image);\n            }\n\n            return image;\n          });\n      })\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        value: 'fetch_error'\n      }, error, context));\n  } else if(imageId !== '') {\n    const update = {\n      description,\n      modified: now\n    };\n    const aqlQry: AqlQuery = aql`UPDATE {_key: ${formatImageId}} WITH ${update} IN images RETURN NEW`;\n\n    return useDb(databaseName).query(aqlQry)\n      .then((cursor) => cursor.next())\n      .catch((error: Error) => logError({\n        action,\n        category: eventCategory,\n        params: {\n          aqlQry,\n          formatImageId,\n          update\n        },\n        value: ErrorTypes.DATABASE_ERROR\n      }, error, context));\n  }\n\n  return null;\n};\n\nexport const deleteImage = async (context: ApiContext, imageId: string): Promise<ImageType> => {\n  const action: string = 'delete';\n  const {databaseName, session: {userId: sessionId}} = context;\n  const formatImageId = parseId(imageId);\n\n  try {\n    const removeEdgeQuery = aql`FOR hi IN hasImage\n    FILTER hi._from == ${formatImageId}\n    REMOVE hi IN hasImage\n    RETURN OLD`;\n\n    await useDb(databaseName).query(removeEdgeQuery);\n\n    const aqlQuery = aql`FOR i IN images\n    FILTER i._key == ${formatImageId} && i.userId == ${sessionId}\n    REMOVE i IN images\n    RETURN OLD`;\n\n    const image = await useDb(databaseName).query(aqlQuery).then((cursor) => cursor.next());\n\n    if(image) {\n      const {_key: imageKey} = image;\n      const params: DeleteObjectsCommandInput = {\n        Bucket: null,\n        Delete: {\n          Objects: [\n            {Key: `users/${sessionId}/images/${imageKey}.jpg`},\n            {Key: `users/${sessionId}/thumbs/${imageKey}.jpg`}\n          ],\n          Quiet: true\n        }\n      };\n\n      return s3DeleteList(params).then(() => image);\n    }\n\n    return null;\n  } catch(error) {\n    return logError({\n      action,\n      category: eventCategory,\n      params: {imageId},\n      value: ErrorTypes.DATABASE_ERROR\n    }, error, context);\n  }\n};\n"],"names":["get","httpGet","parseNum","createHash","parseArangoId","parseChar","parseId","aql","gm","DateTime","Config","ErrorTypes","logError","logException","getDocId","getLimit","selectReactionCountByType","useDb","getGroupDetails","isGrouped","s3DeleteList","s3GetSignedUrl","s3Put","lowerCaseKeys","eventCategory","subClass","imageMagick","parseImageOptions","options","from","to","type","limit","getImageOptional","fields","reduce","selects","field","includes","queries","push","objects","getImagesByUser","context","userId","action","databaseName","formatUserId","aqlQry","query","then","cursor","all","catch","error","category","params","value","DATABASE_ERROR","getImageCountByItem","itemId","formatItemId","next","count","getImagesByItem","selectObjects","selectQueries","join","getImagesByGroup","session","sessionId","filters","groupId","formatGroupId","map","filter","conditional","name","formatCond","group","privacy","filterStr","grouped","isValid","filterList","getImagesByReactions","reactions","formatSessionId","formatReactions","reactionName","filterBy","JSON","stringify","getImage","imageId","formatImageId","getPathUserImages","imageType","dir","filename","getAppImageUrl","bucket","directory","isThumb","typeId","bucketName","imageName","imageUrl","getImageUrl","imageDir","urlType","thumbSuffix","imageKey","defaultBucket","Bucket","Expires","Key","resizeSaveImage","buffer","fileType","s3Options","imgW","imgQ","format","split","imageOptimizedBuffer","Promise","resolve","reject","setFormat","quality","autoOrient","resize","stream","streamError","stdout","IMAGE_SAVE","imageBuffer","Buffer","on","data","concat","imageObj","ACL","Body","ContentType","imageIdentity","identify","bufferStream","imageData","formatValues","make","cameraMake","model","cameraModel","datetimeoriginal","taken","stats","color","red","green","blue","mean","substring","rgb","rgbColor","toString","Orientation","orientationData","Resolution","resolutionData","size","resolution","resolutionNumber","resolutionUnit","resolutionValue","resolutionUnitValue","fileSize","length","height","orientation","undefined","toLowerCase","fromMillis","millisecond","width","thumbOptimizedBuffer","thmW","thmQ","gravity","extent","thumbStdout","thumbBuffer","thumbObj","addImage","image","description","now","Date","resizedImage","insert","_key","added","modified","IMAGE_RESIZE","addImageEdge","imageEdge","itemType","edgeId","formatItemType","itemDocId","fileDocId","edge","_from","_to","collection","save","returnNew","fileEdge","document","updateImage","base64","url","customParams","formatBase64","substr","indexOf","updatedFileType","fileTypeFromBuffer","mime","imgParams","contentType","res","status","IMAGE_REQUEST","headers","update","deleteImage","removeEdgeQuery","aqlQuery","Delete","Objects","Quiet"],"mappings":"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"}
|