@nlabs/reaktor 0.10.2 → 0.10.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/coverage/block-navigation.js +1 -1
- package/coverage/index.html +23 -23
- package/coverage/sorter.js +21 -7
- package/lib/actions/apps.d.ts +1 -0
- package/lib/actions/apps.d.ts.map +1 -0
- package/lib/actions/apps.js +218 -17
- package/lib/actions/connections.d.ts +1 -0
- package/lib/actions/connections.d.ts.map +1 -0
- package/lib/actions/connections.js +61 -6
- package/lib/actions/content.d.ts +1 -0
- package/lib/actions/content.d.ts.map +1 -0
- package/lib/actions/content.js +232 -9
- package/lib/actions/conversations.d.ts +1 -0
- package/lib/actions/conversations.d.ts.map +1 -0
- package/lib/actions/conversations.js +321 -19
- package/lib/actions/dynamodb.d.ts +1 -0
- package/lib/actions/dynamodb.d.ts.map +1 -0
- package/lib/actions/dynamodb.js +119 -2
- package/lib/actions/email.d.ts +1 -0
- package/lib/actions/email.d.ts.map +1 -0
- package/lib/actions/email.js +56 -2
- package/lib/actions/files.d.ts +1 -0
- package/lib/actions/files.d.ts.map +1 -0
- package/lib/actions/files.js +226 -5
- package/lib/actions/groups.d.ts +1 -0
- package/lib/actions/groups.d.ts.map +1 -0
- package/lib/actions/groups.js +234 -19
- package/lib/actions/images.d.ts +1 -0
- package/lib/actions/images.d.ts.map +1 -0
- package/lib/actions/images.js +688 -31
- package/lib/actions/index.d.ts +2 -0
- package/lib/actions/index.d.ts.map +1 -0
- package/lib/actions/index.js +29 -2
- package/lib/actions/ios.d.ts +1 -0
- package/lib/actions/ios.d.ts.map +1 -0
- package/lib/actions/ios.js +228 -9
- package/lib/actions/locations.d.ts +1 -0
- package/lib/actions/locations.d.ts.map +1 -0
- package/lib/actions/locations.js +111 -7
- package/lib/actions/messages.d.ts +1 -0
- package/lib/actions/messages.d.ts.map +1 -0
- package/lib/actions/messages.js +155 -21
- package/lib/actions/notifications.d.ts +1 -0
- package/lib/actions/notifications.d.ts.map +1 -0
- package/lib/actions/notifications.js +37 -2
- package/lib/actions/payments.d.ts +1 -0
- package/lib/actions/payments.d.ts.map +1 -0
- package/lib/actions/payments.js +428 -11
- package/lib/actions/posts.d.ts +1 -0
- package/lib/actions/posts.d.ts.map +1 -0
- package/lib/actions/posts.js +540 -75
- package/lib/actions/profiles.d.ts +1 -0
- package/lib/actions/profiles.d.ts.map +1 -0
- package/lib/actions/profiles.js +58 -8
- package/lib/actions/reactions.d.ts +1 -0
- package/lib/actions/reactions.d.ts.map +1 -0
- package/lib/actions/reactions.js +276 -25
- package/lib/actions/s3.d.ts +1 -0
- package/lib/actions/s3.d.ts.map +1 -0
- package/lib/actions/s3.js +102 -2
- package/lib/actions/search.d.ts +1 -0
- package/lib/actions/search.d.ts.map +1 -0
- package/lib/actions/search.js +81 -5
- package/lib/actions/sms.d.ts +1 -0
- package/lib/actions/sms.d.ts.map +1 -0
- package/lib/actions/sms.js +52 -2
- package/lib/actions/statistics.d.ts +1 -0
- package/lib/actions/statistics.d.ts.map +1 -0
- package/lib/actions/statistics.js +41 -6
- package/lib/actions/subscriptions.d.ts +1 -0
- package/lib/actions/subscriptions.d.ts.map +1 -0
- package/lib/actions/subscriptions.js +196 -6
- package/lib/actions/tags.d.ts +1 -0
- package/lib/actions/tags.d.ts.map +1 -0
- package/lib/actions/tags.js +258 -19
- package/lib/actions/users.d.ts +1 -0
- package/lib/actions/users.d.ts.map +1 -0
- package/lib/actions/users.js +781 -66
- package/lib/actions/videos.d.ts +26 -0
- package/lib/actions/videos.d.ts.map +1 -0
- package/lib/actions/videos.js +423 -0
- package/lib/actions/websockets.d.ts +1 -0
- package/lib/actions/websockets.d.ts.map +1 -0
- package/lib/actions/websockets.js +148 -14
- package/lib/adapters/arangoAdapter.d.ts +1 -0
- package/lib/adapters/arangoAdapter.d.ts.map +1 -0
- package/lib/adapters/arangoAdapter.js +70 -2
- package/lib/adapters/contentAdapter.d.ts +1 -0
- package/lib/adapters/contentAdapter.d.ts.map +1 -0
- package/lib/adapters/contentAdapter.js +108 -2
- package/lib/adapters/fileAdapter.d.ts +1 -0
- package/lib/adapters/fileAdapter.d.ts.map +1 -0
- package/lib/adapters/fileAdapter.js +119 -2
- package/lib/adapters/imageAdapter.d.ts +1 -0
- package/lib/adapters/imageAdapter.d.ts.map +1 -0
- package/lib/adapters/imageAdapter.js +112 -2
- package/lib/adapters/index.d.ts +1 -0
- package/lib/adapters/index.d.ts.map +1 -0
- package/lib/adapters/index.js +11 -2
- package/lib/adapters/messageAdapter.d.ts +1 -0
- package/lib/adapters/messageAdapter.d.ts.map +1 -0
- package/lib/adapters/messageAdapter.js +82 -2
- package/lib/adapters/postAdapter.d.ts +1 -0
- package/lib/adapters/postAdapter.d.ts.map +1 -0
- package/lib/adapters/postAdapter.js +117 -2
- package/lib/adapters/reaktorAdapter.d.ts +1 -0
- package/lib/adapters/reaktorAdapter.d.ts.map +1 -0
- package/lib/adapters/reaktorAdapter.js +59 -2
- package/lib/adapters/tagAdapter.d.ts +1 -0
- package/lib/adapters/tagAdapter.d.ts.map +1 -0
- package/lib/adapters/tagAdapter.js +99 -2
- package/lib/adapters/userAdapter.d.ts +1 -0
- package/lib/adapters/userAdapter.d.ts.map +1 -0
- package/lib/adapters/userAdapter.js +264 -2
- package/lib/config.d.ts +1 -0
- package/lib/config.d.ts.map +1 -0
- package/lib/config.js +161 -2
- package/lib/handlers/graphqlHandler.d.ts +1 -0
- package/lib/handlers/graphqlHandler.d.ts.map +1 -0
- package/lib/handlers/graphqlHandler.js +118 -2
- package/lib/index.d.ts +1 -0
- package/lib/index.d.ts.map +1 -0
- package/lib/index.js +18 -2
- package/lib/lambdas/actions/websockets.d.ts +1 -0
- package/lib/lambdas/actions/websockets.d.ts.map +1 -0
- package/lib/lambdas/actions/websockets.js +89 -14
- package/lib/lambdas/authorizer.d.ts +1 -0
- package/lib/lambdas/authorizer.d.ts.map +1 -0
- package/lib/lambdas/authorizer.js +41 -2
- package/lib/lambdas/connection.d.ts +1 -0
- package/lib/lambdas/connection.d.ts.map +1 -0
- package/lib/lambdas/connection.js +85 -2
- package/lib/lambdas/utils/message.d.ts +1 -0
- package/lib/lambdas/utils/message.d.ts.map +1 -0
- package/lib/lambdas/utils/message.js +20 -2
- package/lib/lambdas/utils/websocket.d.ts +1 -0
- package/lib/lambdas/utils/websocket.d.ts.map +1 -0
- package/lib/lambdas/utils/websocket.js +78 -2
- package/lib/mocks/conversation.d.ts +1 -0
- package/lib/mocks/conversation.d.ts.map +1 -0
- package/lib/mocks/conversation.js +10 -2
- package/lib/mocks/file.d.ts +1 -0
- package/lib/mocks/file.d.ts.map +1 -0
- package/lib/mocks/file.js +13 -2
- package/lib/mocks/group.d.ts +1 -0
- package/lib/mocks/group.d.ts.map +1 -0
- package/lib/mocks/group.js +20 -2
- package/lib/mocks/image.d.ts +1 -0
- package/lib/mocks/image.d.ts.map +1 -0
- package/lib/mocks/image.js +17 -2
- package/lib/mocks/post.d.ts +1 -0
- package/lib/mocks/post.d.ts.map +1 -0
- package/lib/mocks/post.js +28 -2
- package/lib/mocks/tag.d.ts +1 -0
- package/lib/mocks/tag.d.ts.map +1 -0
- package/lib/mocks/tag.js +12 -2
- package/lib/mocks/user.d.ts +1 -0
- package/lib/mocks/user.d.ts.map +1 -0
- package/lib/mocks/user.js +61 -2
- package/lib/mocks/video.d.ts +4 -0
- package/lib/mocks/video.d.ts.map +1 -0
- package/lib/mocks/video.js +17 -0
- package/lib/mutations/content.d.ts +1 -0
- package/lib/mutations/content.d.ts.map +1 -0
- package/lib/mutations/content.js +27 -2
- package/lib/mutations/index.d.ts +1 -0
- package/lib/mutations/index.d.ts.map +1 -0
- package/lib/mutations/index.js +29 -2
- package/lib/mutations/locations.d.ts +1 -0
- package/lib/mutations/locations.d.ts.map +1 -0
- package/lib/mutations/locations.js +22 -2
- package/lib/mutations/messages.d.ts +1 -0
- package/lib/mutations/messages.d.ts.map +1 -0
- package/lib/mutations/messages.js +75 -2
- package/lib/mutations/posts.d.ts +1 -0
- package/lib/mutations/posts.d.ts.map +1 -0
- package/lib/mutations/posts.js +31 -2
- package/lib/mutations/profiles.d.ts +1 -0
- package/lib/mutations/profiles.d.ts.map +1 -0
- package/lib/mutations/profiles.js +78 -2
- package/lib/mutations/reactions.d.ts +1 -0
- package/lib/mutations/reactions.d.ts.map +1 -0
- package/lib/mutations/reactions.js +29 -2
- package/lib/mutations/statistics.d.ts +1 -0
- package/lib/mutations/statistics.d.ts.map +1 -0
- package/lib/mutations/statistics.js +17 -2
- package/lib/mutations/subscriptions.d.ts +1 -0
- package/lib/mutations/subscriptions.d.ts.map +1 -0
- package/lib/mutations/subscriptions.js +38 -2
- package/lib/mutations/tags.d.ts +1 -0
- package/lib/mutations/tags.d.ts.map +1 -0
- package/lib/mutations/tags.js +109 -2
- package/lib/mutations/users.d.ts +1 -0
- package/lib/mutations/users.d.ts.map +1 -0
- package/lib/mutations/users.js +129 -2
- package/lib/objectTypes/app.d.ts +1 -0
- package/lib/objectTypes/app.d.ts.map +1 -0
- package/lib/objectTypes/app.js +147 -2
- package/lib/objectTypes/bankAccount.d.ts +1 -0
- package/lib/objectTypes/bankAccount.d.ts.map +1 -0
- package/lib/objectTypes/bankAccount.js +54 -2
- package/lib/objectTypes/connection.d.ts +1 -0
- package/lib/objectTypes/connection.d.ts.map +1 -0
- package/lib/objectTypes/connection.js +26 -2
- package/lib/objectTypes/content.d.ts +1 -0
- package/lib/objectTypes/content.d.ts.map +1 -0
- package/lib/objectTypes/content.js +79 -2
- package/lib/objectTypes/conversation.d.ts +1 -0
- package/lib/objectTypes/conversation.d.ts.map +1 -0
- package/lib/objectTypes/conversation.js +53 -2
- package/lib/objectTypes/creditCard.d.ts +1 -0
- package/lib/objectTypes/creditCard.d.ts.map +1 -0
- package/lib/objectTypes/creditCard.js +64 -2
- package/lib/objectTypes/document.d.ts +1 -0
- package/lib/objectTypes/document.d.ts.map +1 -0
- package/lib/objectTypes/document.js +21 -2
- package/lib/objectTypes/error.d.ts +1 -0
- package/lib/objectTypes/error.d.ts.map +1 -0
- package/lib/objectTypes/error.js +24 -2
- package/lib/objectTypes/external.d.ts +1 -0
- package/lib/objectTypes/external.d.ts.map +1 -0
- package/lib/objectTypes/external.js +52 -2
- package/lib/objectTypes/file.d.ts +1 -0
- package/lib/objectTypes/file.d.ts.map +1 -0
- package/lib/objectTypes/file.js +76 -2
- package/lib/objectTypes/filter.d.ts +1 -0
- package/lib/objectTypes/filter.d.ts.map +1 -0
- package/lib/objectTypes/filter.js +21 -2
- package/lib/objectTypes/group.d.ts +1 -0
- package/lib/objectTypes/group.d.ts.map +1 -0
- package/lib/objectTypes/group.js +97 -2
- package/lib/objectTypes/iapSubscription.d.ts +1 -0
- package/lib/objectTypes/iapSubscription.d.ts.map +1 -0
- package/lib/objectTypes/iapSubscription.js +18 -2
- package/lib/objectTypes/image.d.ts +1 -0
- package/lib/objectTypes/image.d.ts.map +1 -0
- package/lib/objectTypes/image.js +105 -2
- package/lib/objectTypes/index.d.ts +1 -0
- package/lib/objectTypes/index.d.ts.map +1 -0
- package/lib/objectTypes/index.js +28 -2
- package/lib/objectTypes/location.d.ts +1 -0
- package/lib/objectTypes/location.d.ts.map +1 -0
- package/lib/objectTypes/location.js +85 -2
- package/lib/objectTypes/message.d.ts +1 -0
- package/lib/objectTypes/message.d.ts.map +1 -0
- package/lib/objectTypes/message.js +72 -2
- package/lib/objectTypes/passcode.d.ts +1 -0
- package/lib/objectTypes/passcode.d.ts.map +1 -0
- package/lib/objectTypes/passcode.js +20 -2
- package/lib/objectTypes/plan.d.ts +1 -0
- package/lib/objectTypes/plan.d.ts.map +1 -0
- package/lib/objectTypes/plan.js +71 -2
- package/lib/objectTypes/post.d.ts +1 -0
- package/lib/objectTypes/post.d.ts.map +1 -0
- package/lib/objectTypes/post.js +101 -2
- package/lib/objectTypes/profile.d.ts +1 -0
- package/lib/objectTypes/profile.d.ts.map +1 -0
- package/lib/objectTypes/profile.js +68 -2
- package/lib/objectTypes/reaction.d.ts +1 -0
- package/lib/objectTypes/reaction.d.ts.map +1 -0
- package/lib/objectTypes/reaction.js +37 -2
- package/lib/objectTypes/relation.d.ts +1 -0
- package/lib/objectTypes/relation.d.ts.map +1 -0
- package/lib/objectTypes/relation.js +27 -2
- package/lib/objectTypes/search.d.ts +1 -0
- package/lib/objectTypes/search.d.ts.map +1 -0
- package/lib/objectTypes/search.js +50 -2
- package/lib/objectTypes/statistics.d.ts +1 -0
- package/lib/objectTypes/statistics.d.ts.map +1 -0
- package/lib/objectTypes/statistics.js +17 -2
- package/lib/objectTypes/subscription.d.ts +1 -0
- package/lib/objectTypes/subscription.d.ts.map +1 -0
- package/lib/objectTypes/subscription.js +102 -2
- package/lib/objectTypes/tag.d.ts +1 -0
- package/lib/objectTypes/tag.d.ts.map +1 -0
- package/lib/objectTypes/tag.js +41 -2
- package/lib/objectTypes/user.d.ts +1 -0
- package/lib/objectTypes/user.d.ts.map +1 -0
- package/lib/objectTypes/user.js +111 -2
- package/lib/queries/content.d.ts +1 -0
- package/lib/queries/content.d.ts.map +1 -0
- package/lib/queries/content.js +50 -2
- package/lib/queries/index.d.ts +1 -0
- package/lib/queries/index.d.ts.map +1 -0
- package/lib/queries/index.js +27 -2
- package/lib/queries/locations.d.ts +1 -0
- package/lib/queries/locations.d.ts.map +1 -0
- package/lib/queries/locations.js +23 -2
- package/lib/queries/messages.d.ts +1 -0
- package/lib/queries/messages.d.ts.map +1 -0
- package/lib/queries/messages.js +35 -2
- package/lib/queries/posts.d.ts +1 -0
- package/lib/queries/posts.d.ts.map +1 -0
- package/lib/queries/posts.js +154 -2
- package/lib/queries/reactions.d.ts +1 -0
- package/lib/queries/reactions.d.ts.map +1 -0
- package/lib/queries/reactions.js +34 -2
- package/lib/queries/statistics.d.ts +1 -0
- package/lib/queries/statistics.d.ts.map +1 -0
- package/lib/queries/statistics.js +17 -2
- package/lib/queries/subscriptions.d.ts +1 -0
- package/lib/queries/subscriptions.d.ts.map +1 -0
- package/lib/queries/subscriptions.js +21 -2
- package/lib/queries/tags.d.ts +1 -0
- package/lib/queries/tags.d.ts.map +1 -0
- package/lib/queries/tags.js +56 -2
- package/lib/queries/users.d.ts +1 -0
- package/lib/queries/users.d.ts.map +1 -0
- package/lib/queries/users.js +39 -2
- package/lib/templates/email/layout.d.ts +1 -0
- package/lib/templates/email/layout.d.ts.map +1 -0
- package/lib/templates/email/layout.js +4 -3
- package/lib/templates/email/passwordForgot.d.ts +1 -0
- package/lib/templates/email/passwordForgot.d.ts.map +1 -0
- package/lib/templates/email/passwordForgot.js +4 -3
- package/lib/templates/email/passwordRecovery.d.ts +1 -0
- package/lib/templates/email/passwordRecovery.d.ts.map +1 -0
- package/lib/templates/email/passwordRecovery.js +4 -3
- package/lib/templates/email/verifyEmail.d.ts +1 -0
- package/lib/templates/email/verifyEmail.d.ts.map +1 -0
- package/lib/templates/email/verifyEmail.js +4 -3
- package/lib/templates/email/welcome.d.ts +1 -0
- package/lib/templates/email/welcome.d.ts.map +1 -0
- package/lib/templates/email/welcome.js +4 -3
- package/lib/templates/sms/passwordForgot.d.ts +1 -0
- package/lib/templates/sms/passwordForgot.d.ts.map +1 -0
- package/lib/templates/sms/passwordForgot.js +3 -2
- package/lib/templates/sms/passwordRecovery.d.ts +1 -0
- package/lib/templates/sms/passwordRecovery.d.ts.map +1 -0
- package/lib/templates/sms/passwordRecovery.js +3 -2
- package/lib/templates/sms/verifyEmail.d.ts +1 -0
- package/lib/templates/sms/verifyEmail.d.ts.map +1 -0
- package/lib/templates/sms/verifyEmail.js +3 -2
- package/lib/templates/sms/verifyPhone.d.ts +1 -0
- package/lib/templates/sms/verifyPhone.d.ts.map +1 -0
- package/lib/templates/sms/verifyPhone.js +3 -2
- package/lib/templates/sms/welcome.d.ts +1 -0
- package/lib/templates/sms/welcome.d.ts.map +1 -0
- package/lib/templates/sms/welcome.js +3 -2
- package/lib/types/apps.types.d.ts +1 -0
- package/lib/types/apps.types.d.ts.map +1 -0
- package/lib/types/apps.types.js +10 -2
- package/lib/types/arangodb.types.d.ts +2 -1
- package/lib/types/arangodb.types.d.ts.map +1 -0
- package/lib/types/arangodb.types.js +6 -1
- package/lib/types/auth.types.d.ts +1 -0
- package/lib/types/auth.types.d.ts.map +1 -0
- package/lib/types/auth.types.js +6 -1
- package/lib/types/connections.types.d.ts +1 -0
- package/lib/types/connections.types.d.ts.map +1 -0
- package/lib/types/connections.types.js +6 -1
- package/lib/types/content.types.d.ts +1 -0
- package/lib/types/content.types.d.ts.map +1 -0
- package/lib/types/content.types.js +6 -1
- package/lib/types/conversations.types.d.ts +1 -0
- package/lib/types/conversations.types.d.ts.map +1 -0
- package/lib/types/conversations.types.js +6 -1
- package/lib/types/email.types.d.ts +1 -0
- package/lib/types/email.types.d.ts.map +1 -0
- package/lib/types/email.types.js +6 -1
- package/lib/types/error.types.d.ts +1 -0
- package/lib/types/error.types.d.ts.map +1 -0
- package/lib/types/error.types.js +20 -2
- package/lib/types/files.types.d.ts +1 -0
- package/lib/types/files.types.d.ts.map +1 -0
- package/lib/types/files.types.js +6 -1
- package/lib/types/google.types.d.ts +1 -0
- package/lib/types/google.types.d.ts.map +1 -0
- package/lib/types/google.types.js +6 -1
- package/lib/types/groups.types.d.ts +1 -0
- package/lib/types/groups.types.d.ts.map +1 -0
- package/lib/types/groups.types.js +6 -1
- package/lib/types/images.types.d.ts +1 -0
- package/lib/types/images.types.d.ts.map +1 -0
- package/lib/types/images.types.js +6 -1
- package/lib/types/index.d.ts +2 -0
- package/lib/types/index.d.ts.map +1 -0
- package/lib/types/index.js +28 -2
- package/lib/types/locations.types.d.ts +1 -0
- package/lib/types/locations.types.d.ts.map +1 -0
- package/lib/types/locations.types.js +6 -1
- package/lib/types/messages.types.d.ts +1 -0
- package/lib/types/messages.types.d.ts.map +1 -0
- package/lib/types/messages.types.js +6 -1
- package/lib/types/notifications.types.d.ts +1 -0
- package/lib/types/notifications.types.d.ts.map +1 -0
- package/lib/types/notifications.types.js +6 -1
- package/lib/types/payments.types.d.ts +1 -0
- package/lib/types/payments.types.d.ts.map +1 -0
- package/lib/types/payments.types.js +6 -1
- package/lib/types/posts.types.d.ts +1 -0
- package/lib/types/posts.types.d.ts.map +1 -0
- package/lib/types/posts.types.js +6 -1
- package/lib/types/profiles.types.d.ts +1 -0
- package/lib/types/profiles.types.d.ts.map +1 -0
- package/lib/types/profiles.types.js +3 -1
- package/lib/types/statistics.types.d.ts +1 -0
- package/lib/types/statistics.types.d.ts.map +1 -0
- package/lib/types/statistics.types.js +3 -1
- package/lib/types/tags.types.d.ts +1 -0
- package/lib/types/tags.types.d.ts.map +1 -0
- package/lib/types/tags.types.js +6 -1
- package/lib/types/users.types.d.ts +1 -0
- package/lib/types/users.types.d.ts.map +1 -0
- package/lib/types/users.types.js +6 -1
- package/lib/types/videos.types.d.ts +47 -0
- package/lib/types/videos.types.d.ts.map +1 -0
- package/lib/types/videos.types.js +6 -0
- package/lib/types/websockets.types.d.ts +1 -0
- package/lib/types/websockets.types.d.ts.map +1 -0
- package/lib/types/websockets.types.js +6 -1
- package/lib/utils/adapterUtils.d.ts +1 -0
- package/lib/utils/adapterUtils.d.ts.map +1 -0
- package/lib/utils/adapterUtils.js +23 -2
- package/lib/utils/analyticsUtils.d.ts +1 -0
- package/lib/utils/analyticsUtils.d.ts.map +1 -0
- package/lib/utils/analyticsUtils.js +45 -2
- package/lib/utils/arangodbUtils.d.ts +1 -0
- package/lib/utils/arangodbUtils.d.ts.map +1 -0
- package/lib/utils/arangodbUtils.js +128 -5
- package/lib/utils/authUtils.d.ts +1 -0
- package/lib/utils/authUtils.d.ts.map +1 -0
- package/lib/utils/authUtils.js +54 -2
- package/lib/utils/contextUtils.d.ts +1 -0
- package/lib/utils/contextUtils.d.ts.map +1 -0
- package/lib/utils/contextUtils.js +10 -2
- package/lib/utils/dbI18n.d.ts +1 -0
- package/lib/utils/dbI18n.d.ts.map +1 -0
- package/lib/utils/dbI18n.example.d.ts +1 -0
- package/lib/utils/dbI18n.example.d.ts.map +1 -0
- package/lib/utils/dbI18n.example.js +92 -5
- package/lib/utils/dbI18n.js +28 -2
- package/lib/utils/googleTranslate.d.ts +1 -0
- package/lib/utils/googleTranslate.d.ts.map +1 -0
- package/lib/utils/googleTranslate.js +70 -2
- package/lib/utils/graphqlUtils.d.ts +1 -0
- package/lib/utils/graphqlUtils.d.ts.map +1 -0
- package/lib/utils/graphqlUtils.js +11 -2
- package/lib/utils/index.d.ts +1 -0
- package/lib/utils/index.d.ts.map +1 -0
- package/lib/utils/index.js +19 -2
- package/lib/utils/languageDetection.d.ts +1 -0
- package/lib/utils/languageDetection.d.ts.map +1 -0
- package/lib/utils/languageDetection.js +120 -2
- package/lib/utils/localeUtils.d.ts +1 -0
- package/lib/utils/localeUtils.d.ts.map +1 -0
- package/lib/utils/localeUtils.example.d.ts +1 -0
- package/lib/utils/localeUtils.example.d.ts.map +1 -0
- package/lib/utils/localeUtils.example.js +124 -2
- package/lib/utils/localeUtils.js +71 -2
- package/lib/utils/middlewareUtils.d.ts +1 -0
- package/lib/utils/middlewareUtils.d.ts.map +1 -0
- package/lib/utils/middlewareUtils.js +10 -2
- package/lib/utils/sessionUtils.d.ts +1 -0
- package/lib/utils/sessionUtils.d.ts.map +1 -0
- package/lib/utils/sessionUtils.js +26 -2
- package/lib/utils/stripeUtils.d.ts +2 -1
- package/lib/utils/stripeUtils.d.ts.map +1 -0
- package/lib/utils/stripeUtils.js +12 -2
- package/lib/utils/templateUtils.d.ts +1 -0
- package/lib/utils/templateUtils.d.ts.map +1 -0
- package/lib/utils/templateUtils.js +11 -2
- package/lib/utils/testUtils.d.ts +1 -0
- package/lib/utils/testUtils.d.ts.map +1 -0
- package/lib/utils/testUtils.js +292 -2
- package/lib/utils/translationQueue.d.ts +1 -0
- package/lib/utils/translationQueue.d.ts.map +1 -0
- package/lib/utils/translationQueue.example.d.ts +1 -0
- package/lib/utils/translationQueue.example.d.ts.map +1 -0
- package/lib/utils/translationQueue.example.js +340 -2
- package/lib/utils/translationQueue.js +113 -2
- package/package.json +47 -29
package/lib/actions/payments.js
CHANGED
|
@@ -1,32 +1,449 @@
|
|
|
1
|
-
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) 2019-Present, Nitrogen Labs, Inc.
|
|
3
|
+
* Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
|
|
4
|
+
*/ import { parseNum } from '@nlabs/utils/parsers/numbers';
|
|
5
|
+
import { createHash, parseChar, parseId, parseString, parseVarChar } from '@nlabs/utils/parsers/strings';
|
|
6
|
+
import { aql } from 'arangojs';
|
|
7
|
+
import { ErrorTypes } from '../types/error.types.js';
|
|
8
|
+
import { logError, UserError } from '../utils/analyticsUtils.js';
|
|
9
|
+
import { useDb } from '../utils/arangodbUtils.js';
|
|
10
|
+
import { getStripeClient } from '../utils/stripeUtils.js';
|
|
11
|
+
import { getUser } from './users.js';
|
|
12
|
+
const eventCategory = 'payments';
|
|
13
|
+
export const addCustomerAccount = (context)=>{
|
|
14
|
+
const action = 'addCustomerAccount';
|
|
15
|
+
const { databaseName, session: { userId: sessionId, username } = {} } = context;
|
|
16
|
+
const stripeClient = getStripeClient();
|
|
17
|
+
return stripeClient.customers.create({
|
|
18
|
+
metadata: {
|
|
19
|
+
userId: sessionId,
|
|
20
|
+
username
|
|
21
|
+
}
|
|
22
|
+
}).then((customer)=>{
|
|
23
|
+
// Create session
|
|
24
|
+
const now = Date.now();
|
|
25
|
+
const update = {
|
|
26
|
+
modified: now,
|
|
27
|
+
stripeCustomerId: customer.id
|
|
28
|
+
};
|
|
29
|
+
const aqlQry = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;
|
|
30
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((updatedUser)=>!!updatedUser).catch((error)=>logError({
|
|
31
|
+
action,
|
|
32
|
+
category: eventCategory,
|
|
33
|
+
label: ErrorTypes.DATABASE_ERROR
|
|
34
|
+
}, error, context).then(()=>null));
|
|
35
|
+
});
|
|
36
|
+
};
|
|
37
|
+
export const addBankAccount = (context, bankAccount)=>{
|
|
38
|
+
const action = 'addPaymentAccountBank';
|
|
39
|
+
const { databaseName, session: { userId: sessionId } = {} } = context;
|
|
40
|
+
// Params
|
|
41
|
+
const { accountNumber, fullName, routing } = bankAccount;
|
|
42
|
+
const formatAccount = parseString(accountNumber, 32);
|
|
43
|
+
if (formatAccount === '') {
|
|
44
|
+
throw new UserError('required_account_number');
|
|
45
|
+
}
|
|
46
|
+
const formatFullName = parseVarChar(fullName, 128);
|
|
47
|
+
if (formatFullName === '') {
|
|
48
|
+
throw new UserError('required_full_name');
|
|
49
|
+
}
|
|
50
|
+
const formatRouting = parseString(routing, 32);
|
|
51
|
+
if (formatRouting === '') {
|
|
52
|
+
throw new UserError('required_routing_number');
|
|
53
|
+
}
|
|
54
|
+
return getUser(context, {
|
|
55
|
+
userId: sessionId
|
|
56
|
+
}).then((user)=>{
|
|
57
|
+
const { stripeAccountId } = user;
|
|
58
|
+
const stripeClient = getStripeClient();
|
|
59
|
+
// Create a token first, then use the token as the source
|
|
60
|
+
return stripeClient.tokens.create({
|
|
61
|
+
bank_account: {
|
|
62
|
+
account_holder_name: formatFullName,
|
|
63
|
+
account_holder_type: 'individual',
|
|
64
|
+
account_number: formatAccount,
|
|
65
|
+
country: 'US',
|
|
66
|
+
currency: 'USD',
|
|
67
|
+
routing_number: formatRouting
|
|
68
|
+
}
|
|
69
|
+
}).then((token)=>stripeClient.customers.createSource(stripeAccountId, {
|
|
70
|
+
source: token.id
|
|
71
|
+
})).then((account)=>{
|
|
72
|
+
// Use type assertion for card properties
|
|
73
|
+
// const cardSource = account as unknown as Card;
|
|
74
|
+
// const brand = cardSource.brand || '';
|
|
75
|
+
// const cvcCheck = cardSource.cvc_check || '';
|
|
76
|
+
// const last4 = cardSource.last4 || '';
|
|
77
|
+
// Create session
|
|
78
|
+
const now = Date.now();
|
|
79
|
+
const update = {
|
|
80
|
+
bankAccount,
|
|
81
|
+
bankFullName: formatFullName,
|
|
82
|
+
bankId: account.id,
|
|
83
|
+
bankRouting: formatRouting,
|
|
84
|
+
modified: now
|
|
85
|
+
};
|
|
86
|
+
const aqlQry = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;
|
|
87
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((updatedUser)=>updatedUser);
|
|
88
|
+
}).catch((error)=>{
|
|
89
|
+
const msg = error.message;
|
|
90
|
+
if (msg === 'A bank account with that routing number and account number ' + 'already exists for this customer.') {
|
|
91
|
+
return logError({
|
|
92
|
+
action,
|
|
93
|
+
category: eventCategory,
|
|
94
|
+
label: 'bank_account_exists'
|
|
95
|
+
}, error, context).then(()=>null);
|
|
96
|
+
}
|
|
97
|
+
return logError({
|
|
98
|
+
action,
|
|
99
|
+
category: eventCategory,
|
|
100
|
+
label: 'payment_error'
|
|
101
|
+
}, error, context).then(()=>null);
|
|
102
|
+
});
|
|
103
|
+
});
|
|
104
|
+
};
|
|
105
|
+
export const addCreditCard = (context, card)=>{
|
|
106
|
+
const action = 'addCreditCard';
|
|
107
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
108
|
+
return getUser(context, {
|
|
109
|
+
userId: sessionId
|
|
110
|
+
}).then((user)=>{
|
|
111
|
+
// User
|
|
112
|
+
const { stripeAccountId } = user;
|
|
113
|
+
// Card
|
|
114
|
+
const { accountNumber, city, country, cvc, expMonth, expYear, fullName, street1, street2, state, zip } = card;
|
|
115
|
+
const formatNumber = parseNum(accountNumber, 16);
|
|
116
|
+
if (!formatNumber) {
|
|
117
|
+
throw new UserError('required_credit_card_number');
|
|
118
|
+
}
|
|
119
|
+
const formatExpMonth = parseNum(expMonth, 2);
|
|
120
|
+
if (!formatExpMonth) {
|
|
121
|
+
throw new UserError('required_credit_card_exp_month');
|
|
122
|
+
}
|
|
123
|
+
const formatExpYear = parseNum(expYear, 2);
|
|
124
|
+
if (!formatExpYear) {
|
|
125
|
+
throw new UserError('required_credit_card_exp_year');
|
|
126
|
+
}
|
|
127
|
+
const formatCvc = parseNum(cvc, 3);
|
|
128
|
+
// Address
|
|
129
|
+
const paymentCard = {};
|
|
130
|
+
const formatCity = parseVarChar(city, 32);
|
|
131
|
+
if (formatCity) {
|
|
132
|
+
paymentCard.city = formatCity;
|
|
133
|
+
}
|
|
134
|
+
const formatCountry = parseChar(country, 2);
|
|
135
|
+
if (formatCountry) {
|
|
136
|
+
paymentCard.country = formatCountry;
|
|
137
|
+
}
|
|
138
|
+
const formatFullName = parseVarChar(fullName, 32);
|
|
139
|
+
if (formatFullName) {
|
|
140
|
+
paymentCard.fullName = formatFullName;
|
|
141
|
+
}
|
|
142
|
+
const formatStreet1 = parseVarChar(street1, 32);
|
|
143
|
+
if (formatStreet1) {
|
|
144
|
+
paymentCard.street1 = formatStreet1;
|
|
145
|
+
}
|
|
146
|
+
const formatStreet2 = parseVarChar(street2, 32);
|
|
147
|
+
if (formatStreet2) {
|
|
148
|
+
paymentCard.street2 = formatStreet2;
|
|
149
|
+
}
|
|
150
|
+
const formatState = parseChar(state, 2);
|
|
151
|
+
if (formatState) {
|
|
152
|
+
paymentCard.state = formatState;
|
|
153
|
+
}
|
|
154
|
+
const formatZip = parseVarChar(zip, 10);
|
|
155
|
+
if (formatZip) {
|
|
156
|
+
paymentCard.zip = formatZip;
|
|
157
|
+
}
|
|
158
|
+
const stripeClient = getStripeClient();
|
|
159
|
+
// Create a token first, then use the token as the source
|
|
160
|
+
return stripeClient.tokens.create({
|
|
161
|
+
card: {
|
|
162
|
+
address_city: formatCity,
|
|
163
|
+
address_country: formatCountry,
|
|
164
|
+
address_line1: formatStreet1,
|
|
165
|
+
address_line2: formatStreet2,
|
|
166
|
+
address_state: formatState,
|
|
167
|
+
address_zip: formatZip,
|
|
168
|
+
cvc: formatCvc.toString(),
|
|
169
|
+
exp_month: formatExpMonth.toString(),
|
|
170
|
+
exp_year: formatExpYear.toString(),
|
|
171
|
+
name: fullName,
|
|
172
|
+
number: formatNumber.toString()
|
|
173
|
+
}
|
|
174
|
+
}).then((token)=>stripeClient.customers.createSource(stripeAccountId, {
|
|
175
|
+
source: token.id
|
|
176
|
+
})).then((newSource)=>{
|
|
177
|
+
// Use type assertion for card properties
|
|
178
|
+
const cardSource = newSource;
|
|
179
|
+
const brand = cardSource.brand || '';
|
|
180
|
+
const cvcCheck = cardSource.cvc_check || '';
|
|
181
|
+
const last4 = cardSource.last4 || '';
|
|
182
|
+
// Create session
|
|
183
|
+
const now = Date.now();
|
|
184
|
+
const insert = {
|
|
185
|
+
...paymentCard,
|
|
186
|
+
_key: createHash(`user-payment-${sessionId}`),
|
|
187
|
+
accountNumber: last4,
|
|
188
|
+
added: now,
|
|
189
|
+
brand,
|
|
190
|
+
cvcCheck,
|
|
191
|
+
expMonth,
|
|
192
|
+
expYear,
|
|
193
|
+
modified: now,
|
|
194
|
+
userId: sessionId
|
|
195
|
+
};
|
|
196
|
+
const insertAqlQry = aql`INSERT ${insert} IN creditCards RETURN NEW`;
|
|
197
|
+
return useDb(databaseName).query(insertAqlQry).then((cursor)=>cursor.next()).then((newCard)=>{
|
|
198
|
+
if (newCard) {
|
|
199
|
+
// Add linked edge
|
|
200
|
+
const { _id: cardId, _key: cardKey } = card;
|
|
201
|
+
const edgeCollection = useDb(databaseName).collection('hasPayment');
|
|
202
|
+
const edgeId = createHash(`payment-${cardKey}`);
|
|
203
|
+
const edge = {
|
|
204
|
+
_from: `users/${sessionId}`,
|
|
205
|
+
_key: edgeId,
|
|
206
|
+
_to: cardId
|
|
207
|
+
};
|
|
208
|
+
return edgeCollection.save(edge, {
|
|
209
|
+
returnNew: true
|
|
210
|
+
}).then(()=>card);
|
|
211
|
+
}
|
|
212
|
+
return newCard;
|
|
213
|
+
}).catch((error)=>logError({
|
|
214
|
+
action,
|
|
215
|
+
category: eventCategory,
|
|
216
|
+
label: 'payment_error'
|
|
217
|
+
}, error, context).then(()=>null));
|
|
218
|
+
});
|
|
219
|
+
});
|
|
220
|
+
};
|
|
221
|
+
export const updateCreditCard = (context, card)=>{
|
|
222
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
223
|
+
const { city, country, expMonth, expYear, fullName, id, street1, state, zip } = card;
|
|
224
|
+
const formatId = parseId(id);
|
|
225
|
+
if (formatId) {
|
|
226
|
+
throw new UserError('required_credit_card_id');
|
|
227
|
+
}
|
|
228
|
+
const paymentCard = {};
|
|
229
|
+
const formatExpMonth = parseNum(expMonth, 2);
|
|
230
|
+
const formatExpYear = parseNum(expYear, 2);
|
|
231
|
+
const formatCity = parseVarChar(city, 32);
|
|
232
|
+
const formatCountry = parseChar(country, 2);
|
|
233
|
+
const formatFullName = parseVarChar(fullName, 32);
|
|
234
|
+
const formatStreet1 = parseString(street1, 32);
|
|
235
|
+
const formatState = parseChar(state, 2);
|
|
236
|
+
const formatZip = parseVarChar(zip, 10);
|
|
237
|
+
if (formatExpMonth) {
|
|
238
|
+
paymentCard.expMonth = formatExpMonth;
|
|
239
|
+
}
|
|
240
|
+
if (formatExpYear) {
|
|
241
|
+
paymentCard.expYear = formatExpYear;
|
|
242
|
+
}
|
|
243
|
+
if (formatCity) {
|
|
244
|
+
paymentCard.city = formatCity;
|
|
245
|
+
}
|
|
246
|
+
if (formatCountry) {
|
|
247
|
+
paymentCard.country = formatCountry;
|
|
248
|
+
}
|
|
249
|
+
if (formatFullName) {
|
|
250
|
+
paymentCard.fullName = formatFullName;
|
|
251
|
+
}
|
|
252
|
+
if (formatStreet1) {
|
|
253
|
+
paymentCard.street1 = formatStreet1;
|
|
254
|
+
}
|
|
255
|
+
if (formatState) {
|
|
256
|
+
paymentCard.state = formatState;
|
|
257
|
+
}
|
|
258
|
+
if (formatZip) {
|
|
259
|
+
paymentCard.zip = formatZip;
|
|
260
|
+
}
|
|
261
|
+
const update = paymentCard;
|
|
262
|
+
const aqlQry = aql`
|
|
2
263
|
LET updatedCard = FIRST(
|
|
3
264
|
FOR c IN creditCards
|
|
4
|
-
FILTER c._key == ${
|
|
5
|
-
UPDATE c WITH ${
|
|
265
|
+
FILTER c._key == ${formatId} && c.userId == ${sessionId}
|
|
266
|
+
UPDATE c WITH ${update} IN creditCards
|
|
6
267
|
LIMIT 1
|
|
7
268
|
RETURN NEW
|
|
8
269
|
)
|
|
9
270
|
LET user = FIRST(
|
|
10
271
|
FOR u IN users
|
|
11
|
-
FILTER u._key == ${
|
|
272
|
+
FILTER u._key == ${sessionId}
|
|
12
273
|
LIMIT 1
|
|
13
274
|
RETURN u
|
|
14
275
|
)
|
|
15
|
-
RETURN {user: user, card: updatedCard}`;
|
|
16
|
-
|
|
17
|
-
|
|
276
|
+
RETURN {user: user, card: updatedCard}`;
|
|
277
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((results = {
|
|
278
|
+
card: {},
|
|
279
|
+
user: {}
|
|
280
|
+
})=>{
|
|
281
|
+
const updatedCard = results.card;
|
|
282
|
+
const { user } = results;
|
|
283
|
+
if (!updatedCard) {
|
|
284
|
+
throw new UserError('not_found');
|
|
285
|
+
}
|
|
286
|
+
const { stripeCustomerId } = user;
|
|
287
|
+
const { stripeId } = card;
|
|
288
|
+
const stripeClient = getStripeClient();
|
|
289
|
+
const update = {
|
|
290
|
+
address_city: formatCity,
|
|
291
|
+
address_country: formatCountry,
|
|
292
|
+
address_line1: formatStreet1,
|
|
293
|
+
address_state: formatState,
|
|
294
|
+
address_zip: formatZip,
|
|
295
|
+
exp_month: formatExpMonth.toString(),
|
|
296
|
+
exp_year: formatExpYear.toString(),
|
|
297
|
+
name: formatFullName
|
|
298
|
+
};
|
|
299
|
+
return stripeClient.customers.updateSource(stripeCustomerId, stripeId, update).then(()=>card).catch((error)=>{
|
|
300
|
+
console.log('payments::updateCard::error', error);
|
|
301
|
+
throw new UserError('payment_error');
|
|
302
|
+
});
|
|
303
|
+
});
|
|
304
|
+
};
|
|
305
|
+
export const getCreditCards = (context)=>{
|
|
306
|
+
const action = 'getCreditCards';
|
|
307
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
308
|
+
const aqlQry = aql`FOR c IN creditCards
|
|
309
|
+
FILTER c.userId == ${sessionId}
|
|
310
|
+
RETURN c`;
|
|
311
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.all()).then((list = [])=>list).catch((error)=>logError({
|
|
312
|
+
action,
|
|
313
|
+
category: eventCategory,
|
|
314
|
+
label: ErrorTypes.DATABASE_ERROR
|
|
315
|
+
}, error, context).then(()=>null));
|
|
316
|
+
};
|
|
317
|
+
export const deleteCreditCard = (context, cardId)=>{
|
|
318
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
319
|
+
const formatCardId = parseId(cardId);
|
|
320
|
+
const aqlQry = aql`
|
|
18
321
|
LET card = FIRST(
|
|
19
322
|
FOR c IN creditCards
|
|
20
|
-
FILTER c._key == ${
|
|
323
|
+
FILTER c._key == ${formatCardId} && c.userId == ${sessionId}
|
|
21
324
|
LIMIT 1
|
|
22
325
|
REMOVE c IN creditCards
|
|
23
326
|
RETURN OLD
|
|
24
327
|
)
|
|
25
328
|
LET user = FIRST(
|
|
26
329
|
FOR u IN users
|
|
27
|
-
FILTER u._key == ${
|
|
330
|
+
FILTER u._key == ${sessionId}
|
|
28
331
|
LIMIT 1
|
|
29
332
|
RETURN u
|
|
30
333
|
)
|
|
31
|
-
RETURN {user: user, card: card}`;
|
|
32
|
-
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vc3JjL2FjdGlvbnMvcGF5bWVudHMudHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbIi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDE5LVByZXNlbnQsIE5pdHJvZ2VuIExhYnMsIEluYy5cbiAqIENvcHlyaWdodHMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIGFjY29tcGFueWluZyBMSUNFTlNFIGZpbGUgZm9yIHRlcm1zLlxuICovXG5pbXBvcnQge3BhcnNlTnVtfSBmcm9tICdAbmxhYnMvdXRpbHMvcGFyc2Vycy9udW1iZXJzJztcbmltcG9ydCB7Y3JlYXRlSGFzaCwgcGFyc2VDaGFyLCBwYXJzZUlkLCBwYXJzZVN0cmluZywgcGFyc2VWYXJDaGFyfSBmcm9tICdAbmxhYnMvdXRpbHMvcGFyc2Vycy9zdHJpbmdzJztcbmltcG9ydCB7YXFsfSBmcm9tICdhcmFuZ29qcyc7XG5pbXBvcnQge0FxbFF1ZXJ5fSBmcm9tICdhcmFuZ29qcy9hcWwnO1xuaW1wb3J0IHN0cmlwZSBmcm9tICdzdHJpcGUnO1xuXG5pbXBvcnQge0FwaUNvbnRleHR9IGZyb20gJy4uL3R5cGVzL2F1dGgudHlwZXMuanMnO1xuaW1wb3J0IHtFcnJvclR5cGVzfSBmcm9tICcuLi90eXBlcy9lcnJvci50eXBlcy5qcyc7XG5pbXBvcnQge1BheW1lbnRCYW5rQWNjb3VudCwgUGF5bWVudENhcmRUeXBlLCBQYXltZW50Q2hhcmdlLCBQYXltZW50VHJhbnNmZXJ9IGZyb20gJy4uL3R5cGVzL3BheW1lbnRzLnR5cGVzLmpzJztcbmltcG9ydCB7VXNlclR5cGV9IGZyb20gJy4uL3R5cGVzL3VzZXJzLnR5cGVzLmpzJztcbmltcG9ydCB7bG9nRXJyb3IsIFVzZXJFcnJvcn0gZnJvbSAnLi4vdXRpbHMvYW5hbHl0aWNzVXRpbHMuanMnO1xuaW1wb3J0IHt1c2VEYn0gZnJvbSAnLi4vdXRpbHMvYXJhbmdvZGJVdGlscy5qcyc7XG5pbXBvcnQge2dldFN0cmlwZUNsaWVudH0gZnJvbSAnLi4vdXRpbHMvc3RyaXBlVXRpbHMuanMnO1xuaW1wb3J0IHtnZXRVc2VyfSBmcm9tICcuL3VzZXJzLmpzJztcblxuaW1wb3J0IHR5cGUge0VkZ2VDb2xsZWN0aW9ufSBmcm9tICdhcmFuZ29qcy9jb2xsZWN0aW9ucyc7XG5cbmNvbnN0IGV2ZW50Q2F0ZWdvcnkgPSAncGF5bWVudHMnO1xuXG5leHBvcnQgY29uc3QgYWRkQ3VzdG9tZXJBY2NvdW50ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQpOiBQcm9taXNlPGJvb2xlYW4+ID0+IHtcbiAgY29uc3QgYWN0aW9uID0gJ2FkZEN1c3RvbWVyQWNjb3VudCc7XG4gIGNvbnN0IHtkYXRhYmFzZU5hbWUsIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZCwgdXNlcm5hbWV9ID0ge319ID0gY29udGV4dDtcblxuICBjb25zdCBzdHJpcGVDbGllbnQgPSBnZXRTdHJpcGVDbGllbnQoKTtcblxuICByZXR1cm4gc3RyaXBlQ2xpZW50LmN1c3RvbWVyc1xuICAgIC5jcmVhdGUoe1xuICAgICAgbWV0YWRhdGE6IHtcbiAgICAgICAgdXNlcklkOiBzZXNzaW9uSWQsXG4gICAgICAgIHVzZXJuYW1lXG4gICAgICB9XG4gICAgfSBhcyBhbnkpXG4gICAgLnRoZW4oKGN1c3RvbWVyKSA9PiB7XG4gICAgICAvLyBDcmVhdGUgc2Vzc2lvblxuICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICAgICAgY29uc3QgdXBkYXRlOiBVc2VyVHlwZSA9IHtcbiAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgc3RyaXBlQ3VzdG9tZXJJZDogY3VzdG9tZXIuaWRcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBEQVRFICR7c2Vzc2lvbklkfSBXSVRIICR7dXBkYXRlfSBJTiB1c2VycyBMSU1JVCAxIFJFVFVSTiBORVdgO1xuXG4gICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2VOYW1lKS5xdWVyeShhcWxRcnkpXG4gICAgICAgIC50aGVuKChjdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAgIC50aGVuKCh1cGRhdGVkVXNlcjogVXNlclR5cGUpID0+ICEhdXBkYXRlZFVzZXIpXG4gICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgICAgIGxhYmVsOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBhZGRCYW5rQWNjb3VudCA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBiYW5rQWNjb3VudDogUGF5bWVudEJhbmtBY2NvdW50KTogUHJvbWlzZTxib29sZWFuPiA9PiB7XG4gIGNvbnN0IGFjdGlvbiA9ICdhZGRQYXltZW50QWNjb3VudEJhbmsnO1xuICBjb25zdCB7ZGF0YWJhc2VOYW1lLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9ID0ge319ID0gY29udGV4dDtcblxuICAvLyBQYXJhbXNcbiAgY29uc3Qge1xuICAgIGFjY291bnROdW1iZXIsXG4gICAgZnVsbE5hbWUsXG4gICAgcm91dGluZ1xuICB9ID0gYmFua0FjY291bnQ7XG5cbiAgY29uc3QgZm9ybWF0QWNjb3VudDogc3RyaW5nID0gcGFyc2VTdHJpbmcoYWNjb3VudE51bWJlciwgMzIpO1xuXG4gIGlmKGZvcm1hdEFjY291bnQgPT09ICcnKSB7XG4gICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfYWNjb3VudF9udW1iZXInKTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdEZ1bGxOYW1lOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoZnVsbE5hbWUsIDEyOCk7XG5cbiAgaWYoZm9ybWF0RnVsbE5hbWUgPT09ICcnKSB7XG4gICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfZnVsbF9uYW1lJyk7XG4gIH1cblxuICBjb25zdCBmb3JtYXRSb3V0aW5nOiBzdHJpbmcgPSBwYXJzZVN0cmluZyhyb3V0aW5nLCAzMik7XG5cbiAgaWYoZm9ybWF0Um91dGluZyA9PT0gJycpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9yb3V0aW5nX251bWJlcicpO1xuICB9XG5cbiAgcmV0dXJuIGdldFVzZXIoY29udGV4dCwge3VzZXJJZDogc2Vzc2lvbklkfSlcbiAgICAudGhlbigodXNlcjogVXNlclR5cGUpID0+IHtcbiAgICAgIGNvbnN0IHtzdHJpcGVBY2NvdW50SWR9ID0gdXNlcjtcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IGdldFN0cmlwZUNsaWVudCgpO1xuXG4gICAgICAvLyBDcmVhdGUgYSB0b2tlbiBmaXJzdCwgdGhlbiB1c2UgdGhlIHRva2VuIGFzIHRoZSBzb3VyY2VcbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQudG9rZW5zLmNyZWF0ZSh7XG4gICAgICAgIGJhbmtfYWNjb3VudDoge1xuICAgICAgICAgIGFjY291bnRfaG9sZGVyX25hbWU6IGZvcm1hdEZ1bGxOYW1lLFxuICAgICAgICAgIGFjY291bnRfaG9sZGVyX3R5cGU6ICdpbmRpdmlkdWFsJyxcbiAgICAgICAgICBhY2NvdW50X251bWJlcjogZm9ybWF0QWNjb3VudCxcbiAgICAgICAgICBjb3VudHJ5OiAnVVMnLFxuICAgICAgICAgIGN1cnJlbmN5OiAnVVNEJyxcbiAgICAgICAgICByb3V0aW5nX251bWJlcjogZm9ybWF0Um91dGluZ1xuICAgICAgICB9XG4gICAgICB9KVxuICAgICAgICAudGhlbigodG9rZW4pID0+IHN0cmlwZUNsaWVudC5jdXN0b21lcnMuY3JlYXRlU291cmNlKFxuICAgICAgICAgIHN0cmlwZUFjY291bnRJZCxcbiAgICAgICAgICB7c291cmNlOiB0b2tlbi5pZH1cbiAgICAgICAgKSlcbiAgICAgICAgLnRoZW4oKGFjY291bnQpID0+IHtcbiAgICAgICAgICAvLyBVc2UgdHlwZSBhc3NlcnRpb24gZm9yIGNhcmQgcHJvcGVydGllc1xuICAgICAgICAgIC8vIGNvbnN0IGNhcmRTb3VyY2UgPSBhY2NvdW50IGFzIHVua25vd24gYXMgQ2FyZDtcbiAgICAgICAgICAvLyBjb25zdCBicmFuZCA9IGNhcmRTb3VyY2UuYnJhbmQgfHwgJyc7XG4gICAgICAgICAgLy8gY29uc3QgY3ZjQ2hlY2sgPSBjYXJkU291cmNlLmN2Y19jaGVjayB8fCAnJztcbiAgICAgICAgICAvLyBjb25zdCBsYXN0NCA9IGNhcmRTb3VyY2UubGFzdDQgfHwgJyc7XG5cbiAgICAgICAgICAvLyBDcmVhdGUgc2Vzc2lvblxuICAgICAgICAgIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgICAgICAgICBjb25zdCB1cGRhdGUgPSB7XG4gICAgICAgICAgICBiYW5rQWNjb3VudCxcbiAgICAgICAgICAgIGJhbmtGdWxsTmFtZTogZm9ybWF0RnVsbE5hbWUsXG4gICAgICAgICAgICBiYW5rSWQ6IGFjY291bnQuaWQsXG4gICAgICAgICAgICBiYW5rUm91dGluZzogZm9ybWF0Um91dGluZyxcbiAgICAgICAgICAgIG1vZGlmaWVkOiBub3dcbiAgICAgICAgICB9O1xuXG4gICAgICAgICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUgJHtzZXNzaW9uSWR9IFdJVEggJHt1cGRhdGV9IElOIHVzZXJzIExJTUlUIDEgUkVUVVJOIE5FV2A7XG5cbiAgICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2VOYW1lKS5xdWVyeShhcWxRcnkpXG4gICAgICAgICAgICAudGhlbigoY3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAgICAgLnRoZW4oKHVwZGF0ZWRVc2VyOiBVc2VyVHlwZSkgPT4gdXBkYXRlZFVzZXIpO1xuICAgICAgICB9KVxuICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgIGNvbnN0IG1zZyA9IGVycm9yLm1lc3NhZ2U7XG5cbiAgICAgICAgICBpZihtc2cgPT09ICdBIGJhbmsgYWNjb3VudCB3aXRoIHRoYXQgcm91dGluZyBudW1iZXIgYW5kIGFjY291bnQgbnVtYmVyICcgK1xuICAgICAgICAgICAgJ2FscmVhZHkgZXhpc3RzIGZvciB0aGlzIGN1c3RvbWVyLicpIHtcbiAgICAgICAgICAgIHJldHVybiBsb2dFcnJvcih7XG4gICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICAgIGxhYmVsOiAnYmFua19hY2NvdW50X2V4aXN0cydcbiAgICAgICAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm4gbG9nRXJyb3Ioe1xuICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICBsYWJlbDogJ3BheW1lbnRfZXJyb3InXG4gICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCk7XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZENyZWRpdENhcmQgPSAoY29udGV4dDogQXBpQ29udGV4dCwgY2FyZDogUGF5bWVudENhcmRUeXBlKTogUHJvbWlzZTxVc2VyVHlwZT4gPT4ge1xuICBjb25zdCBhY3Rpb24gPSAnYWRkQ3JlZGl0Q2FyZCc7XG4gIGNvbnN0IHtkYXRhYmFzZU5hbWUsIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH19ID0gY29udGV4dDtcblxuICByZXR1cm4gZ2V0VXNlcihjb250ZXh0LCB7dXNlcklkOiBzZXNzaW9uSWR9KVxuICAgIC50aGVuKCh1c2VyOiBVc2VyVHlwZSkgPT4ge1xuICAgICAgLy8gVXNlclxuICAgICAgY29uc3Qge3N0cmlwZUFjY291bnRJZH0gPSB1c2VyO1xuXG4gICAgICAvLyBDYXJkXG4gICAgICBjb25zdCB7XG4gICAgICAgIGFjY291bnROdW1iZXIsXG4gICAgICAgIGNpdHksXG4gICAgICAgIGNvdW50cnksXG4gICAgICAgIGN2YyxcbiAgICAgICAgZXhwTW9udGgsXG4gICAgICAgIGV4cFllYXIsXG4gICAgICAgIGZ1bGxOYW1lLFxuICAgICAgICBzdHJlZXQxLFxuICAgICAgICBzdHJlZXQyLFxuICAgICAgICBzdGF0ZSxcbiAgICAgICAgemlwXG4gICAgICB9OiBQYXltZW50Q2FyZFR5cGUgPSBjYXJkO1xuXG4gICAgICBjb25zdCBmb3JtYXROdW1iZXI6IG51bWJlciA9IHBhcnNlTnVtKGFjY291bnROdW1iZXIsIDE2KTtcblxuICAgICAgaWYoIWZvcm1hdE51bWJlcikge1xuICAgICAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9jcmVkaXRfY2FyZF9udW1iZXInKTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgZm9ybWF0RXhwTW9udGg6IG51bWJlciA9IHBhcnNlTnVtKGV4cE1vbnRoLCAyKTtcblxuICAgICAgaWYoIWZvcm1hdEV4cE1vbnRoKSB7XG4gICAgICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2NyZWRpdF9jYXJkX2V4cF9tb250aCcpO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBmb3JtYXRFeHBZZWFyOiBudW1iZXIgPSBwYXJzZU51bShleHBZZWFyLCAyKTtcblxuICAgICAgaWYoIWZvcm1hdEV4cFllYXIpIHtcbiAgICAgICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfY3JlZGl0X2NhcmRfZXhwX3llYXInKTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgZm9ybWF0Q3ZjOiBudW1iZXIgPSBwYXJzZU51bShjdmMsIDMpO1xuXG4gICAgICAvLyBBZGRyZXNzXG4gICAgICBjb25zdCBwYXltZW50Q2FyZDogUGF5bWVudENhcmRUeXBlID0ge307XG4gICAgICBjb25zdCBmb3JtYXRDaXR5OiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoY2l0eSwgMzIpO1xuXG4gICAgICBpZihmb3JtYXRDaXR5KSB7XG4gICAgICAgIHBheW1lbnRDYXJkLmNpdHkgPSBmb3JtYXRDaXR5O1xuICAgICAgfVxuXG4gICAgICBjb25zdCBmb3JtYXRDb3VudHJ5OiBzdHJpbmcgPSBwYXJzZUNoYXIoY291bnRyeSwgMik7XG5cbiAgICAgIGlmKGZvcm1hdENvdW50cnkpIHtcbiAgICAgICAgcGF5bWVudENhcmQuY291bnRyeSA9IGZvcm1hdENvdW50cnk7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGZvcm1hdEZ1bGxOYW1lOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoZnVsbE5hbWUsIDMyKTtcblxuICAgICAgaWYoZm9ybWF0RnVsbE5hbWUpIHtcbiAgICAgICAgcGF5bWVudENhcmQuZnVsbE5hbWUgPSBmb3JtYXRGdWxsTmFtZTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgZm9ybWF0U3RyZWV0MTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKHN0cmVldDEsIDMyKTtcblxuICAgICAgaWYoZm9ybWF0U3RyZWV0MSkge1xuICAgICAgICBwYXltZW50Q2FyZC5zdHJlZXQxID0gZm9ybWF0U3RyZWV0MTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgZm9ybWF0U3RyZWV0Mjogc3RyaW5nID0gcGFyc2VWYXJDaGFyKHN0cmVldDIsIDMyKTtcblxuICAgICAgaWYoZm9ybWF0U3RyZWV0Mikge1xuICAgICAgICBwYXltZW50Q2FyZC5zdHJlZXQyID0gZm9ybWF0U3RyZWV0MjtcbiAgICAgIH1cblxuICAgICAgY29uc3QgZm9ybWF0U3RhdGU6IHN0cmluZyA9IHBhcnNlQ2hhcihzdGF0ZSwgMik7XG5cbiAgICAgIGlmKGZvcm1hdFN0YXRlKSB7XG4gICAgICAgIHBheW1lbnRDYXJkLnN0YXRlID0gZm9ybWF0U3RhdGU7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGZvcm1hdFppcDogc3RyaW5nID0gcGFyc2VWYXJDaGFyKHppcCwgMTApO1xuXG4gICAgICBpZihmb3JtYXRaaXApIHtcbiAgICAgICAgcGF5bWVudENhcmQuemlwID0gZm9ybWF0WmlwO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBnZXRTdHJpcGVDbGllbnQoKTtcblxuICAgICAgLy8gQ3JlYXRlIGEgdG9rZW4gZmlyc3QsIHRoZW4gdXNlIHRoZSB0b2tlbiBhcyB0aGUgc291cmNlXG4gICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LnRva2Vucy5jcmVhdGUoe1xuICAgICAgICBjYXJkOiB7XG4gICAgICAgICAgYWRkcmVzc19jaXR5OiBmb3JtYXRDaXR5LFxuICAgICAgICAgIGFkZHJlc3NfY291bnRyeTogZm9ybWF0Q291bnRyeSxcbiAgICAgICAgICBhZGRyZXNzX2xpbmUxOiBmb3JtYXRTdHJlZXQxLFxuICAgICAgICAgIGFkZHJlc3NfbGluZTI6IGZvcm1hdFN0cmVldDIsXG4gICAgICAgICAgYWRkcmVzc19zdGF0ZTogZm9ybWF0U3RhdGUsXG4gICAgICAgICAgYWRkcmVzc196aXA6IGZvcm1hdFppcCxcbiAgICAgICAgICBjdmM6IGZvcm1hdEN2Yy50b1N0cmluZygpLFxuICAgICAgICAgIGV4cF9tb250aDogZm9ybWF0RXhwTW9udGgudG9TdHJpbmcoKSxcbiAgICAgICAgICBleHBfeWVhcjogZm9ybWF0RXhwWWVhci50b1N0cmluZygpLFxuICAgICAgICAgIG5hbWU6IGZ1bGxOYW1lLFxuICAgICAgICAgIG51bWJlcjogZm9ybWF0TnVtYmVyLnRvU3RyaW5nKClcbiAgICAgICAgfVxuICAgICAgfSlcbiAgICAgICAgLnRoZW4oKHRva2VuKSA9PiBzdHJpcGVDbGllbnQuY3VzdG9tZXJzLmNyZWF0ZVNvdXJjZShcbiAgICAgICAgICBzdHJpcGVBY2NvdW50SWQsXG4gICAgICAgICAge3NvdXJjZTogdG9rZW4uaWR9XG4gICAgICAgICkpXG4gICAgICAgIC50aGVuKChuZXdTb3VyY2UpID0+IHtcbiAgICAgICAgICAvLyBVc2UgdHlwZSBhc3NlcnRpb24gZm9yIGNhcmQgcHJvcGVydGllc1xuICAgICAgICAgIGNvbnN0IGNhcmRTb3VyY2UgPSBuZXdTb3VyY2UgYXMgdW5rbm93biBhcyBDYXJkO1xuICAgICAgICAgIGNvbnN0IGJyYW5kID0gY2FyZFNvdXJjZS5icmFuZCB8fCAnJztcbiAgICAgICAgICBjb25zdCBjdmNDaGVjayA9IGNhcmRTb3VyY2UuY3ZjX2NoZWNrIHx8ICcnO1xuICAgICAgICAgIGNvbnN0IGxhc3Q0ID0gY2FyZFNvdXJjZS5sYXN0NCB8fCAnJztcblxuICAgICAgICAgIC8vIENyZWF0ZSBzZXNzaW9uXG4gICAgICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICAgICAgICAgIGNvbnN0IGluc2VydCA9IHtcbiAgICAgICAgICAgIC4uLnBheW1lbnRDYXJkLFxuICAgICAgICAgICAgX2tleTogY3JlYXRlSGFzaChgdXNlci1wYXltZW50LSR7c2Vzc2lvbklkfWApLFxuICAgICAgICAgICAgYWNjb3VudE51bWJlcjogbGFzdDQsXG4gICAgICAgICAgICBhZGRlZDogbm93LFxuICAgICAgICAgICAgYnJhbmQsXG4gICAgICAgICAgICBjdmNDaGVjayxcbiAgICAgICAgICAgIGV4cE1vbnRoLFxuICAgICAgICAgICAgZXhwWWVhcixcbiAgICAgICAgICAgIG1vZGlmaWVkOiBub3csXG4gICAgICAgICAgICB1c2VySWQ6IHNlc3Npb25JZFxuICAgICAgICAgIH07XG4gICAgICAgICAgY29uc3QgaW5zZXJ0QXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBJTlNFUlQgJHtpbnNlcnR9IElOIGNyZWRpdENhcmRzIFJFVFVSTiBORVdgO1xuXG4gICAgICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlTmFtZSkucXVlcnkoaW5zZXJ0QXFsUXJ5KVxuICAgICAgICAgICAgLnRoZW4oKGN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgICAgIC50aGVuKChuZXdDYXJkOiBQYXltZW50Q2FyZFR5cGUpID0+IHtcbiAgICAgICAgICAgICAgaWYobmV3Q2FyZCkge1xuICAgICAgICAgICAgICAgIC8vIEFkZCBsaW5rZWQgZWRnZVxuICAgICAgICAgICAgICAgIGNvbnN0IHtfaWQ6IGNhcmRJZCwgX2tleTogY2FyZEtleX0gPSBjYXJkO1xuICAgICAgICAgICAgICAgIGNvbnN0IGVkZ2VDb2xsZWN0aW9uOiBFZGdlQ29sbGVjdGlvbiA9IHVzZURiKGRhdGFiYXNlTmFtZSkuY29sbGVjdGlvbignaGFzUGF5bWVudCcpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGVkZ2VJZCA9IGNyZWF0ZUhhc2goYHBheW1lbnQtJHtjYXJkS2V5fWApO1xuICAgICAgICAgICAgICAgIGNvbnN0IGVkZ2UgPSB7XG4gICAgICAgICAgICAgICAgICBfZnJvbTogYHVzZXJzLyR7c2Vzc2lvbklkfWAsXG4gICAgICAgICAgICAgICAgICBfa2V5OiBlZGdlSWQsXG4gICAgICAgICAgICAgICAgICBfdG86IGNhcmRJZFxuICAgICAgICAgICAgICAgIH07XG5cbiAgICAgICAgICAgICAgICByZXR1cm4gZWRnZUNvbGxlY3Rpb24uc2F2ZShlZGdlLCB7cmV0dXJuTmV3OiB0cnVlfSkudGhlbigoKSA9PiBjYXJkKTtcbiAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAgIHJldHVybiBuZXdDYXJkO1xuICAgICAgICAgICAgfSlcbiAgICAgICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICAgICAgICAgIGFjdGlvbixcbiAgICAgICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgICAgIGxhYmVsOiAncGF5bWVudF9lcnJvcidcbiAgICAgICAgICAgIH0sIGVycm9yLCBjb250ZXh0KS50aGVuKCgpID0+IG51bGwpKTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgdXBkYXRlQ3JlZGl0Q2FyZCA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBjYXJkOiBQYXltZW50Q2FyZFR5cGUpOiBQcm9taXNlPFBheW1lbnRDYXJkVHlwZT4gPT4ge1xuICBjb25zdCB7ZGF0YWJhc2VOYW1lLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG5cbiAgY29uc3Qge1xuICAgIGNpdHksXG4gICAgY291bnRyeSxcbiAgICBleHBNb250aCxcbiAgICBleHBZZWFyLFxuICAgIGZ1bGxOYW1lLFxuICAgIGlkLFxuICAgIHN0cmVldDEsXG4gICAgc3RhdGUsXG4gICAgemlwXG4gIH06IFBheW1lbnRDYXJkVHlwZSA9IGNhcmQ7XG5cbiAgY29uc3QgZm9ybWF0SWQ6IHN0cmluZyA9IHBhcnNlSWQoaWQpO1xuXG4gIGlmKGZvcm1hdElkKSB7XG4gICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfY3JlZGl0X2NhcmRfaWQnKTtcbiAgfVxuXG4gIGNvbnN0IHBheW1lbnRDYXJkOiBQYXltZW50Q2FyZFR5cGUgPSB7fTtcbiAgY29uc3QgZm9ybWF0RXhwTW9udGg6IG51bWJlciA9IHBhcnNlTnVtKGV4cE1vbnRoLCAyKTtcbiAgY29uc3QgZm9ybWF0RXhwWWVhcjogbnVtYmVyID0gcGFyc2VOdW0oZXhwWWVhciwgMik7XG4gIGNvbnN0IGZvcm1hdENpdHk6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihjaXR5LCAzMik7XG4gIGNvbnN0IGZvcm1hdENvdW50cnk6IHN0cmluZyA9IHBhcnNlQ2hhcihjb3VudHJ5LCAyKTtcbiAgY29uc3QgZm9ybWF0RnVsbE5hbWU6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihmdWxsTmFtZSwgMzIpO1xuICBjb25zdCBmb3JtYXRTdHJlZXQxOiBzdHJpbmcgPSBwYXJzZVN0cmluZyhzdHJlZXQxLCAzMik7XG4gIGNvbnN0IGZvcm1hdFN0YXRlOiBzdHJpbmcgPSBwYXJzZUNoYXIoc3RhdGUsIDIpO1xuICBjb25zdCBmb3JtYXRaaXA6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcih6aXAsIDEwKTtcblxuICBpZihmb3JtYXRFeHBNb250aCkge1xuICAgIHBheW1lbnRDYXJkLmV4cE1vbnRoID0gZm9ybWF0RXhwTW9udGg7XG4gIH1cblxuICBpZihmb3JtYXRFeHBZZWFyKSB7XG4gICAgcGF5bWVudENhcmQuZXhwWWVhciA9IGZvcm1hdEV4cFllYXI7XG4gIH1cblxuICBpZihmb3JtYXRDaXR5KSB7XG4gICAgcGF5bWVudENhcmQuY2l0eSA9IGZvcm1hdENpdHk7XG4gIH1cblxuICBpZihmb3JtYXRDb3VudHJ5KSB7XG4gICAgcGF5bWVudENhcmQuY291bnRyeSA9IGZvcm1hdENvdW50cnk7XG4gIH1cblxuICBpZihmb3JtYXRGdWxsTmFtZSkge1xuICAgIHBheW1lbnRDYXJkLmZ1bGxOYW1lID0gZm9ybWF0RnVsbE5hbWU7XG4gIH1cblxuICBpZihmb3JtYXRTdHJlZXQxKSB7XG4gICAgcGF5bWVudENhcmQuc3RyZWV0MSA9IGZvcm1hdFN0cmVldDE7XG4gIH1cblxuICBpZihmb3JtYXRTdGF0ZSkge1xuICAgIHBheW1lbnRDYXJkLnN0YXRlID0gZm9ybWF0U3RhdGU7XG4gIH1cblxuICBpZihmb3JtYXRaaXApIHtcbiAgICBwYXltZW50Q2FyZC56aXAgPSBmb3JtYXRaaXA7XG4gIH1cblxuICBjb25zdCB1cGRhdGUgPSBwYXltZW50Q2FyZDtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBcbiAgICAgIExFVCB1cGRhdGVkQ2FyZCA9IEZJUlNUKFxuICAgICAgICBGT1IgYyBJTiBjcmVkaXRDYXJkc1xuICAgICAgICBGSUxURVIgYy5fa2V5ID09ICR7Zm9ybWF0SWR9ICYmIGMudXNlcklkID09ICR7c2Vzc2lvbklkfVxuICAgICAgICBVUERBVEUgYyBXSVRIICR7dXBkYXRlfSBJTiBjcmVkaXRDYXJkc1xuICAgICAgICBMSU1JVCAxXG4gICAgICAgIFJFVFVSTiBORVdcbiAgICAgIClcbiAgICAgIExFVCB1c2VyID0gRklSU1QoXG4gICAgICAgIEZPUiB1IElOIHVzZXJzXG4gICAgICAgIEZJTFRFUiB1Ll9rZXkgPT0gJHtzZXNzaW9uSWR9XG4gICAgICAgIExJTUlUIDFcbiAgICAgICAgUkVUVVJOIHVcbiAgICAgIClcbiAgICAgIFJFVFVSTiB7dXNlcjogdXNlciwgY2FyZDogdXBkYXRlZENhcmR9YDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2VOYW1lKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigocmVzdWx0cyA9IHtjYXJkOiB7fSwgdXNlcjoge319KSA9PiB7XG4gICAgICBjb25zdCB1cGRhdGVkQ2FyZDogUGF5bWVudENhcmRUeXBlID0gcmVzdWx0cy5jYXJkO1xuICAgICAgY29uc3Qge3VzZXJ9ID0gcmVzdWx0cztcblxuICAgICAgaWYoIXVwZGF0ZWRDYXJkKSB7XG4gICAgICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ25vdF9mb3VuZCcpO1xuICAgICAgfVxuXG4gICAgICBjb25zdCB7c3RyaXBlQ3VzdG9tZXJJZH0gPSB1c2VyO1xuICAgICAgY29uc3Qge3N0cmlwZUlkfSA9IGNhcmQ7XG4gICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBnZXRTdHJpcGVDbGllbnQoKTtcbiAgICAgIGNvbnN0IHVwZGF0ZTogc3RyaXBlLkN1c3RvbWVyVXBkYXRlU291cmNlUGFyYW1zID0ge1xuICAgICAgICBhZGRyZXNzX2NpdHk6IGZvcm1hdENpdHksXG4gICAgICAgIGFkZHJlc3NfY291bnRyeTogZm9ybWF0Q291bnRyeSxcbiAgICAgICAgYWRkcmVzc19saW5lMTogZm9ybWF0U3RyZWV0MSxcbiAgICAgICAgYWRkcmVzc19zdGF0ZTogZm9ybWF0U3RhdGUsXG4gICAgICAgIGFkZHJlc3NfemlwOiBmb3JtYXRaaXAsXG4gICAgICAgIGV4cF9tb250aDogZm9ybWF0RXhwTW9udGgudG9TdHJpbmcoKSxcbiAgICAgICAgZXhwX3llYXI6IGZvcm1hdEV4cFllYXIudG9TdHJpbmcoKSxcbiAgICAgICAgbmFtZTogZm9ybWF0RnVsbE5hbWVcbiAgICAgIH07XG5cbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQuY3VzdG9tZXJzXG4gICAgICAgIC51cGRhdGVTb3VyY2Uoc3RyaXBlQ3VzdG9tZXJJZCwgc3RyaXBlSWQsIHVwZGF0ZSlcbiAgICAgICAgLnRoZW4oKCkgPT4gY2FyZClcbiAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgICAgICBjb25zb2xlLmxvZygncGF5bWVudHM6OnVwZGF0ZUNhcmQ6OmVycm9yJywgZXJyb3IpO1xuICAgICAgICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3BheW1lbnRfZXJyb3InKTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZ2V0Q3JlZGl0Q2FyZHMgPSAoY29udGV4dDogQXBpQ29udGV4dCk6IFByb21pc2U8UGF5bWVudENhcmRUeXBlW10+ID0+IHtcbiAgY29uc3QgYWN0aW9uID0gJ2dldENyZWRpdENhcmRzJztcbiAgY29uc3Qge2RhdGFiYXNlTmFtZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYEZPUiBjIElOIGNyZWRpdENhcmRzXG4gICAgRklMVEVSIGMudXNlcklkID09ICR7c2Vzc2lvbklkfVxuICAgIFJFVFVSTiBjYDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2VOYW1lKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcikgPT4gY3Vyc29yLmFsbCgpKVxuICAgIC50aGVuKChsaXN0OiBQYXltZW50Q2FyZFR5cGVbXSA9IFtdKSA9PiBsaXN0KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiBsb2dFcnJvcih7XG4gICAgICBhY3Rpb24sXG4gICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgIGxhYmVsOiBFcnJvclR5cGVzLkRBVEFCQVNFX0VSUk9SXG4gICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xufTtcblxuZXhwb3J0IGNvbnN0IGRlbGV0ZUNyZWRpdENhcmQgPSAoY29udGV4dDogQXBpQ29udGV4dCwgY2FyZElkOiBzdHJpbmcpOiBQcm9taXNlPGJvb2xlYW4+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlTmFtZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCBmb3JtYXRDYXJkSWQ6IHN0cmluZyA9IHBhcnNlSWQoY2FyZElkKTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBcbiAgICBMRVQgY2FyZCA9IEZJUlNUKFxuICAgICAgRk9SIGMgSU4gY3JlZGl0Q2FyZHNcbiAgICAgIEZJTFRFUiBjLl9rZXkgPT0gJHtmb3JtYXRDYXJkSWR9ICYmIGMudXNlcklkID09ICR7c2Vzc2lvbklkfVxuICAgICAgTElNSVQgMVxuICAgICAgUkVNT1ZFIGMgSU4gY3JlZGl0Q2FyZHNcbiAgICAgIFJFVFVSTiBPTERcbiAgICApXG4gICAgTEVUIHVzZXIgPSBGSVJTVChcbiAgICAgIEZPUiB1IElOIHVzZXJzXG4gICAgICBGSUxURVIgdS5fa2V5ID09ICR7c2Vzc2lvbklkfVxuICAgICAgTElNSVQgMVxuICAgICAgUkVUVVJOIHVcbiAgICApXG4gICAgUkVUVVJOIHt1c2VyOiB1c2VyLCBjYXJkOiBjYXJkfWA7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlTmFtZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKHJlc3VsdCA9IHtjYXJkOiB7fSwgdXNlcjoge319KSA9PiB7XG4gICAgICBpZighcmVzdWx0KSB7XG4gICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgIH1cblxuICAgICAgY29uc3Qge2NhcmQsIHVzZXJ9ID0gcmVzdWx0O1xuICAgICAgY29uc3Qge19rZXk6IGNhcmRLZXl9ID0gY2FyZDtcblxuICAgICAgLy8gUmVtb3ZlIGxpbmtlZCBlZGdlc1xuICAgICAgY29uc3QgZWRnZUNvbGxlY3Rpb24gPSB1c2VEYihkYXRhYmFzZU5hbWUpLmNvbGxlY3Rpb24oJ2hhc1BheW1lbnQnKTtcblxuICAgICAgcmV0dXJuIGVkZ2VDb2xsZWN0aW9uLm91dEVkZ2VzKGNhcmRLZXksIHt9KVxuICAgICAgICAudGhlbihhc3luYyAocmVzcG9uc2UpID0+IHtcbiAgICAgICAgICAvLyBFeHRyYWN0IGVkZ2VzIGZyb20gdGhlIHJlc3BvbnNlXG4gICAgICAgICAgY29uc3QgZWRnZXMgPSBBcnJheS5pc0FycmF5KHJlc3BvbnNlKSA/IHJlc3BvbnNlIDogW107XG5cbiAgICAgICAgICBpZihlZGdlcy5sZW5ndGgpIHtcbiAgICAgICAgICAgIGF3YWl0IFByb21pc2UuYWxsKFxuICAgICAgICAgICAgICBlZGdlcy5tYXAoKGVkZ2UpID0+IHtcbiAgICAgICAgICAgICAgICBjb25zdCB7X2tleTogZWRnZUtleX0gPSBlZGdlO1xuICAgICAgICAgICAgICAgIGNvbnN0IHJlbW92ZUFxbFFyeTogQXFsUXVlcnkgPSBhcWxgUkVNT1ZFIHtfa2V5OiR7ZWRnZUtleX19IElOIGhhc1BheW1lbnRgO1xuICAgICAgICAgICAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZU5hbWUpLnF1ZXJ5KHJlbW92ZUFxbFFyeSk7XG4gICAgICAgICAgICAgIH0pKVxuICAgICAgICAgICAgICAudGhlbigoKSA9PiB7XG4gICAgICAgICAgICAgICAgLy8gU3RyaXBlXG4gICAgICAgICAgICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gZ2V0U3RyaXBlQ2xpZW50KCk7XG5cbiAgICAgICAgICAgICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LmN1c3RvbWVyc1xuICAgICAgICAgICAgICAgICAgLmRlbGV0ZVNvdXJjZSh1c2VyLnN0cmlwZUN1c3RvbWVySWQsIGNhcmQuc3RyaXBlSWQpXG4gICAgICAgICAgICAgICAgICAudGhlbigoKSA9PiB0cnVlKVxuICAgICAgICAgICAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgICAgICAgICAgICAgICAgY29uc29sZS5sb2coJ3BheW1lbnRzOjpkZWxldGVDYXJkOjplcnJvcicsIGVycm9yKTtcbiAgICAgICAgICAgICAgICAgICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncGF5bWVudF9lcnJvcicpO1xuICAgICAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgICByZXR1cm4gdHJ1ZTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICByZXR1cm4gZmFsc2U7XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGRlbGV0ZUJhbmtBY2NvdW50ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGJhbmtJZDogc3RyaW5nKTogUHJvbWlzZTxib29sZWFuPiA9PiB7XG4gIGNvbnN0IHtkYXRhYmFzZU5hbWUsIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH19ID0gY29udGV4dDtcblxuICAvLyBDbGVhbiBkYlxuICBjb25zdCB1cGRhdGU6IFVzZXJUeXBlID0ge1xuICAgIGJhbmtBY2NvdW50OiAnJyxcbiAgICBiYW5rRnVsbE5hbWU6ICcnLFxuICAgIGJhbmtJZDogJycsXG4gICAgYmFua1JvdXRpbmc6ICcnLFxuICAgIG1vZGlmaWVkOiBEYXRlLm5vdygpXG4gIH07XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgVVBEQVRFICR7c2Vzc2lvbklkfSBXSVRIICR7dXBkYXRlfSBJTiB1c2VycyBMSU1JVCAxIFJFVFVSTiBORVdgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZU5hbWUpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKCh1c2VyOiBVc2VyVHlwZSkgPT4ge1xuICAgICAgY29uc3Qge3N0cmlwZUFjY291bnRJZH0gPSB1c2VyO1xuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gZ2V0U3RyaXBlQ2xpZW50KCk7XG5cbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQuY3VzdG9tZXJzXG4gICAgICAgIC5kZWxldGVTb3VyY2Uoc3RyaXBlQWNjb3VudElkLCBiYW5rSWQpXG4gICAgICAgIC50aGVuKCgpID0+IHRydWUpXG4gICAgICAgIC5jYXRjaCgoKSA9PiBQcm9taXNlLnJlc29sdmUoZmFsc2UpKTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBjcmVhdGVQYXltZW50VHJhbnNmZXIgPSAoY29udGV4dDogQXBpQ29udGV4dCwgdHJhbnNmZXI6IFBheW1lbnRUcmFuc2Zlcik6IFByb21pc2U8UGF5bWVudFRyYW5zZmVyPiA9PiB7XG4gIGNvbnN0IHtkYXRhYmFzZU5hbWUsIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH19ID0gY29udGV4dDtcbiAgY29uc3Qge2Ftb3VudCwgY3VycmVuY3l9ID0gdHJhbnNmZXI7XG4gIGNvbnN0IGZvcm1hdEFtb3VudDogbnVtYmVyID0gcGFyc2VOdW0oYW1vdW50KTtcbiAgY29uc3QgZm9ybWF0Q3VycmVuY3k6IHN0cmluZyA9IHBhcnNlQ2hhcihjdXJyZW5jeSwgMywgJ1VTRCcpLnRvVXBwZXJDYXNlKCk7XG5cbiAgcmV0dXJuIGdldFVzZXIoY29udGV4dCwge3VzZXJJZDogc2Vzc2lvbklkfSlcbiAgICAudGhlbigodXNlcjogVXNlclR5cGUpID0+IHtcbiAgICAgIGNvbnN0IHtzdHJpcGVBY2NvdW50SWR9ID0gdXNlcjtcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IGdldFN0cmlwZUNsaWVudCgpO1xuXG4gICAgICByZXR1cm4gc3RyaXBlQ2xpZW50LnRyYW5zZmVyc1xuICAgICAgICAuY3JlYXRlKHtcbiAgICAgICAgICBhbW91bnQ6IGZvcm1hdEFtb3VudCxcbiAgICAgICAgICBjdXJyZW5jeTogZm9ybWF0Q3VycmVuY3ksXG4gICAgICAgICAgZGVzdGluYXRpb246IHN0cmlwZUFjY291bnRJZFxuICAgICAgICB9KVxuICAgICAgICAudGhlbigoc3RyaXBlVHJhbnNmZXIpID0+IHtcbiAgICAgICAgICBjb25zb2xlLmxvZyhzdHJpcGVUcmFuc2Zlcik7XG4gICAgICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICAgICAgICAgIGNvbnN0IGluc2VydDogUGF5bWVudFRyYW5zZmVyID0ge1xuICAgICAgICAgICAgYWRkZWQ6IG5vdyxcbiAgICAgICAgICAgIGFtb3VudDogZm9ybWF0QW1vdW50LFxuICAgICAgICAgICAgY3VycmVuY3k6IGZvcm1hdEN1cnJlbmN5LFxuICAgICAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICAgICAgfTtcbiAgICAgICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYElOU0VSVCAke2luc2VydH0gSU4gdHJhbnNmZXJzIFJFVFVSTiBORVdgO1xuXG4gICAgICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlTmFtZSkucXVlcnkoYXFsUXJ5KVxuICAgICAgICAgICAgLnRoZW4oKGN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgICAgIC50aGVuKChuZXdUcmFuc2ZlcjogUGF5bWVudFRyYW5zZmVyKSA9PiBuZXdUcmFuc2Zlcik7XG4gICAgICAgIH0pO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGNyZWF0ZVBheW1lbnRIb2xkID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIHBheW1lbnQ6IFBheW1lbnRDaGFyZ2UpOiBQcm9taXNlPFBheW1lbnRDaGFyZ2U+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlTmFtZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCB7YW1vdW50LCBjYXB0dXJlLCBjYXJkSWQsIGN1cnJlbmN5LCBkZXNjcmlwdGlvbn0gPSBwYXltZW50O1xuICBjb25zdCBmb3JtYXRDdXJyZW5jeSA9IHBhcnNlQ2hhcihjdXJyZW5jeSwgMywgJ1VTRCcpLnRvVXBwZXJDYXNlKCk7XG4gIGNvbnN0IHN0cmlwZUNsaWVudCA9IGdldFN0cmlwZUNsaWVudCgpO1xuXG4gIHJldHVybiBzdHJpcGVDbGllbnQuY2hhcmdlc1xuICAgIC5jcmVhdGUoe1xuICAgICAgYW1vdW50LFxuICAgICAgY2FwdHVyZSxcbiAgICAgIGN1cnJlbmN5OiBmb3JtYXRDdXJyZW5jeSxcbiAgICAgIGRlc2NyaXB0aW9uLFxuICAgICAgc291cmNlOiBjYXJkSWRcbiAgICB9KVxuICAgIC50aGVuKChzdHJpcGVDaGFyZ2UpID0+IHtcbiAgICAgIGNvbnN0IG5vdzogbnVtYmVyID0gRGF0ZS5ub3coKTtcbiAgICAgIGNvbnN0IGluc2VydDogUGF5bWVudENoYXJnZSA9IHtcbiAgICAgICAgYWRkZWQ6IG5vdyxcbiAgICAgICAgYW1vdW50LFxuICAgICAgICBjYXB0dXJlLFxuICAgICAgICBjYXJkSWQsXG4gICAgICAgIGNoYXJnZUZhaWxDb2RlOiBzdHJpcGVDaGFyZ2UuZmFpbHVyZV9jb2RlLFxuICAgICAgICBjaGFyZ2VGYWlsTXNnOiBzdHJpcGVDaGFyZ2UuZmFpbHVyZV9tZXNzYWdlLFxuICAgICAgICBjaGFyZ2VJZDogc3RyaXBlQ2hhcmdlLmlkLFxuICAgICAgICBjaGFyZ2VTdGF0dXM6IHN0cmlwZUNoYXJnZS5zdGF0dXMsXG4gICAgICAgIGN1cnJlbmN5OiBmb3JtYXRDdXJyZW5jeSxcbiAgICAgICAgZGVzY3JpcHRpb24sXG4gICAgICAgIG1vZGlmaWVkOiBub3csXG4gICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICB9O1xuICAgICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBJTlNFUlQgJHtpbnNlcnR9IElOIHBheW1lbnRzIFJFVFVSTiBORVdgO1xuXG4gICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2VOYW1lKS5xdWVyeShhcWxRcnkpXG4gICAgICAgIC50aGVuKChjdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAgIC50aGVuKChuZXdQYXltZW50OiBQYXltZW50Q2hhcmdlKSA9PiBuZXdQYXltZW50KTtcbiAgICB9KVxuICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICBjb25zb2xlLmxvZygncGF5bWVudHM6OmNyZWF0ZUhvbGQ6OmVycm9yJywgZXJyb3IpO1xuICAgICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncGF5bWVudF9lcnJvcicpO1xuICAgIH0pO1xufTtcblxuaW50ZXJmYWNlIENhcmQge1xuICBpZDogc3RyaW5nO1xuICBicmFuZD86IHN0cmluZztcbiAgY3ZjX2NoZWNrPzogc3RyaW5nO1xuICBsYXN0ND86IHN0cmluZztcbn1cbiJdLAogICJtYXBwaW5ncyI6ICJBQUlBLE9BQVEsWUFBQUEsTUFBZSwrQkFDdkIsT0FBUSxjQUFBQyxFQUFZLGFBQUFDLEVBQVcsV0FBQUMsRUFBUyxlQUFBQyxFQUFhLGdCQUFBQyxNQUFtQiwrQkFDeEUsT0FBUSxPQUFBQyxNQUFVLFdBS2xCLE9BQVEsY0FBQUMsTUFBaUIsMEJBR3pCLE9BQVEsWUFBQUMsRUFBVSxhQUFBQyxNQUFnQiw2QkFDbEMsT0FBUSxTQUFBQyxNQUFZLDRCQUNwQixPQUFRLG1CQUFBQyxNQUFzQiwwQkFDOUIsT0FBUSxXQUFBQyxNQUFjLGFBSXRCLE1BQU1DLEVBQWdCLFdBRVRDLEdBQXNCQyxHQUEwQyxDQUMzRSxNQUFNQyxFQUFTLHFCQUNULENBQUMsYUFBQUMsRUFBYyxRQUFTLENBQUMsT0FBUUMsRUFBVyxTQUFBQyxDQUFRLEVBQUksQ0FBQyxDQUFDLEVBQUlKLEVBSXBFLE9BRnFCSixFQUFnQixFQUVqQixVQUNqQixPQUFPLENBQ04sU0FBVSxDQUNSLE9BQVFPLEVBQ1IsU0FBQUMsQ0FDRixDQUNGLENBQVEsRUFDUCxLQUFNQyxHQUFhLENBR2xCLE1BQU1DLEVBQW1CLENBQ3ZCLFNBRmtCLEtBQUssSUFBSSxFQUczQixpQkFBa0JELEVBQVMsRUFDN0IsRUFFTUUsRUFBbUJoQixXQUFhWSxDQUFTLFNBQVNHLENBQU0sK0JBRTlELE9B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QVVBLEVBQWEsR0FDdkIsYUFBY0EsRUFBYSxPQUMzQixTQUFVUCxFQUNWLFlBQUFNLEVBQ0EsU0FBVWpFLEVBQ1YsT0FBUXJCLENBQ1YsRUFDTUksRUFBbUJoQixXQUFhNkQsQ0FBTSwwQkFFNUMsT0FBT3pELEVBQU1PLENBQVksRUFBRSxNQUFNSyxDQUFNLEVBQ3BDLEtBQU1DLEdBQVdBLEVBQU8sS0FBSyxDQUFDLEVBQzlCLEtBQU1tRixHQUE4QkEsQ0FBVSxDQUNuRCxDQUFDLEVBQ0EsTUFBT2pGLEdBQWlCLENBRXZCLE1BQU0sSUFBSWhCLEVBQVUsZUFBZSxDQUNyQyxDQUFDLENBQ0wiLAogICJuYW1lcyI6IFsicGFyc2VOdW0iLCAiY3JlYXRlSGFzaCIsICJwYXJzZUNoYXIiLCAicGFyc2VJZCIsICJwYXJzZVN0cmluZyIsICJwYXJzZVZhckNoYXIiLCAiYXFsIiwgIkVycm9yVHlwZXMiLCAibG9nRXJyb3IiLCAiVXNlckVycm9yIiwgInVzZURiIiwgImdldFN0cmlwZUNsaWVudCIsICJnZXRVc2VyIiwgImV2ZW50Q2F0ZWdvcnkiLCAiYWRkQ3VzdG9tZXJBY2NvdW50IiwgImNvbnRleHQiLCAiYWN0aW9uIiwgImRhdGFiYXNlTmFtZSIsICJzZXNzaW9uSWQiLCAidXNlcm5hbWUiLCAiY3VzdG9tZXIiLCAidXBkYXRlIiwgImFxbFFyeSIsICJjdXJzb3IiLCAidXBkYXRlZFVzZXIiLCAiZXJyb3IiLCAiYWRkQmFua0FjY291bnQiLCAiYmFua0FjY291bnQiLCAiYWNjb3VudE51bWJlciIsICJmdWxsTmFtZSIsICJyb3V0aW5nIiwgImZvcm1hdEFjY291bnQiLCAiZm9ybWF0RnVsbE5hbWUiLCAiZm9ybWF0Um91dGluZyIsICJ1c2VyIiwgInN0cmlwZUFjY291bnRJZCIsICJzdHJpcGVDbGllbnQiLCAidG9rZW4iLCAiYWNjb3VudCIsICJub3ciLCAiYWRkQ3JlZGl0Q2FyZCIsICJjYXJkIiwgImNpdHkiLCAiY291bnRyeSIsICJjdmMiLCAiZXhwTW9udGgiLCAiZXhwWWVhciIsICJzdHJlZXQxIiwgInN0cmVldDIiLCAic3RhdGUiLCAiemlwIiwgImZvcm1hdE51bWJlciIsICJmb3JtYXRFeHBNb250aCIsICJmb3JtYXRFeHBZZWFyIiwgImZvcm1hdEN2YyIsICJwYXltZW50Q2FyZCIsICJmb3JtYXRDaXR5IiwgImZvcm1hdENvdW50cnkiLCAiZm9ybWF0U3RyZWV0MSIsICJmb3JtYXRTdHJlZXQyIiwgImZvcm1hdFN0YXRlIiwgImZvcm1hdFppcCIsICJuZXdTb3VyY2UiLCAiY2FyZFNvdXJjZSIsICJicmFuZCIsICJjdmNDaGVjayIsICJsYXN0NCIsICJpbnNlcnQiLCAiaW5zZXJ0QXFsUXJ5IiwgIm5ld0NhcmQiLCAiY2FyZElkIiwgImNhcmRLZXkiLCAiZWRnZUNvbGxlY3Rpb24iLCAiZWRnZUlkIiwgImVkZ2UiLCAidXBkYXRlQ3JlZGl0Q2FyZCIsICJpZCIsICJmb3JtYXRJZCIsICJyZXN1bHRzIiwgInVwZGF0ZWRDYXJkIiwgInN0cmlwZUN1c3RvbWVySWQiLCAic3RyaXBlSWQiLCAiZ2V0Q3JlZGl0Q2FyZHMiLCAibGlzdCIsICJkZWxldGVDcmVkaXRDYXJkIiwgImZvcm1hdENhcmRJZCIsICJyZXN1bHQiLCAicmVzcG9uc2UiLCAiZWRnZXMiLCAiZWRnZUtleSIsICJyZW1vdmVBcWxRcnkiLCAiZGVsZXRlQmFua0FjY291bnQiLCAiYmFua0lkIiwgImNyZWF0ZVBheW1lbnRUcmFuc2ZlciIsICJ0cmFuc2ZlciIsICJhbW91bnQiLCAiY3VycmVuY3kiLCAiZm9ybWF0QW1vdW50IiwgImZvcm1hdEN1cnJlbmN5IiwgInN0cmlwZVRyYW5zZmVyIiwgIm5ld1RyYW5zZmVyIiwgImNyZWF0ZVBheW1lbnRIb2xkIiwgInBheW1lbnQiLCAiY2FwdHVyZSIsICJkZXNjcmlwdGlvbiIsICJzdHJpcGVDaGFyZ2UiLCAibmV3UGF5bWVudCJdCn0K
|
|
334
|
+
RETURN {user: user, card: card}`;
|
|
335
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((result = {
|
|
336
|
+
card: {},
|
|
337
|
+
user: {}
|
|
338
|
+
})=>{
|
|
339
|
+
if (!result) {
|
|
340
|
+
return false;
|
|
341
|
+
}
|
|
342
|
+
const { card, user } = result;
|
|
343
|
+
const { _key: cardKey } = card;
|
|
344
|
+
// Remove linked edges
|
|
345
|
+
const edgeCollection = useDb(databaseName).collection('hasPayment');
|
|
346
|
+
return edgeCollection.outEdges(cardKey, {}).then(async (response)=>{
|
|
347
|
+
// Extract edges from the response
|
|
348
|
+
const edges = Array.isArray(response) ? response : [];
|
|
349
|
+
if (edges.length) {
|
|
350
|
+
await Promise.all(edges.map((edge)=>{
|
|
351
|
+
const { _key: edgeKey } = edge;
|
|
352
|
+
const removeAqlQry = aql`REMOVE {_key:${edgeKey}} IN hasPayment`;
|
|
353
|
+
return useDb(databaseName).query(removeAqlQry);
|
|
354
|
+
})).then(()=>{
|
|
355
|
+
// Stripe
|
|
356
|
+
const stripeClient = getStripeClient();
|
|
357
|
+
return stripeClient.customers.deleteSource(user.stripeCustomerId, card.stripeId).then(()=>true).catch((error)=>{
|
|
358
|
+
console.log('payments::deleteCard::error', error);
|
|
359
|
+
throw new UserError('payment_error');
|
|
360
|
+
});
|
|
361
|
+
});
|
|
362
|
+
return true;
|
|
363
|
+
}
|
|
364
|
+
return false;
|
|
365
|
+
});
|
|
366
|
+
});
|
|
367
|
+
};
|
|
368
|
+
export const deleteBankAccount = (context, bankId)=>{
|
|
369
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
370
|
+
// Clean db
|
|
371
|
+
const update = {
|
|
372
|
+
bankAccount: '',
|
|
373
|
+
bankFullName: '',
|
|
374
|
+
bankId: '',
|
|
375
|
+
bankRouting: '',
|
|
376
|
+
modified: Date.now()
|
|
377
|
+
};
|
|
378
|
+
const aqlQry = aql`UPDATE ${sessionId} WITH ${update} IN users LIMIT 1 RETURN NEW`;
|
|
379
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((user)=>{
|
|
380
|
+
const { stripeAccountId } = user;
|
|
381
|
+
const stripeClient = getStripeClient();
|
|
382
|
+
return stripeClient.customers.deleteSource(stripeAccountId, bankId).then(()=>true).catch(()=>Promise.resolve(false));
|
|
383
|
+
});
|
|
384
|
+
};
|
|
385
|
+
export const createPaymentTransfer = (context, transfer)=>{
|
|
386
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
387
|
+
const { amount, currency } = transfer;
|
|
388
|
+
const formatAmount = parseNum(amount);
|
|
389
|
+
const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();
|
|
390
|
+
return getUser(context, {
|
|
391
|
+
userId: sessionId
|
|
392
|
+
}).then((user)=>{
|
|
393
|
+
const { stripeAccountId } = user;
|
|
394
|
+
const stripeClient = getStripeClient();
|
|
395
|
+
return stripeClient.transfers.create({
|
|
396
|
+
amount: formatAmount,
|
|
397
|
+
currency: formatCurrency,
|
|
398
|
+
destination: stripeAccountId
|
|
399
|
+
}).then((stripeTransfer)=>{
|
|
400
|
+
console.log(stripeTransfer);
|
|
401
|
+
const now = Date.now();
|
|
402
|
+
const insert = {
|
|
403
|
+
added: now,
|
|
404
|
+
amount: formatAmount,
|
|
405
|
+
currency: formatCurrency,
|
|
406
|
+
modified: now,
|
|
407
|
+
userId: sessionId
|
|
408
|
+
};
|
|
409
|
+
const aqlQry = aql`INSERT ${insert} IN transfers RETURN NEW`;
|
|
410
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((newTransfer)=>newTransfer);
|
|
411
|
+
});
|
|
412
|
+
});
|
|
413
|
+
};
|
|
414
|
+
export const createPaymentHold = (context, payment)=>{
|
|
415
|
+
const { databaseName, session: { userId: sessionId } } = context;
|
|
416
|
+
const { amount, capture, cardId, currency, description } = payment;
|
|
417
|
+
const formatCurrency = parseChar(currency, 3, 'USD').toUpperCase();
|
|
418
|
+
const stripeClient = getStripeClient();
|
|
419
|
+
return stripeClient.charges.create({
|
|
420
|
+
amount,
|
|
421
|
+
capture,
|
|
422
|
+
currency: formatCurrency,
|
|
423
|
+
description,
|
|
424
|
+
source: cardId
|
|
425
|
+
}).then((stripeCharge)=>{
|
|
426
|
+
const now = Date.now();
|
|
427
|
+
const insert = {
|
|
428
|
+
added: now,
|
|
429
|
+
amount,
|
|
430
|
+
capture,
|
|
431
|
+
cardId,
|
|
432
|
+
chargeFailCode: stripeCharge.failure_code,
|
|
433
|
+
chargeFailMsg: stripeCharge.failure_message,
|
|
434
|
+
chargeId: stripeCharge.id,
|
|
435
|
+
chargeStatus: stripeCharge.status,
|
|
436
|
+
currency: formatCurrency,
|
|
437
|
+
description,
|
|
438
|
+
modified: now,
|
|
439
|
+
userId: sessionId
|
|
440
|
+
};
|
|
441
|
+
const aqlQry = aql`INSERT ${insert} IN payments RETURN NEW`;
|
|
442
|
+
return useDb(databaseName).query(aqlQry).then((cursor)=>cursor.next()).then((newPayment)=>newPayment);
|
|
443
|
+
}).catch((error)=>{
|
|
444
|
+
console.log('payments::createHold::error', error);
|
|
445
|
+
throw new UserError('payment_error');
|
|
446
|
+
});
|
|
447
|
+
};
|
|
448
|
+
|
|
449
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi9Vc2Vycy9uaXRyb2c3L0RldmVsb3BtZW50L3JlYWt0b3Ivc3JjL2FjdGlvbnMvcGF5bWVudHMudHMiXSwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTktUHJlc2VudCwgTml0cm9nZW4gTGFicywgSW5jLlxuICogQ29weXJpZ2h0cyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuIFNlZSB0aGUgYWNjb21wYW55aW5nIExJQ0VOU0UgZmlsZSBmb3IgdGVybXMuXG4gKi9cbmltcG9ydCB7cGFyc2VOdW19IGZyb20gJ0BubGFicy91dGlscy9wYXJzZXJzL251bWJlcnMnO1xuaW1wb3J0IHtjcmVhdGVIYXNoLCBwYXJzZUNoYXIsIHBhcnNlSWQsIHBhcnNlU3RyaW5nLCBwYXJzZVZhckNoYXJ9IGZyb20gJ0BubGFicy91dGlscy9wYXJzZXJzL3N0cmluZ3MnO1xuaW1wb3J0IHthcWx9IGZyb20gJ2FyYW5nb2pzJztcbmltcG9ydCB7QXFsUXVlcnl9IGZyb20gJ2FyYW5nb2pzL2FxbCc7XG5pbXBvcnQgc3RyaXBlIGZyb20gJ3N0cmlwZSc7XG5cbmltcG9ydCB7QXBpQ29udGV4dH0gZnJvbSAnLi4vdHlwZXMvYXV0aC50eXBlcy5qcyc7XG5pbXBvcnQge0Vycm9yVHlwZXN9IGZyb20gJy4uL3R5cGVzL2Vycm9yLnR5cGVzLmpzJztcbmltcG9ydCB7UGF5bWVudEJhbmtBY2NvdW50LCBQYXltZW50Q2FyZFR5cGUsIFBheW1lbnRDaGFyZ2UsIFBheW1lbnRUcmFuc2Zlcn0gZnJvbSAnLi4vdHlwZXMvcGF5bWVudHMudHlwZXMuanMnO1xuaW1wb3J0IHtVc2VyVHlwZX0gZnJvbSAnLi4vdHlwZXMvdXNlcnMudHlwZXMuanMnO1xuaW1wb3J0IHtsb2dFcnJvciwgVXNlckVycm9yfSBmcm9tICcuLi91dGlscy9hbmFseXRpY3NVdGlscy5qcyc7XG5pbXBvcnQge3VzZURifSBmcm9tICcuLi91dGlscy9hcmFuZ29kYlV0aWxzLmpzJztcbmltcG9ydCB7Z2V0U3RyaXBlQ2xpZW50fSBmcm9tICcuLi91dGlscy9zdHJpcGVVdGlscy5qcyc7XG5pbXBvcnQge2dldFVzZXJ9IGZyb20gJy4vdXNlcnMuanMnO1xuXG5pbXBvcnQgdHlwZSB7RWRnZUNvbGxlY3Rpb259IGZyb20gJ2FyYW5nb2pzL2NvbGxlY3Rpb25zJztcblxuY29uc3QgZXZlbnRDYXRlZ29yeSA9ICdwYXltZW50cyc7XG5cbmV4cG9ydCBjb25zdCBhZGRDdXN0b21lckFjY291bnQgPSAoY29udGV4dDogQXBpQ29udGV4dCk6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICBjb25zdCBhY3Rpb24gPSAnYWRkQ3VzdG9tZXJBY2NvdW50JztcbiAgY29uc3Qge2RhdGFiYXNlTmFtZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkLCB1c2VybmFtZX0gPSB7fX0gPSBjb250ZXh0O1xuXG4gIGNvbnN0IHN0cmlwZUNsaWVudCA9IGdldFN0cmlwZUNsaWVudCgpO1xuXG4gIHJldHVybiBzdHJpcGVDbGllbnQuY3VzdG9tZXJzXG4gICAgLmNyZWF0ZSh7XG4gICAgICBtZXRhZGF0YToge1xuICAgICAgICB1c2VySWQ6IHNlc3Npb25JZCxcbiAgICAgICAgdXNlcm5hbWVcbiAgICAgIH1cbiAgICB9IGFzIGFueSlcbiAgICAudGhlbigoY3VzdG9tZXIpID0+IHtcbiAgICAgIC8vIENyZWF0ZSBzZXNzaW9uXG4gICAgICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gICAgICBjb25zdCB1cGRhdGU6IFVzZXJUeXBlID0ge1xuICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICBzdHJpcGVDdXN0b21lcklkOiBjdXN0b21lci5pZFxuICAgICAgfTtcblxuICAgICAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUgJHtzZXNzaW9uSWR9IFdJVEggJHt1cGRhdGV9IElOIHVzZXJzIExJTUlUIDEgUkVUVVJOIE5FV2A7XG5cbiAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZU5hbWUpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgLnRoZW4oKGN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgLnRoZW4oKHVwZGF0ZWRVc2VyOiBVc2VyVHlwZSkgPT4gISF1cGRhdGVkVXNlcilcbiAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgY2F0ZWdvcnk6IGV2ZW50Q2F0ZWdvcnksXG4gICAgICAgICAgbGFiZWw6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1JcbiAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGFkZEJhbmtBY2NvdW50ID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGJhbmtBY2NvdW50OiBQYXltZW50QmFua0FjY291bnQpOiBQcm9taXNlPGJvb2xlYW4+ID0+IHtcbiAgY29uc3QgYWN0aW9uID0gJ2FkZFBheW1lbnRBY2NvdW50QmFuayc7XG4gIGNvbnN0IHtkYXRhYmFzZU5hbWUsIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH0gPSB7fX0gPSBjb250ZXh0O1xuXG4gIC8vIFBhcmFtc1xuICBjb25zdCB7XG4gICAgYWNjb3VudE51bWJlcixcbiAgICBmdWxsTmFtZSxcbiAgICByb3V0aW5nXG4gIH0gPSBiYW5rQWNjb3VudDtcblxuICBjb25zdCBmb3JtYXRBY2NvdW50OiBzdHJpbmcgPSBwYXJzZVN0cmluZyhhY2NvdW50TnVtYmVyLCAzMik7XG5cbiAgaWYoZm9ybWF0QWNjb3VudCA9PT0gJycpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9hY2NvdW50X251bWJlcicpO1xuICB9XG5cbiAgY29uc3QgZm9ybWF0RnVsbE5hbWU6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihmdWxsTmFtZSwgMTI4KTtcblxuICBpZihmb3JtYXRGdWxsTmFtZSA9PT0gJycpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9mdWxsX25hbWUnKTtcbiAgfVxuXG4gIGNvbnN0IGZvcm1hdFJvdXRpbmc6IHN0cmluZyA9IHBhcnNlU3RyaW5nKHJvdXRpbmcsIDMyKTtcblxuICBpZihmb3JtYXRSb3V0aW5nID09PSAnJykge1xuICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX3JvdXRpbmdfbnVtYmVyJyk7XG4gIH1cblxuICByZXR1cm4gZ2V0VXNlcihjb250ZXh0LCB7dXNlcklkOiBzZXNzaW9uSWR9KVxuICAgIC50aGVuKCh1c2VyOiBVc2VyVHlwZSkgPT4ge1xuICAgICAgY29uc3Qge3N0cmlwZUFjY291bnRJZH0gPSB1c2VyO1xuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gZ2V0U3RyaXBlQ2xpZW50KCk7XG5cbiAgICAgIC8vIENyZWF0ZSBhIHRva2VuIGZpcnN0LCB0aGVuIHVzZSB0aGUgdG9rZW4gYXMgdGhlIHNvdXJjZVxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC50b2tlbnMuY3JlYXRlKHtcbiAgICAgICAgYmFua19hY2NvdW50OiB7XG4gICAgICAgICAgYWNjb3VudF9ob2xkZXJfbmFtZTogZm9ybWF0RnVsbE5hbWUsXG4gICAgICAgICAgYWNjb3VudF9ob2xkZXJfdHlwZTogJ2luZGl2aWR1YWwnLFxuICAgICAgICAgIGFjY291bnRfbnVtYmVyOiBmb3JtYXRBY2NvdW50LFxuICAgICAgICAgIGNvdW50cnk6ICdVUycsXG4gICAgICAgICAgY3VycmVuY3k6ICdVU0QnLFxuICAgICAgICAgIHJvdXRpbmdfbnVtYmVyOiBmb3JtYXRSb3V0aW5nXG4gICAgICAgIH1cbiAgICAgIH0pXG4gICAgICAgIC50aGVuKCh0b2tlbikgPT4gc3RyaXBlQ2xpZW50LmN1c3RvbWVycy5jcmVhdGVTb3VyY2UoXG4gICAgICAgICAgc3RyaXBlQWNjb3VudElkLFxuICAgICAgICAgIHtzb3VyY2U6IHRva2VuLmlkfVxuICAgICAgICApKVxuICAgICAgICAudGhlbigoYWNjb3VudCkgPT4ge1xuICAgICAgICAgIC8vIFVzZSB0eXBlIGFzc2VydGlvbiBmb3IgY2FyZCBwcm9wZXJ0aWVzXG4gICAgICAgICAgLy8gY29uc3QgY2FyZFNvdXJjZSA9IGFjY291bnQgYXMgdW5rbm93biBhcyBDYXJkO1xuICAgICAgICAgIC8vIGNvbnN0IGJyYW5kID0gY2FyZFNvdXJjZS5icmFuZCB8fCAnJztcbiAgICAgICAgICAvLyBjb25zdCBjdmNDaGVjayA9IGNhcmRTb3VyY2UuY3ZjX2NoZWNrIHx8ICcnO1xuICAgICAgICAgIC8vIGNvbnN0IGxhc3Q0ID0gY2FyZFNvdXJjZS5sYXN0NCB8fCAnJztcblxuICAgICAgICAgIC8vIENyZWF0ZSBzZXNzaW9uXG4gICAgICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICAgICAgICAgIGNvbnN0IHVwZGF0ZSA9IHtcbiAgICAgICAgICAgIGJhbmtBY2NvdW50LFxuICAgICAgICAgICAgYmFua0Z1bGxOYW1lOiBmb3JtYXRGdWxsTmFtZSxcbiAgICAgICAgICAgIGJhbmtJZDogYWNjb3VudC5pZCxcbiAgICAgICAgICAgIGJhbmtSb3V0aW5nOiBmb3JtYXRSb3V0aW5nLFxuICAgICAgICAgICAgbW9kaWZpZWQ6IG5vd1xuICAgICAgICAgIH07XG5cbiAgICAgICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFVQREFURSAke3Nlc3Npb25JZH0gV0lUSCAke3VwZGF0ZX0gSU4gdXNlcnMgTElNSVQgMSBSRVRVUk4gTkVXYDtcblxuICAgICAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZU5hbWUpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgICAgIC50aGVuKChjdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgICAgICAgICAudGhlbigodXBkYXRlZFVzZXI6IFVzZXJUeXBlKSA9PiB1cGRhdGVkVXNlcik7XG4gICAgICAgIH0pXG4gICAgICAgIC5jYXRjaCgoZXJyb3I6IEVycm9yKSA9PiB7XG4gICAgICAgICAgY29uc3QgbXNnID0gZXJyb3IubWVzc2FnZTtcblxuICAgICAgICAgIGlmKG1zZyA9PT0gJ0EgYmFuayBhY2NvdW50IHdpdGggdGhhdCByb3V0aW5nIG51bWJlciBhbmQgYWNjb3VudCBudW1iZXIgJyArXG4gICAgICAgICAgICAnYWxyZWFkeSBleGlzdHMgZm9yIHRoaXMgY3VzdG9tZXIuJykge1xuICAgICAgICAgICAgcmV0dXJuIGxvZ0Vycm9yKHtcbiAgICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgICAgbGFiZWw6ICdiYW5rX2FjY291bnRfZXhpc3RzJ1xuICAgICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCk7XG4gICAgICAgICAgfVxuICAgICAgICAgIHJldHVybiBsb2dFcnJvcih7XG4gICAgICAgICAgICBhY3Rpb24sXG4gICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgIGxhYmVsOiAncGF5bWVudF9lcnJvcidcbiAgICAgICAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgYWRkQ3JlZGl0Q2FyZCA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBjYXJkOiBQYXltZW50Q2FyZFR5cGUpOiBQcm9taXNlPFVzZXJUeXBlPiA9PiB7XG4gIGNvbnN0IGFjdGlvbiA9ICdhZGRDcmVkaXRDYXJkJztcbiAgY29uc3Qge2RhdGFiYXNlTmFtZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuXG4gIHJldHVybiBnZXRVc2VyKGNvbnRleHQsIHt1c2VySWQ6IHNlc3Npb25JZH0pXG4gICAgLnRoZW4oKHVzZXI6IFVzZXJUeXBlKSA9PiB7XG4gICAgICAvLyBVc2VyXG4gICAgICBjb25zdCB7c3RyaXBlQWNjb3VudElkfSA9IHVzZXI7XG5cbiAgICAgIC8vIENhcmRcbiAgICAgIGNvbnN0IHtcbiAgICAgICAgYWNjb3VudE51bWJlcixcbiAgICAgICAgY2l0eSxcbiAgICAgICAgY291bnRyeSxcbiAgICAgICAgY3ZjLFxuICAgICAgICBleHBNb250aCxcbiAgICAgICAgZXhwWWVhcixcbiAgICAgICAgZnVsbE5hbWUsXG4gICAgICAgIHN0cmVldDEsXG4gICAgICAgIHN0cmVldDIsXG4gICAgICAgIHN0YXRlLFxuICAgICAgICB6aXBcbiAgICAgIH06IFBheW1lbnRDYXJkVHlwZSA9IGNhcmQ7XG5cbiAgICAgIGNvbnN0IGZvcm1hdE51bWJlcjogbnVtYmVyID0gcGFyc2VOdW0oYWNjb3VudE51bWJlciwgMTYpO1xuXG4gICAgICBpZighZm9ybWF0TnVtYmVyKSB7XG4gICAgICAgIHRocm93IG5ldyBVc2VyRXJyb3IoJ3JlcXVpcmVkX2NyZWRpdF9jYXJkX251bWJlcicpO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBmb3JtYXRFeHBNb250aDogbnVtYmVyID0gcGFyc2VOdW0oZXhwTW9udGgsIDIpO1xuXG4gICAgICBpZighZm9ybWF0RXhwTW9udGgpIHtcbiAgICAgICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncmVxdWlyZWRfY3JlZGl0X2NhcmRfZXhwX21vbnRoJyk7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGZvcm1hdEV4cFllYXI6IG51bWJlciA9IHBhcnNlTnVtKGV4cFllYXIsIDIpO1xuXG4gICAgICBpZighZm9ybWF0RXhwWWVhcikge1xuICAgICAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9jcmVkaXRfY2FyZF9leHBfeWVhcicpO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBmb3JtYXRDdmM6IG51bWJlciA9IHBhcnNlTnVtKGN2YywgMyk7XG5cbiAgICAgIC8vIEFkZHJlc3NcbiAgICAgIGNvbnN0IHBheW1lbnRDYXJkOiBQYXltZW50Q2FyZFR5cGUgPSB7fTtcbiAgICAgIGNvbnN0IGZvcm1hdENpdHk6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihjaXR5LCAzMik7XG5cbiAgICAgIGlmKGZvcm1hdENpdHkpIHtcbiAgICAgICAgcGF5bWVudENhcmQuY2l0eSA9IGZvcm1hdENpdHk7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGZvcm1hdENvdW50cnk6IHN0cmluZyA9IHBhcnNlQ2hhcihjb3VudHJ5LCAyKTtcblxuICAgICAgaWYoZm9ybWF0Q291bnRyeSkge1xuICAgICAgICBwYXltZW50Q2FyZC5jb3VudHJ5ID0gZm9ybWF0Q291bnRyeTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgZm9ybWF0RnVsbE5hbWU6IHN0cmluZyA9IHBhcnNlVmFyQ2hhcihmdWxsTmFtZSwgMzIpO1xuXG4gICAgICBpZihmb3JtYXRGdWxsTmFtZSkge1xuICAgICAgICBwYXltZW50Q2FyZC5mdWxsTmFtZSA9IGZvcm1hdEZ1bGxOYW1lO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBmb3JtYXRTdHJlZXQxOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoc3RyZWV0MSwgMzIpO1xuXG4gICAgICBpZihmb3JtYXRTdHJlZXQxKSB7XG4gICAgICAgIHBheW1lbnRDYXJkLnN0cmVldDEgPSBmb3JtYXRTdHJlZXQxO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBmb3JtYXRTdHJlZXQyOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoc3RyZWV0MiwgMzIpO1xuXG4gICAgICBpZihmb3JtYXRTdHJlZXQyKSB7XG4gICAgICAgIHBheW1lbnRDYXJkLnN0cmVldDIgPSBmb3JtYXRTdHJlZXQyO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBmb3JtYXRTdGF0ZTogc3RyaW5nID0gcGFyc2VDaGFyKHN0YXRlLCAyKTtcblxuICAgICAgaWYoZm9ybWF0U3RhdGUpIHtcbiAgICAgICAgcGF5bWVudENhcmQuc3RhdGUgPSBmb3JtYXRTdGF0ZTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgZm9ybWF0WmlwOiBzdHJpbmcgPSBwYXJzZVZhckNoYXIoemlwLCAxMCk7XG5cbiAgICAgIGlmKGZvcm1hdFppcCkge1xuICAgICAgICBwYXltZW50Q2FyZC56aXAgPSBmb3JtYXRaaXA7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IGdldFN0cmlwZUNsaWVudCgpO1xuXG4gICAgICAvLyBDcmVhdGUgYSB0b2tlbiBmaXJzdCwgdGhlbiB1c2UgdGhlIHRva2VuIGFzIHRoZSBzb3VyY2VcbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQudG9rZW5zLmNyZWF0ZSh7XG4gICAgICAgIGNhcmQ6IHtcbiAgICAgICAgICBhZGRyZXNzX2NpdHk6IGZvcm1hdENpdHksXG4gICAgICAgICAgYWRkcmVzc19jb3VudHJ5OiBmb3JtYXRDb3VudHJ5LFxuICAgICAgICAgIGFkZHJlc3NfbGluZTE6IGZvcm1hdFN0cmVldDEsXG4gICAgICAgICAgYWRkcmVzc19saW5lMjogZm9ybWF0U3RyZWV0MixcbiAgICAgICAgICBhZGRyZXNzX3N0YXRlOiBmb3JtYXRTdGF0ZSxcbiAgICAgICAgICBhZGRyZXNzX3ppcDogZm9ybWF0WmlwLFxuICAgICAgICAgIGN2YzogZm9ybWF0Q3ZjLnRvU3RyaW5nKCksXG4gICAgICAgICAgZXhwX21vbnRoOiBmb3JtYXRFeHBNb250aC50b1N0cmluZygpLFxuICAgICAgICAgIGV4cF95ZWFyOiBmb3JtYXRFeHBZZWFyLnRvU3RyaW5nKCksXG4gICAgICAgICAgbmFtZTogZnVsbE5hbWUsXG4gICAgICAgICAgbnVtYmVyOiBmb3JtYXROdW1iZXIudG9TdHJpbmcoKVxuICAgICAgICB9XG4gICAgICB9KVxuICAgICAgICAudGhlbigodG9rZW4pID0+IHN0cmlwZUNsaWVudC5jdXN0b21lcnMuY3JlYXRlU291cmNlKFxuICAgICAgICAgIHN0cmlwZUFjY291bnRJZCxcbiAgICAgICAgICB7c291cmNlOiB0b2tlbi5pZH1cbiAgICAgICAgKSlcbiAgICAgICAgLnRoZW4oKG5ld1NvdXJjZSkgPT4ge1xuICAgICAgICAgIC8vIFVzZSB0eXBlIGFzc2VydGlvbiBmb3IgY2FyZCBwcm9wZXJ0aWVzXG4gICAgICAgICAgY29uc3QgY2FyZFNvdXJjZSA9IG5ld1NvdXJjZSBhcyB1bmtub3duIGFzIENhcmQ7XG4gICAgICAgICAgY29uc3QgYnJhbmQgPSBjYXJkU291cmNlLmJyYW5kIHx8ICcnO1xuICAgICAgICAgIGNvbnN0IGN2Y0NoZWNrID0gY2FyZFNvdXJjZS5jdmNfY2hlY2sgfHwgJyc7XG4gICAgICAgICAgY29uc3QgbGFzdDQgPSBjYXJkU291cmNlLmxhc3Q0IHx8ICcnO1xuXG4gICAgICAgICAgLy8gQ3JlYXRlIHNlc3Npb25cbiAgICAgICAgICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gICAgICAgICAgY29uc3QgaW5zZXJ0ID0ge1xuICAgICAgICAgICAgLi4ucGF5bWVudENhcmQsXG4gICAgICAgICAgICBfa2V5OiBjcmVhdGVIYXNoKGB1c2VyLXBheW1lbnQtJHtzZXNzaW9uSWR9YCksXG4gICAgICAgICAgICBhY2NvdW50TnVtYmVyOiBsYXN0NCxcbiAgICAgICAgICAgIGFkZGVkOiBub3csXG4gICAgICAgICAgICBicmFuZCxcbiAgICAgICAgICAgIGN2Y0NoZWNrLFxuICAgICAgICAgICAgZXhwTW9udGgsXG4gICAgICAgICAgICBleHBZZWFyLFxuICAgICAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgICAgIHVzZXJJZDogc2Vzc2lvbklkXG4gICAgICAgICAgfTtcbiAgICAgICAgICBjb25zdCBpbnNlcnRBcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYElOU0VSVCAke2luc2VydH0gSU4gY3JlZGl0Q2FyZHMgUkVUVVJOIE5FV2A7XG5cbiAgICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2VOYW1lKS5xdWVyeShpbnNlcnRBcWxRcnkpXG4gICAgICAgICAgICAudGhlbigoY3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAgICAgLnRoZW4oKG5ld0NhcmQ6IFBheW1lbnRDYXJkVHlwZSkgPT4ge1xuICAgICAgICAgICAgICBpZihuZXdDYXJkKSB7XG4gICAgICAgICAgICAgICAgLy8gQWRkIGxpbmtlZCBlZGdlXG4gICAgICAgICAgICAgICAgY29uc3Qge19pZDogY2FyZElkLCBfa2V5OiBjYXJkS2V5fSA9IGNhcmQ7XG4gICAgICAgICAgICAgICAgY29uc3QgZWRnZUNvbGxlY3Rpb246IEVkZ2VDb2xsZWN0aW9uID0gdXNlRGIoZGF0YWJhc2VOYW1lKS5jb2xsZWN0aW9uKCdoYXNQYXltZW50Jyk7XG4gICAgICAgICAgICAgICAgY29uc3QgZWRnZUlkID0gY3JlYXRlSGFzaChgcGF5bWVudC0ke2NhcmRLZXl9YCk7XG4gICAgICAgICAgICAgICAgY29uc3QgZWRnZSA9IHtcbiAgICAgICAgICAgICAgICAgIF9mcm9tOiBgdXNlcnMvJHtzZXNzaW9uSWR9YCxcbiAgICAgICAgICAgICAgICAgIF9rZXk6IGVkZ2VJZCxcbiAgICAgICAgICAgICAgICAgIF90bzogY2FyZElkXG4gICAgICAgICAgICAgICAgfTtcblxuICAgICAgICAgICAgICAgIHJldHVybiBlZGdlQ29sbGVjdGlvbi5zYXZlKGVkZ2UsIHtyZXR1cm5OZXc6IHRydWV9KS50aGVuKCgpID0+IGNhcmQpO1xuICAgICAgICAgICAgICB9XG5cbiAgICAgICAgICAgICAgcmV0dXJuIG5ld0NhcmQ7XG4gICAgICAgICAgICB9KVxuICAgICAgICAgICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgICAgICAgICAgYWN0aW9uLFxuICAgICAgICAgICAgICBjYXRlZ29yeTogZXZlbnRDYXRlZ29yeSxcbiAgICAgICAgICAgICAgbGFiZWw6ICdwYXltZW50X2Vycm9yJ1xuICAgICAgICAgICAgfSwgZXJyb3IsIGNvbnRleHQpLnRoZW4oKCkgPT4gbnVsbCkpO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCB1cGRhdGVDcmVkaXRDYXJkID0gKGNvbnRleHQ6IEFwaUNvbnRleHQsIGNhcmQ6IFBheW1lbnRDYXJkVHlwZSk6IFByb21pc2U8UGF5bWVudENhcmRUeXBlPiA9PiB7XG4gIGNvbnN0IHtkYXRhYmFzZU5hbWUsIHNlc3Npb246IHt1c2VySWQ6IHNlc3Npb25JZH19ID0gY29udGV4dDtcblxuICBjb25zdCB7XG4gICAgY2l0eSxcbiAgICBjb3VudHJ5LFxuICAgIGV4cE1vbnRoLFxuICAgIGV4cFllYXIsXG4gICAgZnVsbE5hbWUsXG4gICAgaWQsXG4gICAgc3RyZWV0MSxcbiAgICBzdGF0ZSxcbiAgICB6aXBcbiAgfTogUGF5bWVudENhcmRUeXBlID0gY2FyZDtcblxuICBjb25zdCBmb3JtYXRJZDogc3RyaW5nID0gcGFyc2VJZChpZCk7XG5cbiAgaWYoZm9ybWF0SWQpIHtcbiAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdyZXF1aXJlZF9jcmVkaXRfY2FyZF9pZCcpO1xuICB9XG5cbiAgY29uc3QgcGF5bWVudENhcmQ6IFBheW1lbnRDYXJkVHlwZSA9IHt9O1xuICBjb25zdCBmb3JtYXRFeHBNb250aDogbnVtYmVyID0gcGFyc2VOdW0oZXhwTW9udGgsIDIpO1xuICBjb25zdCBmb3JtYXRFeHBZZWFyOiBudW1iZXIgPSBwYXJzZU51bShleHBZZWFyLCAyKTtcbiAgY29uc3QgZm9ybWF0Q2l0eTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKGNpdHksIDMyKTtcbiAgY29uc3QgZm9ybWF0Q291bnRyeTogc3RyaW5nID0gcGFyc2VDaGFyKGNvdW50cnksIDIpO1xuICBjb25zdCBmb3JtYXRGdWxsTmFtZTogc3RyaW5nID0gcGFyc2VWYXJDaGFyKGZ1bGxOYW1lLCAzMik7XG4gIGNvbnN0IGZvcm1hdFN0cmVldDE6IHN0cmluZyA9IHBhcnNlU3RyaW5nKHN0cmVldDEsIDMyKTtcbiAgY29uc3QgZm9ybWF0U3RhdGU6IHN0cmluZyA9IHBhcnNlQ2hhcihzdGF0ZSwgMik7XG4gIGNvbnN0IGZvcm1hdFppcDogc3RyaW5nID0gcGFyc2VWYXJDaGFyKHppcCwgMTApO1xuXG4gIGlmKGZvcm1hdEV4cE1vbnRoKSB7XG4gICAgcGF5bWVudENhcmQuZXhwTW9udGggPSBmb3JtYXRFeHBNb250aDtcbiAgfVxuXG4gIGlmKGZvcm1hdEV4cFllYXIpIHtcbiAgICBwYXltZW50Q2FyZC5leHBZZWFyID0gZm9ybWF0RXhwWWVhcjtcbiAgfVxuXG4gIGlmKGZvcm1hdENpdHkpIHtcbiAgICBwYXltZW50Q2FyZC5jaXR5ID0gZm9ybWF0Q2l0eTtcbiAgfVxuXG4gIGlmKGZvcm1hdENvdW50cnkpIHtcbiAgICBwYXltZW50Q2FyZC5jb3VudHJ5ID0gZm9ybWF0Q291bnRyeTtcbiAgfVxuXG4gIGlmKGZvcm1hdEZ1bGxOYW1lKSB7XG4gICAgcGF5bWVudENhcmQuZnVsbE5hbWUgPSBmb3JtYXRGdWxsTmFtZTtcbiAgfVxuXG4gIGlmKGZvcm1hdFN0cmVldDEpIHtcbiAgICBwYXltZW50Q2FyZC5zdHJlZXQxID0gZm9ybWF0U3RyZWV0MTtcbiAgfVxuXG4gIGlmKGZvcm1hdFN0YXRlKSB7XG4gICAgcGF5bWVudENhcmQuc3RhdGUgPSBmb3JtYXRTdGF0ZTtcbiAgfVxuXG4gIGlmKGZvcm1hdFppcCkge1xuICAgIHBheW1lbnRDYXJkLnppcCA9IGZvcm1hdFppcDtcbiAgfVxuXG4gIGNvbnN0IHVwZGF0ZSA9IHBheW1lbnRDYXJkO1xuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFxuICAgICAgTEVUIHVwZGF0ZWRDYXJkID0gRklSU1QoXG4gICAgICAgIEZPUiBjIElOIGNyZWRpdENhcmRzXG4gICAgICAgIEZJTFRFUiBjLl9rZXkgPT0gJHtmb3JtYXRJZH0gJiYgYy51c2VySWQgPT0gJHtzZXNzaW9uSWR9XG4gICAgICAgIFVQREFURSBjIFdJVEggJHt1cGRhdGV9IElOIGNyZWRpdENhcmRzXG4gICAgICAgIExJTUlUIDFcbiAgICAgICAgUkVUVVJOIE5FV1xuICAgICAgKVxuICAgICAgTEVUIHVzZXIgPSBGSVJTVChcbiAgICAgICAgRk9SIHUgSU4gdXNlcnNcbiAgICAgICAgRklMVEVSIHUuX2tleSA9PSAke3Nlc3Npb25JZH1cbiAgICAgICAgTElNSVQgMVxuICAgICAgICBSRVRVUk4gdVxuICAgICAgKVxuICAgICAgUkVUVVJOIHt1c2VyOiB1c2VyLCBjYXJkOiB1cGRhdGVkQ2FyZH1gO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZU5hbWUpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgIC50aGVuKChyZXN1bHRzID0ge2NhcmQ6IHt9LCB1c2VyOiB7fX0pID0+IHtcbiAgICAgIGNvbnN0IHVwZGF0ZWRDYXJkOiBQYXltZW50Q2FyZFR5cGUgPSByZXN1bHRzLmNhcmQ7XG4gICAgICBjb25zdCB7dXNlcn0gPSByZXN1bHRzO1xuXG4gICAgICBpZighdXBkYXRlZENhcmQpIHtcbiAgICAgICAgdGhyb3cgbmV3IFVzZXJFcnJvcignbm90X2ZvdW5kJyk7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHtzdHJpcGVDdXN0b21lcklkfSA9IHVzZXI7XG4gICAgICBjb25zdCB7c3RyaXBlSWR9ID0gY2FyZDtcbiAgICAgIGNvbnN0IHN0cmlwZUNsaWVudCA9IGdldFN0cmlwZUNsaWVudCgpO1xuICAgICAgY29uc3QgdXBkYXRlOiBzdHJpcGUuQ3VzdG9tZXJVcGRhdGVTb3VyY2VQYXJhbXMgPSB7XG4gICAgICAgIGFkZHJlc3NfY2l0eTogZm9ybWF0Q2l0eSxcbiAgICAgICAgYWRkcmVzc19jb3VudHJ5OiBmb3JtYXRDb3VudHJ5LFxuICAgICAgICBhZGRyZXNzX2xpbmUxOiBmb3JtYXRTdHJlZXQxLFxuICAgICAgICBhZGRyZXNzX3N0YXRlOiBmb3JtYXRTdGF0ZSxcbiAgICAgICAgYWRkcmVzc196aXA6IGZvcm1hdFppcCxcbiAgICAgICAgZXhwX21vbnRoOiBmb3JtYXRFeHBNb250aC50b1N0cmluZygpLFxuICAgICAgICBleHBfeWVhcjogZm9ybWF0RXhwWWVhci50b1N0cmluZygpLFxuICAgICAgICBuYW1lOiBmb3JtYXRGdWxsTmFtZVxuICAgICAgfTtcblxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5jdXN0b21lcnNcbiAgICAgICAgLnVwZGF0ZVNvdXJjZShzdHJpcGVDdXN0b21lcklkLCBzdHJpcGVJZCwgdXBkYXRlKVxuICAgICAgICAudGhlbigoKSA9PiBjYXJkKVxuICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgIGNvbnNvbGUubG9nKCdwYXltZW50czo6dXBkYXRlQ2FyZDo6ZXJyb3InLCBlcnJvcik7XG4gICAgICAgICAgdGhyb3cgbmV3IFVzZXJFcnJvcigncGF5bWVudF9lcnJvcicpO1xuICAgICAgICB9KTtcbiAgICB9KTtcbn07XG5cbmV4cG9ydCBjb25zdCBnZXRDcmVkaXRDYXJkcyA9IChjb250ZXh0OiBBcGlDb250ZXh0KTogUHJvbWlzZTxQYXltZW50Q2FyZFR5cGVbXT4gPT4ge1xuICBjb25zdCBhY3Rpb24gPSAnZ2V0Q3JlZGl0Q2FyZHMnO1xuICBjb25zdCB7ZGF0YWJhc2VOYW1lLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgRk9SIGMgSU4gY3JlZGl0Q2FyZHNcbiAgICBGSUxURVIgYy51c2VySWQgPT0gJHtzZXNzaW9uSWR9XG4gICAgUkVUVVJOIGNgO1xuXG4gIHJldHVybiB1c2VEYihkYXRhYmFzZU5hbWUpLnF1ZXJ5KGFxbFFyeSlcbiAgICAudGhlbigoY3Vyc29yKSA9PiBjdXJzb3IuYWxsKCkpXG4gICAgLnRoZW4oKGxpc3Q6IFBheW1lbnRDYXJkVHlwZVtdID0gW10pID0+IGxpc3QpXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IGxvZ0Vycm9yKHtcbiAgICAgIGFjdGlvbixcbiAgICAgIGNhdGVnb3J5OiBldmVudENhdGVnb3J5LFxuICAgICAgbGFiZWw6IEVycm9yVHlwZXMuREFUQUJBU0VfRVJST1JcbiAgICB9LCBlcnJvciwgY29udGV4dCkudGhlbigoKSA9PiBudWxsKSk7XG59O1xuXG5leHBvcnQgY29uc3QgZGVsZXRlQ3JlZGl0Q2FyZCA9IChjb250ZXh0OiBBcGlDb250ZXh0LCBjYXJkSWQ6IHN0cmluZyk6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICBjb25zdCB7ZGF0YWJhc2VOYW1lLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IGZvcm1hdENhcmRJZDogc3RyaW5nID0gcGFyc2VJZChjYXJkSWQpO1xuICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYFxuICAgIExFVCBjYXJkID0gRklSU1QoXG4gICAgICBGT1IgYyBJTiBjcmVkaXRDYXJkc1xuICAgICAgRklMVEVSIGMuX2tleSA9PSAke2Zvcm1hdENhcmRJZH0gJiYgYy51c2VySWQgPT0gJHtzZXNzaW9uSWR9XG4gICAgICBMSU1JVCAxXG4gICAgICBSRU1PVkUgYyBJTiBjcmVkaXRDYXJkc1xuICAgICAgUkVUVVJOIE9MRFxuICAgIClcbiAgICBMRVQgdXNlciA9IEZJUlNUKFxuICAgICAgRk9SIHUgSU4gdXNlcnNcbiAgICAgIEZJTFRFUiB1Ll9rZXkgPT0gJHtzZXNzaW9uSWR9XG4gICAgICBMSU1JVCAxXG4gICAgICBSRVRVUk4gdVxuICAgIClcbiAgICBSRVRVUk4ge3VzZXI6IHVzZXIsIGNhcmQ6IGNhcmR9YDtcblxuICByZXR1cm4gdXNlRGIoZGF0YWJhc2VOYW1lKS5xdWVyeShhcWxRcnkpXG4gICAgLnRoZW4oKGN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAudGhlbigocmVzdWx0ID0ge2NhcmQ6IHt9LCB1c2VyOiB7fX0pID0+IHtcbiAgICAgIGlmKCFyZXN1bHQpIHtcbiAgICAgICAgcmV0dXJuIGZhbHNlO1xuICAgICAgfVxuXG4gICAgICBjb25zdCB7Y2FyZCwgdXNlcn0gPSByZXN1bHQ7XG4gICAgICBjb25zdCB7X2tleTogY2FyZEtleX0gPSBjYXJkO1xuXG4gICAgICAvLyBSZW1vdmUgbGlua2VkIGVkZ2VzXG4gICAgICBjb25zdCBlZGdlQ29sbGVjdGlvbiA9IHVzZURiKGRhdGFiYXNlTmFtZSkuY29sbGVjdGlvbignaGFzUGF5bWVudCcpO1xuXG4gICAgICByZXR1cm4gZWRnZUNvbGxlY3Rpb24ub3V0RWRnZXMoY2FyZEtleSwge30pXG4gICAgICAgIC50aGVuKGFzeW5jIChyZXNwb25zZSkgPT4ge1xuICAgICAgICAgIC8vIEV4dHJhY3QgZWRnZXMgZnJvbSB0aGUgcmVzcG9uc2VcbiAgICAgICAgICBjb25zdCBlZGdlcyA9IEFycmF5LmlzQXJyYXkocmVzcG9uc2UpID8gcmVzcG9uc2UgOiBbXTtcblxuICAgICAgICAgIGlmKGVkZ2VzLmxlbmd0aCkge1xuICAgICAgICAgICAgYXdhaXQgUHJvbWlzZS5hbGwoXG4gICAgICAgICAgICAgIGVkZ2VzLm1hcCgoZWRnZSkgPT4ge1xuICAgICAgICAgICAgICAgIGNvbnN0IHtfa2V5OiBlZGdlS2V5fSA9IGVkZ2U7XG4gICAgICAgICAgICAgICAgY29uc3QgcmVtb3ZlQXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBSRU1PVkUge19rZXk6JHtlZGdlS2V5fX0gSU4gaGFzUGF5bWVudGA7XG4gICAgICAgICAgICAgICAgcmV0dXJuIHVzZURiKGRhdGFiYXNlTmFtZSkucXVlcnkocmVtb3ZlQXFsUXJ5KTtcbiAgICAgICAgICAgICAgfSkpXG4gICAgICAgICAgICAgIC50aGVuKCgpID0+IHtcbiAgICAgICAgICAgICAgICAvLyBTdHJpcGVcbiAgICAgICAgICAgICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBnZXRTdHJpcGVDbGllbnQoKTtcblxuICAgICAgICAgICAgICAgIHJldHVybiBzdHJpcGVDbGllbnQuY3VzdG9tZXJzXG4gICAgICAgICAgICAgICAgICAuZGVsZXRlU291cmNlKHVzZXIuc3RyaXBlQ3VzdG9tZXJJZCwgY2FyZC5zdHJpcGVJZClcbiAgICAgICAgICAgICAgICAgIC50aGVuKCgpID0+IHRydWUpXG4gICAgICAgICAgICAgICAgICAuY2F0Y2goKGVycm9yOiBFcnJvcikgPT4ge1xuICAgICAgICAgICAgICAgICAgICBjb25zb2xlLmxvZygncGF5bWVudHM6OmRlbGV0ZUNhcmQ6OmVycm9yJywgZXJyb3IpO1xuICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdwYXltZW50X2Vycm9yJyk7XG4gICAgICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgZGVsZXRlQmFua0FjY291bnQgPSAoY29udGV4dDogQXBpQ29udGV4dCwgYmFua0lkOiBzdHJpbmcpOiBQcm9taXNlPGJvb2xlYW4+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlTmFtZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuXG4gIC8vIENsZWFuIGRiXG4gIGNvbnN0IHVwZGF0ZTogVXNlclR5cGUgPSB7XG4gICAgYmFua0FjY291bnQ6ICcnLFxuICAgIGJhbmtGdWxsTmFtZTogJycsXG4gICAgYmFua0lkOiAnJyxcbiAgICBiYW5rUm91dGluZzogJycsXG4gICAgbW9kaWZpZWQ6IERhdGUubm93KClcbiAgfTtcbiAgY29uc3QgYXFsUXJ5OiBBcWxRdWVyeSA9IGFxbGBVUERBVEUgJHtzZXNzaW9uSWR9IFdJVEggJHt1cGRhdGV9IElOIHVzZXJzIExJTUlUIDEgUkVUVVJOIE5FV2A7XG5cbiAgcmV0dXJuIHVzZURiKGRhdGFiYXNlTmFtZSkucXVlcnkoYXFsUXJ5KVxuICAgIC50aGVuKChjdXJzb3IpID0+IGN1cnNvci5uZXh0KCkpXG4gICAgLnRoZW4oKHVzZXI6IFVzZXJUeXBlKSA9PiB7XG4gICAgICBjb25zdCB7c3RyaXBlQWNjb3VudElkfSA9IHVzZXI7XG4gICAgICBjb25zdCBzdHJpcGVDbGllbnQgPSBnZXRTdHJpcGVDbGllbnQoKTtcblxuICAgICAgcmV0dXJuIHN0cmlwZUNsaWVudC5jdXN0b21lcnNcbiAgICAgICAgLmRlbGV0ZVNvdXJjZShzdHJpcGVBY2NvdW50SWQsIGJhbmtJZClcbiAgICAgICAgLnRoZW4oKCkgPT4gdHJ1ZSlcbiAgICAgICAgLmNhdGNoKCgpID0+IFByb21pc2UucmVzb2x2ZShmYWxzZSkpO1xuICAgIH0pO1xufTtcblxuZXhwb3J0IGNvbnN0IGNyZWF0ZVBheW1lbnRUcmFuc2ZlciA9IChjb250ZXh0OiBBcGlDb250ZXh0LCB0cmFuc2ZlcjogUGF5bWVudFRyYW5zZmVyKTogUHJvbWlzZTxQYXltZW50VHJhbnNmZXI+ID0+IHtcbiAgY29uc3Qge2RhdGFiYXNlTmFtZSwgc2Vzc2lvbjoge3VzZXJJZDogc2Vzc2lvbklkfX0gPSBjb250ZXh0O1xuICBjb25zdCB7YW1vdW50LCBjdXJyZW5jeX0gPSB0cmFuc2ZlcjtcbiAgY29uc3QgZm9ybWF0QW1vdW50OiBudW1iZXIgPSBwYXJzZU51bShhbW91bnQpO1xuICBjb25zdCBmb3JtYXRDdXJyZW5jeTogc3RyaW5nID0gcGFyc2VDaGFyKGN1cnJlbmN5LCAzLCAnVVNEJykudG9VcHBlckNhc2UoKTtcblxuICByZXR1cm4gZ2V0VXNlcihjb250ZXh0LCB7dXNlcklkOiBzZXNzaW9uSWR9KVxuICAgIC50aGVuKCh1c2VyOiBVc2VyVHlwZSkgPT4ge1xuICAgICAgY29uc3Qge3N0cmlwZUFjY291bnRJZH0gPSB1c2VyO1xuICAgICAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gZ2V0U3RyaXBlQ2xpZW50KCk7XG5cbiAgICAgIHJldHVybiBzdHJpcGVDbGllbnQudHJhbnNmZXJzXG4gICAgICAgIC5jcmVhdGUoe1xuICAgICAgICAgIGFtb3VudDogZm9ybWF0QW1vdW50LFxuICAgICAgICAgIGN1cnJlbmN5OiBmb3JtYXRDdXJyZW5jeSxcbiAgICAgICAgICBkZXN0aW5hdGlvbjogc3RyaXBlQWNjb3VudElkXG4gICAgICAgIH0pXG4gICAgICAgIC50aGVuKChzdHJpcGVUcmFuc2ZlcikgPT4ge1xuICAgICAgICAgIGNvbnNvbGUubG9nKHN0cmlwZVRyYW5zZmVyKTtcbiAgICAgICAgICBjb25zdCBub3c6IG51bWJlciA9IERhdGUubm93KCk7XG4gICAgICAgICAgY29uc3QgaW5zZXJ0OiBQYXltZW50VHJhbnNmZXIgPSB7XG4gICAgICAgICAgICBhZGRlZDogbm93LFxuICAgICAgICAgICAgYW1vdW50OiBmb3JtYXRBbW91bnQsXG4gICAgICAgICAgICBjdXJyZW5jeTogZm9ybWF0Q3VycmVuY3ksXG4gICAgICAgICAgICBtb2RpZmllZDogbm93LFxuICAgICAgICAgICAgdXNlcklkOiBzZXNzaW9uSWRcbiAgICAgICAgICB9O1xuICAgICAgICAgIGNvbnN0IGFxbFFyeTogQXFsUXVlcnkgPSBhcWxgSU5TRVJUICR7aW5zZXJ0fSBJTiB0cmFuc2ZlcnMgUkVUVVJOIE5FV2A7XG5cbiAgICAgICAgICByZXR1cm4gdXNlRGIoZGF0YWJhc2VOYW1lKS5xdWVyeShhcWxRcnkpXG4gICAgICAgICAgICAudGhlbigoY3Vyc29yKSA9PiBjdXJzb3IubmV4dCgpKVxuICAgICAgICAgICAgLnRoZW4oKG5ld1RyYW5zZmVyOiBQYXltZW50VHJhbnNmZXIpID0+IG5ld1RyYW5zZmVyKTtcbiAgICAgICAgfSk7XG4gICAgfSk7XG59O1xuXG5leHBvcnQgY29uc3QgY3JlYXRlUGF5bWVudEhvbGQgPSAoY29udGV4dDogQXBpQ29udGV4dCwgcGF5bWVudDogUGF5bWVudENoYXJnZSk6IFByb21pc2U8UGF5bWVudENoYXJnZT4gPT4ge1xuICBjb25zdCB7ZGF0YWJhc2VOYW1lLCBzZXNzaW9uOiB7dXNlcklkOiBzZXNzaW9uSWR9fSA9IGNvbnRleHQ7XG4gIGNvbnN0IHthbW91bnQsIGNhcHR1cmUsIGNhcmRJZCwgY3VycmVuY3ksIGRlc2NyaXB0aW9ufSA9IHBheW1lbnQ7XG4gIGNvbnN0IGZvcm1hdEN1cnJlbmN5ID0gcGFyc2VDaGFyKGN1cnJlbmN5LCAzLCAnVVNEJykudG9VcHBlckNhc2UoKTtcbiAgY29uc3Qgc3RyaXBlQ2xpZW50ID0gZ2V0U3RyaXBlQ2xpZW50KCk7XG5cbiAgcmV0dXJuIHN0cmlwZUNsaWVudC5jaGFyZ2VzXG4gICAgLmNyZWF0ZSh7XG4gICAgICBhbW91bnQsXG4gICAgICBjYXB0dXJlLFxuICAgICAgY3VycmVuY3k6IGZvcm1hdEN1cnJlbmN5LFxuICAgICAgZGVzY3JpcHRpb24sXG4gICAgICBzb3VyY2U6IGNhcmRJZFxuICAgIH0pXG4gICAgLnRoZW4oKHN0cmlwZUNoYXJnZSkgPT4ge1xuICAgICAgY29uc3Qgbm93OiBudW1iZXIgPSBEYXRlLm5vdygpO1xuICAgICAgY29uc3QgaW5zZXJ0OiBQYXltZW50Q2hhcmdlID0ge1xuICAgICAgICBhZGRlZDogbm93LFxuICAgICAgICBhbW91bnQsXG4gICAgICAgIGNhcHR1cmUsXG4gICAgICAgIGNhcmRJZCxcbiAgICAgICAgY2hhcmdlRmFpbENvZGU6IHN0cmlwZUNoYXJnZS5mYWlsdXJlX2NvZGUsXG4gICAgICAgIGNoYXJnZUZhaWxNc2c6IHN0cmlwZUNoYXJnZS5mYWlsdXJlX21lc3NhZ2UsXG4gICAgICAgIGNoYXJnZUlkOiBzdHJpcGVDaGFyZ2UuaWQsXG4gICAgICAgIGNoYXJnZVN0YXR1czogc3RyaXBlQ2hhcmdlLnN0YXR1cyxcbiAgICAgICAgY3VycmVuY3k6IGZvcm1hdEN1cnJlbmN5LFxuICAgICAgICBkZXNjcmlwdGlvbixcbiAgICAgICAgbW9kaWZpZWQ6IG5vdyxcbiAgICAgICAgdXNlcklkOiBzZXNzaW9uSWRcbiAgICAgIH07XG4gICAgICBjb25zdCBhcWxRcnk6IEFxbFF1ZXJ5ID0gYXFsYElOU0VSVCAke2luc2VydH0gSU4gcGF5bWVudHMgUkVUVVJOIE5FV2A7XG5cbiAgICAgIHJldHVybiB1c2VEYihkYXRhYmFzZU5hbWUpLnF1ZXJ5KGFxbFFyeSlcbiAgICAgICAgLnRoZW4oKGN1cnNvcikgPT4gY3Vyc29yLm5leHQoKSlcbiAgICAgICAgLnRoZW4oKG5ld1BheW1lbnQ6IFBheW1lbnRDaGFyZ2UpID0+IG5ld1BheW1lbnQpO1xuICAgIH0pXG4gICAgLmNhdGNoKChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgIGNvbnNvbGUubG9nKCdwYXltZW50czo6Y3JlYXRlSG9sZDo6ZXJyb3InLCBlcnJvcik7XG4gICAgICB0aHJvdyBuZXcgVXNlckVycm9yKCdwYXltZW50X2Vycm9yJyk7XG4gICAgfSk7XG59O1xuXG5pbnRlcmZhY2UgQ2FyZCB7XG4gIGlkOiBzdHJpbmc7XG4gIGJyYW5kPzogc3RyaW5nO1xuICBjdmNfY2hlY2s/OiBzdHJpbmc7XG4gIGxhc3Q0Pzogc3RyaW5nO1xufVxuIl0sIm5hbWVzIjpbInBhcnNlTnVtIiwiY3JlYXRlSGFzaCIsInBhcnNlQ2hhciIsInBhcnNlSWQiLCJwYXJzZVN0cmluZyIsInBhcnNlVmFyQ2hhciIsImFxbCIsIkVycm9yVHlwZXMiLCJsb2dFcnJvciIsIlVzZXJFcnJvciIsInVzZURiIiwiZ2V0U3RyaXBlQ2xpZW50IiwiZ2V0VXNlciIsImV2ZW50Q2F0ZWdvcnkiLCJhZGRDdXN0b21lckFjY291bnQiLCJjb250ZXh0IiwiYWN0aW9uIiwiZGF0YWJhc2VOYW1lIiwic2Vzc2lvbiIsInVzZXJJZCIsInNlc3Npb25JZCIsInVzZXJuYW1lIiwic3RyaXBlQ2xpZW50IiwiY3VzdG9tZXJzIiwiY3JlYXRlIiwibWV0YWRhdGEiLCJ0aGVuIiwiY3VzdG9tZXIiLCJub3ciLCJEYXRlIiwidXBkYXRlIiwibW9kaWZpZWQiLCJzdHJpcGVDdXN0b21lcklkIiwiaWQiLCJhcWxRcnkiLCJxdWVyeSIsImN1cnNvciIsIm5leHQiLCJ1cGRhdGVkVXNlciIsImNhdGNoIiwiZXJyb3IiLCJjYXRlZ29yeSIsImxhYmVsIiwiREFUQUJBU0VfRVJST1IiLCJhZGRCYW5rQWNjb3VudCIsImJhbmtBY2NvdW50IiwiYWNjb3VudE51bWJlciIsImZ1bGxOYW1lIiwicm91dGluZyIsImZvcm1hdEFjY291bnQiLCJmb3JtYXRGdWxsTmFtZSIsImZvcm1hdFJvdXRpbmciLCJ1c2VyIiwic3RyaXBlQWNjb3VudElkIiwidG9rZW5zIiwiYmFua19hY2NvdW50IiwiYWNjb3VudF9ob2xkZXJfbmFtZSIsImFjY291bnRfaG9sZGVyX3R5cGUiLCJhY2NvdW50X251bWJlciIsImNvdW50cnkiLCJjdXJyZW5jeSIsInJvdXRpbmdfbnVtYmVyIiwidG9rZW4iLCJjcmVhdGVTb3VyY2UiLCJzb3VyY2UiLCJhY2NvdW50IiwiYmFua0Z1bGxOYW1lIiwiYmFua0lkIiwiYmFua1JvdXRpbmciLCJtc2ciLCJtZXNzYWdlIiwiYWRkQ3JlZGl0Q2FyZCIsImNhcmQiLCJjaXR5IiwiY3ZjIiwiZXhwTW9udGgiLCJleHBZZWFyIiwic3RyZWV0MSIsInN0cmVldDIiLCJzdGF0ZSIsInppcCIsImZvcm1hdE51bWJlciIsImZvcm1hdEV4cE1vbnRoIiwiZm9ybWF0RXhwWWVhciIsImZvcm1hdEN2YyIsInBheW1lbnRDYXJkIiwiZm9ybWF0Q2l0eSIsImZvcm1hdENvdW50cnkiLCJmb3JtYXRTdHJlZXQxIiwiZm9ybWF0U3RyZWV0MiIsImZvcm1hdFN0YXRlIiwiZm9ybWF0WmlwIiwiYWRkcmVzc19jaXR5IiwiYWRkcmVzc19jb3VudHJ5IiwiYWRkcmVzc19saW5lMSIsImFkZHJlc3NfbGluZTIiLCJhZGRyZXNzX3N0YXRlIiwiYWRkcmVzc196aXAiLCJ0b1N0cmluZyIsImV4cF9tb250aCIsImV4cF95ZWFyIiwibmFtZSIsIm51bWJlciIsIm5ld1NvdXJjZSIsImNhcmRTb3VyY2UiLCJicmFuZCIsImN2Y0NoZWNrIiwiY3ZjX2NoZWNrIiwibGFzdDQiLCJpbnNlcnQiLCJfa2V5IiwiYWRkZWQiLCJpbnNlcnRBcWxRcnkiLCJuZXdDYXJkIiwiX2lkIiwiY2FyZElkIiwiY2FyZEtleSIsImVkZ2VDb2xsZWN0aW9uIiwiY29sbGVjdGlvbiIsImVkZ2VJZCIsImVkZ2UiLCJfZnJvbSIsIl90byIsInNhdmUiLCJyZXR1cm5OZXciLCJ1cGRhdGVDcmVkaXRDYXJkIiwiZm9ybWF0SWQiLCJyZXN1bHRzIiwidXBkYXRlZENhcmQiLCJzdHJpcGVJZCIsInVwZGF0ZVNvdXJjZSIsImNvbnNvbGUiLCJsb2ciLCJnZXRDcmVkaXRDYXJkcyIsImFsbCIsImxpc3QiLCJkZWxldGVDcmVkaXRDYXJkIiwiZm9ybWF0Q2FyZElkIiwicmVzdWx0Iiwib3V0RWRnZXMiLCJyZXNwb25zZSIsImVkZ2VzIiwiQXJyYXkiLCJpc0FycmF5IiwibGVuZ3RoIiwiUHJvbWlzZSIsIm1hcCIsImVkZ2VLZXkiLCJyZW1vdmVBcWxRcnkiLCJkZWxldGVTb3VyY2UiLCJkZWxldGVCYW5rQWNjb3VudCIsInJlc29sdmUiLCJjcmVhdGVQYXltZW50VHJhbnNmZXIiLCJ0cmFuc2ZlciIsImFtb3VudCIsImZvcm1hdEFtb3VudCIsImZvcm1hdEN1cnJlbmN5IiwidG9VcHBlckNhc2UiLCJ0cmFuc2ZlcnMiLCJkZXN0aW5hdGlvbiIsInN0cmlwZVRyYW5zZmVyIiwibmV3VHJhbnNmZXIiLCJjcmVhdGVQYXltZW50SG9sZCIsInBheW1lbnQiLCJjYXB0dXJlIiwiZGVzY3JpcHRpb24iLCJjaGFyZ2VzIiwic3RyaXBlQ2hhcmdlIiwiY2hhcmdlRmFpbENvZGUiLCJmYWlsdXJlX2NvZGUiLCJjaGFyZ2VGYWlsTXNnIiwiZmFpbHVyZV9tZXNzYWdlIiwiY2hhcmdlSWQiLCJjaGFyZ2VTdGF0dXMiLCJzdGF0dXMiLCJuZXdQYXltZW50Il0sIm1hcHBpbmdzIjoiQUFBQTs7O0NBR0MsR0FDRCxTQUFRQSxRQUFRLFFBQU8sK0JBQStCO0FBQ3RELFNBQVFDLFVBQVUsRUFBRUMsU0FBUyxFQUFFQyxPQUFPLEVBQUVDLFdBQVcsRUFBRUMsWUFBWSxRQUFPLCtCQUErQjtBQUN2RyxTQUFRQyxHQUFHLFFBQU8sV0FBVztBQUs3QixTQUFRQyxVQUFVLFFBQU8sMEJBQTBCO0FBR25ELFNBQVFDLFFBQVEsRUFBRUMsU0FBUyxRQUFPLDZCQUE2QjtBQUMvRCxTQUFRQyxLQUFLLFFBQU8sNEJBQTRCO0FBQ2hELFNBQVFDLGVBQWUsUUFBTywwQkFBMEI7QUFDeEQsU0FBUUMsT0FBTyxRQUFPLGFBQWE7QUFJbkMsTUFBTUMsZ0JBQWdCO0FBRXRCLE9BQU8sTUFBTUMscUJBQXFCLENBQUNDO0lBQ2pDLE1BQU1DLFNBQVM7SUFDZixNQUFNLEVBQUNDLFlBQVksRUFBRUMsU0FBUyxFQUFDQyxRQUFRQyxTQUFTLEVBQUVDLFFBQVEsRUFBQyxHQUFHLENBQUMsQ0FBQyxFQUFDLEdBQUdOO0lBRXBFLE1BQU1PLGVBQWVYO0lBRXJCLE9BQU9XLGFBQWFDLFNBQVMsQ0FDMUJDLE1BQU0sQ0FBQztRQUNOQyxVQUFVO1lBQ1JOLFFBQVFDO1lBQ1JDO1FBQ0Y7SUFDRixHQUNDSyxJQUFJLENBQUMsQ0FBQ0M7UUFDTCxpQkFBaUI7UUFDakIsTUFBTUMsTUFBY0MsS0FBS0QsR0FBRztRQUM1QixNQUFNRSxTQUFtQjtZQUN2QkMsVUFBVUg7WUFDVkksa0JBQWtCTCxTQUFTTSxFQUFFO1FBQy9CO1FBRUEsTUFBTUMsU0FBbUI1QixHQUFHLENBQUMsT0FBTyxFQUFFYyxVQUFVLE1BQU0sRUFBRVUsT0FBTyw0QkFBNEIsQ0FBQztRQUU1RixPQUFPcEIsTUFBTU8sY0FBY2tCLEtBQUssQ0FBQ0QsUUFDOUJSLElBQUksQ0FBQyxDQUFDVSxTQUFXQSxPQUFPQyxJQUFJLElBQzVCWCxJQUFJLENBQUMsQ0FBQ1ksY0FBMEIsQ0FBQyxDQUFDQSxhQUNsQ0MsS0FBSyxDQUFDLENBQUNDLFFBQWlCaEMsU0FBUztnQkFDaENRO2dCQUNBeUIsVUFBVTVCO2dCQUNWNkIsT0FBT25DLFdBQVdvQyxjQUFjO1lBQ2xDLEdBQUdILE9BQU96QixTQUFTVyxJQUFJLENBQUMsSUFBTTtJQUNsQztBQUNKLEVBQUU7QUFFRixPQUFPLE1BQU1rQixpQkFBaUIsQ0FBQzdCLFNBQXFCOEI7SUFDbEQsTUFBTTdCLFNBQVM7SUFDZixNQUFNLEVBQUNDLFlBQVksRUFBRUMsU0FBUyxFQUFDQyxRQUFRQyxTQUFTLEVBQUMsR0FBRyxDQUFDLENBQUMsRUFBQyxHQUFHTDtJQUUxRCxTQUFTO0lBQ1QsTUFBTSxFQUNKK0IsYUFBYSxFQUNiQyxRQUFRLEVBQ1JDLE9BQU8sRUFDUixHQUFHSDtJQUVKLE1BQU1JLGdCQUF3QjdDLFlBQVkwQyxlQUFlO0lBRXpELElBQUdHLGtCQUFrQixJQUFJO1FBQ3ZCLE1BQU0sSUFBSXhDLFVBQVU7SUFDdEI7SUFFQSxNQUFNeUMsaUJBQXlCN0MsYUFBYTBDLFVBQVU7SUFFdEQsSUFBR0csbUJBQW1CLElBQUk7UUFDeEIsTUFBTSxJQUFJekMsVUFBVTtJQUN0QjtJQUVBLE1BQU0wQyxnQkFBd0IvQyxZQUFZNEMsU0FBUztJQUVuRCxJQUFHRyxrQkFBa0IsSUFBSTtRQUN2QixNQUFNLElBQUkxQyxVQUFVO0lBQ3RCO0lBRUEsT0FBT0csUUFBUUcsU0FBUztRQUFDSSxRQUFRQztJQUFTLEdBQ3ZDTSxJQUFJLENBQUMsQ0FBQzBCO1FBQ0wsTUFBTSxFQUFDQyxlQUFlLEVBQUMsR0FBR0Q7UUFDMUIsTUFBTTlCLGVBQWVYO1FBRXJCLHlEQUF5RDtRQUN6RCxPQUFPVyxhQUFhZ0MsTUFBTSxDQUFDOUIsTUFBTSxDQUFDO1lBQ2hDK0IsY0FBYztnQkFDWkMscUJBQXFCTjtnQkFDckJPLHFCQUFxQjtnQkFDckJDLGdCQUFnQlQ7Z0JBQ2hCVSxTQUFTO2dCQUNUQyxVQUFVO2dCQUNWQyxnQkFBZ0JWO1lBQ2xCO1FBQ0YsR0FDR3pCLElBQUksQ0FBQyxDQUFDb0MsUUFBVXhDLGFBQWFDLFNBQVMsQ0FBQ3dDLFlBQVksQ0FDbERWLGlCQUNBO2dCQUFDVyxRQUFRRixNQUFNN0IsRUFBRTtZQUFBLElBRWxCUCxJQUFJLENBQUMsQ0FBQ3VDO1lBQ0wseUNBQXlDO1lBQ3pDLGlEQUFpRDtZQUNqRCx3Q0FBd0M7WUFDeEMsK0NBQStDO1lBQy9DLHdDQUF3QztZQUV4QyxpQkFBaUI7WUFDakIsTUFBTXJDLE1BQWNDLEtBQUtELEdBQUc7WUFDNUIsTUFBTUUsU0FBUztnQkFDYmU7Z0JBQ0FxQixjQUFjaEI7Z0JBQ2RpQixRQUFRRixRQUFRaEMsRUFBRTtnQkFDbEJtQyxhQUFhakI7Z0JBQ2JwQixVQUFVSDtZQUNaO1lBRUEsTUFBTU0sU0FBbUI1QixHQUFHLENBQUMsT0FBTyxFQUFFYyxVQUFVLE1BQU0sRUFBRVUsT0FBTyw0QkFBNEIsQ0FBQztZQUU1RixPQUFPcEIsTUFBTU8sY0FBY2tCLEtBQUssQ0FBQ0QsUUFDOUJSLElBQUksQ0FBQyxDQUFDVSxTQUFXQSxPQUFPQyxJQUFJLElBQzVCWCxJQUFJLENBQUMsQ0FBQ1ksY0FBMEJBO1FBQ3JDLEdBQ0NDLEtBQUssQ0FBQyxDQUFDQztZQUNOLE1BQU02QixNQUFNN0IsTUFBTThCLE9BQU87WUFFekIsSUFBR0QsUUFBUSxnRUFDVCxxQ0FBcUM7Z0JBQ3JDLE9BQU83RCxTQUFTO29CQUNkUTtvQkFDQXlCLFVBQVU1QjtvQkFDVjZCLE9BQU87Z0JBQ1QsR0FBR0YsT0FBT3pCLFNBQVNXLElBQUksQ0FBQyxJQUFNO1lBQ2hDO1lBQ0EsT0FBT2xCLFNBQVM7Z0JBQ2RRO2dCQUNBeUIsVUFBVTVCO2dCQUNWNkIsT0FBTztZQUNULEdBQUdGLE9BQU96QixTQUFTVyxJQUFJLENBQUMsSUFBTTtRQUNoQztJQUNKO0FBQ0osRUFBRTtBQUVGLE9BQU8sTUFBTTZDLGdCQUFnQixDQUFDeEQsU0FBcUJ5RDtJQUNqRCxNQUFNeEQsU0FBUztJQUNmLE1BQU0sRUFBQ0MsWUFBWSxFQUFFQyxTQUFTLEVBQUNDLFFBQVFDLFNBQVMsRUFBQyxFQUFDLEdBQUdMO0lBRXJELE9BQU9ILFFBQVFHLFNBQVM7UUFBQ0ksUUFBUUM7SUFBUyxHQUN2Q00sSUFBSSxDQUFDLENBQUMwQjtRQUNMLE9BQU87UUFDUCxNQUFNLEVBQUNDLGVBQWUsRUFBQyxHQUFHRDtRQUUxQixPQUFPO1FBQ1AsTUFBTSxFQUNKTixhQUFhLEVBQ2IyQixJQUFJLEVBQ0pkLE9BQU8sRUFDUGUsR0FBRyxFQUNIQyxRQUFRLEVBQ1JDLE9BQU8sRUFDUDdCLFFBQVEsRUFDUjhCLE9BQU8sRUFDUEMsT0FBTyxFQUNQQyxLQUFLLEVBQ0xDLEdBQUcsRUFDSixHQUFvQlI7UUFFckIsTUFBTVMsZUFBdUJqRixTQUFTOEMsZUFBZTtRQUVyRCxJQUFHLENBQUNtQyxjQUFjO1lBQ2hCLE1BQU0sSUFBSXhFLFVBQVU7UUFDdEI7UUFFQSxNQUFNeUUsaUJBQXlCbEYsU0FBUzJFLFVBQVU7UUFFbEQsSUFBRyxDQUFDTyxnQkFBZ0I7WUFDbEIsTUFBTSxJQUFJekUsVUFBVTtRQUN0QjtRQUVBLE1BQU0wRSxnQkFBd0JuRixTQUFTNEUsU0FBUztRQUVoRCxJQUFHLENBQUNPLGVBQWU7WUFDakIsTUFBTSxJQUFJMUUsVUFBVTtRQUN0QjtRQUVBLE1BQU0yRSxZQUFvQnBGLFNBQVMwRSxLQUFLO1FBRXhDLFVBQVU7UUFDVixNQUFNVyxjQUErQixDQUFDO1FBQ3RDLE1BQU1DLGFBQXFCakYsYUFBYW9FLE1BQU07UUFFOUMsSUFBR2EsWUFBWTtZQUNiRCxZQUFZWixJQUFJLEdBQUdhO1FBQ3JCO1FBRUEsTUFBTUMsZ0JBQXdCckYsVUFBVXlELFNBQVM7UUFFakQsSUFBRzRCLGVBQWU7WUFDaEJGLFlBQVkxQixPQUFPLEdBQUc0QjtRQUN4QjtRQUVBLE1BQU1yQyxpQkFBeUI3QyxhQUFhMEMsVUFBVTtRQUV0RCxJQUFHRyxnQkFBZ0I7WUFDakJtQyxZQUFZdEMsUUFBUSxHQUFHRztRQUN6QjtRQUVBLE1BQU1zQyxnQkFBd0JuRixhQUFhd0UsU0FBUztRQUVwRCxJQUFHVyxlQUFlO1lBQ2hCSCxZQUFZUixPQUFPLEdBQUdXO1FBQ3hCO1FBRUEsTUFBTUMsZ0JBQXdCcEYsYUFBYXlFLFNBQVM7UUFFcEQsSUFBR1csZUFBZTtZQUNoQkosWUFBWVAsT0FBTyxHQUFHVztRQUN4QjtRQUVBLE1BQU1DLGNBQXNCeEYsVUFBVTZFLE9BQU87UUFFN0MsSUFBR1csYUFBYTtZQUNkTCxZQUFZTixLQUFLLEdBQUdXO1FBQ3RCO1FBRUEsTUFBTUMsWUFBb0J0RixhQUFhMkUsS0FBSztRQUU1QyxJQUFHVyxXQUFXO1lBQ1pOLFlBQVlMLEdBQUcsR0FBR1c7UUFDcEI7UUFFQSxNQUFNckUsZUFBZVg7UUFFckIseURBQXlEO1FBQ3pELE9BQU9XLGFBQWFnQyxNQUFNLENBQUM5QixNQUFNLENBQUM7WUFDaENnRCxNQUFNO2dCQUNKb0IsY0FBY047Z0JBQ2RPLGlCQUFpQk47Z0JBQ2pCTyxlQUFlTjtnQkFDZk8sZUFBZU47Z0JBQ2ZPLGVBQWVOO2dCQUNmTyxhQUFhTjtnQkFDYmpCLEtBQUtVLFVBQVVjLFFBQVE7Z0JBQ3ZCQyxXQUFXakIsZUFBZWdCLFFBQVE7Z0JBQ2xDRSxVQUFVakIsY0FBY2UsUUFBUTtnQkFDaENHLE1BQU10RDtnQkFDTnVELFFBQVFyQixhQUFhaUIsUUFBUTtZQUMvQjtRQUNGLEdBQ0d4RSxJQUFJLENBQUMsQ0FBQ29DLFFBQVV4QyxhQUFhQyxTQUFTLENBQUN3QyxZQUFZLENBQ2xEVixpQkFDQTtnQkFBQ1csUUFBUUYsTUFBTTdCLEVBQUU7WUFBQSxJQUVsQlAsSUFBSSxDQUFDLENBQUM2RTtZQUNMLHlDQUF5QztZQUN6QyxNQUFNQyxhQUFhRDtZQUNuQixNQUFNRSxRQUFRRCxXQUFXQyxLQUFLLElBQUk7WUFDbEMsTUFBTUMsV0FBV0YsV0FBV0csU0FBUyxJQUFJO1lBQ3pDLE1BQU1DLFFBQVFKLFdBQVdJLEtBQUssSUFBSTtZQUVsQyxpQkFBaUI7WUFDakIsTUFBTWhGLE1BQWNDLEtBQUtELEdBQUc7WUFDNUIsTUFBTWlGLFNBQVM7Z0JBQ2IsR0FBR3hCLFdBQVc7Z0JBQ2R5QixNQUFNN0csV0FBVyxDQUFDLGFBQWEsRUFBRW1CLFdBQVc7Z0JBQzVDMEIsZUFBZThEO2dCQUNmRyxPQUFPbkY7Z0JBQ1A2RTtnQkFDQUM7Z0JBQ0EvQjtnQkFDQUM7Z0JBQ0E3QyxVQUFVSDtnQkFDVlQsUUFBUUM7WUFDVjtZQUNBLE1BQU00RixlQUF5QjFHLEdBQUcsQ0FBQyxPQUFPLEVBQUV1RyxPQUFPLDBCQUEwQixDQUFDO1lBRTlFLE9BQU9uRyxNQUFNTyxjQUFja0IsS0FBSyxDQUFDNkUsY0FDOUJ0RixJQUFJLENBQUMsQ0FBQ1UsU0FBV0EsT0FBT0MsSUFBSSxJQUM1QlgsSUFBSSxDQUFDLENBQUN1RjtnQkFDTCxJQUFHQSxTQUFTO29CQUNWLGtCQUFrQjtvQkFDbEIsTUFBTSxFQUFDQyxLQUFLQyxNQUFNLEVBQUVMLE1BQU1NLE9BQU8sRUFBQyxHQUFHNUM7b0JBQ3JDLE1BQU02QyxpQkFBaUMzRyxNQUFNTyxjQUFjcUcsVUFBVSxDQUFDO29CQUN0RSxNQUFNQyxTQUFTdEgsV0FBVyxDQUFDLFFBQVEsRUFBRW1ILFNBQVM7b0JBQzlDLE1BQU1JLE9BQU87d0JBQ1hDLE9BQU8sQ0FBQyxNQUFNLEVBQUVyRyxXQUFXO3dCQUMzQjBGLE1BQU1TO3dCQUNORyxLQUFLUDtvQkFDUDtvQkFFQSxPQUFPRSxlQUFlTSxJQUFJLENBQUNILE1BQU07d0JBQUNJLFdBQVc7b0JBQUksR0FBR2xHLElBQUksQ0FBQyxJQUFNOEM7Z0JBQ2pFO2dCQUVBLE9BQU95QztZQUNULEdBQ0MxRSxLQUFLLENBQUMsQ0FBQ0MsUUFBaUJoQyxTQUFTO29CQUNoQ1E7b0JBQ0F5QixVQUFVNUI7b0JBQ1Y2QixPQUFPO2dCQUNULEdBQUdGLE9BQU96QixTQUFTVyxJQUFJLENBQUMsSUFBTTtRQUNsQztJQUNKO0FBQ0osRUFBRTtBQUVGLE9BQU8sTUFBTW1HLG1CQUFtQixDQUFDOUcsU0FBcUJ5RDtJQUNwRCxNQUFNLEVBQUN2RCxZQUFZLEVBQUVDLFNBQVMsRUFBQ0MsUUFBUUMsU0FBUyxFQUFDLEVBQUMsR0FBR0w7SUFFckQsTUFBTSxFQUNKMEQsSUFBSSxFQUNKZCxPQUFPLEVBQ1BnQixRQUFRLEVBQ1JDLE9BQU8sRUFDUDdCLFFBQVEsRUFDUmQsRUFBRSxFQUNGNEMsT0FBTyxFQUNQRSxLQUFLLEVBQ0xDLEdBQUcsRUFDSixHQUFvQlI7SUFFckIsTUFBTXNELFdBQW1CM0gsUUFBUThCO0lBRWpDLElBQUc2RixVQUFVO1FBQ1gsTUFBTSxJQUFJckgsVUFBVTtJQUN0QjtJQUVBLE1BQU00RSxjQUErQixDQUFDO0lBQ3RDLE1BQU1ILGlCQUF5QmxGLFNBQVMyRSxVQUFVO0lBQ2xELE1BQU1RLGdCQUF3Qm5GLFNBQVM0RSxTQUFTO0lBQ2hELE1BQU1VLGFBQXFCakYsYUFBYW9FLE1BQU07SUFDOUMsTUFBTWMsZ0JBQXdCckYsVUFBVXlELFNBQVM7SUFDakQsTUFBTVQsaUJBQXlCN0MsYUFBYTBDLFVBQVU7SUFDdEQsTUFBTXlDLGdCQUF3QnBGLFlBQVl5RSxTQUFTO0lBQ25ELE1BQU1hLGNBQXNCeEYsVUFBVTZFLE9BQU87SUFDN0MsTUFBTVksWUFBb0J0RixhQUFhMkUsS0FBSztJQUU1QyxJQUFHRSxnQkFBZ0I7UUFDakJHLFlBQVlWLFFBQVEsR0FBR087SUFDekI7SUFFQSxJQUFHQyxlQUFlO1FBQ2hCRSxZQUFZVCxPQUFPLEdBQUdPO0lBQ3hCO0lBRUEsSUFBR0csWUFBWTtRQUNiRCxZQUFZWixJQUFJLEdBQUdhO0lBQ3JCO0lBRUEsSUFBR0MsZUFBZTtRQUNoQkYsWUFBWTFCLE9BQU8sR0FBRzRCO0lBQ3hCO0lBRUEsSUFBR3JDLGdCQUFnQjtRQUNqQm1DLFlBQVl0QyxRQUFRLEdBQUdHO0lBQ3pCO0lBRUEsSUFBR3NDLGVBQWU7UUFDaEJILFlBQVlSLE9BQU8sR0FBR1c7SUFDeEI7SUFFQSxJQUFHRSxhQUFhO1FBQ2RMLFlBQVlOLEtBQUssR0FBR1c7SUFDdEI7SUFFQSxJQUFHQyxXQUFXO1FBQ1pOLFlBQVlMLEdBQUcsR0FBR1c7SUFDcEI7SUFFQSxNQUFNN0QsU0FBU3VEO0lBQ2YsTUFBTW5ELFNBQW1CNUIsR0FBRyxDQUFDOzs7eUJBR04sRUFBRXdILFNBQVMsZ0JBQWdCLEVBQUUxRyxVQUFVO3NCQUMxQyxFQUFFVSxPQUFPOzs7Ozs7eUJBTU4sRUFBRVYsVUFBVTs7Ozs0Q0FJTyxDQUFDO0lBRTNDLE9BQU9WLE1BQU1PLGNBQWNrQixLQUFLLENBQUNELFFBQzlCUixJQUFJLENBQUMsQ0FBQ1UsU0FBV0EsT0FBT0MsSUFBSSxJQUM1QlgsSUFBSSxDQUFDLENBQUNxRyxVQUFVO1FBQUN2RCxNQUFNLENBQUM7UUFBR3BCLE1BQU0sQ0FBQztJQUFDLENBQUM7UUFDbkMsTUFBTTRFLGNBQStCRCxRQUFRdkQsSUFBSTtRQUNqRCxNQUFNLEVBQUNwQixJQUFJLEVBQUMsR0FBRzJFO1FBRWYsSUFBRyxDQUFDQyxhQUFhO1lBQ2YsTUFBTSxJQUFJdkgsVUFBVTtRQUN0QjtRQUVBLE1BQU0sRUFBQ3VCLGdCQUFnQixFQUFDLEdBQUdvQjtRQUMzQixNQUFNLEVBQUM2RSxRQUFRLEVBQUMsR0FBR3pEO1FBQ25CLE1BQU1sRCxlQUFlWDtRQUNyQixNQUFNbUIsU0FBNEM7WUFDaEQ4RCxjQUFjTjtZQUNkTyxpQkFBaUJOO1lBQ2pCTyxlQUFlTjtZQUNmUSxlQUFlTjtZQUNmTyxhQUFhTjtZQUNiUSxXQUFXakIsZUFBZWdCLFFBQVE7WUFDbENFLFVBQVVqQixjQUFjZSxRQUFRO1lBQ2hDRyxNQUFNbkQ7UUFDUjtRQUVBLE9BQU81QixhQUFhQyxTQUFTLENBQzFCMkcsWUFBWSxDQUFDbEcsa0JBQWtCaUcsVUFBVW5HLFFBQ3pDSixJQUFJLENBQUMsSUFBTThDLE1BQ1hqQyxLQUFLLENBQUMsQ0FBQ0M7WUFDTjJGLFFBQVFDLEdBQUcsQ0FBQywrQkFBK0I1RjtZQUMzQyxNQUFNLElBQUkvQixVQUFVO1FBQ3RCO0lBQ0o7QUFDSixFQUFFO0FBRUYsT0FBTyxNQUFNNEgsaUJBQWlCLENBQUN0SDtJQUM3QixNQUFNQyxTQUFTO0lBQ2YsTUFBTSxFQUFDQyxZQUFZLEVBQUVDLFNBQVMsRUFBQ0MsUUFBUUMsU0FBUyxFQUFDLEVBQUMsR0FBR0w7SUFDckQsTUFBTW1CLFNBQW1CNUIsR0FBRyxDQUFDO3VCQUNSLEVBQUVjLFVBQVU7WUFDdkIsQ0FBQztJQUVYLE9BQU9WLE1BQU1PLGNBQWNrQixLQUFLLENBQUNELFFBQzlCUixJQUFJLENBQUMsQ0FBQ1UsU0FBV0EsT0FBT2tHLEdBQUcsSUFDM0I1RyxJQUFJLENBQUMsQ0FBQzZHLE9BQTBCLEVBQUUsR0FBS0EsTUFDdkNoRyxLQUFLLENBQUMsQ0FBQ0MsUUFBaUJoQyxTQUFTO1lBQ2hDUTtZQUNBeUIsVUFBVTVCO1lBQ1Y2QixPQUFPbkMsV0FBV29DLGNBQWM7UUFDbEMsR0FBR0gsT0FBT3pCLFNBQVNXLElBQUksQ0FBQyxJQUFNO0FBQ2xDLEVBQUU7QUFFRixPQUFPLE1BQU04RyxtQkFBbUIsQ0FBQ3pILFNBQXFCb0c7SUFDcEQsTUFBTSxFQUFDbEcsWUFBWSxFQUFFQyxTQUFTLEVBQUNDLFFBQVFDLFNBQVMsRUFBQyxFQUFDLEdBQUdMO0lBQ3JELE1BQU0wSCxlQUF1QnRJLFFBQVFnSDtJQUNyQyxNQUFNakYsU0FBbUI1QixHQUFHLENBQUM7Ozt1QkFHUixFQUFFbUksYUFBYSxnQkFBZ0IsRUFBRXJILFVBQVU7Ozs7Ozs7dUJBTzNDLEVBQUVBLFVBQVU7Ozs7bUNBSUEsQ0FBQztJQUVsQyxPQUFPVixNQUFNTyxjQUFja0IsS0FBSyxDQUFDRCxRQUM5QlIsSUFBSSxDQUFDLENBQUNVLFNBQVdBLE9BQU9DLElBQUksSUFDNUJYLElBQUksQ0FBQyxDQUFDZ0gsU0FBUztRQUFDbEUsTUFBTSxDQUFDO1FBQUdwQixNQUFNLENBQUM7SUFBQyxDQUFDO1FBQ2xDLElBQUcsQ0FBQ3NGLFFBQVE7WUFDVixPQUFPO1FBQ1Q7UUFFQSxNQUFNLEVBQUNsRSxJQUFJLEVBQUVwQixJQUFJLEVBQUMsR0FBR3NGO1FBQ3JCLE1BQU0sRUFBQzVCLE1BQU1NLE9BQU8sRUFBQyxHQUFHNUM7UUFFeEIsc0JBQXNCO1FBQ3RCLE1BQU02QyxpQkFBaUIzRyxNQUFNTyxjQUFjcUcsVUFBVSxDQUFDO1FBRXRELE9BQU9ELGVBQWVzQixRQUFRLENBQUN2QixTQUFTLENBQUMsR0FDdEMxRixJQUFJLENBQUMsT0FBT2tIO1lBQ1gsa0NBQWtDO1lBQ2xDLE1BQU1DLFFBQVFDLE1BQU1DLE9BQU8sQ0FBQ0gsWUFBWUEsV0FBVyxFQUFFO1lBRXJELElBQUdDLE1BQU1HLE1BQU0sRUFBRTtnQkFDZixNQUFNQyxRQUFRWCxHQUFHLENBQ2ZPLE1BQU1LLEdBQUcsQ0FBQyxDQUFDMUI7b0JBQ1QsTUFBTSxFQUFDVixNQUFNcUMsT0FBTyxFQUFDLEdBQUczQjtvQkFDeEIsTUFBTTRCLGVBQXlCOUksR0FBRyxDQUFDLGFBQWEsRUFBRTZJLFFBQVEsZUFBZSxDQUFDO29CQUMxRSxPQUFPekksTUFBTU8sY0FBY2tCLEtBQUssQ0FBQ2lIO2dCQUNuQyxJQUNDMUgsSUFBSSxDQUFDO29CQUNKLFNBQVM7b0JBQ1QsTUFBTUosZUFBZVg7b0JBRXJCLE9BQU9XLGFBQWFDLFNBQVMsQ0FDMUI4SCxZQUFZLENBQUNqRyxLQUFLcEIsZ0JBQWdCLEVBQUV3QyxLQUFLeUQsUUFBUSxFQUNqRHZHLElBQUksQ0FBQyxJQUFNLE1BQ1hhLEtBQUssQ0FBQyxDQUFDQzt3QkFDTjJGLFFBQVFDLEdBQUcsQ0FBQywrQkFBK0I1Rjt3QkFDM0MsTUFBTSxJQUFJL0IsVUFBVTtvQkFDdEI7Z0JBQ0o7Z0JBRUYsT0FBTztZQUNUO1lBRUEsT0FBTztRQUNUO0lBQ0o7QUFDSixFQUFFO0FBRUYsT0FBTyxNQUFNNkksb0JBQW9CLENBQUN2SSxTQUFxQm9EO0lBQ3JELE1BQU0sRUFBQ2xELFlBQVksRUFBRUMsU0FBUyxFQUFDQyxRQUFRQyxTQUFTLEVBQUMsRUFBQyxHQUFHTDtJQUVyRCxXQUFXO0lBQ1gsTUFBTWUsU0FBbUI7UUFDdkJlLGFBQWE7UUFDYnFCLGNBQWM7UUFDZEMsUUFBUTtRQUNSQyxhQUFhO1FBQ2JyQyxVQUFVRixLQUFLRCxHQUFHO0lBQ3BCO0lBQ0EsTUFBTU0sU0FBbUI1QixHQUFHLENBQUMsT0FBTyxFQUFFYyxVQUFVLE1BQU0sRUFBRVUsT0FBTyw0QkFBNEIsQ0FBQztJQUU1RixPQUFPcEIsTUFBTU8sY0FBY2tCLEtBQUssQ0FBQ0QsUUFDOUJSLElBQUksQ0FBQyxDQUFDVSxTQUFXQSxPQUFPQyxJQUFJLElBQzVCWCxJQUFJLENBQUMsQ0FBQzBCO1FBQ0wsTUFBTSxFQUFDQyxlQUFlLEVBQUMsR0FBR0Q7UUFDMUIsTUFBTTlCLGVBQWVYO1FBRXJCLE9BQU9XLGFBQWFDLFNBQVMsQ0FDMUI4SCxZQUFZLENBQUNoRyxpQkFBaUJjLFFBQzlCekMsSUFBSSxDQUFDLElBQU0sTUFDWGEsS0FBSyxDQUFDLElBQU0wRyxRQUFRTSxPQUFPLENBQUM7SUFDakM7QUFDSixFQUFFO0FBRUYsT0FBTyxNQUFNQyx3QkFBd0IsQ0FBQ3pJLFNBQXFCMEk7SUFDekQsTUFBTSxFQUFDeEksWUFBWSxFQUFFQyxTQUFTLEVBQUNDLFFBQVFDLFNBQVMsRUFBQyxFQUFDLEdBQUdMO0lBQ3JELE1BQU0sRUFBQzJJLE1BQU0sRUFBRTlGLFFBQVEsRUFBQyxHQUFHNkY7SUFDM0IsTUFBTUUsZUFBdUIzSixTQUFTMEo7SUFDdEMsTUFBTUUsaUJBQXlCMUosVUFBVTBELFVBQVUsR0FBRyxPQUFPaUcsV0FBVztJQUV4RSxPQUFPakosUUFBUUcsU0FBUztRQUFDSSxRQUFRQztJQUFTLEdBQ3ZDTSxJQUFJLENBQUMsQ0FBQzBCO1FBQ0wsTUFBTSxFQUFDQyxlQUFlLEVBQUMsR0FBR0Q7UUFDMUIsTUFBTTlCLGVBQWVYO1FBRXJCLE9BQU9XLGFBQWF3SSxTQUFTLENBQzFCdEksTUFBTSxDQUFDO1lBQ05rSSxRQUFRQztZQUNSL0YsVUFBVWdHO1lBQ1ZHLGFBQWExRztRQUNmLEdBQ0MzQixJQUFJLENBQUMsQ0FBQ3NJO1lBQ0w3QixRQUFRQyxHQUFHLENBQUM0QjtZQUNaLE1BQU1wSSxNQUFjQyxLQUFLRCxHQUFHO1lBQzVCLE1BQU1pRixTQUEwQjtnQkFDOUJFLE9BQU9uRjtnQkFDUDhILFFBQVFDO2dCQUNSL0YsVUFBVWdHO2dCQUNWN0gsVUFBVUg7Z0JBQ1ZULFFBQVFDO1lBQ1Y7WUFDQSxNQUFNYyxTQUFtQjVCLEdBQUcsQ0FBQyxPQUFPLEVBQUV1RyxPQUFPLHdCQUF3QixDQUFDO1lBRXRFLE9BQU9uRyxNQUFNTyxjQUFja0IsS0FBSyxDQUFDRCxRQUM5QlIsSUFBSSxDQUFDLENBQUNVLFNBQVdBLE9BQU9DLElBQUksSUFDNUJYLElBQUksQ0FBQyxDQUFDdUksY0FBaUNBO1FBQzVDO0lBQ0o7QUFDSixFQUFFO0FBRUYsT0FBTyxNQUFNQyxvQkFBb0IsQ0FBQ25KLFNBQXFCb0o7SUFDckQsTUFBTSxFQUFDbEosWUFBWSxFQUFFQyxTQUFTLEVBQUNDLFFBQVFDLFNBQVMsRUFBQyxFQUFDLEdBQUdMO0lBQ3JELE1BQU0sRUFBQzJJLE1BQU0sRUFBRVUsT0FBTyxFQUFFakQsTUFBTSxFQUFFdkQsUUFBUSxFQUFFeUcsV0FBVyxFQUFDLEdBQUdGO0lBQ3pELE1BQU1QLGlCQUFpQjFKLFVBQVUwRCxVQUFVLEdBQUcsT0FBT2lHLFdBQVc7SUFDaEUsTUFBTXZJLGVBQWVYO0lBRXJCLE9BQU9XLGFBQWFnSixPQUFPLENBQ3hCOUksTUFBTSxDQUFDO1FBQ05rSTtRQUNBVTtRQUNBeEcsVUFBVWdHO1FBQ1ZTO1FBQ0FyRyxRQUFRbUQ7SUFDVixHQUNDekYsSUFBSSxDQUFDLENBQUM2STtRQUNMLE1BQU0zSSxNQUFjQyxLQUFLRCxHQUFHO1FBQzVCLE1BQU1pRixTQUF3QjtZQUM1QkUsT0FBT25GO1lBQ1A4SDtZQUNBVTtZQUNBakQ7WUFDQXFELGdCQUFnQkQsYUFBYUUsWUFBWTtZQUN6Q0MsZUFBZUgsYUFBYUksZUFBZTtZQUMzQ0MsVUFBVUwsYUFBYXRJLEVBQUU7WUFDekI0SSxjQUFjTixhQUFhTyxNQUFNO1lBQ2pDbEgsVUFBVWdHO1lBQ1ZTO1lBQ0F0SSxVQUFVSDtZQUNWVCxRQUFRQztRQUNWO1FBQ0EsTUFBTWMsU0FBbUI1QixHQUFHLENBQUMsT0FBTyxFQUFFdUcsT0FBTyx1QkFBdUIsQ0FBQztRQUVyRSxPQUFPbkcsTUFBTU8sY0FBY2tCLEtBQUssQ0FBQ0QsUUFDOUJSLElBQUksQ0FBQyxDQUFDVSxTQUFXQSxPQUFPQyxJQUFJLElBQzVCWCxJQUFJLENBQUMsQ0FBQ3FKLGFBQThCQTtJQUN6QyxHQUNDeEksS0FBSyxDQUFDLENBQUNDO1FBQ04yRixRQUFRQyxHQUFHLENBQUMsK0JBQStCNUY7UUFDM0MsTUFBTSxJQUFJL0IsVUFBVTtJQUN0QjtBQUNKLEVBQUUifQ==
|