@decaf-ts/for-nest 0.11.0 → 0.11.2-refactor
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/README.md +1 -1
- package/dist/for-nest.cjs +2636 -2
- package/dist/for-nest.js +2738 -2
- package/lib/cjs/auth/AuthInterceptor.cjs +88 -0
- package/lib/cjs/auth/DecafAuthHandler.cjs +37 -0
- package/lib/cjs/auth/DecafAuthModule.cjs +56 -0
- package/lib/cjs/auth/constants.cjs +9 -0
- package/lib/cjs/auth/decorators.cjs +28 -0
- package/lib/cjs/auth/index.cjs +23 -0
- package/lib/cjs/bin/cli.cjs +2 -2
- package/lib/cjs/cli-module.cjs +48 -6
- package/lib/cjs/constants.cjs +5 -5
- package/lib/cjs/controllers.cjs +10 -102
- package/lib/cjs/core-module.cjs +8 -8
- package/lib/cjs/decaf-model/DecafModelModule.cjs +19 -4
- package/lib/cjs/decaf-model/FromModelController.cjs +732 -785
- package/lib/cjs/decaf-model/decorators/ApiOperationFromModel.cjs +53 -16
- package/lib/cjs/decaf-model/decorators/ApiParamsFromModel.cjs +2 -2
- package/lib/cjs/decaf-model/decorators/DecafBody.cjs +2 -2
- package/lib/cjs/decaf-model/decorators/DecafParams.cjs +2 -2
- package/lib/cjs/decaf-model/decorators/controller-config.cjs +29 -0
- package/lib/cjs/decaf-model/decorators/decorators.cjs +5 -45
- package/lib/cjs/decaf-model/decorators/expose.cjs +13 -0
- package/lib/cjs/decaf-model/decorators/index.cjs +3 -2
- package/lib/cjs/decaf-model/decorators/types.cjs +2 -2
- package/lib/cjs/decaf-model/decorators/utils.cjs +2 -2
- package/lib/cjs/decaf-model/index.cjs +28 -19
- package/lib/cjs/decaf-model/types.cjs +2 -2
- package/lib/cjs/decaf-model/utils.cjs +2 -2
- package/lib/cjs/decoration.cjs +2 -2
- package/lib/cjs/decorators.cjs +85 -0
- package/lib/cjs/events-module/DecafStreamModule.cjs +2 -2
- package/lib/cjs/events-module/EventsController.cjs +2 -2
- package/lib/cjs/events-module/constant.cjs +2 -2
- package/lib/cjs/events-module/index.cjs +2 -2
- package/lib/cjs/events-module/utils.cjs +2 -2
- package/lib/cjs/factory/NestBootstraper.cjs +2 -2
- package/lib/cjs/factory/errors/cors.cjs +2 -2
- package/lib/cjs/factory/errors/index.cjs +2 -2
- package/lib/cjs/factory/errors/throttling.cjs +2 -2
- package/lib/cjs/factory/exceptions/DecafErrorFilter.cjs +2 -2
- package/lib/cjs/factory/exceptions/decorators.cjs +2 -2
- package/lib/cjs/factory/exceptions/index.cjs +2 -2
- package/lib/cjs/factory/index.cjs +2 -2
- package/lib/cjs/factory/openapi/DtoBuilder.cjs +37 -6
- package/lib/cjs/factory/openapi/SwaggerBuilder.cjs +2 -2
- package/lib/cjs/factory/openapi/SwaggerCustomUI.cjs +2 -2
- package/lib/cjs/factory/openapi/constants.cjs +2 -2
- package/lib/cjs/factory/openapi/index.cjs +2 -2
- package/lib/cjs/index.cjs +8 -7
- package/lib/cjs/interceptors/DecafRequestHandlerInterceptor.cjs +27 -21
- package/lib/cjs/interceptors/index.cjs +3 -4
- package/lib/cjs/module.cjs +4 -2
- package/lib/cjs/overrides/Adapter.cjs +2 -2
- package/lib/cjs/overrides/ModelBuilderExtensions.cjs +2 -2
- package/lib/cjs/overrides/constants.cjs +2 -2
- package/lib/cjs/overrides/decoration.cjs +2 -2
- package/lib/cjs/overrides/helpers.cjs +2 -2
- package/lib/cjs/overrides/index.cjs +3 -2
- package/lib/cjs/overrides/logging.cjs +5 -0
- package/lib/cjs/overrides/overrides.cjs +2 -2
- package/lib/cjs/request/DecafAuthHandler.cjs +5 -39
- package/lib/cjs/request/DecafHandlerExecutor.cjs +2 -2
- package/lib/cjs/request/DecafRequestContext.cjs +11 -20
- package/lib/cjs/request/DecafResponseInterceptor.cjs +3 -3
- package/lib/cjs/request/index.cjs +2 -2
- package/lib/cjs/swagger-types.cjs +2 -2
- package/lib/cjs/types.cjs +2 -2
- package/lib/cjs/utils.cjs +2 -2
- package/lib/cjs/webhooks/DecafWebhookModule.cjs +11 -12
- package/lib/cjs/webhooks/controllers.cjs +10 -10
- package/lib/cjs/webhooks/index.cjs +2 -15
- package/lib/cjs/webhooks/types.cjs +2 -2
- package/lib/esm/auth/AuthInterceptor.js +79 -0
- package/lib/esm/auth/DecafAuthHandler.js +31 -0
- package/lib/esm/auth/DecafAuthModule.js +48 -0
- package/lib/esm/auth/constants.js +5 -0
- package/lib/esm/auth/decorators.js +22 -0
- package/lib/esm/auth/index.js +6 -0
- package/lib/esm/bin/cli.js +1 -1
- package/lib/esm/cli-module.js +47 -5
- package/lib/esm/constants.js +3 -3
- package/lib/esm/controllers.js +10 -102
- package/lib/esm/core-module.js +7 -7
- package/lib/esm/decaf-model/DecafModelModule.js +18 -3
- package/lib/esm/decaf-model/FromModelController.js +733 -786
- package/lib/esm/decaf-model/decorators/ApiOperationFromModel.js +53 -15
- package/lib/esm/decaf-model/decorators/ApiParamsFromModel.js +1 -1
- package/lib/esm/decaf-model/decorators/DecafBody.js +1 -1
- package/lib/esm/decaf-model/decorators/DecafParams.js +1 -1
- package/lib/esm/decaf-model/decorators/controller-config.js +25 -0
- package/lib/esm/decaf-model/decorators/decorators.js +2 -43
- package/lib/esm/decaf-model/decorators/expose.js +9 -0
- package/lib/esm/decaf-model/decorators/index.js +2 -1
- package/lib/esm/decaf-model/decorators/types.js +1 -1
- package/lib/esm/decaf-model/decorators/utils.js +1 -1
- package/lib/esm/decaf-model/index.js +11 -4
- package/lib/esm/decaf-model/types.js +1 -1
- package/lib/esm/decaf-model/utils.js +1 -1
- package/lib/esm/decoration.js +1 -1
- package/lib/esm/decorators.js +79 -0
- package/lib/esm/events-module/DecafStreamModule.js +1 -1
- package/lib/esm/events-module/EventsController.js +1 -1
- package/lib/esm/events-module/constant.js +1 -1
- package/lib/esm/events-module/index.js +1 -1
- package/lib/esm/events-module/utils.js +1 -1
- package/lib/esm/factory/NestBootstraper.js +1 -1
- package/lib/esm/factory/errors/cors.js +1 -1
- package/lib/esm/factory/errors/index.js +1 -1
- package/lib/esm/factory/errors/throttling.js +1 -1
- package/lib/esm/factory/exceptions/DecafErrorFilter.js +1 -1
- package/lib/esm/factory/exceptions/decorators.js +1 -1
- package/lib/esm/factory/exceptions/index.js +1 -1
- package/lib/esm/factory/index.js +1 -1
- package/lib/esm/factory/openapi/DtoBuilder.js +36 -5
- package/lib/esm/factory/openapi/SwaggerBuilder.js +1 -1
- package/lib/esm/factory/openapi/SwaggerCustomUI.js +1 -1
- package/lib/esm/factory/openapi/constants.js +1 -1
- package/lib/esm/factory/openapi/index.js +1 -1
- package/lib/esm/index.js +7 -6
- package/lib/esm/interceptors/DecafRequestHandlerInterceptor.js +27 -21
- package/lib/esm/interceptors/index.js +2 -3
- package/lib/esm/module.js +3 -1
- package/lib/esm/overrides/Adapter.js +1 -1
- package/lib/esm/overrides/ModelBuilderExtensions.js +1 -1
- package/lib/esm/overrides/constants.js +1 -1
- package/lib/esm/overrides/decoration.js +1 -1
- package/lib/esm/overrides/helpers.js +1 -1
- package/lib/esm/overrides/index.js +2 -1
- package/lib/esm/overrides/logging.js +2 -0
- package/lib/esm/overrides/overrides.js +1 -1
- package/lib/esm/request/DecafAuthHandler.js +2 -36
- package/lib/esm/request/DecafHandlerExecutor.js +1 -1
- package/lib/esm/request/DecafRequestContext.js +10 -19
- package/lib/esm/request/DecafResponseInterceptor.js +2 -2
- package/lib/esm/request/index.js +1 -1
- package/lib/esm/swagger-types.js +1 -1
- package/lib/esm/types.js +1 -1
- package/lib/esm/utils.js +1 -1
- package/lib/esm/webhooks/DecafWebhookModule.js +5 -6
- package/lib/esm/webhooks/controllers.js +2 -2
- package/lib/esm/webhooks/index.js +1 -2
- package/lib/esm/webhooks/types.js +1 -1
- package/lib/types/{interceptors → auth}/AuthInterceptor.d.cts +4 -1
- package/lib/types/{interceptors → auth}/AuthInterceptor.d.mts +4 -1
- package/lib/types/auth/DecafAuthHandler.d.cts +22 -0
- package/lib/types/auth/DecafAuthHandler.d.mts +22 -0
- package/lib/types/auth/DecafAuthModule.d.cts +9 -0
- package/lib/types/auth/DecafAuthModule.d.mts +9 -0
- package/lib/types/auth/constants.d.cts +4 -0
- package/lib/types/auth/constants.d.mts +4 -0
- package/lib/types/auth/decorators.d.cts +4 -0
- package/lib/types/auth/decorators.d.mts +4 -0
- package/lib/types/auth/index.d.cts +6 -0
- package/lib/types/auth/index.d.mts +6 -0
- package/lib/types/cli-module.d.cts +1 -4
- package/lib/types/cli-module.d.mts +1 -4
- package/lib/types/constants.d.cts +2 -2
- package/lib/types/constants.d.mts +2 -2
- package/lib/types/controllers.d.cts +7 -12
- package/lib/types/controllers.d.mts +7 -12
- package/lib/types/decaf-model/DecafModelModule.d.cts +1 -0
- package/lib/types/decaf-model/DecafModelModule.d.mts +1 -0
- package/lib/types/decaf-model/FromModelController.d.cts +40 -52
- package/lib/types/decaf-model/FromModelController.d.mts +40 -52
- package/lib/types/decaf-model/decorators/ApiOperationFromModel.d.cts +0 -2
- package/lib/types/decaf-model/decorators/ApiOperationFromModel.d.mts +0 -2
- package/lib/types/decaf-model/decorators/controller-config.d.cts +18 -0
- package/lib/types/decaf-model/decorators/controller-config.d.mts +18 -0
- package/lib/types/decaf-model/decorators/decorators.d.cts +1 -29
- package/lib/types/decaf-model/decorators/decorators.d.mts +1 -29
- package/lib/types/decaf-model/decorators/expose.d.cts +1 -0
- package/lib/types/decaf-model/decorators/expose.d.mts +1 -0
- package/lib/types/decaf-model/decorators/index.d.cts +1 -0
- package/lib/types/decaf-model/decorators/index.d.mts +1 -0
- package/lib/types/decaf-model/decorators/types.d.cts +8 -21
- package/lib/types/decaf-model/decorators/types.d.mts +8 -21
- package/lib/types/decaf-model/index.d.cts +10 -3
- package/lib/types/decaf-model/index.d.mts +10 -3
- package/lib/types/decaf-model/utils.d.cts +2 -1
- package/lib/types/decaf-model/utils.d.mts +2 -1
- package/lib/types/decorators.d.cts +34 -0
- package/lib/types/decorators.d.mts +34 -0
- package/lib/types/factory/openapi/DtoBuilder.d.cts +7 -3
- package/lib/types/factory/openapi/DtoBuilder.d.mts +7 -3
- package/lib/types/index.d.cts +5 -5
- package/lib/types/index.d.mts +5 -5
- package/lib/types/interceptors/DecafRequestHandlerInterceptor.d.cts +1 -2
- package/lib/types/interceptors/DecafRequestHandlerInterceptor.d.mts +1 -2
- package/lib/types/interceptors/index.d.cts +1 -2
- package/lib/types/interceptors/index.d.mts +1 -2
- package/lib/types/overrides/Adapter.d.cts +1 -1
- package/lib/types/overrides/Adapter.d.mts +1 -1
- package/lib/types/overrides/index.d.cts +1 -0
- package/lib/types/overrides/index.d.mts +1 -0
- package/lib/types/overrides/logging.d.cts +1 -0
- package/lib/types/overrides/logging.d.mts +1 -0
- package/lib/types/request/DecafAuthHandler.d.cts +1 -10
- package/lib/types/request/DecafAuthHandler.d.mts +1 -10
- package/lib/types/request/DecafRequestContext.d.cts +4 -5
- package/lib/types/request/DecafRequestContext.d.mts +4 -5
- package/lib/types/types.d.cts +12 -24
- package/lib/types/types.d.mts +12 -24
- package/lib/types/webhooks/controllers.d.cts +1 -1
- package/lib/types/webhooks/controllers.d.mts +1 -1
- package/lib/types/webhooks/index.d.cts +0 -1
- package/lib/types/webhooks/index.d.mts +0 -1
- package/package.json +5 -13
- package/dist/for-nest.cjs.map +0 -1
- package/dist/for-nest.js.map +0 -1
- package/lib/cjs/bin/cli.cjs.map +0 -1
- package/lib/cjs/cli-module.cjs.map +0 -1
- package/lib/cjs/constants.cjs.map +0 -1
- package/lib/cjs/controllers.cjs.map +0 -1
- package/lib/cjs/core-module.cjs.map +0 -1
- package/lib/cjs/decaf-model/DecafModelModule.cjs.map +0 -1
- package/lib/cjs/decaf-model/FromModelController.cjs.map +0 -1
- package/lib/cjs/decaf-model/decorators/ApiOperationFromModel.cjs.map +0 -1
- package/lib/cjs/decaf-model/decorators/ApiParamsFromModel.cjs.map +0 -1
- package/lib/cjs/decaf-model/decorators/DecafBody.cjs.map +0 -1
- package/lib/cjs/decaf-model/decorators/DecafParams.cjs.map +0 -1
- package/lib/cjs/decaf-model/decorators/decorators.cjs.map +0 -1
- package/lib/cjs/decaf-model/decorators/index.cjs.map +0 -1
- package/lib/cjs/decaf-model/decorators/types.cjs.map +0 -1
- package/lib/cjs/decaf-model/decorators/utils.cjs.map +0 -1
- package/lib/cjs/decaf-model/index.cjs.map +0 -1
- package/lib/cjs/decaf-model/types.cjs.map +0 -1
- package/lib/cjs/decaf-model/utils.cjs.map +0 -1
- package/lib/cjs/decoration.cjs.map +0 -1
- package/lib/cjs/events-module/DecafStreamModule.cjs.map +0 -1
- package/lib/cjs/events-module/EventsController.cjs.map +0 -1
- package/lib/cjs/events-module/constant.cjs.map +0 -1
- package/lib/cjs/events-module/index.cjs.map +0 -1
- package/lib/cjs/events-module/utils.cjs.map +0 -1
- package/lib/cjs/factory/NestBootstraper.cjs.map +0 -1
- package/lib/cjs/factory/errors/cors.cjs.map +0 -1
- package/lib/cjs/factory/errors/index.cjs.map +0 -1
- package/lib/cjs/factory/errors/throttling.cjs.map +0 -1
- package/lib/cjs/factory/exceptions/DecafErrorFilter.cjs.map +0 -1
- package/lib/cjs/factory/exceptions/decorators.cjs.map +0 -1
- package/lib/cjs/factory/exceptions/index.cjs.map +0 -1
- package/lib/cjs/factory/index.cjs.map +0 -1
- package/lib/cjs/factory/openapi/DtoBuilder.cjs.map +0 -1
- package/lib/cjs/factory/openapi/SwaggerBuilder.cjs.map +0 -1
- package/lib/cjs/factory/openapi/SwaggerCustomUI.cjs.map +0 -1
- package/lib/cjs/factory/openapi/constants.cjs.map +0 -1
- package/lib/cjs/factory/openapi/index.cjs.map +0 -1
- package/lib/cjs/index.cjs.map +0 -1
- package/lib/cjs/interceptors/AuthInterceptor.cjs +0 -52
- package/lib/cjs/interceptors/AuthInterceptor.cjs.map +0 -1
- package/lib/cjs/interceptors/DecafRequestHandlerInterceptor.cjs.map +0 -1
- package/lib/cjs/interceptors/context.cjs +0 -18
- package/lib/cjs/interceptors/context.cjs.map +0 -1
- package/lib/cjs/interceptors/index.cjs.map +0 -1
- package/lib/cjs/migrations/index.cjs +0 -35
- package/lib/cjs/migrations/index.cjs.map +0 -1
- package/lib/cjs/migrations/migration-module.cjs +0 -60
- package/lib/cjs/migrations/migration-module.cjs.map +0 -1
- package/lib/cjs/migrations/migration-options.cjs +0 -4
- package/lib/cjs/migrations/migration-options.cjs.map +0 -1
- package/lib/cjs/module.cjs.map +0 -1
- package/lib/cjs/overrides/Adapter.cjs.map +0 -1
- package/lib/cjs/overrides/ModelBuilderExtensions.cjs.map +0 -1
- package/lib/cjs/overrides/constants.cjs.map +0 -1
- package/lib/cjs/overrides/decoration.cjs.map +0 -1
- package/lib/cjs/overrides/helpers.cjs.map +0 -1
- package/lib/cjs/overrides/index.cjs.map +0 -1
- package/lib/cjs/overrides/overrides.cjs.map +0 -1
- package/lib/cjs/ram/RamRequestTransformer.cjs +0 -26
- package/lib/cjs/ram/RamRequestTransformer.cjs.map +0 -1
- package/lib/cjs/ram/index.cjs +0 -19
- package/lib/cjs/ram/index.cjs.map +0 -1
- package/lib/cjs/request/DecafAuthHandler.cjs.map +0 -1
- package/lib/cjs/request/DecafHandlerExecutor.cjs.map +0 -1
- package/lib/cjs/request/DecafRequestContext.cjs.map +0 -1
- package/lib/cjs/request/DecafResponseInterceptor.cjs.map +0 -1
- package/lib/cjs/request/index.cjs.map +0 -1
- package/lib/cjs/swagger-types.cjs.map +0 -1
- package/lib/cjs/types.cjs.map +0 -1
- package/lib/cjs/utils.cjs.map +0 -1
- package/lib/cjs/webhooks/DecafWebhookModule.cjs.map +0 -1
- package/lib/cjs/webhooks/controllers.cjs.map +0 -1
- package/lib/cjs/webhooks/index.cjs.map +0 -1
- package/lib/cjs/webhooks/types.cjs.map +0 -1
- package/lib/esm/bin/cli.js.map +0 -1
- package/lib/esm/cli-module.js.map +0 -1
- package/lib/esm/constants.js.map +0 -1
- package/lib/esm/controllers.js.map +0 -1
- package/lib/esm/core-module.js.map +0 -1
- package/lib/esm/decaf-model/DecafModelModule.js.map +0 -1
- package/lib/esm/decaf-model/FromModelController.js.map +0 -1
- package/lib/esm/decaf-model/decorators/ApiOperationFromModel.js.map +0 -1
- package/lib/esm/decaf-model/decorators/ApiParamsFromModel.js.map +0 -1
- package/lib/esm/decaf-model/decorators/DecafBody.js.map +0 -1
- package/lib/esm/decaf-model/decorators/DecafParams.js.map +0 -1
- package/lib/esm/decaf-model/decorators/decorators.js.map +0 -1
- package/lib/esm/decaf-model/decorators/index.js.map +0 -1
- package/lib/esm/decaf-model/decorators/types.js.map +0 -1
- package/lib/esm/decaf-model/decorators/utils.js.map +0 -1
- package/lib/esm/decaf-model/index.js.map +0 -1
- package/lib/esm/decaf-model/types.js.map +0 -1
- package/lib/esm/decaf-model/utils.js.map +0 -1
- package/lib/esm/decoration.js.map +0 -1
- package/lib/esm/events-module/DecafStreamModule.js.map +0 -1
- package/lib/esm/events-module/EventsController.js.map +0 -1
- package/lib/esm/events-module/constant.js.map +0 -1
- package/lib/esm/events-module/index.js.map +0 -1
- package/lib/esm/events-module/utils.js.map +0 -1
- package/lib/esm/factory/NestBootstraper.js.map +0 -1
- package/lib/esm/factory/errors/cors.js.map +0 -1
- package/lib/esm/factory/errors/index.js.map +0 -1
- package/lib/esm/factory/errors/throttling.js.map +0 -1
- package/lib/esm/factory/exceptions/DecafErrorFilter.js.map +0 -1
- package/lib/esm/factory/exceptions/decorators.js.map +0 -1
- package/lib/esm/factory/exceptions/index.js.map +0 -1
- package/lib/esm/factory/index.js.map +0 -1
- package/lib/esm/factory/openapi/DtoBuilder.js.map +0 -1
- package/lib/esm/factory/openapi/SwaggerBuilder.js.map +0 -1
- package/lib/esm/factory/openapi/SwaggerCustomUI.js.map +0 -1
- package/lib/esm/factory/openapi/constants.js.map +0 -1
- package/lib/esm/factory/openapi/index.js.map +0 -1
- package/lib/esm/index.js.map +0 -1
- package/lib/esm/interceptors/AuthInterceptor.js +0 -43
- package/lib/esm/interceptors/AuthInterceptor.js.map +0 -1
- package/lib/esm/interceptors/DecafRequestHandlerInterceptor.js.map +0 -1
- package/lib/esm/interceptors/context.js +0 -12
- package/lib/esm/interceptors/context.js.map +0 -1
- package/lib/esm/interceptors/index.js.map +0 -1
- package/lib/esm/migrations/index.js +0 -18
- package/lib/esm/migrations/index.js.map +0 -1
- package/lib/esm/migrations/migration-module.js +0 -52
- package/lib/esm/migrations/migration-module.js.map +0 -1
- package/lib/esm/migrations/migration-options.js +0 -7
- package/lib/esm/migrations/migration-options.js.map +0 -1
- package/lib/esm/module.js.map +0 -1
- package/lib/esm/overrides/Adapter.js.map +0 -1
- package/lib/esm/overrides/ModelBuilderExtensions.js.map +0 -1
- package/lib/esm/overrides/constants.js.map +0 -1
- package/lib/esm/overrides/decoration.js.map +0 -1
- package/lib/esm/overrides/helpers.js.map +0 -1
- package/lib/esm/overrides/index.js.map +0 -1
- package/lib/esm/overrides/overrides.js.map +0 -1
- package/lib/esm/ram/RamRequestTransformer.js +0 -21
- package/lib/esm/ram/RamRequestTransformer.js.map +0 -1
- package/lib/esm/ram/index.js +0 -2
- package/lib/esm/ram/index.js.map +0 -1
- package/lib/esm/request/DecafAuthHandler.js.map +0 -1
- package/lib/esm/request/DecafHandlerExecutor.js.map +0 -1
- package/lib/esm/request/DecafRequestContext.js.map +0 -1
- package/lib/esm/request/DecafResponseInterceptor.js.map +0 -1
- package/lib/esm/request/index.js.map +0 -1
- package/lib/esm/swagger-types.js.map +0 -1
- package/lib/esm/types.js.map +0 -1
- package/lib/esm/utils.js.map +0 -1
- package/lib/esm/webhooks/DecafWebhookModule.js.map +0 -1
- package/lib/esm/webhooks/controllers.js.map +0 -1
- package/lib/esm/webhooks/index.js.map +0 -1
- package/lib/esm/webhooks/types.js.map +0 -1
- package/lib/types/interceptors/context.d.cts +0 -6
- package/lib/types/interceptors/context.d.mts +0 -6
- package/lib/types/migrations/index.d.cts +0 -17
- package/lib/types/migrations/index.d.mts +0 -17
- package/lib/types/migrations/migration-module.d.cts +0 -15
- package/lib/types/migrations/migration-module.d.mts +0 -15
- package/lib/types/migrations/migration-options.d.cts +0 -11
- package/lib/types/migrations/migration-options.d.mts +0 -11
- package/lib/types/ram/RamRequestTransformer.d.cts +0 -5
- package/lib/types/ram/RamRequestTransformer.d.mts +0 -5
- package/lib/types/ram/index.d.cts +0 -1
- package/lib/types/ram/index.d.mts +0 -1
package/dist/for-nest.js
CHANGED
|
@@ -1,2 +1,2738 @@
|
|
|
1
|
-
import{apply as e,Metadata as t,Decoration as r,DecorationKeys as o,metadata as s,uses as i}from"@decaf-ts/decoration";import{InjectablesKeys as n,DefaultInjectablesConfig as a}from"@decaf-ts/injectable-decorators";import{Delete as c,Patch as p,Put as d,Post as l,Get as u,applyDecorators as h,createParamDecorator as f,InternalServerErrorException as g,Injectable as m,Optional as y,Inject as b,UseInterceptors as v,SetMetadata as w,Scope as A,Query as E,Param as P,Response as R,Controller as T,Module as O,Catch as x,NotFoundException as S,NotAcceptableException as $,UseFilters as _,Logger as C,Global as k,Sse as I}from"@nestjs/common";import{Model as N,ValidationKeys as F,ModelBuilder as D}from"@decaf-ts/decorator-validation";import{AuthorizationError as q,PersistenceKeys as M,UUID as L,OrderDirection as j,ModelService as U,Service as B,Repository as G,ContextualLoggedClass as z,TransactionOperationKeys as H,Adapter as Y,Context as J,PreparedStatementKeys as X,ForbiddenError as W,UnsupportedError as Q,DefaultAdapterFlags as V,PersistenceService as K}from"@decaf-ts/core";import{isArray as Z,isString as ee,pickBy as te,negate as re}from"lodash";import{ApiExcludeEndpoint as oe,ApiParam as se,ApiBearerAuth as ie,ApiQuery as ne,ApiOperation as ae,ApiOkResponse as ce,ApiNoContentResponse as pe,ApiNotFoundResponse as de,ApiBody as le,getSchemaPath as ue,ApiCreatedResponse as he,ApiBadRequestResponse as fe,ApiUnprocessableEntityResponse as ge,ApiTags as me,ApiExtraModels as ye,DocumentBuilder as be,SwaggerModule as ve}from"@nestjs/swagger";import{OperationKeys as we,BulkCrudOperationKeys as Ae,InternalError as Ee,DBKeys as Pe,ValidationError as Re,BaseError as Te}from"@decaf-ts/db-decorators";import{__decorate as Oe,__param as xe,__metadata as Se}from"tslib";import{Reflector as $e,REQUEST as _e,ModuleRef as Ce,APP_INTERCEPTOR as ke,RouterModule as Ie}from"@nestjs/core";import{Logging as Ne,toPascalCase as Fe,toKebabCase as De,LoggedEnvironment as qe}from"@decaf-ts/logging";import{tap as Me,Observable as Le,interval as je,merge as Ue}from"rxjs";import{readFileSync as Be}from"fs";import*as Ge from"path";import ze from"yaml";import{tap as He,map as Ye}from"rxjs/operators";import{MigrationService as Je}from"@decaf-ts/core/migrations";import{WebhookSubscription as Xe,WebhookEventRecord as We,WebhookDelivery as Qe,collectPagedResults as Ve,WebhookStatus as Ke}from"@decaf-ts/for-http/server";export{WebhookDelivery,WebhookDeliveryService,WebhookEventRecord,WebhookPublisherService,WebhookSignatureMiddleware,WebhookStatus,WebhookSubscription,WebhookSubscriptionService,computeNextAttempt,signWebhookPayload,verifyWebhookSignature}from"@decaf-ts/for-http/server";const Ze="swagger",et={API_OPERATION:Ze+"/apiOperation",API_RESPONSE:Ze+"/apiResponse",API_PRODUCES:Ze+"/apiProduces",API_CONSUMES:Ze+"/apiConsumes",API_TAGS:Ze+"/apiUseTags",API_CALLBACKS:Ze+"/apiCallbacks",API_PARAMETERS:Ze+"/apiParameters",API_HEADERS:Ze+"/apiHeaders",API_MODEL_PROPERTIES:Ze+"/apiModelProperties",API_MODEL_PROPERTIES_ARRAY:Ze+"/apiModelPropertiesArray",API_SECURITY:Ze+"/apiSecurity",API_EXCLUDE_ENDPOINT:Ze+"/apiExcludeEndpoint",API_EXCLUDE_CONTROLLER:Ze+"/apiExcludeController",API_EXTRA_MODELS:Ze+"/apiExtraModels",API_EXTENSION:Ze+"/apiExtension",API_SCHEMA:Ze+"/apiSchema",API_DEFAULT_GETTER:Ze+"/apiDefaultGetter",API_LINK:Ze+"/apiLink"};function tt(e,t){if(!e)return[e,t];if(t)return[e,t];const r=Z(e);return[r?e[0]:e,r]}const rt=e=>void 0===e;function ot(e){if("function"==typeof e)return ot(e());if(Array.isArray(e))return e;if("object"!=typeof e)return[];const t=Object.values(e).filter(e=>"number"==typeof e).map(e=>e.toString());return Object.keys(e).filter(e=>!t.includes(e)).map(t=>e[t])}function st(e){return e.filter(ee).length>0?"string":"number"}function it(e,t,r=!0){return(o,s)=>{const i=Reflect.getMetadata(et.API_MODEL_PROPERTIES_ARRAY,o)||[],n=":"+s;i.includes(n)||Reflect.defineMetadata(et.API_MODEL_PROPERTIES_ARRAY,[...i,":"+s],o);const a=Reflect.getMetadata(e,o,s);if(a){const i=te(t,re(rt)),n=r?{...a,...i}:{...i,...a};Reflect.defineMetadata(e,n,o,s)}else{const r=o?.constructor?._OPENAPI_METADATA_FACTORY?.()[s]?.type??Reflect.getMetadata("design:type",o,s);Reflect.defineMetadata(e,{type:r,required:!1,...te(t,re(rt))},o,s)}}}function nt(e={}){return at(e)}function at(e={},t=!0){const[r,o]=tt(e.type,e.isArray);if((s=e={...e,type:r,isArray:o}).isArray&&"enum"in s&&void 0!==s.enum){e.type="array";const t=ot(e.enum);e.items={type:st(t),enum:t},delete e.enum}else if("enum"in e&&void 0!==e.enum){const t=ot(e.enum);e.enum=t,e.type=st(t)}var s;return Array.isArray(e.type)&&(e.type="array",e.items={type:"array",items:{type:e.type[0]}}),it(et.API_MODEL_PROPERTIES,e,t)}function ct(e,r){const{handler:o,args:s}=t.get(e,we.REFLECT+we.BLOCK)||{};return!!o&&(o(...s,r)??!1)}function pt(t,r,o){const s={GET:[we.READ,u],POST:[we.CREATE,l],PUT:[we.UPDATE,d],PATCH:[we.UPDATE,p],DELETE:[we.DELETE,c]},[i,n]=s[r];return ct(t,i)?e(oe()):e(n(o))}function dt(t,r,o){const s={GET:[Ae.READ_ALL,u],POST:[Ae.CREATE_ALL,l],PUT:[Ae.UPDATE_ALL,d],PATCH:[Ae.UPDATE_ALL,p],DELETE:[Ae.DELETE_ALL,c]},[i,n]=s[r];return ct(t,i)?e(oe()):e(n(o))}function lt(e=[]){const t=e.map(e=>se({name:e.name,description:e.description??"Path parameter: "+e.name,required:e.required??!0,type:e.type??String}));return h(...t)}const ut=f((e,t)=>{const r=t.switchToHttp().getRequest().body,o=t.getClass(),s=o.class;if(!s)throw new g(`ModelConstructor not found on controller ${o.name}. Ensure the controller was created via FromModelController.`);return r?Array.isArray(r)?r.map(e=>new s(e)):new s(r):r}),ht=f((e,t)=>{const r=t.switchToHttp().getRequest(),o=r?.params??{},s=e??Object.keys(o),i=s.map(e=>o[e]);return{raw:o,valuesInOrder:i,keysInOrder:s}});function ft(e=[]){const t=e.map(e=>e.name);return ht(t)}const gt=f((e,t)=>{const r={...t.switchToHttp().getRequest().query??{}};if(void 0!==r.limit){const e=Number(r.limit);Number.isNaN(e)||(r.limit=e)}if(void 0!==r.offset){const e=Number(r.offset);Number.isNaN(e)||(r.offset=e)}if(void 0!==r.bookmark){const e=Number(r.bookmark);r.bookmark=Number.isNaN(e)?r.bookmark:e}return r});function mt(e){const t={GET:u,POST:l,PUT:d,PATCH:p,DELETE:c}[e];if(!t)throw Error(`Unsupported HTTP verb "${e}". No NestJS decorator mapping was found.`);return t}const yt="DecafModuleOptions",bt="DecafAdapter",vt="DecafTaskService",wt="DecafRoute",At=Symbol("DecafHandlers"),Et=Symbol("DecafAdapterForOptions"),Pt=Symbol("AUTH_HANDLER"),Rt="auth:meta",Tt=Symbol("decaf:context");let Ot=class{constructor(e,t){this.reflector=e,this.authHandler=t}async intercept(e,t){const r=Ne.for(this).for(this.intercept),o=this.reflector.get(Rt,e.getHandler())??this.reflector.get(Rt,e.getClass());return r.verbose("Intercepted request"+(o?" for "+o:"")),this.authHandler?await this.authHandler.authorize(e,o):r.debug("No auth handler/model"),t.handle()}};function xt(e){const t=e?"string"==typeof e?e:e.name:void 0,r=[ie(),v(Ot)];return t&&r.push(w(Rt,t)),h(...r)}Ot=Oe([m(),xe(1,y()),xe(1,b(Pt)),Se("design:paramtypes",[$e,Object])],Ot);class St{parseRequest(e){const t=e.headers.authorization?.split(" ")[1];return t}async authorize(e,r){const o=e.switchToHttp().getRequest(),s=this.parseRequest(o);if(!s)throw new q("Unauthenticated");if(!t.get(N.get(r),M.AUTH_ROLE).includes(s))throw new q("Missing role: "+s)}}class $t extends St{constructor(){super()}async authorize(e,r){const o=e.switchToHttp().getRequest(),s=this.parseRequest(o);if(!s)throw new q("Unauthenticated");const i=t.get(N.get(r),M.AUTH_ROLE);if(!i.includes(s))throw new q("Missing role: "+s);return i}}let _t=class{constructor(e){this.req=e,this.uuid=L.instance.generate()}get request(){return this.req}put(e){let t;try{t=this.ctx.get("overrides")}catch(e){t={}}this._ctx=this.ctx.accumulate({overrides:Object.assign(t,e)})}applyCtx(e){if(this._ctx)throw new Ee("Trying to overwrite context");this._ctx=e}get ctx(){if(!this._ctx)throw new Ee("Context not initialized for request");return this._ctx}};var Ct;_t=Oe([m({scope:A.REQUEST}),xe(0,b(_e)),Se("design:paramtypes",[Request])],_t);let kt=Ct=class{constructor(e,t){this.handlers=e,this.context=t}async exec(e,t){Ne.for(Ct.name).for(this.exec).debug(`CONTEXT ${this.context.uuid} running ${this.handlers.length} handlers for request ${e.method} ${e.url}`);for(const r of this.handlers)await r.handle(this.context,e,t)}};kt=Ct=Oe([m({scope:A.REQUEST}),xe(0,b(At)),Se("design:paramtypes",[Array,_t])],kt);let It=class{constructor(e){this.ctx=e}intercept(e,t){let r=e.switchToHttp().getResponse();return t.handle().pipe(Me(e=>{r=this.ctx.ctx.toResponse(r)}))}};It=Oe([m(),Se("design:paramtypes",[_t])],It);const Nt=e=>({name:e,description:e+" parameter for the query",required:!0,type:String});function Ft(e,t,r,o=!1){const s=mt(r),i=(e=>e.split("/").filter(e=>e.startsWith(":")).map(e=>e.slice(1)))(t).map(Nt),n=[];return"GET"===r&&o&&n.push(ne({name:"direction",required:!1,enum:j,description:"the sort order when applicable"}),ne({name:"limit",required:!1,description:"limit or page size when applicable"}),ne({name:"offset",required:!1,description:"offset or bookmark when applicable"})),{method:[s(t),...i.map(se),...n,ae({summary:`Retrieve records using according to "${e}".`}),ce({description:"Result successfully retrieved."}),pe({description:"No content returned by the method."})],params:[ft(i),E()]}}function Dt(e,t,r,o){const s=e?.prototype??e;o.method.forEach(e=>e(s,t,r)),o.params?.forEach((e,r)=>e(s,t,r))}function qt(e,t,...r){if(e instanceof U)return"function"==typeof e[t]?e[t](...r):e.statement(t,...r);if("function"==typeof e[t])return e[t](...r);throw Error(`Persistence method "${t}" not found on ${e?.constructor?.name}`)}function Mt(e){return async function(t,r){const o=this.log.for(e);try{o.debug(`Invoking persistence method "${e}" given parameters: ${JSON.stringify(t.valuesInOrder)}`);const{direction:s,limit:i,offset:n}=r;return await qt(this.persistence(),e,...t.valuesInOrder,s,i,n)}catch(t){throw o.error(`Custom query "${e}" failed`,t),t}}}function Lt(e,t,r){return Object.defineProperty(e.prototype||e,t,{value:r,writable:!1}),Object.getOwnPropertyDescriptor(e.prototype||e,t)}class jt extends B{constructor(e,t){super(),this.clientContext=e}logCtx(e,t,r=!1,o){const s=this.clientContext.ctx;let i;e=e.filter(e=>void 0!==e),o&&(o.headers||o.ip)&&(i=o,o={});const n=z.logCtx.call(this,t,o||{},r,...e,s);return this.bindLoggerToRequest(n,i)}bindLoggerToRequest(e,t){const r=this;function o(e,t){return e.log=((e,t)=>{const o=(e=>{if(!e)return;const t=e.headers,r=Bt(t?.["x-forwarded-for"])??Bt(t?.["x-real-ip"])??Bt(t?.["X-Forwarded-For"])??Bt(t?.["X-Real-IP"]);if(r)return r;if("string"==typeof e.ip&&e.ip.length)return e.ip;const o=e.socket||e.connection;return o&&"string"==typeof o.remoteAddress&&o.remoteAddress.length?o.remoteAddress:void 0})(t??r.clientContext.request);if(!o)return e;const s={ip:o};return e.for(s)})(e.log,t),e}return(e=>"object"==typeof e&&null!==e&&"function"==typeof e.then)(e)?e.then(e=>o(e,t)):o(e,t)}}class Ut extends jt{persistence(e){if(!this._persistence)try{this._persistence=B.get(this.class)}catch(e){try{this._persistence=U.getService(this.class)}catch(e){this._persistence=G.forModel(this.class)}}const t=this.clientContext.request[Et]||{};return e&&this.clientContext.put(t),e?this._persistence instanceof G?this._persistence.override(t):this._persistence.for(t):this._persistence}constructor(e,t){super(e,t)}logCtx(e,t,r=!1,o){const s=this.clientContext.ctx;let i;o&&(o.headers||o.ip)&&(i=o,o={});try{o=s.get("overrides")}catch(e){}const n=this.persistence(s);let a,c;return n instanceof U?a=n.repo._adapter:n instanceof G?a=n._adapter:n.context&&(a=n),r?a.logCtx(e,t,!0,o).then(e=>this.bindLoggerToRequest(e,i)):(c=a.logCtx(e,t,!1,o),this.bindLoggerToRequest(c,i))}}function Bt(e){if(e)return(Array.isArray(e)?e[0]:e).split(",").map(e=>e.trim()).filter(Boolean)[0]}const Gt=new Map;function zt(e,r){if(!H.includes(e))return r;const o=(e=>(Gt.has(e)||Gt.set(e,new WeakMap),Gt.get(e)))(e),s=o.get(r);if(s)return s;const i=[we.UPDATE,Ae.UPDATE_ALL].includes(e);class n{}o.set(r,n),Object.defineProperty(n,"name",{value:`${Fe(r.name)}${Fe(e)}DTO`});const a=(()=>{try{return N.pk(r)}catch{return}})(),c=a?N.pkProps(r):void 0,p=a?Reflect.getMetadata("design:type",r.prototype,a):void 0,d=p===Number||p===BigInt,l=!!c?.generated||!!a&&(N.generatedBySequence(r,a)||Jt(r,a)||d),u=Array.from(new Set(t.properties(r)||[])),h=new Set(N.relations(r)||[]),f=[];for(const e of u)e&&(h.has(e)||e===a&&!i&&l||e!==a&&Jt(r,e)||f.push(e));for(const e of f){const t=Xt(r,e),o=!!t?.[F.REQUIRED],s=Wt(r,e)??Reflect.getMetadata("design:type",r.prototype,e),i={required:o};s&&(i.type=s),nt(i)(n.prototype,e);const a=Reflect.getMetadata("design:type",r.prototype,e)??s;void 0!==a&&Reflect.defineMetadata("design:type",a,n.prototype,e),Object.defineProperty(n.prototype,e,{value:void 0,writable:!0,enumerable:!0,configurable:!0})}for(const o of h){const s=t.relations(r,o);if(!s)throw new Ee(`Metadata for relation ${o} not found`);let a=s.class;if("function"!=typeof a||a.name||(a=a()),!a||"function"!=typeof a)throw new Ee(`Type for relation ${o} not found`);if(!N.get(a.name))continue;const c=t.validationFor(r,o),p=!!c?.[F.LIST],d=!!c?.[F.REQUIRED],l=zt(e,a);i?Yt(n,o,a,l,p,d):Ht(n,o,l,p,d)}return n}function Ht(e,t,r,o,s){nt({type:r,required:s,isArray:o})(e.prototype,t),Reflect.defineMetadata("design:type",o?Array:r,e.prototype,t),Object.defineProperty(e.prototype,t,{value:void 0,writable:!0,enumerable:!0,configurable:!0})}function Yt(e,t,r,o,s,i){const n=Reflect.getMetadata(et.API_EXTRA_MODELS,e)||[];n.includes(o)||Reflect.defineMetadata(et.API_EXTRA_MODELS,[...n,o],e);const a="#/components/schemas/"+o.name,c=(e=>{try{const t=N.pkProps(e),r=t?.type;return r===Number||r===BigInt?"integer":"string"}catch{return"string"}})(r),p=[{$ref:a},{type:c}];nt(s?{type:"array",required:i,oneOf:p}:{type:Object,required:i,oneOf:p})(e.prototype,t),Reflect.defineMetadata("design:type",s?Array:Object,e.prototype,t),Object.defineProperty(e.prototype,t,{value:void 0,writable:!0,enumerable:!0,configurable:!0})}function Jt(e,t){let r=e;for(;r&&r!==Object&&r!==Function;){if(N.generated(r,t))return!0;r=Object.getPrototypeOf(r)}return!1}function Xt(e,r){let o=e;for(;o&&o!==Object&&o!==Function;){const e=t.validationFor(o,r);if(e)return e;o=Object.getPrototypeOf(o)}}function Wt(e,r){let o=e;for(;o&&o!==Object&&o!==Function;){const e=t.type(o,r);if(e)return e;o=Object.getPrototypeOf(o)}}Y.transformerFor=(e=>{const r="string"==typeof e?e:e.alias;return t.innerGet(Symbol.for("transformers"),r)}).bind(Y),Y.flavoursToTransform=(()=>{const e=t.innerGet(Symbol.for("transformers"));if(e)return Object.keys(e)}).bind(Y),J.prototype.toResponse=function(e){const t=this.pending();return t&&e.header("x-pending-task",JSON.stringify(t)),e};const Qt=D.prototype;if(Qt.decorateClass||(Qt.decorateClass=function(e){return this._classDecorators||(this._classDecorators=[]),this._classDecorators.push(e),this}),Qt.Auth=function(e){return this.decorateClass(xt(e))},!Qt.__hasClassDecoratorSupport){const e=Qt.build;Qt.build=function(){let t=e.call(this);const r=this._classDecorators;if(r?.length)for(const e of r){const r=e(t);r&&(t=r)}return t},Qt.__hasClassDecoratorSupport=!0}class Vt{static{this.log=Ne.for(Vt.name)}static getPersistence(e){try{return B.get(e)}catch(t){try{return U.getService(e)}catch(t){return G.forModel(e)}}}static createQueryRoutesFromRepository(e,r=M.QUERY){const o=Vt.log.for(Vt.createQueryRoutesFromRepository),s=e instanceof U?e.repo:e,i=s.class,n=t.get(s.constructor,t.key(M.QUERY))??{},a=t.get(e.constructor,t.key(wt))??{};class c extends Ut{get class(){throw Error("Method not implemented.")}constructor(e,t){super(e,t)}}for(const[e,t]of Object.entries(a)){const r=[t.path.replace(/^\/+|\/+$/g,"")].filter(e=>e&&e.trim()).join("/"),s=Mt(e);if(!s){const t=`Invalid or missing handler for model ${i.name} on decorated method ${e}`;throw o.error(t),Error(t)}const n=Lt(c,e,s);n&&Dt(c,e,n,Ft(e,r,t.httpMethod))}for(const[e,t]of Object.entries(n)){const o=[r,e,...(t.fields??[]).map(e=>":"+e)].filter(e=>e&&e.trim()).join("/"),s=Lt(c,e,Mt(e));s&&Dt(c,e,s,Ft(e,o,"GET",!0))}return c}static create(e){const t=Vt.log.for(Vt.create),r=N.tableName(e),o=De(r),s=e.name,i=Vt.getPersistence(e),{description:n,getPK:a,apiProperties:c,path:p}=Vt.getRouteParametersFromModel(e);t.debug("Creating controller for model: "+s);const d=Vt.createQueryRoutesFromRepository(i);let l=class extends d{static get class(){return e}get class(){return e}constructor(r){super(r),this.pk=N.pk(e),t.info(`Registering dynamic controller for model: ${this.class.name} route: /${o}`)}async listBy(e,t){const{ctx:r}=(await this.logCtx([],X.LIST_BY,!0)).for(this.listBy);return this.persistence(r).listBy(e,t.direction,r)}async paginateBy(e,t){const{ctx:r}=(await this.logCtx([],X.PAGE_BY,!0)).for(this.paginateBy);return this.persistence(r).paginateBy(e,t.direction,t,r)}async find(e,t){const{ctx:r}=(await this.logCtx([],X.FIND,!0)).for(this.find),o=t.direction??j.ASC;return qt(this.persistence(r),this.find.name,e,o,r)}async page(e,t){const{ctx:r}=(await this.logCtx([],X.PAGE,!0)).for(this.page),{direction:o=j.ASC,limit:s,offset:i,bookmark:n}=t,a={offset:i??1,limit:s??10,bookmark:n};return qt(this.persistence(r),this.page.name,e,o,a,r)}async findOneBy(e,t){const{ctx:r}=(await this.logCtx([],X.FIND_ONE_BY,!0)).for(this.findOneBy);return this.persistence(r).findOneBy(e,t,r)}async findBy(e,t,r){const{ctx:o}=(await this.logCtx([],X.FIND_BY,!0)).for(this.findBy);return this.persistence(o).for(o.toOverrides()).findBy(e,t,o)}async statement(e,t,r){const{ctx:o}=(await this.logCtx([],M.STATEMENT,!0)).for(this.statement),{direction:s,offset:i,limit:n,bookmark:a}=r,c=(t=t.map(e=>("string"==typeof e?parseInt(e):e)||e)).length>1?t[1]:void 0,p=s??c;switch(p&&t.length>1&&(t[1]=p),e){case X.FIND:case X.FIND_BY:break;case X.LIST_BY:t.push(s);break;case X.PAGE:case X.PAGE_BY:t=[t[0],p,{limit:n,offset:i,bookmark:a}];case X.FIND_ONE_BY:case X.COUNT_OF:case X.MAX_OF:case X.MIN_OF:case X.AVG_OF:case X.SUM_OF:case X.DISTINCT_OF:case X.GROUP_OF:}return this.persistence(o).statement(e,...t,o)}async createAll(t,r){const{ctx:o,log:i}=(await this.logCtx([],Ae.CREATE_ALL,!0)).for(this.createAll);let n;i.verbose("creating new "+s);try{n=await this.persistence(o).createAll(t.map(t=>new e(t)),o)}catch(e){throw i.error("Failed to create new "+s,e),e}return i.info(`created new ${s} with id ${n[this.pk]}`),o.toResponse(r),n}async create(e,t){const{ctx:r,log:o}=(await this.logCtx([],we.CREATE,!0)).for(this.create);let i;o.verbose("creating new "+s);try{const t=this.persistence(r);i=await t.create(e,r)}catch(e){throw o.error("Failed to create new "+s,e),e}return o.info(`created new ${s} with id ${i[this.pk]}`),r.toResponse(t),i}async readAll(e){const{ctx:t,log:r}=(await this.logCtx([],Ae.READ_ALL,!0)).for(this.readAll);let o;try{r.debug(`reading ${e.length} ${s}: ${e}`);const i=this.persistence(t);o=await i.readAll(e,t)}catch(t){throw r.error(`Failed to ${s} with id ${e}`,t),t}return r.info(`read ${o.length} ${s}`),o}async read(e){const{ctx:t,log:r}=(await this.logCtx([],we.READ,!0)).for(this.read),o=a(...e.valuesInOrder);if(void 0===o)throw new Re(`No ${this.pk} provided`);let i;try{r.debug(`reading ${s} with ${this.pk} ${o}`);const e=this.persistence(t);i=await e.read(o,t)}catch(e){throw r.error(`Failed to read ${s} with id ${o}`,e),e}return r.info(`read ${s} with id ${i[this.pk]}`),i}async updateAll(e,t){const{ctx:r,log:o}=(await this.logCtx([],Ae.UPDATE_ALL,!0)).for(this.updateAll);let i;try{o.info(`updating ${e.length} ${s}`),i=await this.persistence(r).updateAll(e,r)}catch(e){throw o.error(e),e}return r.toResponse(t),i}async update(t,r,o){const{ctx:i,log:n}=(await this.logCtx([],we.UPDATE,!0)).for(this.update),c=a(...t.valuesInOrder);if(void 0===c)throw new Re(`No ${this.pk} provided`);let p;try{n.info(`updating ${s} with ${this.pk} ${c}`);const t=JSON.parse(JSON.stringify(r)),o=this.persistence(i);p=await o.update(new e({...t,[this.pk]:c}),i)}catch(e){throw n.error(e),e}return i.toResponse(o),p}async deleteAll(e,t){const{ctx:r,log:o}=(await this.logCtx([],Ae.DELETE_ALL,!0)).for(this.deleteAll);let i;try{o.debug(`deleting ${e.length} ${s}: ${e}`),i=await this.persistence(r).deleteAll(e,r)}catch(t){throw o.error(`Failed to delete ${s} with id ${e}`,t),t}return o.info(`deleted ${i.length} ${s}`),r.toResponse(t),i}async delete(e,t){const{ctx:r,log:o}=(await this.logCtx([],we.DELETE,!0)).for(this.delete),i=a(...e.valuesInOrder);if(void 0===i)throw new Re(`No ${this.pk} provided`);let n;try{o.debug(`deleting ${s} with ${this.pk} ${i}`),n=await this.persistence(r).delete(i,r)}catch(e){throw o.error(`Failed to delete ${s} with id ${i}`,e),e}return o.info(`deleted ${s} with id ${i}`),r.toResponse(t),n}};return Oe([pt(e,"GET","listBy/:key"),ae({summary:`Retrieve ${s} records by query.`}),se({name:"key",description:"the model key to sort by"}),ne({name:"direction",required:!0,enum:j}),ce({description:s+" listed successfully."}),xe(0,P("key")),xe(1,gt()),Se("design:type",Function),Se("design:paramtypes",[String,Object]),Se("design:returntype",Promise)],l.prototype,"listBy",null),Oe([pt(e,"GET","paginateBy/:key/:page"),ae({summary:`Retrieve ${s} records by query.`}),se({name:"key",description:"the model key to sort by"}),se({name:"page",description:"the page to retrieve or the bookmark"}),ne({name:"direction",required:!0,enum:j,description:"the sort order"}),ne({name:"limit",required:!0,description:"the page size"}),ne({name:"offset",description:"the bookmark when necessary"}),ce({description:s+" listed paginated."}),xe(0,P("key")),xe(1,gt()),Se("design:type",Function),Se("design:paramtypes",[String,Object]),Se("design:returntype",Promise)],l.prototype,"paginateBy",null),Oe([pt(e,"GET","find/:value"),ae({summary:`Find ${s} records using the default query attributes.`}),se({name:"value",description:"The string to match against the default query attributes"}),ne({name:"direction",required:!1,enum:j,description:"the sort order for the matching results"}),ce({description:s+" records matching the provided prefix."}),xe(0,P("value")),xe(1,gt()),Se("design:type",Function),Se("design:paramtypes",[String,Object]),Se("design:returntype",Promise)],l.prototype,"find",null),Oe([pt(e,"GET","page/:value"),ae({summary:`Page ${s} records using the default query attributes.`}),se({name:"value",description:"The string to match against the default query attributes"}),ne({name:"direction",required:!1,enum:j,description:"the sort order for the paged results"}),ne({name:"limit",required:!1,description:"page size"}),ne({name:"offset",required:!1,description:"page number"}),ne({name:"bookmark",required:!1,description:"bookmark for cursor pagination"}),ce({description:s+" records paged by the provided prefix."}),xe(0,P("value")),xe(1,gt()),Se("design:type",Function),Se("design:paramtypes",[String,Object]),Se("design:returntype",Promise)],l.prototype,"page",null),Oe([pt(e,"GET","findOneBy/:key"),ae({summary:`Retrieve ${s} records by query.`}),se({name:"key",description:"the model key to sort by"}),ce({description:s+" listed found."}),de({description:`No ${s} record matches the provided identifier.`}),xe(0,P("key")),xe(1,P("value")),Se("design:type",Function),Se("design:paramtypes",[String,Object]),Se("design:returntype",Promise)],l.prototype,"findOneBy",null),Oe([pt(e,"GET","findBy/:key/:value"),ae({summary:`Retrieve ${s} records by query.`}),se({name:"key",description:"the model key to compare"}),se({name:"value",description:"the value to match"}),ne({name:"direction",required:!0,enum:j,description:"the sort order when applicable"}),ce({description:s+" listed found."}),de({description:`No ${s} record matches the provided identifier.`}),xe(0,P("key")),xe(1,P("value")),xe(2,gt()),Se("design:type",Function),Se("design:paramtypes",[String,Object,Object]),Se("design:returntype",Promise)],l.prototype,"findBy",null),Oe([pt(e,"GET","statement/:method/*args"),ae({summary:`Executes a prepared statement on ${s}.`}),se({name:"method",description:"the prepared statement to execute"}),se({name:"args",description:"concatenated list of arguments the prepared statement can accept"}),ne({name:"direction",required:!0,enum:j,description:"the sort order when applicable"}),ne({name:"limit",required:!0,description:"limit or page size when applicable"}),ne({name:"offset",required:!0,description:"offset or bookmark when applicable"}),ce({description:s+" listed found."}),de({description:`No ${s} record matches the provided identifier.`}),xe(0,P("method")),xe(1,P("args")),xe(2,gt()),Se("design:type",Function),Se("design:paramtypes",[String,Array,Object]),Se("design:returntype",Promise)],l.prototype,"statement",null),Oe([pt(e,"POST","bulk"),ae({summary:`Create a new ${s}.`}),le({description:"Payload for "+s,schema:{type:"array",items:{$ref:ue(e)}}}),he({description:s+" created successfully.",schema:{type:"array",items:{$ref:ue(e)}}}),fe({description:"Payload validation failed."}),ge({description:"Repository rejected the provided payload."}),xe(0,ut()),xe(1,R({passthrough:!0})),Se("design:type",Function),Se("design:paramtypes",[Array,Object]),Se("design:returntype",Promise)],l.prototype,"createAll",null),Oe([pt(e,"POST"),ae({summary:`Create a new ${s}.`}),le({description:"Payload for "+s,type:zt(we.CREATE,e)}),he({description:s+" created successfully.",schema:{$ref:ue(e)}}),fe({description:"Payload validation failed."}),ge({description:"Repository rejected the provided payload."}),xe(0,ut()),xe(1,R({passthrough:!0})),Se("design:type",Function),Se("design:paramtypes",[Object,Object]),Se("design:returntype",Promise)],l.prototype,"create",null),Oe([pt(e,"GET","bulk"),ae({summary:`Retrieve a ${s} record by id.`}),ne({name:"ids",required:!0,type:"array"}),ce({description:s+" retrieved successfully.",schema:{type:"array",items:{$ref:ue(e)}}}),de({description:`No ${s} record matches the provided identifier.`}),xe(0,E("ids")),Se("design:type",Function),Se("design:paramtypes",[Array]),Se("design:returntype",Promise)],l.prototype,"readAll",null),Oe([pt(e,"GET",p),lt(c),ae({summary:`Retrieve a ${s} record by id.`}),ce({description:s+" retrieved successfully.",schema:{$ref:ue(e)}}),de({description:`No ${s} record matches the provided identifier.`}),xe(0,ft(c)),Se("design:type",Function),Se("design:paramtypes",[Object]),Se("design:returntype",Promise)],l.prototype,"read",null),Oe([pt(e,"PUT","bulk"),lt(c),ae({summary:`Replace an existing ${s} record with a new payload.`}),le({description:"Payload for replace a existing record of "+s,schema:{type:"array",$ref:ue(zt(we.UPDATE,e))}}),ce({description:s+" updated successfully.",schema:{type:"array",items:{$ref:ue(e)}}}),de({description:`No ${s} record matches the provided identifier.`}),fe({description:"Payload validation failed."}),xe(0,ut()),xe(1,R({passthrough:!0})),Se("design:type",Function),Se("design:paramtypes",[Array,Object]),Se("design:returntype",Promise)],l.prototype,"updateAll",null),Oe([pt(e,"PUT",p),lt(c),ae({summary:`Replace an existing ${s} record with a new payload.`}),le({description:"Payload for replace a existing record of "+s,type:zt(we.UPDATE,e)}),ce({description:s+" updated successfully.",schema:{$ref:ue(e)}}),de({description:`No ${s} record matches the provided identifier.`}),fe({description:"Payload validation failed."}),xe(0,ft(c)),xe(1,ut()),xe(2,R({passthrough:!0})),Se("design:type",Function),Se("design:paramtypes",[Object,Object,Object]),Se("design:returntype",Promise)],l.prototype,"update",null),Oe([pt(e,"DELETE","bulk"),lt(c),ae({summary:`Retrieve a ${s} record by id.`}),ne({name:"ids",required:!0,type:"array"}),ce({description:s+" deleted successfully.",schema:{type:"array",items:{$ref:ue(e)}}}),de({description:`No ${s} record matches the provided identifier.`}),xe(0,E("ids")),xe(1,R({passthrough:!0})),Se("design:type",Function),Se("design:paramtypes",[Array,Object]),Se("design:returntype",Promise)],l.prototype,"deleteAll",null),Oe([pt(e,"DELETE",p),lt(c),ae({summary:`Delete a ${s} record by id.`}),ce({description:s+" deleted successfully.",schema:{$ref:ue(e)}}),de({description:`No ${s} record matches the provided identifier.`}),xe(0,ft(c)),xe(1,R({passthrough:!0})),Se("design:type",Function),Se("design:paramtypes",[Object,Object]),Se("design:returntype",Promise)],l.prototype,"delete",null),l=Oe([T(o),me(s),ye(e),xt(e),Se("design:paramtypes",[_t])],l),l}static getRouteParametersFromModel(e){const r=N.pk(e),o=t.get(e,t.key(Pe.COMPOSED,r)),s=o?.args??[],i=Array.isArray(s)&&s.length>0?Array.from(new Set([...s])):Array.from(new Set([r])),n=t.description(e);return{path:":"+i.join("/:"),description:n,apiProperties:i.map(r=>({name:r,description:t.description(e,r),required:!0,type:String})),getPK:(...e)=>o?.separator?e.join(o.separator):e.join("")}}}function Kt(e){var t;let r=class{static{t=this}static{this.log=Ne.for(t.name).for(e)}static createModelServices(e){return e.map(e=>({provide:e.name+"Service",useFactory:()=>U.forModel(e)}))}static forRoot(e,r={}){const o=this.log.for(this.forRoot);o.info("Generating controllers for flavour...");const s=Y.models(e);let i=[];r.autoServices&&(o.info("Auto-services enabled. Initializing service generation."),i=this.createModelServices(s),o.info(`Auto-services completed. ${i.length} services initialized.`));const n=s.map(Vt.create);return o.info(`Generated ${n.length} controllers`),{module:t,controllers:n,providers:[...i]}}};return r=t=Oe([O({})],r),Object.assign(r,"name",{value:"DecafModule"+e}),r}r.for(n.INJECTABLE).extend({decorator:(e,t)=>(t=t||("object"==typeof e?Object.assign(e,a):a),m({scope:t.singleton?A.DEFAULT:A.REQUEST,durable:!t.singleton||void 0}))}).apply(),r.for(n.INJECT).extend({decorator:(e,t)=>(t,r,o)=>b(e||t)(t,r,o)}).apply(),r.for(F.REQUIRED).extend(nt({required:!0})).apply(),r.for(F.MAX).extend({decorator:e=>nt({maximum:e,required:!1})}).apply(),r.for(F.MIN).extend({decorator:e=>nt({minimum:e,required:!1})}).apply(),r.for(F.MAX_LENGTH).extend({decorator:e=>nt({maxLength:e,required:!1})}).apply(),r.for(F.MIN_LENGTH).extend({decorator:e=>nt({minLength:e,required:!1})}).apply(),r.for(F.TYPE).extend({decorator:e=>(t,r)=>("function"!=typeof(e=Array.isArray(e)?e[0]:e)||e.name||(e=e()),nt({type:e,required:!1})(t,r))}).apply(),r.for(F.DATE).extend({decorator:e=>nt({type:String,format:"date-time",required:!1})}).apply(),r.for(F.ENUM).extend({decorator:e=>nt({enum:Array.isArray(e)?e:Object.values(e),required:!1})}).apply(),r.for(F.PATTERN).extend({decorator:e=>nt({pattern:"string"==typeof e?e:e.source,required:!1})}).apply(),r.for(M.COLUMN).extend({decorator:e=>nt({name:e,required:!1})}).apply(),r.for(o.DESCRIPTION).extend({decorator:e=>nt({description:e,required:!1})}).apply(),r.for(M.AUTH).extend({decorator:xt}).apply();class Zt extends W{constructor(e){super(e,Zt.name)}}class er extends Te{constructor(e){super(er.name,e,429)}}let tr=class{catch(e,t){const r=t.switchToHttp(),o=r.getResponse(),s=r.getRequest(),i="production"===qe.env;let n;e instanceof S||e instanceof Q?n=(e=new $(e.message)).getStatus():e instanceof Te||(e=429===e.status?new er(e.message):new Ee(e.message)),o.status(e.code||n).json({status:e.code||n,error:i?e.name:e.message,timestamp:(new Date).toISOString(),path:s.url,method:s.method})}};function rr(){return _(new tr)}function or(){return v(It)}tr=Oe([x()],tr);const sr={path:"docs",auth:{type:"http",scheme:"bearer",bearerFormat:"JWT",name:"Authorization",description:"Enter JWT token",in:"header"},topbarBgColor:"#000000"};class ir{constructor(e){this.options={...e}}customCSS(){let e="";return this.options.topbarIconPath&&(e+=`.topbar-wrapper { content: url('data:image/png;base64,${this.b64(this.options.topbarIconPath)}'); width: 200px; height: auto; }\n`),e+`\n .topbar-wrapper svg { visibility: hidden; }\n .swagger-ui .topbar { background-color: ${this.options.topbarBgColor||sr.topbarBgColor}; }\n `}getCustomOptions(){const e={};return this.options.faviconPath&&(e.customfavIcon=this.b64(this.options.faviconPath,!0)),{customSiteTitle:this.options.title,...e,customCss:this.customCSS(),swaggerOptions:{persistAuthorization:this.options.persistAuthorization}}}b64(e,t=!1){const r=Ge.join(this.options.assetsPath||"",e),o=Be(r,{encoding:"base64"});return t?"data:image/png;base64,"+o:o}}class nr{constructor(e,t){this.app=e,this.options=t}createDocument(){const e=this.options.path?this.options.description+""+`<br><br><a href="${this.options.openApiJsonPath}">OpenAPI JSON Spec</a> | `+`<a href="${this.options.openApiYamlPath}">OpenAPI YAML Spec</a>`:this.options.description,t=(new be).setTitle(this.options.title).setDescription(e).setVersion(this.options.version||"0.0.1").addBearerAuth(this.options.auth||sr.auth).build();return ve.createDocument(this.app,t,{extraModels:this.options.extraModels||[]})}registerOpenApiRoute(e,t,r){if(!e)return;const o=this.app.getHttpAdapter();e=e.startsWith("/")?e:"/"+e,o.get(e,(e,s)=>{o.reply(s,r(),200,{"Content-Type":t})})}setupSwagger(){const e=this.createDocument(),t=new ir({title:this.options.title,path:this.options.path||sr.path,persistAuthorization:this.options.persistAuthorization??!0,assetsPath:this.options.assetsPath,faviconPath:this.options.faviconFilePath,topbarIconPath:this.options.topbarIconFilePath,topbarBgColor:this.options.topbarBgColor});ve.setup(this.options.path||sr.path,this.app,e,{...t.getCustomOptions(),jsonDocumentUrl:this.options.openApiJsonPath?""+this.options.openApiJsonPath:void 0,yamlDocumentUrl:this.options.openApiYamlPath?""+this.options.openApiYamlPath:void 0}),this.registerOpenApiRoute(this.options.openApiJsonPath,"application/json",()=>e),this.registerOpenApiRoute(this.options.openApiYamlPath,"application/x-yaml",()=>ze.stringify(e))}}class ar{static get logger(){return this._logger||(this._logger=new C("NestBootstrap")),this._logger}static initialize(e){return this.app=e,this}static enableLogger(e){return this._logger=e||new C("NestBootstrap"),this.app.useLogger(this._logger),this}static enableCors(e=[],t=["GET","POST","PUT","DELETE"]){const r="*"===e?"*":e.map(e=>e.trim().toLowerCase()),o={origin:(e,t)=>e?"*"===r||Array.isArray(r)&&r.includes(e.toLowerCase())?t(null,!0):void t(new Zt(`Origin ${e} not allowed`)):t(null,!0),credentials:!0,methods:t.join(",")};return this.app.enableCors(o),this}static useHelmet(e){try{const t=require("helmet");this.app.use(t(e)),this.logger.log("Helmet middleware enabled successfully.")}catch(e){this.logger.warn("Helmet not installed. Skipping middleware.")}return this}static setupSwagger(e){return new nr(this.app,{title:e.title,description:e.description,version:e.version,path:e.path||"api",persistAuthorization:e.persistAuthorization??!0,assetsPath:e.assetsPath,faviconFilePath:e.faviconPath,topbarIconFilePath:e.topbarIconPath,topbarBgColor:e.topbarBgColor,openApiJsonPath:e.openApiJsonPath,openApiYamlPath:e.openApiYamlPath}).setupSwagger(),this}static useGlobalPipes(...e){return e.length>0&&this.app.useGlobalPipes(...e),this}static useGlobalFilters(...e){return this.app.useGlobalFilters(...e.length>0?e:[new tr]),this}static useGlobalInterceptors(...e){return e.length>0&&this.app.useGlobalInterceptors(...e),this}static async start(e=Number(process.env.PORT)||3e3,t=void 0,r=!0){this.app.listen(e,t).then(async()=>{if(r){const e=await this.app.getUrl();this.logger.log("\ud83d\ude80 Application is running at: "+e)}})}}var cr;let pr=cr=class{constructor(e,t){this.requestContext=e,this.executor=t}async contextualize(e){const t={headers:e.headers,overrides:{}},r=Y.flavoursToTransform();if(r)for(const o of r)try{const r=Y.transformerFor(o),s=await r.from(e);Object.assign(t.overrides,s)}catch(e){throw new Ee("Failed to contextualize request: "+e)}return(new J).accumulate(Object.assign({},V,{logger:Ne.get(),timestamp:new Date},t))}async intercept(e,t){const r=e.switchToHttp().getRequest(),o=e.switchToHttp().getResponse(),s=Ne.for(cr).for(this.intercept);s.debug(`CONTEXT ${this.requestContext.uuid} - request: ${r.method} ${r.url}`);const i=await this.contextualize(r);return s.debug(`CONTEXT ${this.requestContext.uuid} contextualized - request: ${r.method} ${r.url}`),this.requestContext.applyCtx(i),s.debug(`CONTEXT ${this.requestContext.uuid} applied - request: ${r.method} ${r.url}`),await this.executor.exec(r,o),s.debug(`CONTEXT ${this.requestContext.uuid} executors finished - request: ${r.method} ${r.url}`),t.handle()}};pr=cr=Oe([m({scope:A.REQUEST}),Se("design:paramtypes",[_t,kt])],pr);class dr{}function lr(e){return r=>(t.set("transformers",e,r),"function"==typeof r?s("transformers",e)(r):r)}var ur;let hr=class{static{ur=this}static get persistence(){if(!this._persistence)throw new Ee("Persistence service not initialized");return this._persistence}static get log(){return this._logger||(this._logger=Ne.for(ur)),this._logger}constructor(e,t){this.options=e,this.moduleRef=t}static forRoot(e){const t=this.log.for(this.forRoot);return{module:ur,providers:[{provide:yt,useValue:e},{provide:bt,useValue:this.persistence?.client},{provide:At,useFactory:()=>e.handlers?.map(e=>(t.info("Registered request handler: "+e.name),new e))??[]},Ot,{provide:ke,useExisting:Ot},_t,kt,{provide:ke,useClass:pr}],exports:[yt,bt,At,_t,kt]}}static async bootPersistence(e){const t=this.log.for(this.bootPersistence);if(!this._persistence){const r=e.conf.map(([e,t,...r])=>r.pop()?[e,t,...r]:[e,t]);this._persistence=new K,await this._persistence.boot(r);const o=this._persistence.client;for(let t=0;t<o.length;t++){const r=e.conf[t];let s=r.slice(2,r.length).pop();if(!s||!s.from){const e=Y.transformerFor(o[t].flavour);if(!e)throw new Ee(`No transformer found for flavour ${o[t].flavour}. you should either @requestToContextTransformer or provide a transformer in the config`);try{s=e.from?e:new e}catch(e){throw new Ee(`Failed to boot transformer for ${o[t].flavour}: ${e}`)}}lr(o[t].flavour)(s)}if(t.info("persistence layer created successfully!"),e.initialization)try{await e.initialization()}catch(e){throw new Ee("Failed to initialize application: "+e)}}return this.persistence.client}async onApplicationShutdown(){const e=ur.log.for(this.onApplicationShutdown),t=this.moduleRef.get(bt);for(const r of t)try{r&&(e.info("Shutting down "+r.toString()),await r.shutdown())}catch(t){e.error("Failed to shutdown application",t)}try{await B.shutdown()}catch(t){e.error("Failed to shutdown services",t)}}};hr=ur=Oe([k(),O({}),xe(0,b(yt)),Se("design:paramtypes",[Object,Ce])],hr);const fr=Symbol("LISTENING_ADAPTERS_FLAVOURS");var gr;let mr=gr=class extends jt{constructor(e,t){super(e,gr.name),this.adapters=t.map(e=>Y.get(e))}listen(){const e=Ne.for(gr.name),t=new Le(t=>{const r=("B-"+Math.random().toString(36).slice(2,8)).toUpperCase();e.info(`Creating SSE observer: ${r} for client ${this.clientContext.uuid}`);const o=new class{constructor(){this.observerId=r}refresh(...r){return e.debug(`SSE observer ${this.observerId} received refresh event`),Promise.resolve().then(()=>{const o=(e=>{const[t,r,o,s]=e;return[t?.name??t,r,o,Array.isArray(s)?s.map(e=>{try{return"function"==typeof e.serialize?e.serialize():"string"==typeof e?e:JSON.stringify(e)}catch(e){return}}):s&&"function"==typeof s.serialize?s.serialize():"string"==typeof s?s:JSON.stringify(s)]})(r);t.next({type:"message",data:o}),e.debug(`SSE observer ${this.observerId} event pushed to client`)})}};e.verbose(`Registering observer ${r} across ${this.adapters.length} adapter(s)`);for(const t of this.adapters){const s=t?.constructor?.name??"UnknownAdapter";try{e.debug(`Registering observer ${r} in adapter ${s}`),t.observe(o)}catch(t){e.debug(`Failed to register observer ${r} in adapter ${s}: ${t?.message||t}`),e.error(t)}}return()=>{e.debug("Cleaning up SSE observer "+r);for(const t of this.adapters){const s=t?.constructor?.name??"UnknownAdapter";try{e.debug(`Unregistering observer ${r} from adapter ${s}`),t.unObserve(o)}catch(t){e.debug(`Failed during cleanup of observer ${r} in adapter ${s}: ${t?.message||t}`),e.error(t)}}}}),r=je(15e3).pipe(He(()=>{e.debug("Sending heartbeat")}),Ye(()=>({type:"heartbeat",data:{ts:(new Date).toISOString()}})));return Ue(t,r)}listenForModel(e){const t=Ne.for(gr.name);return new Le(e=>{const r=new class{refresh(...t){return Promise.resolve().then(()=>{e.next({data:t})})}};try{for(const e of this.adapters)e.observe(r)}catch(t){e.error("Failed to observe event: "+(t.message||t))}return()=>{try{for(const e of this.adapters)e.unObserve(r)}catch(e){t.error(e)}}})}};var yr;Oe([I(),Se("design:type",Function),Se("design:paramtypes",[]),Se("design:returntype",Le)],mr.prototype,"listen",null),Oe([I("/:model"),xe(0,E("model")),Se("design:type",Function),Se("design:paramtypes",[String]),Se("design:returntype",Le)],mr.prototype,"listenForModel",null),mr=gr=Oe([T(),xe(1,b(fr)),Se("design:paramtypes",[_t,Array])],mr);let br=yr=class{static forFlavours(e,t="events"){return{module:yr,controllers:[mr],imports:[Ie.register([{path:t.replace(/^\//,""),module:yr}])],providers:[_t,{provide:fr,useValue:e??[]}]}}};var vr;br=yr=Oe([O({})],br);let wr=vr=class{static async forRootAsync(e){const{autoControllers:t,autoServices:r}=e,o=(await hr.bootPersistence(e)).map(e=>e.flavour),s=[hr.forRoot(e)];return t&&o.forEach(e=>{s.push(Kt(e).forRoot(e,{autoServices:r}))}),e.observerOptions?.enableObserverEvents&&s.push(br.forFlavours(e.observerOptions.observerFlavours||o)),{module:vr,imports:s}}};async function Ar(){}var Er;wr=vr=Oe([O({})],wr);const Pr=Ne.for("DecafMigrationModule");let Rr=Er=class{static forRoot(){return{module:Er}}static async migrate(e,t){const r=Pr.for(this.migrate);if(!t||0===t.length)throw new Ee("No adapters provided. Make sure DecafCoreModule is configured and adapters are available.");return r.info("Running migrations with config: "+JSON.stringify(e)),Je.migrateAdapters(t,e||{})}};function Tr(e){const t=N.get(e);if(!t)throw new Ee("Failed to find repository for "+e);return G.forModel(t)}var Or,xr;Rr=Er=Oe([k(),O({providers:[{provide:"MIGRATION_ADAPTERS",useFactory:e=>e,inject:[bt]}],exports:["MIGRATION_ADAPTERS"]})],Rr);let Sr=Or=class extends jt{constructor(e){super(e,Or.name)}async deactivate(e){const{ctx:t}=(await this.logCtx([],"deactivate",!0)).for(this.deactivate),r=G.forModel(Xe),o=await r.read(e,t);return o.active=!1,r.update(o,t)}async reactivate(e){const{ctx:t}=(await this.logCtx([],"reactivate",!0)).for(this.reactivate),r=G.forModel(Xe),o=await r.read(e,t);return o.active=!0,r.update(o,t)}};Oe([l(":id/deactivate"),xe(0,P("id")),Se("design:type",Function),Se("design:paramtypes",[String]),Se("design:returntype",Promise)],Sr.prototype,"deactivate",null),Oe([l(":id/reactivate"),xe(0,P("id")),Se("design:type",Function),Se("design:paramtypes",[String]),Se("design:returntype",Promise)],Sr.prototype,"reactivate",null),Sr=Or=Oe([T("webhook-subscriptions"),Se("design:paramtypes",[_t])],Sr);let $r=xr=class extends jt{constructor(e){super(e,xr.name)}async replay(e){const{ctx:t}=(await this.logCtx([],"replay",!0)).for(this.replay),r=G.forModel(We),o=G.forModel(Qe);try{let s;try{s=await r.read(e,t)}catch(o){const i=await r.select().where(r.attr("id").eq(e)).limit(1).execute(t);if(!i.length)throw o;s=i[0]}let i=[];try{i=await Ve(()=>o.select().where(o.attr("eventId").eq(s.id)).orderBy("createdAt",j.ASC).thenBy("id",j.ASC).paginate(250,t),250,t)}catch{try{i=await o.select().where(o.attr("eventId").eq(s.id)).execute(t)}catch{i=[]}}const n=new Date;for(const e of i)e.status=Ke.PENDING,e.attempts=0,e.nextAttemptAt=n,e.lastAttemptAt=null,e.errorMessage=void 0,e.responseStatus=void 0,e.responseBody=void 0;if(s.status=Ke.PENDING,s.deliveriesSucceeded=0,s.deliveriesFailed=0,s.nextAttemptAt=n,s.updatedAt=n,i.length)try{await o.updateAll(i,t.override({applyUpdateValidation:!1}))}catch{}return r.update(s,t)}catch(e){throw e}}};var _r;Oe([l(":id/replay"),xe(0,P("id")),Se("design:type",Function),Se("design:paramtypes",[String]),Se("design:returntype",Promise)],$r.prototype,"replay",null),$r=xr=Oe([T("webhook-events"),Se("design:paramtypes",[_t])],$r);let Cr=class{static{_r=this}static{this._logger=Ne.for(_r)}static get log(){return this._logger}static async bootPersistence(e){const t=this.log.for(this.bootPersistence);if(!this._persistence){const r=e.conf.map(([e,t,...r])=>r.pop()?[e,t,...r]:[e,t]);this._persistence=new K,await this._persistence.boot(r);const o=this._persistence.client;for(let t=0;t<o.length;t++){const r=Y._cache||(Y._cache={}),s=[o[t].flavour,o[t].alias,"webhook_deliveries","webhook_events","webhook_subscriptions"].filter(e=>!!e);for(const e of s)r[e]=o[t];const n=e.conf[t];let a=n.slice(2,n.length).pop();if(!a||!a.from){const e=Y.transformerFor(o[t].flavour);if(!e)throw new Ee(`No transformer found for flavour ${o[t].flavour}.`);try{a=e.from?e:new e}catch(e){throw new Ee(`Failed to boot transformer for ${o[t].flavour}: ${e}`)}}lr(o[t].flavour)(a),i(o[t].flavour)(Xe),i(o[t].flavour)(We),i(o[t].flavour)(Qe)}if(t.info("persistence layer created successfully!"),e.initialization)try{await e.initialization()}catch(e){throw new Ee("Failed to initialize webhook module: "+e)}}return this._persistence.client}static async forRoot(e){return this.forRootAsync(e)}static async forRootAsync(e){await this.bootPersistence(e);const t=[Vt.create(Xe),Vt.create(We),Vt.create(Qe),Sr,$r];return{module:_r,controllers:t,imports:[Ie.register([{path:(e.webhookApiPath||"webhooks").replace(/^\//,""),module:_r}])],providers:[_t,kt,{provide:At,useFactory:()=>e.handlers?.map(e=>new e)??[]},{provide:ke,useClass:pr}],exports:[_t,kt]}}};Cr=_r=Oe([O({})],Cr);const kr=Cr;async function Ir(){}const Nr="##VERSION##",Fr="##COMMIT##",Dr="##FULL_VERSION##",qr="##PACKAGE##";t.allowReregistration(!0),t.registerLibrary(qr,Nr),t.allowReregistration(!1);export{Pt as AUTH_HANDLER,Rt as AUTH_META_KEY,pt as ApiOperationFromModel,lt as ApiParamsFromModel,nt as ApiProperty,xt as Auth,Ot as AuthInterceptor,dt as BulkApiOperationFromModel,Fr as COMMIT,Zt as CorsError,bt as DECAF_ADAPTER_ID,Et as DECAF_ADAPTER_OPTIONS,Tt as DECAF_CONTEXT_KEY,At as DECAF_HANDLERS,yt as DECAF_MODULE_OPTIONS,wt as DECAF_ROUTE,vt as DECAF_TASK_SERVICE_ID,et as DECORATORS,Ze as DECORATORS_PREFIX,St as DecafAuthHandler,ut as DecafBody,jt as DecafController,hr as DecafCoreModule,tr as DecafExceptionFilter,kt as DecafHandlerExecutor,Rr as DecafMigrationModule,Ut as DecafModelController,wr as DecafModule,ft as DecafParams,gt as DecafQuery,_t as DecafRequestContext,pr as DecafRequestHandlerInterceptor,It as DecafResponseInterceptor,$t as DecafRoleAuthHandler,br as DecafStreamModule,Cr as DecafWebhookModule,kr as DecafWebhooksModule,zt as DtoFor,mr as EventsController,Dr as FULL_VERSION,Vt as FromModelController,mt as HttpVerbToDecorator,ar as NestBootstraper,qr as PACKAGE_NAME,dr as RequestToContextTransformer,nr as SwaggerBuilder,rr as UseDecafFilter,or as UseDecafHeaders,Nr as VERSION,$r as WebhookEventActionsController,Sr as WebhookSubscriptionActionsController,at as createApiPropertyDecorator,it as createPropertyDecorator,st as getEnumType,ot as getEnumValues,Kt as getModuleFor,ct as isOperationBlocked,rt as isUndefined,Tr as repoForModel,lr as requestToContextTransformer,Ar as runMigrations,Ir as runWebhooksMigrations};
|
|
2
|
-
|
|
1
|
+
import { apply, Metadata, Decoration, DecorationKeys } from "@decaf-ts/decoration";
|
|
2
|
+
|
|
3
|
+
import { InjectablesKeys, DefaultInjectablesConfig } from "@decaf-ts/injectable-decorators";
|
|
4
|
+
|
|
5
|
+
import { Delete, Patch, Put, Post, Get, applyDecorators, createParamDecorator as createParamDecorator$1, InternalServerErrorException, Injectable, Scope, Inject, Optional, UseInterceptors, SetMetadata, Module, Query, Controller, Response, Param, NotFoundException, NotAcceptableException, Catch, UseFilters, Logger, Global, Sse } from "@nestjs/common";
|
|
6
|
+
|
|
7
|
+
import { ModelBuilder, Model, ValidationKeys } from "@decaf-ts/decorator-validation";
|
|
8
|
+
|
|
9
|
+
import { isOperationBlocked, PreparedStatementKeys, UUID, Adapter, AuthorizationError, Context, DefaultAdapterFlags, OrderDirection, ModelService, Service as Service$1, Repository as Repository$1, TransactionOperationKeys, PersistenceKeys, ForbiddenError, UnsupportedError, injectableServiceKey, PersistenceService } from "@decaf-ts/core";
|
|
10
|
+
|
|
11
|
+
import { pickBy, negate, isUndefined as isUndefined$1, isArray, isString } from "lodash";
|
|
12
|
+
|
|
13
|
+
import { ApiExcludeEndpoint, ApiParam, ApiBearerAuth, ApiSecurity, ApiQuery, ApiOperation, ApiOkResponse, ApiNoContentResponse, ApiTags, ApiExtraModels, ApiBody, ApiCreatedResponse, getSchemaPath, ApiBadRequestResponse, ApiUnprocessableEntityResponse, ApiNotFoundResponse, DocumentBuilder, SwaggerModule } from "@nestjs/swagger";
|
|
14
|
+
|
|
15
|
+
import { OperationKeys, BulkCrudOperationKeys, InternalError, DBKeys, ValidationError, BaseError } from "@decaf-ts/db-decorators";
|
|
16
|
+
|
|
17
|
+
import { __decorate, __param, __metadata } from "tslib";
|
|
18
|
+
|
|
19
|
+
import { REQUEST, Reflector, APP_INTERCEPTOR, ModuleRef, RouterModule } from "@nestjs/core";
|
|
20
|
+
|
|
21
|
+
import { Logging, toPascalCase, toKebabCase, LoggedEnvironment } from "@decaf-ts/logging";
|
|
22
|
+
|
|
23
|
+
import { RequestContext, AuthHandler, DecafController as DecafController$1, ModelControllerFactory, requestToContextTransformer } from "@decaf-ts/for-http/server";
|
|
24
|
+
|
|
25
|
+
import { tap, Observable, interval, merge } from "rxjs";
|
|
26
|
+
|
|
27
|
+
import { readFileSync } from "fs";
|
|
28
|
+
|
|
29
|
+
import * as path from "path";
|
|
30
|
+
|
|
31
|
+
import YAML from "yaml";
|
|
32
|
+
|
|
33
|
+
import { tap as tap$1, map } from "rxjs/operators";
|
|
34
|
+
|
|
35
|
+
const DECORATORS_PREFIX = "swagger";
|
|
36
|
+
|
|
37
|
+
const DECORATORS = {
|
|
38
|
+
API_OPERATION: `${DECORATORS_PREFIX}/apiOperation`,
|
|
39
|
+
API_RESPONSE: `${DECORATORS_PREFIX}/apiResponse`,
|
|
40
|
+
API_PRODUCES: `${DECORATORS_PREFIX}/apiProduces`,
|
|
41
|
+
API_CONSUMES: `${DECORATORS_PREFIX}/apiConsumes`,
|
|
42
|
+
API_TAGS: `${DECORATORS_PREFIX}/apiUseTags`,
|
|
43
|
+
API_CALLBACKS: `${DECORATORS_PREFIX}/apiCallbacks`,
|
|
44
|
+
API_PARAMETERS: `${DECORATORS_PREFIX}/apiParameters`,
|
|
45
|
+
API_HEADERS: `${DECORATORS_PREFIX}/apiHeaders`,
|
|
46
|
+
API_MODEL_PROPERTIES: `${DECORATORS_PREFIX}/apiModelProperties`,
|
|
47
|
+
API_MODEL_PROPERTIES_ARRAY: `${DECORATORS_PREFIX}/apiModelPropertiesArray`,
|
|
48
|
+
API_SECURITY: `${DECORATORS_PREFIX}/apiSecurity`,
|
|
49
|
+
API_EXCLUDE_ENDPOINT: `${DECORATORS_PREFIX}/apiExcludeEndpoint`,
|
|
50
|
+
API_EXCLUDE_CONTROLLER: `${DECORATORS_PREFIX}/apiExcludeController`,
|
|
51
|
+
API_EXTRA_MODELS: `${DECORATORS_PREFIX}/apiExtraModels`,
|
|
52
|
+
API_EXTENSION: `${DECORATORS_PREFIX}/apiExtension`,
|
|
53
|
+
API_SCHEMA: `${DECORATORS_PREFIX}/apiSchema`,
|
|
54
|
+
API_DEFAULT_GETTER: `${DECORATORS_PREFIX}/apiDefaultGetter`,
|
|
55
|
+
API_LINK: `${DECORATORS_PREFIX}/apiLink`
|
|
56
|
+
};
|
|
57
|
+
|
|
58
|
+
const isConstructor = val => val === "constructor";
|
|
59
|
+
|
|
60
|
+
const METADATA_FACTORY_NAME = "_OPENAPI_METADATA_FACTORY";
|
|
61
|
+
|
|
62
|
+
const METHOD_METADATA = "method";
|
|
63
|
+
|
|
64
|
+
function createMethodDecorator(metakey, metadata, {overrideExisting: overrideExisting} = {
|
|
65
|
+
overrideExisting: true
|
|
66
|
+
}) {
|
|
67
|
+
return (target, key, descriptor) => {
|
|
68
|
+
if (typeof metadata === "object") {
|
|
69
|
+
const prevValue = Reflect.getMetadata(metakey, descriptor.value);
|
|
70
|
+
if (prevValue && !overrideExisting) {
|
|
71
|
+
return descriptor;
|
|
72
|
+
}
|
|
73
|
+
Reflect.defineMetadata(metakey, {
|
|
74
|
+
...prevValue,
|
|
75
|
+
...metadata
|
|
76
|
+
}, descriptor.value);
|
|
77
|
+
return descriptor;
|
|
78
|
+
}
|
|
79
|
+
Reflect.defineMetadata(metakey, metadata, descriptor.value);
|
|
80
|
+
return descriptor;
|
|
81
|
+
};
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
function createClassDecorator(metakey, metadata = []) {
|
|
85
|
+
return target => {
|
|
86
|
+
const prevValue = Reflect.getMetadata(metakey, target) || [];
|
|
87
|
+
Reflect.defineMetadata(metakey, [ ...prevValue, ...metadata ], target);
|
|
88
|
+
return target;
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
function createPropertyDecorator$1(metakey, metadata, overrideExisting = true) {
|
|
93
|
+
return (target, propertyKey) => {
|
|
94
|
+
const properties = Reflect.getMetadata(DECORATORS.API_MODEL_PROPERTIES_ARRAY, target) || [];
|
|
95
|
+
const key = `:${propertyKey}`;
|
|
96
|
+
if (!properties.includes(key)) {
|
|
97
|
+
Reflect.defineMetadata(DECORATORS.API_MODEL_PROPERTIES_ARRAY, [ ...properties, `:${propertyKey}` ], target);
|
|
98
|
+
}
|
|
99
|
+
const existingMetadata = Reflect.getMetadata(metakey, target, propertyKey);
|
|
100
|
+
if (existingMetadata) {
|
|
101
|
+
const newMetadata = pickBy(metadata, negate(isUndefined$1));
|
|
102
|
+
const metadataToSave = overrideExisting ? {
|
|
103
|
+
...existingMetadata,
|
|
104
|
+
...newMetadata
|
|
105
|
+
} : {
|
|
106
|
+
...newMetadata,
|
|
107
|
+
...existingMetadata
|
|
108
|
+
};
|
|
109
|
+
Reflect.defineMetadata(metakey, metadataToSave, target, propertyKey);
|
|
110
|
+
} else {
|
|
111
|
+
const type = target?.constructor?.[METADATA_FACTORY_NAME]?.()[propertyKey]?.type ?? Reflect.getMetadata("design:type", target, propertyKey);
|
|
112
|
+
Reflect.defineMetadata(metakey, {
|
|
113
|
+
type: type,
|
|
114
|
+
...pickBy(metadata, negate(isUndefined$1))
|
|
115
|
+
}, target, propertyKey);
|
|
116
|
+
}
|
|
117
|
+
};
|
|
118
|
+
}
|
|
119
|
+
|
|
120
|
+
function createMixedDecorator(metakey, metadata) {
|
|
121
|
+
return (target, key, descriptor) => {
|
|
122
|
+
if (descriptor) {
|
|
123
|
+
let metadatas;
|
|
124
|
+
if (Array.isArray(metadata)) {
|
|
125
|
+
const previousMetadata = Reflect.getMetadata(metakey, descriptor.value) || [];
|
|
126
|
+
metadatas = [ ...previousMetadata, ...metadata ];
|
|
127
|
+
} else {
|
|
128
|
+
const previousMetadata = Reflect.getMetadata(metakey, descriptor.value) || {};
|
|
129
|
+
metadatas = {
|
|
130
|
+
...previousMetadata,
|
|
131
|
+
...metadata
|
|
132
|
+
};
|
|
133
|
+
}
|
|
134
|
+
Reflect.defineMetadata(metakey, metadatas, descriptor.value);
|
|
135
|
+
return descriptor;
|
|
136
|
+
}
|
|
137
|
+
let metadatas;
|
|
138
|
+
if (Array.isArray(metadata)) {
|
|
139
|
+
const previousMetadata = Reflect.getMetadata(metakey, target) || [];
|
|
140
|
+
metadatas = [ ...previousMetadata, ...metadata ];
|
|
141
|
+
} else {
|
|
142
|
+
const previousMetadata = Reflect.getMetadata(metakey, target) || {};
|
|
143
|
+
metadatas = Object.assign(Object.assign({}, previousMetadata), metadata);
|
|
144
|
+
}
|
|
145
|
+
Reflect.defineMetadata(metakey, metadatas, target);
|
|
146
|
+
return target;
|
|
147
|
+
};
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
function createParamDecorator(metadata, initial) {
|
|
151
|
+
return (target, key, descriptor) => {
|
|
152
|
+
const paramOptions = {
|
|
153
|
+
...initial,
|
|
154
|
+
...pickBy(metadata, negate(isUndefined$1))
|
|
155
|
+
};
|
|
156
|
+
if (descriptor) {
|
|
157
|
+
const parameters = Reflect.getMetadata(DECORATORS.API_PARAMETERS, descriptor.value) || [];
|
|
158
|
+
Reflect.defineMetadata(DECORATORS.API_PARAMETERS, [ ...parameters, paramOptions ], descriptor.value);
|
|
159
|
+
return descriptor;
|
|
160
|
+
}
|
|
161
|
+
if (typeof target === "object") {
|
|
162
|
+
return target;
|
|
163
|
+
}
|
|
164
|
+
const propertyKeys = Object.getOwnPropertyNames(target.prototype);
|
|
165
|
+
for (const propertyKey of propertyKeys) {
|
|
166
|
+
if (isConstructor(propertyKey)) {
|
|
167
|
+
continue;
|
|
168
|
+
}
|
|
169
|
+
const methodDescriptor = Object.getOwnPropertyDescriptor(target.prototype, propertyKey);
|
|
170
|
+
if (!methodDescriptor) {
|
|
171
|
+
continue;
|
|
172
|
+
}
|
|
173
|
+
const isApiMethod = Reflect.hasMetadata(METHOD_METADATA, methodDescriptor.value);
|
|
174
|
+
if (!isApiMethod) {
|
|
175
|
+
continue;
|
|
176
|
+
}
|
|
177
|
+
const parameters = Reflect.getMetadata(DECORATORS.API_PARAMETERS, methodDescriptor.value) || [];
|
|
178
|
+
Reflect.defineMetadata(DECORATORS.API_PARAMETERS, [ ...parameters, paramOptions ], methodDescriptor.value);
|
|
179
|
+
}
|
|
180
|
+
};
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
function getTypeIsArrayTuple(input, isArrayFlag) {
|
|
184
|
+
if (!input) {
|
|
185
|
+
return [ input, isArrayFlag ];
|
|
186
|
+
}
|
|
187
|
+
if (isArrayFlag) {
|
|
188
|
+
return [ input, isArrayFlag ];
|
|
189
|
+
}
|
|
190
|
+
const isInputArray = isArray(input);
|
|
191
|
+
const type = isInputArray ? input[0] : input;
|
|
192
|
+
return [ type, isInputArray ];
|
|
193
|
+
}
|
|
194
|
+
|
|
195
|
+
const isUndefined = obj => typeof obj === "undefined";
|
|
196
|
+
|
|
197
|
+
function getEnumValues(enumType) {
|
|
198
|
+
if (typeof enumType === "function") {
|
|
199
|
+
return getEnumValues(enumType());
|
|
200
|
+
}
|
|
201
|
+
if (Array.isArray(enumType)) {
|
|
202
|
+
return enumType;
|
|
203
|
+
}
|
|
204
|
+
if (typeof enumType !== "object") {
|
|
205
|
+
return [];
|
|
206
|
+
}
|
|
207
|
+
const numericValues = Object.values(enumType).filter(value => typeof value === "number").map(value => value.toString());
|
|
208
|
+
return Object.keys(enumType).filter(key => !numericValues.includes(key)).map(key => enumType[key]);
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
function getEnumType(values) {
|
|
212
|
+
const hasString = values.filter(isString).length > 0;
|
|
213
|
+
return hasString ? "string" : "number";
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
function createPropertyDecorator(metakey, metadata, overrideExisting = true) {
|
|
217
|
+
return (target, propertyKey) => {
|
|
218
|
+
const properties = Reflect.getMetadata(DECORATORS.API_MODEL_PROPERTIES_ARRAY, target) || [];
|
|
219
|
+
const key = `:${propertyKey}`;
|
|
220
|
+
if (!properties.includes(key)) {
|
|
221
|
+
Reflect.defineMetadata(DECORATORS.API_MODEL_PROPERTIES_ARRAY, [ ...properties, `:${propertyKey}` ], target);
|
|
222
|
+
}
|
|
223
|
+
const existingMetadata = Reflect.getMetadata(metakey, target, propertyKey);
|
|
224
|
+
if (existingMetadata) {
|
|
225
|
+
const newMetadata = pickBy(metadata, negate(isUndefined));
|
|
226
|
+
const metadataToSave = overrideExisting ? {
|
|
227
|
+
...existingMetadata,
|
|
228
|
+
...newMetadata
|
|
229
|
+
} : {
|
|
230
|
+
...newMetadata,
|
|
231
|
+
...existingMetadata
|
|
232
|
+
};
|
|
233
|
+
Reflect.defineMetadata(metakey, metadataToSave, target, propertyKey);
|
|
234
|
+
} else {
|
|
235
|
+
const type = target?.constructor?.[METADATA_FACTORY_NAME]?.()[propertyKey]?.type ?? Reflect.getMetadata("design:type", target, propertyKey);
|
|
236
|
+
Reflect.defineMetadata(metakey, {
|
|
237
|
+
type: type,
|
|
238
|
+
required: false,
|
|
239
|
+
...pickBy(metadata, negate(isUndefined))
|
|
240
|
+
}, target, propertyKey);
|
|
241
|
+
}
|
|
242
|
+
};
|
|
243
|
+
}
|
|
244
|
+
|
|
245
|
+
const isEnumArray = opts => opts.isArray && "enum" in opts && opts.enum !== undefined;
|
|
246
|
+
|
|
247
|
+
function ApiProperty(options = {}) {
|
|
248
|
+
return createApiPropertyDecorator(options);
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
function createApiPropertyDecorator(options = {}, overrideExisting = true) {
|
|
252
|
+
const [type, isArray] = getTypeIsArrayTuple(options.type, options.isArray);
|
|
253
|
+
options = {
|
|
254
|
+
...options,
|
|
255
|
+
type: type,
|
|
256
|
+
isArray: isArray
|
|
257
|
+
};
|
|
258
|
+
if (isEnumArray(options)) {
|
|
259
|
+
options.type = "array";
|
|
260
|
+
const enumValues = getEnumValues(options.enum);
|
|
261
|
+
options.items = {
|
|
262
|
+
type: getEnumType(enumValues),
|
|
263
|
+
enum: enumValues
|
|
264
|
+
};
|
|
265
|
+
delete options.enum;
|
|
266
|
+
} else if ("enum" in options && options.enum !== undefined) {
|
|
267
|
+
const enumValues = getEnumValues(options.enum);
|
|
268
|
+
options.enum = enumValues;
|
|
269
|
+
options.type = getEnumType(enumValues);
|
|
270
|
+
}
|
|
271
|
+
if (Array.isArray(options.type)) {
|
|
272
|
+
options.type = "array";
|
|
273
|
+
options.items = {
|
|
274
|
+
type: "array",
|
|
275
|
+
items: {
|
|
276
|
+
type: options.type[0]
|
|
277
|
+
}
|
|
278
|
+
};
|
|
279
|
+
}
|
|
280
|
+
return createPropertyDecorator(DECORATORS.API_MODEL_PROPERTIES, options, overrideExisting);
|
|
281
|
+
}
|
|
282
|
+
|
|
283
|
+
function ApiOperationFromModel(ModelConstructor, verb, path) {
|
|
284
|
+
const httpToCrud = {
|
|
285
|
+
GET: [ OperationKeys.READ, Get ],
|
|
286
|
+
POST: [ OperationKeys.CREATE, Post ],
|
|
287
|
+
PUT: [ OperationKeys.UPDATE, Put ],
|
|
288
|
+
PATCH: [ OperationKeys.UPDATE, Patch ],
|
|
289
|
+
DELETE: [ OperationKeys.DELETE, Delete ]
|
|
290
|
+
};
|
|
291
|
+
const [crudOp, HttpMethodDecorator] = httpToCrud[verb];
|
|
292
|
+
const target = resolveBlockTarget(verb, path);
|
|
293
|
+
return target ? isOperationBlocked(ModelConstructor, target.kind, target.value) ? apply(ApiExcludeEndpoint()) : apply(HttpMethodDecorator(path)) : isOperationBlocked(ModelConstructor, crudOp) ? apply(ApiExcludeEndpoint()) : apply(HttpMethodDecorator(path));
|
|
294
|
+
}
|
|
295
|
+
|
|
296
|
+
function BulkApiOperationFromModel(ModelConstructor, verb, path) {
|
|
297
|
+
const httpToCrud = {
|
|
298
|
+
GET: [ BulkCrudOperationKeys.READ_ALL, Get ],
|
|
299
|
+
POST: [ BulkCrudOperationKeys.CREATE_ALL, Post ],
|
|
300
|
+
PUT: [ BulkCrudOperationKeys.UPDATE_ALL, Put ],
|
|
301
|
+
PATCH: [ BulkCrudOperationKeys.UPDATE_ALL, Patch ],
|
|
302
|
+
DELETE: [ BulkCrudOperationKeys.DELETE_ALL, Delete ]
|
|
303
|
+
};
|
|
304
|
+
const [crudOp, HttpMethodDecorator] = httpToCrud[verb];
|
|
305
|
+
const target = path ? resolveBlockTarget(verb, path) : undefined;
|
|
306
|
+
return target ? isOperationBlocked(ModelConstructor, target.kind, target.value) ? apply(ApiExcludeEndpoint()) : apply(HttpMethodDecorator(path)) : isOperationBlocked(ModelConstructor, "bulk", crudOp) ? apply(ApiExcludeEndpoint()) : apply(HttpMethodDecorator(path));
|
|
307
|
+
}
|
|
308
|
+
|
|
309
|
+
function resolveBlockTarget(verb, path) {
|
|
310
|
+
if (!path) return undefined;
|
|
311
|
+
const normalized = path.replace(/^\/+|\/+$/g, "");
|
|
312
|
+
const statementTargets = {
|
|
313
|
+
"listBy/:key": PreparedStatementKeys.LIST_BY,
|
|
314
|
+
"paginateBy/:key/:page": PreparedStatementKeys.PAGE_BY,
|
|
315
|
+
"find/:value": PreparedStatementKeys.FIND,
|
|
316
|
+
"page/:value": PreparedStatementKeys.PAGE,
|
|
317
|
+
"findOneBy/:key/:value": PreparedStatementKeys.FIND_ONE_BY,
|
|
318
|
+
"findBy/:key/:value": PreparedStatementKeys.FIND_BY,
|
|
319
|
+
"statement/:method/*args": "statement",
|
|
320
|
+
"countOf/:field": PreparedStatementKeys.COUNT_OF,
|
|
321
|
+
"maxOf/:field": PreparedStatementKeys.MAX_OF,
|
|
322
|
+
"minOf/:field": PreparedStatementKeys.MIN_OF,
|
|
323
|
+
"avgOf/:field": PreparedStatementKeys.AVG_OF,
|
|
324
|
+
"sumOf/:field": PreparedStatementKeys.SUM_OF,
|
|
325
|
+
"distinctOf/:field": PreparedStatementKeys.DISTINCT_OF,
|
|
326
|
+
"groupOf/:field": PreparedStatementKeys.GROUP_OF
|
|
327
|
+
};
|
|
328
|
+
if (normalized.startsWith("query/")) {
|
|
329
|
+
return {
|
|
330
|
+
kind: "query",
|
|
331
|
+
value: normalized.replace(/^query\//, "")
|
|
332
|
+
};
|
|
333
|
+
}
|
|
334
|
+
const statementValue = statementTargets[normalized];
|
|
335
|
+
if (statementValue) {
|
|
336
|
+
return {
|
|
337
|
+
kind: "statement",
|
|
338
|
+
value: statementValue
|
|
339
|
+
};
|
|
340
|
+
}
|
|
341
|
+
if (verb === "GET" && normalized === "") {
|
|
342
|
+
return {
|
|
343
|
+
kind: "crud",
|
|
344
|
+
value: OperationKeys.READ
|
|
345
|
+
};
|
|
346
|
+
}
|
|
347
|
+
return undefined;
|
|
348
|
+
}
|
|
349
|
+
|
|
350
|
+
function ApiParamsFromModel(props = []) {
|
|
351
|
+
const decorators = props.map(p => ApiParam({
|
|
352
|
+
name: p.name,
|
|
353
|
+
description: p.description ?? `Path parameter: ${p.name}`,
|
|
354
|
+
required: p.required ?? true,
|
|
355
|
+
type: p.type ?? String
|
|
356
|
+
}));
|
|
357
|
+
return applyDecorators(...decorators);
|
|
358
|
+
}
|
|
359
|
+
|
|
360
|
+
const DecafBody = createParamDecorator$1((data, ctx) => {
|
|
361
|
+
const request = ctx.switchToHttp().getRequest();
|
|
362
|
+
const body = request.body;
|
|
363
|
+
const controller = ctx.getClass();
|
|
364
|
+
const ModelConstr = controller.class;
|
|
365
|
+
if (!ModelConstr) {
|
|
366
|
+
throw new InternalServerErrorException(`ModelConstructor not found on controller ${controller.name}. Ensure the controller was created via FromModelController.`);
|
|
367
|
+
}
|
|
368
|
+
if (!body) return body;
|
|
369
|
+
if (Array.isArray(body)) {
|
|
370
|
+
return body.map(item => new ModelConstr(item));
|
|
371
|
+
}
|
|
372
|
+
return new ModelConstr(body);
|
|
373
|
+
});
|
|
374
|
+
|
|
375
|
+
const OrderedParams = createParamDecorator$1((order, ctx) => {
|
|
376
|
+
const req = ctx.switchToHttp().getRequest();
|
|
377
|
+
const original = req?.params ?? {};
|
|
378
|
+
const orderList = order ?? Object.keys(original);
|
|
379
|
+
const ordered = orderList.map(k => original[k]);
|
|
380
|
+
return {
|
|
381
|
+
raw: original,
|
|
382
|
+
valuesInOrder: ordered,
|
|
383
|
+
keysInOrder: orderList
|
|
384
|
+
};
|
|
385
|
+
});
|
|
386
|
+
|
|
387
|
+
function DecafParams(props = []) {
|
|
388
|
+
const order = props.map(p => p.name);
|
|
389
|
+
return OrderedParams(order);
|
|
390
|
+
}
|
|
391
|
+
|
|
392
|
+
const DecafQuery = createParamDecorator$1((_, ctx) => {
|
|
393
|
+
const req = ctx.switchToHttp().getRequest();
|
|
394
|
+
const query = req.query ?? {};
|
|
395
|
+
const parsed = {
|
|
396
|
+
...query
|
|
397
|
+
};
|
|
398
|
+
if (parsed.limit !== undefined) {
|
|
399
|
+
const n = Number(parsed.limit);
|
|
400
|
+
if (!Number.isNaN(n)) parsed.limit = n;
|
|
401
|
+
}
|
|
402
|
+
if (parsed.offset !== undefined) {
|
|
403
|
+
const n = Number(parsed.offset);
|
|
404
|
+
if (!Number.isNaN(n)) parsed.offset = n;
|
|
405
|
+
}
|
|
406
|
+
if (parsed.bookmark !== undefined) {
|
|
407
|
+
const n = Number(parsed.bookmark);
|
|
408
|
+
parsed.bookmark = Number.isNaN(n) ? parsed.bookmark : n;
|
|
409
|
+
}
|
|
410
|
+
return parsed;
|
|
411
|
+
});
|
|
412
|
+
|
|
413
|
+
const DECAF_MODULE_OPTIONS = "DecafModuleOptions";
|
|
414
|
+
|
|
415
|
+
const DECAF_ADAPTER_ID = "DecafAdapter";
|
|
416
|
+
|
|
417
|
+
const DECAF_TASK_SERVICE_ID = "DecafTaskService";
|
|
418
|
+
|
|
419
|
+
const DECAF_ROUTE = "DecafRoute";
|
|
420
|
+
|
|
421
|
+
const DECAF_HANDLERS = Symbol("DecafHandlers");
|
|
422
|
+
|
|
423
|
+
const DECAF_ADAPTER_OPTIONS = Symbol("DecafAdapterForOptions");
|
|
424
|
+
|
|
425
|
+
const DECAF_EXPOSE = "DecafExpose";
|
|
426
|
+
|
|
427
|
+
const DECAF_CONTROLLER_CONFIG = "DecafControllerConfig";
|
|
428
|
+
|
|
429
|
+
const DECAF_CONTEXT_KEY = Symbol("decaf:context");
|
|
430
|
+
|
|
431
|
+
function expose(...flavours) {
|
|
432
|
+
return function expose(target) {
|
|
433
|
+
Metadata.set(target, DECAF_EXPOSE, flavours.length ? flavours : true);
|
|
434
|
+
return target;
|
|
435
|
+
};
|
|
436
|
+
}
|
|
437
|
+
|
|
438
|
+
function controllerConfig(config) {
|
|
439
|
+
return function controllerConfigDecorator(target) {
|
|
440
|
+
Metadata.set(target, DECAF_CONTROLLER_CONFIG, config);
|
|
441
|
+
return target;
|
|
442
|
+
};
|
|
443
|
+
}
|
|
444
|
+
|
|
445
|
+
const AUTH_HANDLER = Symbol("AUTH_HANDLER");
|
|
446
|
+
|
|
447
|
+
const AUTH_META_KEY = "auth:meta";
|
|
448
|
+
|
|
449
|
+
const IS_PUBLIC_KEY = "isPublic";
|
|
450
|
+
|
|
451
|
+
const REQUIRED_ROLES_KEY = "requiredRoles";
|
|
452
|
+
|
|
453
|
+
let DecafRequestContext = class DecafRequestContext extends RequestContext {
|
|
454
|
+
constructor(req) {
|
|
455
|
+
super({
|
|
456
|
+
headersOf: request => request?.headers || undefined
|
|
457
|
+
}, req);
|
|
458
|
+
this.req = req;
|
|
459
|
+
this.uuid = UUID.instance.generate();
|
|
460
|
+
this.request = req;
|
|
461
|
+
}
|
|
462
|
+
put(record) {
|
|
463
|
+
let overrides;
|
|
464
|
+
try {
|
|
465
|
+
overrides = this.get("overrides");
|
|
466
|
+
} catch (e) {
|
|
467
|
+
overrides = {};
|
|
468
|
+
}
|
|
469
|
+
this.accumulate({
|
|
470
|
+
overrides: Object.assign(overrides, record)
|
|
471
|
+
});
|
|
472
|
+
}
|
|
473
|
+
};
|
|
474
|
+
|
|
475
|
+
DecafRequestContext = __decorate([ Injectable({
|
|
476
|
+
scope: Scope.REQUEST
|
|
477
|
+
}), __param(0, Inject(REQUEST)), __metadata("design:paramtypes", [ Object ]) ], DecafRequestContext);
|
|
478
|
+
|
|
479
|
+
let AuthInterceptor = class AuthInterceptor {
|
|
480
|
+
constructor(reflector, requestContext, authHandler) {
|
|
481
|
+
this.reflector = reflector;
|
|
482
|
+
this.requestContext = requestContext;
|
|
483
|
+
this.authHandler = authHandler;
|
|
484
|
+
}
|
|
485
|
+
async intercept(ctx, next) {
|
|
486
|
+
const log = Logging.for(this).for(this.intercept);
|
|
487
|
+
const isPublic = this.reflector.getAllAndOverride(IS_PUBLIC_KEY, [ ctx.getHandler(), ctx.getClass() ]);
|
|
488
|
+
const modelName = this.reflector.get(AUTH_META_KEY, ctx.getHandler()) ?? this.reflector.get(AUTH_META_KEY, ctx.getClass());
|
|
489
|
+
const requiredRoles = this.reflector.getAllAndOverride(REQUIRED_ROLES_KEY, [ ctx.getHandler(), ctx.getClass() ]);
|
|
490
|
+
log.verbose(`Intercepted request${modelName ? ` for ${modelName}` : ""}`);
|
|
491
|
+
if (isPublic) {
|
|
492
|
+
log.debug(`Public route — skipping auth`);
|
|
493
|
+
} else if (this.authHandler) {
|
|
494
|
+
await this.authHandler.authorize(ctx, modelName, requiredRoles, this.requestContext);
|
|
495
|
+
} else {
|
|
496
|
+
log.debug(`No auth handler registered`);
|
|
497
|
+
}
|
|
498
|
+
await this.applyTransformers();
|
|
499
|
+
const user = this.requestContext.getOrUndefined("user");
|
|
500
|
+
const organization = this.requestContext.getOrUndefined("organization");
|
|
501
|
+
if (user || organization) {
|
|
502
|
+
const currentLog = this.requestContext.get("logger");
|
|
503
|
+
const childLog = currentLog.for({
|
|
504
|
+
user: user,
|
|
505
|
+
organization: organization
|
|
506
|
+
});
|
|
507
|
+
this.requestContext.accumulate({
|
|
508
|
+
logger: childLog
|
|
509
|
+
});
|
|
510
|
+
}
|
|
511
|
+
return next.handle();
|
|
512
|
+
}
|
|
513
|
+
async applyTransformers() {
|
|
514
|
+
const flavours = Adapter.flavoursToTransform();
|
|
515
|
+
if (!flavours) return;
|
|
516
|
+
for (const flavour of flavours) {
|
|
517
|
+
const transformer = Adapter.transformerFor(flavour);
|
|
518
|
+
if (!transformer) continue;
|
|
519
|
+
const from = await transformer.from(this.requestContext);
|
|
520
|
+
if (from) this.requestContext.accumulate(from);
|
|
521
|
+
}
|
|
522
|
+
}
|
|
523
|
+
};
|
|
524
|
+
|
|
525
|
+
AuthInterceptor = __decorate([ Injectable({
|
|
526
|
+
scope: Scope.REQUEST
|
|
527
|
+
}), __param(2, Optional()), __param(2, Inject(AUTH_HANDLER)), __metadata("design:paramtypes", [ Reflector, DecafRequestContext, Object ]) ], AuthInterceptor);
|
|
528
|
+
|
|
529
|
+
class DecafAuthHandler extends AuthHandler {
|
|
530
|
+
parseRequest(req) {
|
|
531
|
+
const userRole = req.headers.authorization?.split(" ")[1];
|
|
532
|
+
return userRole;
|
|
533
|
+
}
|
|
534
|
+
extractFromAuth(ctx) {
|
|
535
|
+
const req = ctx.switchToHttp().getRequest();
|
|
536
|
+
const userRole = this.parseRequest(req);
|
|
537
|
+
if (!userRole) throw new AuthorizationError("Unauthenticated");
|
|
538
|
+
return {
|
|
539
|
+
user: userRole,
|
|
540
|
+
roles: [ userRole ]
|
|
541
|
+
};
|
|
542
|
+
}
|
|
543
|
+
}
|
|
544
|
+
|
|
545
|
+
class DecafRoleAuthHandler extends DecafAuthHandler {}
|
|
546
|
+
|
|
547
|
+
function Auth(model) {
|
|
548
|
+
const resource = model ? typeof model === "string" ? model : model.name : undefined;
|
|
549
|
+
const decs = [ ApiBearerAuth(), UseInterceptors(AuthInterceptor) ];
|
|
550
|
+
if (resource) decs.push(SetMetadata(AUTH_META_KEY, resource));
|
|
551
|
+
return applyDecorators(...decs);
|
|
552
|
+
}
|
|
553
|
+
|
|
554
|
+
function Public() {
|
|
555
|
+
return applyDecorators(SetMetadata(IS_PUBLIC_KEY, true));
|
|
556
|
+
}
|
|
557
|
+
|
|
558
|
+
function RequireRoles(...roles) {
|
|
559
|
+
return applyDecorators(ApiSecurity("bearer"), SetMetadata(REQUIRED_ROLES_KEY, roles), UseInterceptors(AuthInterceptor));
|
|
560
|
+
}
|
|
561
|
+
|
|
562
|
+
var DecafHandlerExecutor_1;
|
|
563
|
+
|
|
564
|
+
let DecafHandlerExecutor = DecafHandlerExecutor_1 = class DecafHandlerExecutor {
|
|
565
|
+
constructor(handlers, context) {
|
|
566
|
+
this.handlers = handlers;
|
|
567
|
+
this.context = context;
|
|
568
|
+
}
|
|
569
|
+
async exec(req, res) {
|
|
570
|
+
const log = Logging.for(DecafHandlerExecutor_1.name).for(this.exec);
|
|
571
|
+
log.debug(`CONTEXT ${this.context.uuid} running ${this.handlers.length} handlers for request ${req.method} ${req.url}`);
|
|
572
|
+
for (const handler of this.handlers) {
|
|
573
|
+
await handler.handle(this.context, req, res);
|
|
574
|
+
}
|
|
575
|
+
}
|
|
576
|
+
};
|
|
577
|
+
|
|
578
|
+
DecafHandlerExecutor = DecafHandlerExecutor_1 = __decorate([ Injectable({
|
|
579
|
+
scope: Scope.REQUEST
|
|
580
|
+
}), __param(0, Inject(DECAF_HANDLERS)), __metadata("design:paramtypes", [ Array, DecafRequestContext ]) ], DecafHandlerExecutor);
|
|
581
|
+
|
|
582
|
+
let DecafResponseInterceptor = class DecafResponseInterceptor {
|
|
583
|
+
constructor(ctx) {
|
|
584
|
+
this.ctx = ctx;
|
|
585
|
+
}
|
|
586
|
+
intercept(context, next) {
|
|
587
|
+
let response = context.switchToHttp().getResponse();
|
|
588
|
+
return next.handle().pipe(tap(data => {
|
|
589
|
+
response = this.ctx.toResponse(response);
|
|
590
|
+
}));
|
|
591
|
+
}
|
|
592
|
+
};
|
|
593
|
+
|
|
594
|
+
DecafResponseInterceptor = __decorate([ Injectable(), __metadata("design:paramtypes", [ DecafRequestContext ]) ], DecafResponseInterceptor);
|
|
595
|
+
|
|
596
|
+
Adapter.transformerFor = function toContextFlags(adapter) {
|
|
597
|
+
const alias = typeof adapter === "string" ? adapter : adapter.alias;
|
|
598
|
+
return Metadata["innerGet"](Symbol.for("transformers"), alias);
|
|
599
|
+
}.bind(Adapter);
|
|
600
|
+
|
|
601
|
+
Adapter.flavoursToTransform = function requestTransformers() {
|
|
602
|
+
const meta = Metadata["innerGet"](Symbol.for("transformers"));
|
|
603
|
+
if (!meta) return undefined;
|
|
604
|
+
return Object.keys(meta);
|
|
605
|
+
}.bind(Adapter);
|
|
606
|
+
|
|
607
|
+
Context.prototype.toResponse = function toResponse(res) {
|
|
608
|
+
const pending = this.pending();
|
|
609
|
+
if (pending) res.header("x-pending-task", JSON.stringify(pending));
|
|
610
|
+
return res;
|
|
611
|
+
};
|
|
612
|
+
|
|
613
|
+
const prototype = ModelBuilder.prototype;
|
|
614
|
+
|
|
615
|
+
if (!prototype.decorateClass) {
|
|
616
|
+
prototype.decorateClass = function(decorator) {
|
|
617
|
+
if (!this._classDecorators) {
|
|
618
|
+
this._classDecorators = [];
|
|
619
|
+
}
|
|
620
|
+
this._classDecorators.push(decorator);
|
|
621
|
+
return this;
|
|
622
|
+
};
|
|
623
|
+
}
|
|
624
|
+
|
|
625
|
+
prototype.Auth = function(model) {
|
|
626
|
+
return this.decorateClass(Auth(model));
|
|
627
|
+
};
|
|
628
|
+
|
|
629
|
+
if (!prototype.__hasClassDecoratorSupport) {
|
|
630
|
+
const originalBuild = prototype.build;
|
|
631
|
+
prototype.build = function() {
|
|
632
|
+
let result = originalBuild.call(this);
|
|
633
|
+
const decorators = this._classDecorators;
|
|
634
|
+
if (decorators?.length) {
|
|
635
|
+
for (const decorator of decorators) {
|
|
636
|
+
const decorated = decorator(result);
|
|
637
|
+
if (decorated) {
|
|
638
|
+
result = decorated;
|
|
639
|
+
}
|
|
640
|
+
}
|
|
641
|
+
}
|
|
642
|
+
return result;
|
|
643
|
+
};
|
|
644
|
+
prototype.__hasClassDecoratorSupport = true;
|
|
645
|
+
}
|
|
646
|
+
|
|
647
|
+
var DecafRequestHandlerInterceptor_1;
|
|
648
|
+
|
|
649
|
+
let DecafRequestHandlerInterceptor = DecafRequestHandlerInterceptor_1 = class DecafRequestHandlerInterceptor {
|
|
650
|
+
constructor(requestContext, executor) {
|
|
651
|
+
this.requestContext = requestContext;
|
|
652
|
+
this.executor = executor;
|
|
653
|
+
}
|
|
654
|
+
async contextualize(req) {
|
|
655
|
+
const headers = req.headers;
|
|
656
|
+
const flags = {
|
|
657
|
+
headers: headers,
|
|
658
|
+
overrides: {}
|
|
659
|
+
};
|
|
660
|
+
Object.assign(flags.overrides, req?.[DECAF_ADAPTER_OPTIONS] ?? {});
|
|
661
|
+
const ip = extractIp(req);
|
|
662
|
+
const logger = Logging.get().for({
|
|
663
|
+
ip: ip
|
|
664
|
+
});
|
|
665
|
+
this.requestContext.accumulate(Object.assign({}, DefaultAdapterFlags, {
|
|
666
|
+
logger: logger,
|
|
667
|
+
timestamp: new Date,
|
|
668
|
+
operation: `${req.method} ${req.url}`
|
|
669
|
+
}, flags));
|
|
670
|
+
}
|
|
671
|
+
async intercept(context, next) {
|
|
672
|
+
const req = context.switchToHttp().getRequest();
|
|
673
|
+
const res = context.switchToHttp().getResponse();
|
|
674
|
+
const log = Logging.for(DecafRequestHandlerInterceptor_1).for(this.intercept);
|
|
675
|
+
log.debug(`CONTEXT ${this.requestContext.uuid} - request: ${req.method} ${req.url}`);
|
|
676
|
+
await this.contextualize(req);
|
|
677
|
+
log.debug(`CONTEXT ${this.requestContext.uuid} contextualized - request: ${req.method} ${req.url}`);
|
|
678
|
+
await this.executor.exec(req, res);
|
|
679
|
+
log.debug(`CONTEXT ${this.requestContext.uuid} executors finished - request: ${req.method} ${req.url}`);
|
|
680
|
+
return next.handle();
|
|
681
|
+
}
|
|
682
|
+
};
|
|
683
|
+
|
|
684
|
+
DecafRequestHandlerInterceptor = DecafRequestHandlerInterceptor_1 = __decorate([ Injectable({
|
|
685
|
+
scope: Scope.REQUEST
|
|
686
|
+
}), __metadata("design:paramtypes", [ DecafRequestContext, DecafHandlerExecutor ]) ], DecafRequestHandlerInterceptor);
|
|
687
|
+
|
|
688
|
+
function extractIp(req) {
|
|
689
|
+
const headers = req.headers;
|
|
690
|
+
function parseIpHeader(value) {
|
|
691
|
+
if (!value) return undefined;
|
|
692
|
+
const candidate = Array.isArray(value) ? value[0] : value;
|
|
693
|
+
return candidate.split(",").map(segment => segment.trim()).filter(Boolean)[0];
|
|
694
|
+
}
|
|
695
|
+
return parseIpHeader(headers?.["x-forwarded-for"]) ?? parseIpHeader(headers?.["x-real-ip"]) ?? parseIpHeader(headers?.["X-Forwarded-For"]) ?? parseIpHeader(headers?.["X-Real-IP"]) ?? req.ip;
|
|
696
|
+
}
|
|
697
|
+
|
|
698
|
+
var DecafAuthModule_1;
|
|
699
|
+
|
|
700
|
+
let DecafAuthModule = DecafAuthModule_1 = class DecafAuthModule {
|
|
701
|
+
static forRoot(options = {}) {
|
|
702
|
+
const providers = [ AuthInterceptor, DecafRequestHandlerInterceptor, {
|
|
703
|
+
provide: APP_INTERCEPTOR,
|
|
704
|
+
useClass: DecafRequestHandlerInterceptor
|
|
705
|
+
} ];
|
|
706
|
+
if (options.handler) {
|
|
707
|
+
providers.push(options.handler);
|
|
708
|
+
providers.push({
|
|
709
|
+
provide: AUTH_HANDLER,
|
|
710
|
+
useClass: options.handler
|
|
711
|
+
});
|
|
712
|
+
}
|
|
713
|
+
if (options.global) {
|
|
714
|
+
providers.push({
|
|
715
|
+
provide: APP_INTERCEPTOR,
|
|
716
|
+
useExisting: AuthInterceptor
|
|
717
|
+
});
|
|
718
|
+
}
|
|
719
|
+
return {
|
|
720
|
+
module: DecafAuthModule_1,
|
|
721
|
+
global: options.global ?? false,
|
|
722
|
+
providers: providers,
|
|
723
|
+
exports: [ AuthInterceptor, AUTH_HANDLER ]
|
|
724
|
+
};
|
|
725
|
+
}
|
|
726
|
+
};
|
|
727
|
+
|
|
728
|
+
DecafAuthModule = DecafAuthModule_1 = __decorate([ Module({}) ], DecafAuthModule);
|
|
729
|
+
|
|
730
|
+
function HttpVerbToDecorator(verb) {
|
|
731
|
+
const httpToCrud = {
|
|
732
|
+
GET: Get,
|
|
733
|
+
POST: Post,
|
|
734
|
+
PUT: Put,
|
|
735
|
+
PATCH: Patch,
|
|
736
|
+
DELETE: Delete
|
|
737
|
+
};
|
|
738
|
+
const decorator = httpToCrud[verb];
|
|
739
|
+
if (!decorator) {
|
|
740
|
+
throw new Error(`Unsupported HTTP verb "${verb}". No NestJS decorator mapping was found.`);
|
|
741
|
+
}
|
|
742
|
+
return decorator;
|
|
743
|
+
}
|
|
744
|
+
|
|
745
|
+
const extractPathParams = routePath => routePath.split("/").filter(p => p.startsWith(":")).map(p => p.slice(1));
|
|
746
|
+
|
|
747
|
+
const apiParamSpec = name => ({
|
|
748
|
+
name: name,
|
|
749
|
+
description: `${name} parameter for the query`,
|
|
750
|
+
required: true,
|
|
751
|
+
type: String
|
|
752
|
+
});
|
|
753
|
+
|
|
754
|
+
function getApiDecorators(methodName, routePath, httpVerb, includeQueryParams = false) {
|
|
755
|
+
const NestHttpRouteDec = HttpVerbToDecorator(httpVerb);
|
|
756
|
+
const apiPathParams = extractPathParams(routePath).map(apiParamSpec);
|
|
757
|
+
const swaggerQueryParams = [];
|
|
758
|
+
if (httpVerb === "GET" && includeQueryParams) {
|
|
759
|
+
swaggerQueryParams.push(ApiQuery({
|
|
760
|
+
name: "direction",
|
|
761
|
+
required: false,
|
|
762
|
+
enum: OrderDirection,
|
|
763
|
+
description: "the sort order when applicable"
|
|
764
|
+
}), ApiQuery({
|
|
765
|
+
name: "limit",
|
|
766
|
+
required: false,
|
|
767
|
+
description: "limit or page size when applicable"
|
|
768
|
+
}), ApiQuery({
|
|
769
|
+
name: "offset",
|
|
770
|
+
required: false,
|
|
771
|
+
description: "offset or bookmark when applicable"
|
|
772
|
+
}));
|
|
773
|
+
}
|
|
774
|
+
return {
|
|
775
|
+
method: [ NestHttpRouteDec(routePath), ...apiPathParams.map(ApiParam), ...swaggerQueryParams, ApiOperation({
|
|
776
|
+
summary: `Retrieve records using according to "${methodName}".`
|
|
777
|
+
}), ApiOkResponse({
|
|
778
|
+
description: `Result successfully retrieved.`
|
|
779
|
+
}), ApiNoContentResponse({
|
|
780
|
+
description: `No content returned by the method.`
|
|
781
|
+
}) ],
|
|
782
|
+
params: [ DecafParams(apiPathParams), Query() ]
|
|
783
|
+
};
|
|
784
|
+
}
|
|
785
|
+
|
|
786
|
+
function applyApiDecorators(target, methodName, descriptor, decorators) {
|
|
787
|
+
const proto = target?.prototype ?? target;
|
|
788
|
+
decorators.method.forEach(d => d(proto, methodName, descriptor));
|
|
789
|
+
decorators.params?.forEach((d, index) => d(proto, methodName, index));
|
|
790
|
+
}
|
|
791
|
+
|
|
792
|
+
function resolvePersistenceMethod(persistence, methodName, ...args) {
|
|
793
|
+
if (persistence instanceof ModelService) {
|
|
794
|
+
return typeof persistence[methodName] === "function" ? persistence[methodName](...args) : persistence.statement(methodName, ...args);
|
|
795
|
+
}
|
|
796
|
+
if (typeof persistence[methodName] === "function") return persistence[methodName](...args);
|
|
797
|
+
throw new Error(`Persistence method "${methodName}" not found on ${persistence?.constructor?.name}`);
|
|
798
|
+
}
|
|
799
|
+
|
|
800
|
+
function createRouteHandler(methodName) {
|
|
801
|
+
return async function(pathParams, queryParams) {
|
|
802
|
+
const log = this.log.for(methodName);
|
|
803
|
+
try {
|
|
804
|
+
log.debug(`Invoking persistence method "${methodName}" given parameters: ${JSON.stringify(pathParams.valuesInOrder)}`);
|
|
805
|
+
const {direction: direction, limit: limit, offset: offset} = queryParams;
|
|
806
|
+
return await resolvePersistenceMethod(this.persistence(), methodName, ...pathParams.valuesInOrder, direction, limit, offset);
|
|
807
|
+
} catch (e) {
|
|
808
|
+
log.error(`Custom query "${methodName}" failed`, e);
|
|
809
|
+
throw e;
|
|
810
|
+
}
|
|
811
|
+
};
|
|
812
|
+
}
|
|
813
|
+
|
|
814
|
+
function defineRouteMethod(ControllerClass, methodName, handler) {
|
|
815
|
+
Object.defineProperty(ControllerClass.prototype || ControllerClass, methodName, {
|
|
816
|
+
value: handler,
|
|
817
|
+
writable: false
|
|
818
|
+
});
|
|
819
|
+
return Object.getOwnPropertyDescriptor(ControllerClass.prototype || ControllerClass, methodName);
|
|
820
|
+
}
|
|
821
|
+
|
|
822
|
+
class DecafController extends DecafController$1 {
|
|
823
|
+
constructor(clientContext, _name) {
|
|
824
|
+
super(clientContext);
|
|
825
|
+
this.clientContext = clientContext;
|
|
826
|
+
}
|
|
827
|
+
}
|
|
828
|
+
|
|
829
|
+
class DecafModelController extends DecafController {
|
|
830
|
+
constructor(clientContext, _name) {
|
|
831
|
+
super(clientContext, _name);
|
|
832
|
+
this.clientContext = clientContext;
|
|
833
|
+
}
|
|
834
|
+
persistence(ctx) {
|
|
835
|
+
if (!this._persistence) try {
|
|
836
|
+
this._persistence = Service$1.get(this.class);
|
|
837
|
+
} catch (e) {
|
|
838
|
+
try {
|
|
839
|
+
this._persistence = ModelService.getService(this.class);
|
|
840
|
+
} catch (e) {
|
|
841
|
+
this._persistence = Repository$1.forModel(this.class);
|
|
842
|
+
}
|
|
843
|
+
}
|
|
844
|
+
const certs = this.clientContext.request[DECAF_ADAPTER_OPTIONS] || {};
|
|
845
|
+
if (ctx) {
|
|
846
|
+
this.clientContext.put(certs);
|
|
847
|
+
}
|
|
848
|
+
return ctx ? this._persistence instanceof Repository$1 ? this._persistence.override(certs) : this._persistence.for(certs) : this._persistence;
|
|
849
|
+
}
|
|
850
|
+
}
|
|
851
|
+
|
|
852
|
+
const dtoCache = new Map;
|
|
853
|
+
|
|
854
|
+
function DtoFor(op, model, stack = new Set) {
|
|
855
|
+
if (!TransactionOperationKeys.includes(op)) {
|
|
856
|
+
return model;
|
|
857
|
+
}
|
|
858
|
+
const cache = getDtoCache(op);
|
|
859
|
+
const cached = cache.get(model);
|
|
860
|
+
if (cached) return cached;
|
|
861
|
+
const isUpdateOp = [ OperationKeys.UPDATE, BulkCrudOperationKeys.UPDATE_ALL ].includes(op);
|
|
862
|
+
class DynamicDTO {}
|
|
863
|
+
cache.set(model, DynamicDTO);
|
|
864
|
+
Object.defineProperty(DynamicDTO, "name", {
|
|
865
|
+
value: `${toPascalCase(model.name)}${toPascalCase(op)}DTO`
|
|
866
|
+
});
|
|
867
|
+
const pkProp = (() => {
|
|
868
|
+
try {
|
|
869
|
+
return Model.pk(model);
|
|
870
|
+
} catch {
|
|
871
|
+
return undefined;
|
|
872
|
+
}
|
|
873
|
+
})();
|
|
874
|
+
const pkPropsMetadata = pkProp ? Model.pkProps(model) : undefined;
|
|
875
|
+
const pkDesignType = pkProp ? Reflect.getMetadata("design:type", model.prototype, pkProp) : undefined;
|
|
876
|
+
const pkTypeIsNumeric = pkDesignType === Number || pkDesignType === BigInt;
|
|
877
|
+
const pkIsGenerated = !!pkPropsMetadata?.generated || (pkProp ? Model.generatedBySequence(model, pkProp) || isPropertyGeneratedAcrossInheritance(model, pkProp) || pkTypeIsNumeric : false);
|
|
878
|
+
const allProps = Array.from(new Set(Metadata.properties(model) || []));
|
|
879
|
+
const relations = new Set(Model.relations(model) || []);
|
|
880
|
+
const scalarProps = [];
|
|
881
|
+
for (const prop of allProps) {
|
|
882
|
+
if (!prop) continue;
|
|
883
|
+
if (relations.has(prop)) continue;
|
|
884
|
+
if (prop === pkProp && !isUpdateOp && pkIsGenerated) continue;
|
|
885
|
+
if (prop !== pkProp && isPropertyGeneratedAcrossInheritance(model, prop)) continue;
|
|
886
|
+
scalarProps.push(prop);
|
|
887
|
+
}
|
|
888
|
+
for (const prop of scalarProps) {
|
|
889
|
+
const validation = getValidationAcrossInheritance(model, prop);
|
|
890
|
+
const isRequired = !!validation?.[ValidationKeys.REQUIRED];
|
|
891
|
+
const typeHint = getTypeAcrossInheritance(model, prop) ?? Reflect.getMetadata("design:type", model.prototype, prop);
|
|
892
|
+
const apiOptions = {
|
|
893
|
+
required: isRequired
|
|
894
|
+
};
|
|
895
|
+
if (typeHint) apiOptions.type = typeHint;
|
|
896
|
+
ApiProperty(apiOptions)(DynamicDTO.prototype, prop);
|
|
897
|
+
const designType = Reflect.getMetadata("design:type", model.prototype, prop) ?? typeHint;
|
|
898
|
+
if (typeof designType !== "undefined") {
|
|
899
|
+
Reflect.defineMetadata("design:type", designType, DynamicDTO.prototype, prop);
|
|
900
|
+
}
|
|
901
|
+
Object.defineProperty(DynamicDTO.prototype, prop, {
|
|
902
|
+
value: undefined,
|
|
903
|
+
writable: true,
|
|
904
|
+
enumerable: true,
|
|
905
|
+
configurable: true
|
|
906
|
+
});
|
|
907
|
+
}
|
|
908
|
+
for (const relation of relations) {
|
|
909
|
+
const relationMeta = Metadata.relations(model, relation);
|
|
910
|
+
if (!relationMeta) {
|
|
911
|
+
throw new InternalError(`Metadata for relation ${relation} not found`);
|
|
912
|
+
}
|
|
913
|
+
let relationType = relationMeta.class;
|
|
914
|
+
if (typeof relationType === "function" && !relationType.name) {
|
|
915
|
+
relationType = relationType();
|
|
916
|
+
}
|
|
917
|
+
if (!relationType || typeof relationType !== "function") {
|
|
918
|
+
throw new InternalError(`Type for relation ${relation} not found`);
|
|
919
|
+
}
|
|
920
|
+
if (!Model.get(relationType.name)) {
|
|
921
|
+
continue;
|
|
922
|
+
}
|
|
923
|
+
const meta = Metadata.validationFor(model, relation);
|
|
924
|
+
const isArray = !!meta?.[ValidationKeys.LIST];
|
|
925
|
+
const isRequired = !!meta?.[ValidationKeys.REQUIRED];
|
|
926
|
+
const isCircular = stack.has(relationType);
|
|
927
|
+
if (isCircular) {
|
|
928
|
+
const pkTypeName = getPkOpenApiType(relationType);
|
|
929
|
+
addRelationPkRef(DynamicDTO, relation, pkTypeName, isArray, isRequired);
|
|
930
|
+
continue;
|
|
931
|
+
}
|
|
932
|
+
const relationDto = DtoFor(op, relationType, new Set(stack).add(model));
|
|
933
|
+
if (isUpdateOp) {
|
|
934
|
+
addRelationUpdate(DynamicDTO, relation, relationType, relationDto, isArray, isRequired);
|
|
935
|
+
} else {
|
|
936
|
+
addRelation(DynamicDTO, relation, relationDto, isArray, isRequired);
|
|
937
|
+
}
|
|
938
|
+
}
|
|
939
|
+
return DynamicDTO;
|
|
940
|
+
}
|
|
941
|
+
|
|
942
|
+
function addRelation(DtoClass, relation, relationDto, isArray, isRequired) {
|
|
943
|
+
const apiOptions = {
|
|
944
|
+
type: relationDto,
|
|
945
|
+
required: isRequired,
|
|
946
|
+
isArray: isArray
|
|
947
|
+
};
|
|
948
|
+
ApiProperty(apiOptions)(DtoClass.prototype, relation);
|
|
949
|
+
Reflect.defineMetadata("design:type", isArray ? Array : relationDto, DtoClass.prototype, relation);
|
|
950
|
+
Object.defineProperty(DtoClass.prototype, relation, {
|
|
951
|
+
value: undefined,
|
|
952
|
+
writable: true,
|
|
953
|
+
enumerable: true,
|
|
954
|
+
configurable: true
|
|
955
|
+
});
|
|
956
|
+
}
|
|
957
|
+
|
|
958
|
+
function addRelationPkRef(DtoClass, relation, pkTypeName, isArray, isRequired) {
|
|
959
|
+
const apiOptions = isArray ? {
|
|
960
|
+
type: "array",
|
|
961
|
+
items: {
|
|
962
|
+
type: pkTypeName
|
|
963
|
+
},
|
|
964
|
+
required: isRequired
|
|
965
|
+
} : {
|
|
966
|
+
type: pkTypeName === "integer" ? Number : String,
|
|
967
|
+
required: isRequired
|
|
968
|
+
};
|
|
969
|
+
ApiProperty(apiOptions)(DtoClass.prototype, relation);
|
|
970
|
+
Reflect.defineMetadata("design:type", isArray ? Array : pkTypeName === "integer" ? Number : String, DtoClass.prototype, relation);
|
|
971
|
+
Object.defineProperty(DtoClass.prototype, relation, {
|
|
972
|
+
value: undefined,
|
|
973
|
+
writable: true,
|
|
974
|
+
enumerable: true,
|
|
975
|
+
configurable: true
|
|
976
|
+
});
|
|
977
|
+
}
|
|
978
|
+
|
|
979
|
+
function addRelationUpdate(DtoClass, relation, relationType, relationDto, isArray, isRequired) {
|
|
980
|
+
const extraModels = Reflect.getMetadata(DECORATORS.API_EXTRA_MODELS, DtoClass) || [];
|
|
981
|
+
if (!extraModels.includes(relationDto)) {
|
|
982
|
+
Reflect.defineMetadata(DECORATORS.API_EXTRA_MODELS, [ ...extraModels, relationDto ], DtoClass);
|
|
983
|
+
}
|
|
984
|
+
const dtoRef = `#/components/schemas/${relationDto.name}`;
|
|
985
|
+
const pkTypeName = getPkOpenApiType(relationType);
|
|
986
|
+
const oneOfItems = [ {
|
|
987
|
+
$ref: dtoRef
|
|
988
|
+
}, {
|
|
989
|
+
type: pkTypeName
|
|
990
|
+
} ];
|
|
991
|
+
const apiOptions = isArray ? {
|
|
992
|
+
type: "array",
|
|
993
|
+
required: isRequired,
|
|
994
|
+
oneOf: oneOfItems
|
|
995
|
+
} : {
|
|
996
|
+
type: Object,
|
|
997
|
+
required: isRequired,
|
|
998
|
+
oneOf: oneOfItems
|
|
999
|
+
};
|
|
1000
|
+
ApiProperty(apiOptions)(DtoClass.prototype, relation);
|
|
1001
|
+
Reflect.defineMetadata("design:type", isArray ? Array : Object, DtoClass.prototype, relation);
|
|
1002
|
+
Object.defineProperty(DtoClass.prototype, relation, {
|
|
1003
|
+
value: undefined,
|
|
1004
|
+
writable: true,
|
|
1005
|
+
enumerable: true,
|
|
1006
|
+
configurable: true
|
|
1007
|
+
});
|
|
1008
|
+
}
|
|
1009
|
+
|
|
1010
|
+
function getPkOpenApiType(relationType) {
|
|
1011
|
+
try {
|
|
1012
|
+
const pkPropsMetadata = Model.pkProps(relationType);
|
|
1013
|
+
const pkType = pkPropsMetadata?.type;
|
|
1014
|
+
if (pkType === Number || pkType === BigInt) return "integer";
|
|
1015
|
+
return "string";
|
|
1016
|
+
} catch {
|
|
1017
|
+
return "string";
|
|
1018
|
+
}
|
|
1019
|
+
}
|
|
1020
|
+
|
|
1021
|
+
function isPropertyGeneratedAcrossInheritance(model, prop) {
|
|
1022
|
+
let current = model;
|
|
1023
|
+
while (current && current !== Object && current !== Function) {
|
|
1024
|
+
if (Model.generated(current, prop)) return true;
|
|
1025
|
+
current = Object.getPrototypeOf(current);
|
|
1026
|
+
}
|
|
1027
|
+
return false;
|
|
1028
|
+
}
|
|
1029
|
+
|
|
1030
|
+
function getValidationAcrossInheritance(model, prop) {
|
|
1031
|
+
let current = model;
|
|
1032
|
+
while (current && current !== Object && current !== Function) {
|
|
1033
|
+
const validation = Metadata.validationFor(current, prop);
|
|
1034
|
+
if (validation) return validation;
|
|
1035
|
+
current = Object.getPrototypeOf(current);
|
|
1036
|
+
}
|
|
1037
|
+
return undefined;
|
|
1038
|
+
}
|
|
1039
|
+
|
|
1040
|
+
function getTypeAcrossInheritance(model, prop) {
|
|
1041
|
+
let current = model;
|
|
1042
|
+
while (current && current !== Object && current !== Function) {
|
|
1043
|
+
const type = Metadata.type(current, prop);
|
|
1044
|
+
if (type) return type;
|
|
1045
|
+
current = Object.getPrototypeOf(current);
|
|
1046
|
+
}
|
|
1047
|
+
return undefined;
|
|
1048
|
+
}
|
|
1049
|
+
|
|
1050
|
+
function getDtoCache(op) {
|
|
1051
|
+
if (!dtoCache.has(op)) {
|
|
1052
|
+
dtoCache.set(op, new WeakMap);
|
|
1053
|
+
}
|
|
1054
|
+
return dtoCache.get(op);
|
|
1055
|
+
}
|
|
1056
|
+
|
|
1057
|
+
var DynamicModelController_1;
|
|
1058
|
+
|
|
1059
|
+
class FromModelController {
|
|
1060
|
+
static {
|
|
1061
|
+
this.log = Logging.for(FromModelController.name);
|
|
1062
|
+
}
|
|
1063
|
+
static getPersistence(ModelClazz) {
|
|
1064
|
+
try {
|
|
1065
|
+
return Service$1.get(ModelClazz);
|
|
1066
|
+
} catch (e) {
|
|
1067
|
+
try {
|
|
1068
|
+
return ModelService.getService(ModelClazz);
|
|
1069
|
+
} catch (e2) {
|
|
1070
|
+
return Repository$1.forModel(ModelClazz);
|
|
1071
|
+
}
|
|
1072
|
+
}
|
|
1073
|
+
}
|
|
1074
|
+
static createQueryRoutesFromRepository(persistence, prefix = PersistenceKeys.QUERY) {
|
|
1075
|
+
const repo = persistence instanceof ModelService ? persistence.repo : persistence;
|
|
1076
|
+
const ModelConstr = repo.class;
|
|
1077
|
+
const queryMethods = Metadata.get(repo.constructor, Metadata.key(PersistenceKeys.QUERY)) ?? {};
|
|
1078
|
+
const routeMethods = Metadata.get(persistence.constructor, Metadata.key(DECAF_ROUTE)) ?? {};
|
|
1079
|
+
class QueryController extends DecafModelController {
|
|
1080
|
+
get class() {
|
|
1081
|
+
throw new Error("Method not implemented.");
|
|
1082
|
+
}
|
|
1083
|
+
constructor(clientContext, name) {
|
|
1084
|
+
super(clientContext, name);
|
|
1085
|
+
}
|
|
1086
|
+
}
|
|
1087
|
+
for (const [methodName, params] of Object.entries(routeMethods)) {
|
|
1088
|
+
const routePath = [ params.path.replace(/^\/+|\/+$/g, "") ].filter(segment => segment && segment.trim()).join("/");
|
|
1089
|
+
const handler = FromModelController.createComplexQueryHandler(methodName);
|
|
1090
|
+
FromModelController.defineMethod(QueryController, methodName, handler);
|
|
1091
|
+
const httpDecorator = HttpVerbToDecorator(params.httpMethod)(routePath || undefined);
|
|
1092
|
+
const decorators = FromModelController.getQueryDecorators(methodName, routePath, params.httpMethod);
|
|
1093
|
+
FromModelController.applyDecorators(QueryController, methodName, [ httpDecorator, ...decorators ]);
|
|
1094
|
+
}
|
|
1095
|
+
for (const [methodName, objValues] of Object.entries(queryMethods)) {
|
|
1096
|
+
const fields = objValues.fields ?? [];
|
|
1097
|
+
const routePath = [ prefix, methodName, ...fields.map(f => `:${f}`) ].filter(segment => segment && segment.trim()).join("/");
|
|
1098
|
+
const handler = FromModelController.createComplexQueryHandler(methodName);
|
|
1099
|
+
FromModelController.defineMethod(QueryController, methodName, handler);
|
|
1100
|
+
const httpDecorator = HttpVerbToDecorator("GET")(routePath || undefined);
|
|
1101
|
+
const decorators = FromModelController.getQueryDecorators(methodName, routePath, "GET", true);
|
|
1102
|
+
FromModelController.applyDecorators(QueryController, methodName, [ httpDecorator, ...decorators ]);
|
|
1103
|
+
}
|
|
1104
|
+
return QueryController;
|
|
1105
|
+
}
|
|
1106
|
+
static create(ModelConstr, moduleConfigOverrides) {
|
|
1107
|
+
const log = FromModelController.log.for(FromModelController.create);
|
|
1108
|
+
const tableName = Model.tableName(ModelConstr);
|
|
1109
|
+
const routePath = toKebabCase(tableName);
|
|
1110
|
+
const modelClazzName = ModelConstr.name;
|
|
1111
|
+
const persistence = FromModelController.getPersistence(ModelConstr);
|
|
1112
|
+
const factoryPersistence = persistence instanceof ModelService ? persistence.repo : persistence;
|
|
1113
|
+
const decoratorConfig = Metadata.get(ModelConstr, Metadata.key(DECAF_CONTROLLER_CONFIG));
|
|
1114
|
+
const moduleOverride = moduleConfigOverrides?.[ModelConstr.name];
|
|
1115
|
+
const mergedConfig = {
|
|
1116
|
+
...decoratorConfig || {},
|
|
1117
|
+
...moduleOverride || {}
|
|
1118
|
+
};
|
|
1119
|
+
const FactoryController = ModelControllerFactory.create(ModelConstr, factoryPersistence, mergedConfig);
|
|
1120
|
+
const factoryRoutes = FactoryController.__routes__;
|
|
1121
|
+
const {getPK: getPK, apiProperties: apiProperties, path: pkPath} = FromModelController.getRouteParametersFromModel(ModelConstr);
|
|
1122
|
+
log.debug(`Creating controller for model: ${modelClazzName} with ${factoryRoutes?.length ?? 0} factory routes`);
|
|
1123
|
+
let DynamicModelController = DynamicModelController_1 = class DynamicModelController extends DecafModelController {
|
|
1124
|
+
static get class() {
|
|
1125
|
+
return ModelConstr;
|
|
1126
|
+
}
|
|
1127
|
+
get class() {
|
|
1128
|
+
return ModelConstr;
|
|
1129
|
+
}
|
|
1130
|
+
constructor(clientContext) {
|
|
1131
|
+
super(clientContext, DynamicModelController_1.name);
|
|
1132
|
+
this.pk = Model.pk(ModelConstr);
|
|
1133
|
+
log.info(`Registering dynamic controller for model: ${this.class.name} route: /${routePath}`);
|
|
1134
|
+
}
|
|
1135
|
+
};
|
|
1136
|
+
DynamicModelController = DynamicModelController_1 = __decorate([ Controller(routePath), ApiTags(modelClazzName), ApiExtraModels(ModelConstr), Auth(ModelConstr), __metadata("design:paramtypes", [ DecafRequestContext ]) ], DynamicModelController);
|
|
1137
|
+
if (factoryRoutes) {
|
|
1138
|
+
const sortedRoutes = [ ...factoryRoutes ].sort((a, b) => {
|
|
1139
|
+
const aSegments = a.path.split("/").filter(Boolean);
|
|
1140
|
+
const bSegments = b.path.split("/").filter(Boolean);
|
|
1141
|
+
const aParamCount = aSegments.filter(s => s.startsWith(":")).length;
|
|
1142
|
+
const bParamCount = bSegments.filter(s => s.startsWith(":")).length;
|
|
1143
|
+
const aLiteralCount = aSegments.length - aParamCount;
|
|
1144
|
+
const bLiteralCount = bSegments.length - bParamCount;
|
|
1145
|
+
if (aLiteralCount !== bLiteralCount) return bLiteralCount - aLiteralCount;
|
|
1146
|
+
if (aParamCount !== bParamCount) return aParamCount - bParamCount;
|
|
1147
|
+
return 0;
|
|
1148
|
+
});
|
|
1149
|
+
for (const route of sortedRoutes) {
|
|
1150
|
+
const registration = FromModelController.matchRoute(route, pkPath, apiProperties, getPK, ModelConstr, modelClazzName, factoryPersistence);
|
|
1151
|
+
if (!registration) continue;
|
|
1152
|
+
const {methodName: methodName, handler: handler, decorators: decorators, paramDecorators: paramDecorators} = registration;
|
|
1153
|
+
const descriptor = FromModelController.defineMethod(DynamicModelController, methodName, handler);
|
|
1154
|
+
if (descriptor) {
|
|
1155
|
+
const httpDecorator = HttpVerbToDecorator(route.method)(route.path.replace(/^\/+|\/+$/g, "") || undefined);
|
|
1156
|
+
FromModelController.applyDecorators(DynamicModelController, methodName, [ httpDecorator, ...decorators ], paramDecorators);
|
|
1157
|
+
}
|
|
1158
|
+
}
|
|
1159
|
+
}
|
|
1160
|
+
return DynamicModelController;
|
|
1161
|
+
}
|
|
1162
|
+
static getRouteParametersFromModel(ModelClazz) {
|
|
1163
|
+
const pk = Model.pk(ModelClazz);
|
|
1164
|
+
const composed = Metadata.get(ModelClazz, Metadata.key(DBKeys.COMPOSED, pk));
|
|
1165
|
+
const composedKeys = composed?.args ?? [];
|
|
1166
|
+
const uniqueKeys = Array.isArray(composedKeys) && composedKeys.length > 0 ? Array.from(new Set([ ...composedKeys ])) : Array.from(new Set([ pk ]));
|
|
1167
|
+
const description = Metadata.description(ModelClazz) ?? "";
|
|
1168
|
+
const path = `:${uniqueKeys.join("/:")}`;
|
|
1169
|
+
const apiProperties = uniqueKeys.map(key => ({
|
|
1170
|
+
name: key,
|
|
1171
|
+
description: Metadata.description(ModelClazz, key),
|
|
1172
|
+
required: true,
|
|
1173
|
+
type: String
|
|
1174
|
+
}));
|
|
1175
|
+
return {
|
|
1176
|
+
path: path,
|
|
1177
|
+
description: description,
|
|
1178
|
+
apiProperties: apiProperties,
|
|
1179
|
+
getPK: (...params) => composed?.separator ? params.join(composed.separator) : params.join("")
|
|
1180
|
+
};
|
|
1181
|
+
}
|
|
1182
|
+
static defineMethod(target, methodName, handler) {
|
|
1183
|
+
Object.defineProperty(target.prototype || target, methodName, {
|
|
1184
|
+
value: handler,
|
|
1185
|
+
writable: false,
|
|
1186
|
+
configurable: true,
|
|
1187
|
+
enumerable: false
|
|
1188
|
+
});
|
|
1189
|
+
return Object.getOwnPropertyDescriptor(target.prototype || target, methodName);
|
|
1190
|
+
}
|
|
1191
|
+
static applyDecorators(target, methodName, methodDecorators, paramDecorators = []) {
|
|
1192
|
+
const proto = target?.prototype ?? target;
|
|
1193
|
+
const descriptor = Object.getOwnPropertyDescriptor(proto, methodName);
|
|
1194
|
+
methodDecorators.forEach(d => d(proto, methodName, descriptor));
|
|
1195
|
+
paramDecorators.forEach(({decorator: decorator, index: index}) => decorator(proto, methodName, index));
|
|
1196
|
+
}
|
|
1197
|
+
static matchRoute(route, pkPath, apiProperties, getPK, ModelConstr, modelClazzName, persistence) {
|
|
1198
|
+
const {method: method, path: path} = route;
|
|
1199
|
+
const normalizedPath = path.replace(/^\/+|\/+$/g, "");
|
|
1200
|
+
if (method === "POST" && normalizedPath === "") {
|
|
1201
|
+
return FromModelController.createRegistration("create", FromModelController.createCreateHandler(ModelConstr, modelClazzName), FromModelController.createCreateDecorators(ModelConstr, modelClazzName), [ {
|
|
1202
|
+
decorator: DecafBody(),
|
|
1203
|
+
index: 0
|
|
1204
|
+
}, {
|
|
1205
|
+
decorator: Response({
|
|
1206
|
+
passthrough: true
|
|
1207
|
+
}),
|
|
1208
|
+
index: 1
|
|
1209
|
+
} ]);
|
|
1210
|
+
}
|
|
1211
|
+
if (method === "POST" && normalizedPath === "bulk") {
|
|
1212
|
+
return FromModelController.createRegistration("createAll", FromModelController.createBulkCreateHandler(ModelConstr, modelClazzName), FromModelController.bulkCreateDecorators(ModelConstr, modelClazzName), [ {
|
|
1213
|
+
decorator: DecafBody(),
|
|
1214
|
+
index: 0
|
|
1215
|
+
}, {
|
|
1216
|
+
decorator: Response({
|
|
1217
|
+
passthrough: true
|
|
1218
|
+
}),
|
|
1219
|
+
index: 1
|
|
1220
|
+
} ]);
|
|
1221
|
+
}
|
|
1222
|
+
if (method === "GET" && normalizedPath === "bulk") {
|
|
1223
|
+
return FromModelController.createRegistration("readAll", FromModelController.createBulkReadHandler(modelClazzName), FromModelController.bulkReadDecorators(ModelConstr, modelClazzName), [ {
|
|
1224
|
+
decorator: Query("ids"),
|
|
1225
|
+
index: 0
|
|
1226
|
+
} ]);
|
|
1227
|
+
}
|
|
1228
|
+
if (method === "PUT" && normalizedPath === "bulk") {
|
|
1229
|
+
return FromModelController.createRegistration("updateAll", FromModelController.createBulkUpdateHandler(modelClazzName), FromModelController.bulkUpdateDecorators(ModelConstr, modelClazzName, apiProperties), [ {
|
|
1230
|
+
decorator: DecafBody(),
|
|
1231
|
+
index: 0
|
|
1232
|
+
}, {
|
|
1233
|
+
decorator: Response({
|
|
1234
|
+
passthrough: true
|
|
1235
|
+
}),
|
|
1236
|
+
index: 1
|
|
1237
|
+
} ]);
|
|
1238
|
+
}
|
|
1239
|
+
if (method === "DELETE" && normalizedPath === "bulk") {
|
|
1240
|
+
return FromModelController.createRegistration("deleteAll", FromModelController.createBulkDeleteHandler(modelClazzName), FromModelController.bulkDeleteDecorators(ModelConstr, modelClazzName, apiProperties), [ {
|
|
1241
|
+
decorator: Query("ids"),
|
|
1242
|
+
index: 0
|
|
1243
|
+
}, {
|
|
1244
|
+
decorator: Response({
|
|
1245
|
+
passthrough: true
|
|
1246
|
+
}),
|
|
1247
|
+
index: 1
|
|
1248
|
+
} ]);
|
|
1249
|
+
}
|
|
1250
|
+
if (method === "GET" && normalizedPath === pkPath) {
|
|
1251
|
+
return FromModelController.createRegistration("read", FromModelController.createReadHandler(getPK, modelClazzName), FromModelController.readDecorators(ModelConstr, modelClazzName, apiProperties, pkPath), [ {
|
|
1252
|
+
decorator: DecafParams(apiProperties),
|
|
1253
|
+
index: 0
|
|
1254
|
+
} ]);
|
|
1255
|
+
}
|
|
1256
|
+
if (method === "PUT" && normalizedPath === pkPath) {
|
|
1257
|
+
return FromModelController.createRegistration("update", FromModelController.createUpdateHandler(getPK, ModelConstr, modelClazzName), FromModelController.updateDecorators(ModelConstr, modelClazzName, apiProperties, pkPath), [ {
|
|
1258
|
+
decorator: DecafParams(apiProperties),
|
|
1259
|
+
index: 0
|
|
1260
|
+
}, {
|
|
1261
|
+
decorator: DecafBody(),
|
|
1262
|
+
index: 1
|
|
1263
|
+
}, {
|
|
1264
|
+
decorator: Response({
|
|
1265
|
+
passthrough: true
|
|
1266
|
+
}),
|
|
1267
|
+
index: 2
|
|
1268
|
+
} ]);
|
|
1269
|
+
}
|
|
1270
|
+
if (method === "DELETE" && normalizedPath === pkPath) {
|
|
1271
|
+
return FromModelController.createRegistration("delete", FromModelController.createDeleteHandler(getPK, modelClazzName), FromModelController.deleteDecorators(ModelConstr, modelClazzName, apiProperties, pkPath), [ {
|
|
1272
|
+
decorator: DecafParams(apiProperties),
|
|
1273
|
+
index: 0
|
|
1274
|
+
}, {
|
|
1275
|
+
decorator: Response({
|
|
1276
|
+
passthrough: true
|
|
1277
|
+
}),
|
|
1278
|
+
index: 1
|
|
1279
|
+
} ]);
|
|
1280
|
+
}
|
|
1281
|
+
const fallbackSegments = normalizedPath.split("/").filter(Boolean);
|
|
1282
|
+
const isAllParams = fallbackSegments.length > 0 && fallbackSegments.every(s => s.startsWith(":"));
|
|
1283
|
+
if (isAllParams && normalizedPath !== pkPath) {
|
|
1284
|
+
const fallbackApiProps = fallbackSegments.map(s => s.slice(1)).map(name => ({
|
|
1285
|
+
name: name,
|
|
1286
|
+
description: `${name} parameter`,
|
|
1287
|
+
required: true,
|
|
1288
|
+
type: String
|
|
1289
|
+
}));
|
|
1290
|
+
const suffix = fallbackSegments.map(s => s.slice(1)).join("And");
|
|
1291
|
+
if (method === "GET") {
|
|
1292
|
+
return FromModelController.createRegistration(`readBy${suffix}`, FromModelController.createReadHandler(getPK, modelClazzName), FromModelController.readDecorators(ModelConstr, modelClazzName, fallbackApiProps, normalizedPath), [ {
|
|
1293
|
+
decorator: DecafParams(fallbackApiProps),
|
|
1294
|
+
index: 0
|
|
1295
|
+
} ]);
|
|
1296
|
+
}
|
|
1297
|
+
if (method === "PUT") {
|
|
1298
|
+
return FromModelController.createRegistration(`updateBy${suffix}`, FromModelController.createUpdateHandler(getPK, ModelConstr, modelClazzName), FromModelController.updateDecorators(ModelConstr, modelClazzName, fallbackApiProps, normalizedPath), [ {
|
|
1299
|
+
decorator: DecafParams(fallbackApiProps),
|
|
1300
|
+
index: 0
|
|
1301
|
+
}, {
|
|
1302
|
+
decorator: DecafBody(),
|
|
1303
|
+
index: 1
|
|
1304
|
+
}, {
|
|
1305
|
+
decorator: Response({
|
|
1306
|
+
passthrough: true
|
|
1307
|
+
}),
|
|
1308
|
+
index: 2
|
|
1309
|
+
} ]);
|
|
1310
|
+
}
|
|
1311
|
+
if (method === "DELETE") {
|
|
1312
|
+
return FromModelController.createRegistration(`deleteBy${suffix}`, FromModelController.createDeleteHandler(getPK, modelClazzName), FromModelController.deleteDecorators(ModelConstr, modelClazzName, fallbackApiProps, normalizedPath), [ {
|
|
1313
|
+
decorator: DecafParams(fallbackApiProps),
|
|
1314
|
+
index: 0
|
|
1315
|
+
}, {
|
|
1316
|
+
decorator: Response({
|
|
1317
|
+
passthrough: true
|
|
1318
|
+
}),
|
|
1319
|
+
index: 1
|
|
1320
|
+
} ]);
|
|
1321
|
+
}
|
|
1322
|
+
}
|
|
1323
|
+
if (method === "GET" && normalizedPath === "statement/:method/*args") {
|
|
1324
|
+
return FromModelController.createRegistration("statement", FromModelController.createStatementHandler(modelClazzName), FromModelController.statementDecorators(ModelConstr, modelClazzName), [ {
|
|
1325
|
+
decorator: Param("method"),
|
|
1326
|
+
index: 0
|
|
1327
|
+
}, {
|
|
1328
|
+
decorator: Param("args"),
|
|
1329
|
+
index: 1
|
|
1330
|
+
}, {
|
|
1331
|
+
decorator: DecafQuery(),
|
|
1332
|
+
index: 2
|
|
1333
|
+
} ]);
|
|
1334
|
+
}
|
|
1335
|
+
const statementRoutes = {
|
|
1336
|
+
"listBy/:key": PreparedStatementKeys.LIST_BY,
|
|
1337
|
+
"paginateBy/:key/:page": PreparedStatementKeys.PAGE_BY,
|
|
1338
|
+
"find/:value": PreparedStatementKeys.FIND,
|
|
1339
|
+
"page/:value": PreparedStatementKeys.PAGE,
|
|
1340
|
+
"findOneBy/:key/:value": PreparedStatementKeys.FIND_ONE_BY,
|
|
1341
|
+
"findBy/:key/:value": PreparedStatementKeys.FIND_BY,
|
|
1342
|
+
"countOf/:field": PreparedStatementKeys.COUNT_OF,
|
|
1343
|
+
"maxOf/:field": PreparedStatementKeys.MAX_OF,
|
|
1344
|
+
"minOf/:field": PreparedStatementKeys.MIN_OF,
|
|
1345
|
+
"avgOf/:field": PreparedStatementKeys.AVG_OF,
|
|
1346
|
+
"sumOf/:field": PreparedStatementKeys.SUM_OF,
|
|
1347
|
+
"distinctOf/:field": PreparedStatementKeys.DISTINCT_OF,
|
|
1348
|
+
"groupOf/:field": PreparedStatementKeys.GROUP_OF
|
|
1349
|
+
};
|
|
1350
|
+
const statementKey = statementRoutes[normalizedPath];
|
|
1351
|
+
if (statementKey && method === "GET") {
|
|
1352
|
+
return FromModelController.createRegistration(FromModelController.statementMethodName(normalizedPath), FromModelController.createStatementShortcutHandler(statementKey, modelClazzName), FromModelController.statementShortcutDecorators(ModelConstr, modelClazzName, normalizedPath, statementKey), FromModelController.statementShortcutParams(normalizedPath));
|
|
1353
|
+
}
|
|
1354
|
+
if (method === "GET" && normalizedPath.startsWith("query/")) {
|
|
1355
|
+
const queryMethod = normalizedPath.replace(/^query\//, "").split("/")[0];
|
|
1356
|
+
return FromModelController.createRegistration(queryMethod, FromModelController.createComplexQueryHandler(queryMethod), FromModelController.getQueryDecorators(queryMethod, normalizedPath, "GET", true), FromModelController.complexQueryParams(normalizedPath));
|
|
1357
|
+
}
|
|
1358
|
+
const pathSegments = normalizedPath.split("/").filter(Boolean);
|
|
1359
|
+
const knownPrefixes = new Set([ "listBy", "findBy", "findByPaginate", "findOneBy", "paginateBy", "find", "page", "countOf", "maxOf", "minOf", "avgOf", "sumOf", "distinctOf", "groupOf", "statement", "bulk", "query" ]);
|
|
1360
|
+
if (pathSegments.length > 0 && !normalizedPath.startsWith("query/") && !knownPrefixes.has(pathSegments[0])) {
|
|
1361
|
+
const routeMetadata = Metadata.get(persistence?.constructor, Metadata.key(DECAF_ROUTE)) ?? {};
|
|
1362
|
+
const matchedEntry = Object.entries(routeMetadata).find(([, info]) => info && typeof info === "object" && info.path?.replace(/^\/+|\/+$/g, "") === normalizedPath);
|
|
1363
|
+
const actualMethodName = matchedEntry?.[0] || pathSegments[0];
|
|
1364
|
+
const paramSegments = pathSegments.filter(s => s.startsWith(":"));
|
|
1365
|
+
const apiPathParams = paramSegments.map(s => s.slice(1)).map(name => ({
|
|
1366
|
+
name: name,
|
|
1367
|
+
description: `${name} parameter for the query`,
|
|
1368
|
+
required: true,
|
|
1369
|
+
type: String
|
|
1370
|
+
}));
|
|
1371
|
+
return FromModelController.createRegistration(actualMethodName, FromModelController.createCustomRouteHandler(actualMethodName), [ ...apiPathParams.map(p => ApiParam(p)), ApiOperation({
|
|
1372
|
+
summary: `Retrieve records using "${actualMethodName}".`
|
|
1373
|
+
}), ApiOkResponse({
|
|
1374
|
+
description: "Result successfully retrieved."
|
|
1375
|
+
}), ApiNoContentResponse({
|
|
1376
|
+
description: "No content returned by the method."
|
|
1377
|
+
}) ], FromModelController.complexQueryParams(normalizedPath));
|
|
1378
|
+
}
|
|
1379
|
+
return undefined;
|
|
1380
|
+
}
|
|
1381
|
+
static createRegistration(methodName, handler, decorators, paramDecorators) {
|
|
1382
|
+
return {
|
|
1383
|
+
methodName: methodName,
|
|
1384
|
+
handler: handler,
|
|
1385
|
+
decorators: decorators,
|
|
1386
|
+
paramDecorators: paramDecorators
|
|
1387
|
+
};
|
|
1388
|
+
}
|
|
1389
|
+
static statementMethodName(path) {
|
|
1390
|
+
const firstSegment = path.split("/")[0];
|
|
1391
|
+
return firstSegment;
|
|
1392
|
+
}
|
|
1393
|
+
static statementShortcutParams(path) {
|
|
1394
|
+
const segments = path.split("/").filter(s => s.startsWith(":"));
|
|
1395
|
+
const params = [];
|
|
1396
|
+
segments.forEach((seg, i) => {
|
|
1397
|
+
const name = seg.replace(":", "");
|
|
1398
|
+
params.push({
|
|
1399
|
+
decorator: Param(name),
|
|
1400
|
+
index: i
|
|
1401
|
+
});
|
|
1402
|
+
});
|
|
1403
|
+
if (path.startsWith("listBy/") || path.startsWith("paginateBy/") || path.startsWith("find/") || path.startsWith("page/")) {
|
|
1404
|
+
params.push({
|
|
1405
|
+
decorator: DecafQuery(),
|
|
1406
|
+
index: segments.length
|
|
1407
|
+
});
|
|
1408
|
+
}
|
|
1409
|
+
return params;
|
|
1410
|
+
}
|
|
1411
|
+
static complexQueryParams(path) {
|
|
1412
|
+
const segments = path.split("/").filter(s => s.startsWith(":"));
|
|
1413
|
+
const params = [];
|
|
1414
|
+
segments.forEach((seg, i) => {
|
|
1415
|
+
const name = seg.replace(":", "");
|
|
1416
|
+
params.push({
|
|
1417
|
+
decorator: Param(name),
|
|
1418
|
+
index: i
|
|
1419
|
+
});
|
|
1420
|
+
});
|
|
1421
|
+
if (path.startsWith("query/")) {
|
|
1422
|
+
params.push({
|
|
1423
|
+
decorator: DecafQuery(),
|
|
1424
|
+
index: segments.length
|
|
1425
|
+
});
|
|
1426
|
+
}
|
|
1427
|
+
return params;
|
|
1428
|
+
}
|
|
1429
|
+
static createCreateHandler(ModelConstr, modelClazzName) {
|
|
1430
|
+
return async function create(data, resp) {
|
|
1431
|
+
const {ctx: ctx, log: log} = (await this.logCtx([], OperationKeys.CREATE, true)).for(create);
|
|
1432
|
+
log.verbose(`creating new ${modelClazzName}`);
|
|
1433
|
+
let created;
|
|
1434
|
+
try {
|
|
1435
|
+
created = await this.persistence(ctx).create(data, ctx);
|
|
1436
|
+
} catch (e) {
|
|
1437
|
+
log.error(`Failed to create new ${modelClazzName}`, e);
|
|
1438
|
+
throw e;
|
|
1439
|
+
}
|
|
1440
|
+
log.info(`created new ${modelClazzName} with id ${created[this.pk]}`);
|
|
1441
|
+
if (resp) ctx.toResponse(resp);
|
|
1442
|
+
return created;
|
|
1443
|
+
};
|
|
1444
|
+
}
|
|
1445
|
+
static createBulkCreateHandler(ModelConstr, modelClazzName) {
|
|
1446
|
+
return async function createAll(data, resp) {
|
|
1447
|
+
const {ctx: ctx, log: log} = (await this.logCtx([], BulkCrudOperationKeys.CREATE_ALL, true)).for(createAll);
|
|
1448
|
+
log.verbose(`creating new ${modelClazzName}`);
|
|
1449
|
+
let created;
|
|
1450
|
+
try {
|
|
1451
|
+
created = await this.persistence(ctx).createAll(data.map(d => new ModelConstr(d)), ctx);
|
|
1452
|
+
} catch (e) {
|
|
1453
|
+
log.error(`Failed to create new ${modelClazzName}`, e);
|
|
1454
|
+
throw e;
|
|
1455
|
+
}
|
|
1456
|
+
log.info(`created new ${modelClazzName} with id ${created[this.pk]}`);
|
|
1457
|
+
if (resp) ctx.toResponse(resp);
|
|
1458
|
+
return created;
|
|
1459
|
+
};
|
|
1460
|
+
}
|
|
1461
|
+
static createBulkReadHandler(modelClazzName) {
|
|
1462
|
+
return async function readAll(ids) {
|
|
1463
|
+
const {ctx: ctx, log: log} = (await this.logCtx([], BulkCrudOperationKeys.READ_ALL, true)).for(readAll);
|
|
1464
|
+
const normalizedIds = Array.isArray(ids) ? ids : [ ids ];
|
|
1465
|
+
let read;
|
|
1466
|
+
try {
|
|
1467
|
+
log.debug(`reading ${normalizedIds} ${modelClazzName}`);
|
|
1468
|
+
read = await this.persistence(ctx).readAll(normalizedIds, ctx);
|
|
1469
|
+
} catch (e) {
|
|
1470
|
+
log.error(`Failed to read ${modelClazzName}`, e);
|
|
1471
|
+
throw e;
|
|
1472
|
+
}
|
|
1473
|
+
log.info(`read ${read.length} ${modelClazzName}`);
|
|
1474
|
+
return read;
|
|
1475
|
+
};
|
|
1476
|
+
}
|
|
1477
|
+
static createBulkUpdateHandler(modelClazzName) {
|
|
1478
|
+
return async function updateAll(body, resp) {
|
|
1479
|
+
const {ctx: ctx, log: log} = (await this.logCtx([], BulkCrudOperationKeys.UPDATE_ALL, true)).for(updateAll);
|
|
1480
|
+
let updated;
|
|
1481
|
+
try {
|
|
1482
|
+
log.info(`updating ${body.length} ${modelClazzName}`);
|
|
1483
|
+
updated = await this.persistence(ctx).updateAll(body, ctx);
|
|
1484
|
+
} catch (e) {
|
|
1485
|
+
log.error(e);
|
|
1486
|
+
throw e;
|
|
1487
|
+
}
|
|
1488
|
+
if (resp) ctx.toResponse(resp);
|
|
1489
|
+
return updated;
|
|
1490
|
+
};
|
|
1491
|
+
}
|
|
1492
|
+
static createBulkDeleteHandler(modelClazzName) {
|
|
1493
|
+
return async function deleteAll(ids, resp) {
|
|
1494
|
+
const {ctx: ctx, log: log} = (await this.logCtx([], BulkCrudOperationKeys.DELETE_ALL, true)).for(deleteAll);
|
|
1495
|
+
const normalizedIds = Array.isArray(ids) ? ids : [ ids ];
|
|
1496
|
+
let read;
|
|
1497
|
+
try {
|
|
1498
|
+
log.debug(`deleting ${normalizedIds.length} ${modelClazzName}`);
|
|
1499
|
+
read = await this.persistence(ctx).deleteAll(normalizedIds, ctx);
|
|
1500
|
+
} catch (e) {
|
|
1501
|
+
log.error(`Failed to delete ${modelClazzName}`, e);
|
|
1502
|
+
throw e;
|
|
1503
|
+
}
|
|
1504
|
+
log.info(`deleted ${read.length} ${modelClazzName}`);
|
|
1505
|
+
if (resp) ctx.toResponse(resp);
|
|
1506
|
+
return read;
|
|
1507
|
+
};
|
|
1508
|
+
}
|
|
1509
|
+
static createReadHandler(getPK, modelClazzName) {
|
|
1510
|
+
return async function read(routeParams) {
|
|
1511
|
+
const {ctx: ctx, log: log} = (await this.logCtx([], OperationKeys.READ, true)).for(read);
|
|
1512
|
+
const id = getPK(...routeParams.valuesInOrder);
|
|
1513
|
+
if (typeof id === "undefined") throw new ValidationError(`No ${this.pk} provided`);
|
|
1514
|
+
let readResult;
|
|
1515
|
+
try {
|
|
1516
|
+
log.debug(`reading ${modelClazzName} with ${this.pk} ${id}`);
|
|
1517
|
+
readResult = await this.persistence(ctx).read(id, ctx);
|
|
1518
|
+
} catch (e) {
|
|
1519
|
+
log.error(`Failed to read ${modelClazzName} with id ${id}`, e);
|
|
1520
|
+
throw e;
|
|
1521
|
+
}
|
|
1522
|
+
log.info(`read ${modelClazzName} with id ${readResult[this.pk]}`);
|
|
1523
|
+
return readResult;
|
|
1524
|
+
};
|
|
1525
|
+
}
|
|
1526
|
+
static createUpdateHandler(getPK, ModelConstr, modelClazzName) {
|
|
1527
|
+
return async function update(routeParams, body, resp) {
|
|
1528
|
+
const {ctx: ctx, log: log} = (await this.logCtx([], OperationKeys.UPDATE, true)).for(update);
|
|
1529
|
+
const id = getPK(...routeParams.valuesInOrder);
|
|
1530
|
+
if (typeof id === "undefined") throw new ValidationError(`No ${this.pk} provided`);
|
|
1531
|
+
let updated;
|
|
1532
|
+
try {
|
|
1533
|
+
log.info(`updating ${modelClazzName} with ${this.pk} ${id}`);
|
|
1534
|
+
const payload = JSON.parse(JSON.stringify(body));
|
|
1535
|
+
updated = await this.persistence(ctx).update(new ModelConstr({
|
|
1536
|
+
...payload,
|
|
1537
|
+
[this.pk]: id
|
|
1538
|
+
}), ctx);
|
|
1539
|
+
} catch (e) {
|
|
1540
|
+
log.error(e);
|
|
1541
|
+
throw e;
|
|
1542
|
+
}
|
|
1543
|
+
if (resp) ctx.toResponse(resp);
|
|
1544
|
+
return updated;
|
|
1545
|
+
};
|
|
1546
|
+
}
|
|
1547
|
+
static createDeleteHandler(getPK, modelClazzName) {
|
|
1548
|
+
return async function remove(routeParams, resp) {
|
|
1549
|
+
const {ctx: ctx, log: log} = (await this.logCtx([], OperationKeys.DELETE, true)).for(remove);
|
|
1550
|
+
const id = getPK(...routeParams.valuesInOrder);
|
|
1551
|
+
if (typeof id === "undefined") throw new ValidationError(`No ${this.pk} provided`);
|
|
1552
|
+
let del;
|
|
1553
|
+
try {
|
|
1554
|
+
log.debug(`deleting ${modelClazzName} with ${this.pk} ${id}`);
|
|
1555
|
+
del = await this.persistence(ctx).delete(id, ctx);
|
|
1556
|
+
} catch (e) {
|
|
1557
|
+
log.error(`Failed to delete ${modelClazzName} with id ${id}`, e);
|
|
1558
|
+
throw e;
|
|
1559
|
+
}
|
|
1560
|
+
log.info(`deleted ${modelClazzName} with id ${id}`);
|
|
1561
|
+
if (resp) ctx.toResponse(resp);
|
|
1562
|
+
return del;
|
|
1563
|
+
};
|
|
1564
|
+
}
|
|
1565
|
+
static createStatementHandler(modelClazzName) {
|
|
1566
|
+
return async function statement(name, args, details) {
|
|
1567
|
+
const {ctx: ctx} = (await this.logCtx([], PersistenceKeys.STATEMENT, true)).for(statement);
|
|
1568
|
+
const {direction: direction, offset: offset, limit: limit, bookmark: bookmark} = details;
|
|
1569
|
+
args = args.map(a => (typeof a === "string" ? parseInt(a) : a) || a);
|
|
1570
|
+
const pathDirection = args.length > 1 ? args[1] : undefined;
|
|
1571
|
+
const resolvedDirection = direction ?? pathDirection;
|
|
1572
|
+
if (resolvedDirection && args.length > 1) args[1] = resolvedDirection;
|
|
1573
|
+
switch (name) {
|
|
1574
|
+
case PreparedStatementKeys.FIND:
|
|
1575
|
+
case PreparedStatementKeys.FIND_BY:
|
|
1576
|
+
break;
|
|
1577
|
+
|
|
1578
|
+
case PreparedStatementKeys.LIST_BY:
|
|
1579
|
+
args.push(direction);
|
|
1580
|
+
break;
|
|
1581
|
+
|
|
1582
|
+
case PreparedStatementKeys.PAGE:
|
|
1583
|
+
case PreparedStatementKeys.PAGE_BY:
|
|
1584
|
+
args = [ args[0], resolvedDirection, {
|
|
1585
|
+
limit: limit,
|
|
1586
|
+
offset: offset,
|
|
1587
|
+
bookmark: bookmark
|
|
1588
|
+
} ];
|
|
1589
|
+
break;
|
|
1590
|
+
|
|
1591
|
+
case PreparedStatementKeys.FIND_ONE_BY:
|
|
1592
|
+
break;
|
|
1593
|
+
|
|
1594
|
+
case PreparedStatementKeys.COUNT_OF:
|
|
1595
|
+
case PreparedStatementKeys.MAX_OF:
|
|
1596
|
+
case PreparedStatementKeys.MIN_OF:
|
|
1597
|
+
case PreparedStatementKeys.AVG_OF:
|
|
1598
|
+
case PreparedStatementKeys.SUM_OF:
|
|
1599
|
+
case PreparedStatementKeys.DISTINCT_OF:
|
|
1600
|
+
case PreparedStatementKeys.GROUP_OF:
|
|
1601
|
+
break;
|
|
1602
|
+
}
|
|
1603
|
+
return this.persistence(ctx).statement(name, ...args, ctx);
|
|
1604
|
+
};
|
|
1605
|
+
}
|
|
1606
|
+
static createStatementShortcutHandler(statementKey, modelClazzName) {
|
|
1607
|
+
return async function statementShortcut(...args) {
|
|
1608
|
+
const {ctx: ctx} = (await this.logCtx([], statementKey, true)).for(statementShortcut);
|
|
1609
|
+
switch (statementKey) {
|
|
1610
|
+
case PreparedStatementKeys.LIST_BY:
|
|
1611
|
+
{
|
|
1612
|
+
const [key, details] = args;
|
|
1613
|
+
return this.persistence(ctx).listBy(key, details?.direction, ctx);
|
|
1614
|
+
}
|
|
1615
|
+
|
|
1616
|
+
case PreparedStatementKeys.PAGE_BY:
|
|
1617
|
+
{
|
|
1618
|
+
const [key, page, details] = args;
|
|
1619
|
+
return this.persistence(ctx).paginateBy(key, details?.direction, {
|
|
1620
|
+
limit: details?.limit,
|
|
1621
|
+
offset: details?.offset,
|
|
1622
|
+
page: page
|
|
1623
|
+
}, ctx);
|
|
1624
|
+
}
|
|
1625
|
+
|
|
1626
|
+
case PreparedStatementKeys.FIND:
|
|
1627
|
+
{
|
|
1628
|
+
const [value, details] = args;
|
|
1629
|
+
const direction = details?.direction ?? OrderDirection.ASC;
|
|
1630
|
+
const persistence = this.persistence(ctx);
|
|
1631
|
+
if (typeof persistence.find === "function") return persistence.find(value, direction, ctx);
|
|
1632
|
+
return persistence.statement(PreparedStatementKeys.FIND, value, direction, ctx);
|
|
1633
|
+
}
|
|
1634
|
+
|
|
1635
|
+
case PreparedStatementKeys.PAGE:
|
|
1636
|
+
{
|
|
1637
|
+
const [value, details] = args;
|
|
1638
|
+
const ref = {
|
|
1639
|
+
offset: details?.offset ?? 1,
|
|
1640
|
+
limit: details?.limit ?? 10,
|
|
1641
|
+
bookmark: details?.bookmark
|
|
1642
|
+
};
|
|
1643
|
+
const persistence = this.persistence(ctx);
|
|
1644
|
+
const direction = details?.direction ?? OrderDirection.ASC;
|
|
1645
|
+
if (typeof persistence.page === "function") return persistence.page(value, direction, ref, ctx);
|
|
1646
|
+
return persistence.statement(PreparedStatementKeys.PAGE, value, direction, ref, ctx);
|
|
1647
|
+
}
|
|
1648
|
+
|
|
1649
|
+
case PreparedStatementKeys.FIND_ONE_BY:
|
|
1650
|
+
{
|
|
1651
|
+
const [key, value] = args;
|
|
1652
|
+
return this.persistence(ctx).findOneBy(key, value, ctx);
|
|
1653
|
+
}
|
|
1654
|
+
|
|
1655
|
+
case PreparedStatementKeys.FIND_BY:
|
|
1656
|
+
{
|
|
1657
|
+
const [key, value] = args;
|
|
1658
|
+
return this.persistence(ctx).for(ctx.toOverrides()).findBy(key, value, ctx);
|
|
1659
|
+
}
|
|
1660
|
+
|
|
1661
|
+
default:
|
|
1662
|
+
if (statementKey === PreparedStatementKeys.COUNT_OF || statementKey === PreparedStatementKeys.MAX_OF || statementKey === PreparedStatementKeys.MIN_OF || statementKey === PreparedStatementKeys.AVG_OF || statementKey === PreparedStatementKeys.SUM_OF || statementKey === PreparedStatementKeys.DISTINCT_OF || statementKey === PreparedStatementKeys.GROUP_OF) {
|
|
1663
|
+
const [field] = args;
|
|
1664
|
+
return this.persistence(ctx).statement(statementKey, field, ctx);
|
|
1665
|
+
}
|
|
1666
|
+
throw new Error(`Unknown statement: ${statementKey}`);
|
|
1667
|
+
}
|
|
1668
|
+
};
|
|
1669
|
+
}
|
|
1670
|
+
static createComplexQueryHandler(methodName) {
|
|
1671
|
+
return async function complexQuery(...args) {
|
|
1672
|
+
const log = this.log?.for?.(complexQuery);
|
|
1673
|
+
try {
|
|
1674
|
+
if (log) log.debug(`Invoking custom query "${methodName}"`);
|
|
1675
|
+
const {ctx: ctx} = (await this.logCtx([], methodName, true)).for(complexQuery);
|
|
1676
|
+
const persistence = this.persistence(ctx);
|
|
1677
|
+
const spreadArgs = FromModelController.extractQueryArgs(args);
|
|
1678
|
+
if (typeof persistence[methodName] === "function") {
|
|
1679
|
+
return persistence[methodName](...spreadArgs, ctx);
|
|
1680
|
+
}
|
|
1681
|
+
if (typeof persistence.statement === "function") {
|
|
1682
|
+
return persistence.statement(methodName, ...spreadArgs, ctx);
|
|
1683
|
+
}
|
|
1684
|
+
throw new Error(`Persistence method "${methodName}" not found on ${persistence?.constructor?.name}`);
|
|
1685
|
+
} catch (e) {
|
|
1686
|
+
if (log) log.error(`Custom query "${methodName}" failed`, e);
|
|
1687
|
+
throw e;
|
|
1688
|
+
}
|
|
1689
|
+
};
|
|
1690
|
+
}
|
|
1691
|
+
static createCustomRouteHandler(methodName) {
|
|
1692
|
+
return async function customRoute(...args) {
|
|
1693
|
+
const log = this.log?.for?.(customRoute);
|
|
1694
|
+
const {ctx: ctx} = (await this.logCtx([], methodName, true)).for(customRoute);
|
|
1695
|
+
const persistence = this.persistence(ctx);
|
|
1696
|
+
const spreadArgs = FromModelController.extractQueryArgs(args);
|
|
1697
|
+
if (typeof persistence[methodName] === "function") {
|
|
1698
|
+
return persistence[methodName](...spreadArgs, ctx);
|
|
1699
|
+
}
|
|
1700
|
+
if (persistence?.repo && typeof persistence.repo[methodName] === "function") {
|
|
1701
|
+
return persistence.repo[methodName](...spreadArgs, ctx);
|
|
1702
|
+
}
|
|
1703
|
+
if (typeof persistence.statement === "function") {
|
|
1704
|
+
return persistence.statement(methodName, ...spreadArgs, ctx);
|
|
1705
|
+
}
|
|
1706
|
+
throw new Error(`Method "${methodName}" not found on ${persistence?.constructor?.name} or its repo`);
|
|
1707
|
+
};
|
|
1708
|
+
}
|
|
1709
|
+
static extractQueryArgs(args) {
|
|
1710
|
+
if (args.length === 0) return args;
|
|
1711
|
+
const last = args[args.length - 1];
|
|
1712
|
+
if (last && typeof last === "object" && !Array.isArray(last)) {
|
|
1713
|
+
const queryObj = args.pop();
|
|
1714
|
+
const hasDirection = queryObj.direction !== undefined;
|
|
1715
|
+
const hasLimit = queryObj.limit !== undefined;
|
|
1716
|
+
const hasOffset = queryObj.offset !== undefined;
|
|
1717
|
+
if (!hasDirection && !hasLimit && !hasOffset) return args;
|
|
1718
|
+
const extras = [];
|
|
1719
|
+
if (hasDirection) extras.push(queryObj.direction); else if (hasLimit || hasOffset) extras.push(undefined);
|
|
1720
|
+
if (hasLimit) extras.push(queryObj.limit);
|
|
1721
|
+
if (hasOffset) extras.push(queryObj.offset);
|
|
1722
|
+
return [ ...args, ...extras ];
|
|
1723
|
+
}
|
|
1724
|
+
return args;
|
|
1725
|
+
}
|
|
1726
|
+
static createCreateDecorators(ModelConstr, modelClazzName) {
|
|
1727
|
+
return [ ApiOperationFromModel(ModelConstr, "POST"), ApiOperation({
|
|
1728
|
+
summary: `Create a new ${modelClazzName}.`
|
|
1729
|
+
}), ApiBody({
|
|
1730
|
+
description: `Payload for ${modelClazzName}`,
|
|
1731
|
+
type: DtoFor(OperationKeys.CREATE, ModelConstr)
|
|
1732
|
+
}), ApiCreatedResponse({
|
|
1733
|
+
description: `${modelClazzName} created successfully.`,
|
|
1734
|
+
schema: {
|
|
1735
|
+
$ref: getSchemaPath(ModelConstr)
|
|
1736
|
+
}
|
|
1737
|
+
}), ApiBadRequestResponse({
|
|
1738
|
+
description: "Payload validation failed."
|
|
1739
|
+
}), ApiUnprocessableEntityResponse({
|
|
1740
|
+
description: "Repository rejected the provided payload."
|
|
1741
|
+
}) ];
|
|
1742
|
+
}
|
|
1743
|
+
static bulkCreateDecorators(ModelConstr, modelClazzName) {
|
|
1744
|
+
return [ ApiOperationFromModel(ModelConstr, "POST", "bulk"), ApiOperation({
|
|
1745
|
+
summary: `Create a new ${modelClazzName}.`
|
|
1746
|
+
}), ApiBody({
|
|
1747
|
+
description: `Payload for ${modelClazzName}`,
|
|
1748
|
+
schema: {
|
|
1749
|
+
type: "array",
|
|
1750
|
+
items: {
|
|
1751
|
+
$ref: getSchemaPath(ModelConstr)
|
|
1752
|
+
}
|
|
1753
|
+
}
|
|
1754
|
+
}), ApiCreatedResponse({
|
|
1755
|
+
description: `${modelClazzName} created successfully.`,
|
|
1756
|
+
schema: {
|
|
1757
|
+
type: "array",
|
|
1758
|
+
items: {
|
|
1759
|
+
$ref: getSchemaPath(ModelConstr)
|
|
1760
|
+
}
|
|
1761
|
+
}
|
|
1762
|
+
}), ApiBadRequestResponse({
|
|
1763
|
+
description: "Payload validation failed."
|
|
1764
|
+
}), ApiUnprocessableEntityResponse({
|
|
1765
|
+
description: "Repository rejected the provided payload."
|
|
1766
|
+
}) ];
|
|
1767
|
+
}
|
|
1768
|
+
static bulkReadDecorators(ModelConstr, modelClazzName) {
|
|
1769
|
+
return [ ApiOperationFromModel(ModelConstr, "GET", "bulk"), ApiOperation({
|
|
1770
|
+
summary: `Retrieve ${modelClazzName} records by ids.`
|
|
1771
|
+
}), ApiQuery({
|
|
1772
|
+
name: "ids",
|
|
1773
|
+
required: true,
|
|
1774
|
+
type: "array"
|
|
1775
|
+
}), ApiOkResponse({
|
|
1776
|
+
description: `${modelClazzName} retrieved successfully.`,
|
|
1777
|
+
schema: {
|
|
1778
|
+
type: "array",
|
|
1779
|
+
items: {
|
|
1780
|
+
$ref: getSchemaPath(ModelConstr)
|
|
1781
|
+
}
|
|
1782
|
+
}
|
|
1783
|
+
}), ApiNotFoundResponse({
|
|
1784
|
+
description: `No ${modelClazzName} record matches the provided identifier.`
|
|
1785
|
+
}) ];
|
|
1786
|
+
}
|
|
1787
|
+
static bulkUpdateDecorators(ModelConstr, modelClazzName, apiProperties) {
|
|
1788
|
+
return [ ApiOperationFromModel(ModelConstr, "PUT", "bulk"), ApiParamsFromModel(apiProperties), ApiOperation({
|
|
1789
|
+
summary: `Replace existing ${modelClazzName} records with new payloads.`
|
|
1790
|
+
}), ApiBody({
|
|
1791
|
+
description: `Payload for replacing existing records of ${modelClazzName}`,
|
|
1792
|
+
schema: {
|
|
1793
|
+
type: "array",
|
|
1794
|
+
$ref: getSchemaPath(DtoFor(OperationKeys.UPDATE, ModelConstr))
|
|
1795
|
+
}
|
|
1796
|
+
}), ApiOkResponse({
|
|
1797
|
+
description: `${modelClazzName} updated successfully.`,
|
|
1798
|
+
schema: {
|
|
1799
|
+
type: "array",
|
|
1800
|
+
items: {
|
|
1801
|
+
$ref: getSchemaPath(ModelConstr)
|
|
1802
|
+
}
|
|
1803
|
+
}
|
|
1804
|
+
}), ApiNotFoundResponse({
|
|
1805
|
+
description: `No ${modelClazzName} record matches the provided identifier.`
|
|
1806
|
+
}), ApiBadRequestResponse({
|
|
1807
|
+
description: "Payload validation failed."
|
|
1808
|
+
}) ];
|
|
1809
|
+
}
|
|
1810
|
+
static bulkDeleteDecorators(ModelConstr, modelClazzName, apiProperties) {
|
|
1811
|
+
return [ ApiOperationFromModel(ModelConstr, "DELETE", "bulk"), ApiParamsFromModel(apiProperties), ApiOperation({
|
|
1812
|
+
summary: `Delete ${modelClazzName} records by ids.`
|
|
1813
|
+
}), ApiQuery({
|
|
1814
|
+
name: "ids",
|
|
1815
|
+
required: true,
|
|
1816
|
+
type: "array"
|
|
1817
|
+
}), ApiOkResponse({
|
|
1818
|
+
description: `${modelClazzName} deleted successfully.`,
|
|
1819
|
+
schema: {
|
|
1820
|
+
type: "array",
|
|
1821
|
+
items: {
|
|
1822
|
+
$ref: getSchemaPath(ModelConstr)
|
|
1823
|
+
}
|
|
1824
|
+
}
|
|
1825
|
+
}), ApiNotFoundResponse({
|
|
1826
|
+
description: `No ${modelClazzName} record matches the provided identifier.`
|
|
1827
|
+
}) ];
|
|
1828
|
+
}
|
|
1829
|
+
static readDecorators(ModelConstr, modelClazzName, apiProperties, pkPath) {
|
|
1830
|
+
return [ ApiOperationFromModel(ModelConstr, "GET", pkPath), ApiParamsFromModel(apiProperties), ApiOperation({
|
|
1831
|
+
summary: `Retrieve a ${modelClazzName} record by id.`
|
|
1832
|
+
}), ApiOkResponse({
|
|
1833
|
+
description: `${modelClazzName} retrieved successfully.`,
|
|
1834
|
+
schema: {
|
|
1835
|
+
$ref: getSchemaPath(ModelConstr)
|
|
1836
|
+
}
|
|
1837
|
+
}), ApiNotFoundResponse({
|
|
1838
|
+
description: `No ${modelClazzName} record matches the provided identifier.`
|
|
1839
|
+
}) ];
|
|
1840
|
+
}
|
|
1841
|
+
static updateDecorators(ModelConstr, modelClazzName, apiProperties, pkPath) {
|
|
1842
|
+
return [ ApiOperationFromModel(ModelConstr, "PUT", pkPath), ApiParamsFromModel(apiProperties), ApiOperation({
|
|
1843
|
+
summary: `Replace an existing ${modelClazzName} record with a new payload.`
|
|
1844
|
+
}), ApiBody({
|
|
1845
|
+
description: `Payload for replacing an existing record of ${modelClazzName}`,
|
|
1846
|
+
type: DtoFor(OperationKeys.UPDATE, ModelConstr)
|
|
1847
|
+
}), ApiOkResponse({
|
|
1848
|
+
description: `${modelClazzName} updated successfully.`,
|
|
1849
|
+
schema: {
|
|
1850
|
+
$ref: getSchemaPath(ModelConstr)
|
|
1851
|
+
}
|
|
1852
|
+
}), ApiNotFoundResponse({
|
|
1853
|
+
description: `No ${modelClazzName} record matches the provided identifier.`
|
|
1854
|
+
}), ApiBadRequestResponse({
|
|
1855
|
+
description: "Payload validation failed."
|
|
1856
|
+
}) ];
|
|
1857
|
+
}
|
|
1858
|
+
static deleteDecorators(ModelConstr, modelClazzName, apiProperties, pkPath) {
|
|
1859
|
+
return [ ApiOperationFromModel(ModelConstr, "DELETE", pkPath), ApiParamsFromModel(apiProperties), ApiOperation({
|
|
1860
|
+
summary: `Delete a ${modelClazzName} record by id.`
|
|
1861
|
+
}), ApiOkResponse({
|
|
1862
|
+
description: `${modelClazzName} deleted successfully.`,
|
|
1863
|
+
schema: {
|
|
1864
|
+
$ref: getSchemaPath(ModelConstr)
|
|
1865
|
+
}
|
|
1866
|
+
}), ApiNotFoundResponse({
|
|
1867
|
+
description: `No ${modelClazzName} record matches the provided identifier.`
|
|
1868
|
+
}) ];
|
|
1869
|
+
}
|
|
1870
|
+
static statementDecorators(ModelConstr, modelClazzName) {
|
|
1871
|
+
return [ ApiOperationFromModel(ModelConstr, "GET", "statement/:method/*args"), ApiOperation({
|
|
1872
|
+
summary: `Executes a prepared statement on ${modelClazzName}.`
|
|
1873
|
+
}), ApiParam({
|
|
1874
|
+
name: "method",
|
|
1875
|
+
description: "the prepared statement to execute"
|
|
1876
|
+
}), ApiParam({
|
|
1877
|
+
name: "args",
|
|
1878
|
+
description: "concatenated list of arguments the prepared statement can accept"
|
|
1879
|
+
}), ApiQuery({
|
|
1880
|
+
name: "direction",
|
|
1881
|
+
required: true,
|
|
1882
|
+
enum: OrderDirection,
|
|
1883
|
+
description: "the sort order when applicable"
|
|
1884
|
+
}), ApiQuery({
|
|
1885
|
+
name: "limit",
|
|
1886
|
+
required: true,
|
|
1887
|
+
description: "limit or page size when applicable"
|
|
1888
|
+
}), ApiQuery({
|
|
1889
|
+
name: "offset",
|
|
1890
|
+
required: true,
|
|
1891
|
+
description: "offset or bookmark when applicable"
|
|
1892
|
+
}), ApiOkResponse({
|
|
1893
|
+
description: `${modelClazzName} listed found.`
|
|
1894
|
+
}), ApiNotFoundResponse({
|
|
1895
|
+
description: `No ${modelClazzName} record matches the provided identifier.`
|
|
1896
|
+
}) ];
|
|
1897
|
+
}
|
|
1898
|
+
static statementShortcutDecorators(ModelConstr, modelClazzName, path, statementKey) {
|
|
1899
|
+
const base = [ ApiOperationFromModel(ModelConstr, "GET", path), ApiOperation({
|
|
1900
|
+
summary: `Retrieve ${modelClazzName} records.`
|
|
1901
|
+
}), ApiOkResponse({
|
|
1902
|
+
description: `${modelClazzName} retrieved successfully.`
|
|
1903
|
+
}) ];
|
|
1904
|
+
const segments = path.split("/").filter(s => s.startsWith(":"));
|
|
1905
|
+
segments.forEach(seg => {
|
|
1906
|
+
const name = seg.replace(":", "");
|
|
1907
|
+
base.push(ApiParam({
|
|
1908
|
+
name: name,
|
|
1909
|
+
description: `The ${name} parameter`
|
|
1910
|
+
}));
|
|
1911
|
+
});
|
|
1912
|
+
if (path.startsWith("listBy/") || path.startsWith("paginateBy/") || path.startsWith("find/") || path.startsWith("page/")) {
|
|
1913
|
+
base.push(ApiQuery({
|
|
1914
|
+
name: "direction",
|
|
1915
|
+
required: true,
|
|
1916
|
+
enum: OrderDirection,
|
|
1917
|
+
description: "the sort order"
|
|
1918
|
+
}));
|
|
1919
|
+
}
|
|
1920
|
+
if (path.startsWith("paginateBy/") || path.startsWith("page/")) {
|
|
1921
|
+
base.push(ApiQuery({
|
|
1922
|
+
name: "limit",
|
|
1923
|
+
required: false,
|
|
1924
|
+
description: "page size"
|
|
1925
|
+
}), ApiQuery({
|
|
1926
|
+
name: "offset",
|
|
1927
|
+
required: false,
|
|
1928
|
+
description: "page number"
|
|
1929
|
+
}), ApiQuery({
|
|
1930
|
+
name: "bookmark",
|
|
1931
|
+
required: false,
|
|
1932
|
+
description: "bookmark for cursor pagination"
|
|
1933
|
+
}));
|
|
1934
|
+
}
|
|
1935
|
+
if (path.startsWith("findOneBy/") || path.startsWith("findBy/")) {
|
|
1936
|
+
base.push(ApiNotFoundResponse({
|
|
1937
|
+
description: `No ${modelClazzName} record matches the provided identifier.`
|
|
1938
|
+
}));
|
|
1939
|
+
}
|
|
1940
|
+
if (statementKey === PreparedStatementKeys.COUNT_OF || statementKey === PreparedStatementKeys.AVG_OF || statementKey === PreparedStatementKeys.SUM_OF) {
|
|
1941
|
+
base.push(ApiOkResponse({
|
|
1942
|
+
description: `Result for ${modelClazzName}.`,
|
|
1943
|
+
type: Number
|
|
1944
|
+
}));
|
|
1945
|
+
}
|
|
1946
|
+
if (statementKey === PreparedStatementKeys.DISTINCT_OF) {
|
|
1947
|
+
base.push(ApiOkResponse({
|
|
1948
|
+
description: `Distinct values for ${modelClazzName}.`,
|
|
1949
|
+
type: [ String ]
|
|
1950
|
+
}));
|
|
1951
|
+
}
|
|
1952
|
+
return base;
|
|
1953
|
+
}
|
|
1954
|
+
static getQueryDecorators(methodName, routePath, httpVerb, includeQueryParams = false) {
|
|
1955
|
+
const extractPathParams = p => p.split("/").filter(s => s.startsWith(":")).map(s => s.slice(1));
|
|
1956
|
+
const apiPathParams = extractPathParams(routePath).map(name => ({
|
|
1957
|
+
name: name,
|
|
1958
|
+
description: `${name} parameter for the query`,
|
|
1959
|
+
required: true,
|
|
1960
|
+
type: String
|
|
1961
|
+
}));
|
|
1962
|
+
const decorators = [ ...apiPathParams.map(p => ApiParam(p)), ApiOperation({
|
|
1963
|
+
summary: `Retrieve records using "${methodName}".`
|
|
1964
|
+
}), ApiOkResponse({
|
|
1965
|
+
description: "Result successfully retrieved."
|
|
1966
|
+
}), ApiNoContentResponse({
|
|
1967
|
+
description: "No content returned by the method."
|
|
1968
|
+
}) ];
|
|
1969
|
+
if (httpVerb === "GET" && includeQueryParams) {
|
|
1970
|
+
decorators.push(ApiQuery({
|
|
1971
|
+
name: "direction",
|
|
1972
|
+
required: false,
|
|
1973
|
+
enum: OrderDirection,
|
|
1974
|
+
description: "the sort order when applicable"
|
|
1975
|
+
}), ApiQuery({
|
|
1976
|
+
name: "limit",
|
|
1977
|
+
required: false,
|
|
1978
|
+
description: "limit or page size"
|
|
1979
|
+
}), ApiQuery({
|
|
1980
|
+
name: "offset",
|
|
1981
|
+
required: false,
|
|
1982
|
+
description: "offset or bookmark"
|
|
1983
|
+
}));
|
|
1984
|
+
}
|
|
1985
|
+
return decorators;
|
|
1986
|
+
}
|
|
1987
|
+
}
|
|
1988
|
+
|
|
1989
|
+
function getModuleFor(flavour) {
|
|
1990
|
+
var DecafModelModule_1;
|
|
1991
|
+
let DecafModelModule = class DecafModelModule {
|
|
1992
|
+
static {
|
|
1993
|
+
DecafModelModule_1 = this;
|
|
1994
|
+
}
|
|
1995
|
+
static {
|
|
1996
|
+
this.log = Logging.for(DecafModelModule_1.name).for(flavour);
|
|
1997
|
+
}
|
|
1998
|
+
static createModelServices(models) {
|
|
1999
|
+
return models.map(model => ({
|
|
2000
|
+
provide: `${model.name}Service`,
|
|
2001
|
+
useFactory: () => ModelService.forModel(model)
|
|
2002
|
+
}));
|
|
2003
|
+
}
|
|
2004
|
+
static isExposed(model, exposure) {
|
|
2005
|
+
const override = exposure?.[model.name];
|
|
2006
|
+
const value = typeof override !== "undefined" ? override : Metadata.get(model, Metadata.key(DECAF_EXPOSE));
|
|
2007
|
+
if (typeof value === "undefined") return true;
|
|
2008
|
+
if (value === true) return true;
|
|
2009
|
+
if (Array.isArray(value)) return value.includes(flavour);
|
|
2010
|
+
return false;
|
|
2011
|
+
}
|
|
2012
|
+
static forRoot(flavour, options = {}) {
|
|
2013
|
+
const log = this.log.for(this.forRoot);
|
|
2014
|
+
log.info(`Generating controllers for flavour...`);
|
|
2015
|
+
const trackedModels = Adapter.models(flavour).filter(model => this.isExposed(model, options.controllerExposure));
|
|
2016
|
+
let modelServices = [];
|
|
2017
|
+
if (options.autoServices) {
|
|
2018
|
+
log.info("Auto-services enabled. Initializing service generation.");
|
|
2019
|
+
modelServices = this.createModelServices(trackedModels);
|
|
2020
|
+
log.info(`Auto-services completed. ${modelServices.length} services initialized.`);
|
|
2021
|
+
}
|
|
2022
|
+
const controllers = trackedModels.map(model => FromModelController.create(model, options.controllerConfig));
|
|
2023
|
+
log.info(`Generated ${controllers.length} controllers`);
|
|
2024
|
+
return {
|
|
2025
|
+
module: DecafModelModule_1,
|
|
2026
|
+
controllers: controllers,
|
|
2027
|
+
providers: [ ...modelServices ]
|
|
2028
|
+
};
|
|
2029
|
+
}
|
|
2030
|
+
};
|
|
2031
|
+
DecafModelModule = DecafModelModule_1 = __decorate([ Module({}) ], DecafModelModule);
|
|
2032
|
+
Object.assign(DecafModelModule, "name", {
|
|
2033
|
+
value: `DecafModule${flavour}`
|
|
2034
|
+
});
|
|
2035
|
+
return DecafModelModule;
|
|
2036
|
+
}
|
|
2037
|
+
|
|
2038
|
+
Decoration.for(InjectablesKeys.INJECTABLE).extend({
|
|
2039
|
+
decorator: function nestInjectable(category, cfg) {
|
|
2040
|
+
cfg = cfg || (typeof category === "object" ? Object.assign(category, DefaultInjectablesConfig) : DefaultInjectablesConfig);
|
|
2041
|
+
return Injectable({
|
|
2042
|
+
scope: cfg.singleton ? Scope.DEFAULT : Scope.REQUEST,
|
|
2043
|
+
durable: cfg.singleton ? undefined : true
|
|
2044
|
+
});
|
|
2045
|
+
}
|
|
2046
|
+
}).apply();
|
|
2047
|
+
|
|
2048
|
+
Decoration.for(InjectablesKeys.INJECT).extend({
|
|
2049
|
+
decorator: function nestInject(category, cfg) {
|
|
2050
|
+
return function innerNestInject(target, propertyKey, descriptor) {
|
|
2051
|
+
return Inject(category || target)(target, propertyKey, descriptor);
|
|
2052
|
+
};
|
|
2053
|
+
}
|
|
2054
|
+
}).apply();
|
|
2055
|
+
|
|
2056
|
+
Decoration.for(ValidationKeys.REQUIRED).extend(ApiProperty({
|
|
2057
|
+
required: true
|
|
2058
|
+
})).apply();
|
|
2059
|
+
|
|
2060
|
+
Decoration.for(ValidationKeys.MAX).extend({
|
|
2061
|
+
decorator: function maxDec(max) {
|
|
2062
|
+
return ApiProperty({
|
|
2063
|
+
maximum: max,
|
|
2064
|
+
required: false
|
|
2065
|
+
});
|
|
2066
|
+
}
|
|
2067
|
+
}).apply();
|
|
2068
|
+
|
|
2069
|
+
Decoration.for(ValidationKeys.MIN).extend({
|
|
2070
|
+
decorator: function minDec(min) {
|
|
2071
|
+
return ApiProperty({
|
|
2072
|
+
minimum: min,
|
|
2073
|
+
required: false
|
|
2074
|
+
});
|
|
2075
|
+
}
|
|
2076
|
+
}).apply();
|
|
2077
|
+
|
|
2078
|
+
Decoration.for(ValidationKeys.MAX_LENGTH).extend({
|
|
2079
|
+
decorator: function maxLengthDec(max) {
|
|
2080
|
+
return ApiProperty({
|
|
2081
|
+
maxLength: max,
|
|
2082
|
+
required: false
|
|
2083
|
+
});
|
|
2084
|
+
}
|
|
2085
|
+
}).apply();
|
|
2086
|
+
|
|
2087
|
+
Decoration.for(ValidationKeys.MIN_LENGTH).extend({
|
|
2088
|
+
decorator: function minLengthDec(min) {
|
|
2089
|
+
return ApiProperty({
|
|
2090
|
+
minLength: min,
|
|
2091
|
+
required: false
|
|
2092
|
+
});
|
|
2093
|
+
}
|
|
2094
|
+
}).apply();
|
|
2095
|
+
|
|
2096
|
+
Decoration.for(ValidationKeys.TYPE).extend({
|
|
2097
|
+
decorator: function typeDec(type) {
|
|
2098
|
+
return (target, prop) => {
|
|
2099
|
+
type = Array.isArray(type) ? type[0] : type;
|
|
2100
|
+
if (typeof type === "function" && !type.name) type = type();
|
|
2101
|
+
return ApiProperty({
|
|
2102
|
+
type: type,
|
|
2103
|
+
required: false
|
|
2104
|
+
})(target, prop);
|
|
2105
|
+
};
|
|
2106
|
+
}
|
|
2107
|
+
}).apply();
|
|
2108
|
+
|
|
2109
|
+
Decoration.for(ValidationKeys.DATE).extend({
|
|
2110
|
+
decorator: function dateDec(format) {
|
|
2111
|
+
return ApiProperty({
|
|
2112
|
+
type: String,
|
|
2113
|
+
format: "date-time",
|
|
2114
|
+
required: false
|
|
2115
|
+
});
|
|
2116
|
+
}
|
|
2117
|
+
}).apply();
|
|
2118
|
+
|
|
2119
|
+
Decoration.for(ValidationKeys.ENUM).extend({
|
|
2120
|
+
decorator: function optionDec(options) {
|
|
2121
|
+
const opts = Array.isArray(options) ? options : Object.values(options);
|
|
2122
|
+
return ApiProperty({
|
|
2123
|
+
enum: opts,
|
|
2124
|
+
required: false
|
|
2125
|
+
});
|
|
2126
|
+
}
|
|
2127
|
+
}).apply();
|
|
2128
|
+
|
|
2129
|
+
Decoration.for(ValidationKeys.PATTERN).extend({
|
|
2130
|
+
decorator: function patternDec(pat) {
|
|
2131
|
+
return ApiProperty({
|
|
2132
|
+
pattern: typeof pat === "string" ? pat : pat.source,
|
|
2133
|
+
required: false
|
|
2134
|
+
});
|
|
2135
|
+
}
|
|
2136
|
+
}).apply();
|
|
2137
|
+
|
|
2138
|
+
Decoration.for(PersistenceKeys.COLUMN).extend({
|
|
2139
|
+
decorator: function columnDec(name) {
|
|
2140
|
+
return ApiProperty({
|
|
2141
|
+
name: name,
|
|
2142
|
+
required: false
|
|
2143
|
+
});
|
|
2144
|
+
}
|
|
2145
|
+
}).apply();
|
|
2146
|
+
|
|
2147
|
+
Decoration.for(DecorationKeys.DESCRIPTION).extend({
|
|
2148
|
+
decorator: function descriptionDec(description) {
|
|
2149
|
+
return ApiProperty({
|
|
2150
|
+
description: description,
|
|
2151
|
+
required: false
|
|
2152
|
+
});
|
|
2153
|
+
}
|
|
2154
|
+
}).apply();
|
|
2155
|
+
|
|
2156
|
+
Decoration.for(PersistenceKeys.AUTH).extend({
|
|
2157
|
+
decorator: Auth
|
|
2158
|
+
}).apply();
|
|
2159
|
+
|
|
2160
|
+
class CorsError extends ForbiddenError {
|
|
2161
|
+
constructor(msg) {
|
|
2162
|
+
super(msg, CorsError.name);
|
|
2163
|
+
}
|
|
2164
|
+
}
|
|
2165
|
+
|
|
2166
|
+
class ToManyRequestsError extends BaseError {
|
|
2167
|
+
constructor(msg) {
|
|
2168
|
+
super(ToManyRequestsError.name, msg, 429);
|
|
2169
|
+
}
|
|
2170
|
+
}
|
|
2171
|
+
|
|
2172
|
+
let DecafExceptionFilter = class DecafExceptionFilter {
|
|
2173
|
+
catch(exception, host) {
|
|
2174
|
+
const ctx = host.switchToHttp();
|
|
2175
|
+
const response = ctx.getResponse();
|
|
2176
|
+
const request = ctx.getRequest();
|
|
2177
|
+
const isProduction = LoggedEnvironment.env === "production";
|
|
2178
|
+
let statusCode;
|
|
2179
|
+
if (exception instanceof NotFoundException || exception instanceof UnsupportedError) {
|
|
2180
|
+
exception = new NotAcceptableException(exception.message);
|
|
2181
|
+
statusCode = exception.getStatus();
|
|
2182
|
+
} else if (!(exception instanceof BaseError)) {
|
|
2183
|
+
if (exception.status === 429) {
|
|
2184
|
+
exception = new ToManyRequestsError(exception.message);
|
|
2185
|
+
} else {
|
|
2186
|
+
exception = new InternalError(exception.message);
|
|
2187
|
+
}
|
|
2188
|
+
}
|
|
2189
|
+
response.status(exception.code || statusCode).json({
|
|
2190
|
+
status: exception.code || statusCode,
|
|
2191
|
+
error: isProduction ? exception.name : exception.message,
|
|
2192
|
+
timestamp: (new Date).toISOString(),
|
|
2193
|
+
path: request.url,
|
|
2194
|
+
method: request.method
|
|
2195
|
+
});
|
|
2196
|
+
}
|
|
2197
|
+
};
|
|
2198
|
+
|
|
2199
|
+
DecafExceptionFilter = __decorate([ Catch() ], DecafExceptionFilter);
|
|
2200
|
+
|
|
2201
|
+
function UseDecafFilter() {
|
|
2202
|
+
return UseFilters(new DecafExceptionFilter);
|
|
2203
|
+
}
|
|
2204
|
+
|
|
2205
|
+
function UseDecafHeaders() {
|
|
2206
|
+
return UseInterceptors(DecafResponseInterceptor);
|
|
2207
|
+
}
|
|
2208
|
+
|
|
2209
|
+
const SWAGGER_UI_CONSTANTS = {
|
|
2210
|
+
title: "Swagger | OpenAPI Specification (OAS)",
|
|
2211
|
+
description: "Standardized format for describing RESTful APIs",
|
|
2212
|
+
version: "0.0.1",
|
|
2213
|
+
path: "docs",
|
|
2214
|
+
faviconFilePath: "",
|
|
2215
|
+
topbarIconFilePath: "",
|
|
2216
|
+
auth: {
|
|
2217
|
+
type: "http",
|
|
2218
|
+
scheme: "bearer",
|
|
2219
|
+
bearerFormat: "JWT",
|
|
2220
|
+
name: "Authorization",
|
|
2221
|
+
description: "Enter JWT token",
|
|
2222
|
+
in: "header"
|
|
2223
|
+
},
|
|
2224
|
+
persistAuthorization: true,
|
|
2225
|
+
topbarBgColor: "#000000"
|
|
2226
|
+
};
|
|
2227
|
+
|
|
2228
|
+
class SwaggerCustomUI {
|
|
2229
|
+
constructor(options) {
|
|
2230
|
+
this.options = {
|
|
2231
|
+
...options
|
|
2232
|
+
};
|
|
2233
|
+
}
|
|
2234
|
+
customCSS() {
|
|
2235
|
+
let css = "";
|
|
2236
|
+
if (this.options.topbarIconPath) {
|
|
2237
|
+
const img = this.b64(this.options.topbarIconPath);
|
|
2238
|
+
css += `.topbar-wrapper { content: url('data:image/png;base64,${img}'); width: 200px; height: auto; }\n`;
|
|
2239
|
+
}
|
|
2240
|
+
return css + `\n .topbar-wrapper svg { visibility: hidden; }\n .swagger-ui .topbar { background-color: ${this.options.topbarBgColor || SWAGGER_UI_CONSTANTS.topbarBgColor}; }\n `;
|
|
2241
|
+
}
|
|
2242
|
+
getCustomOptions() {
|
|
2243
|
+
const favicon = {};
|
|
2244
|
+
if (this.options.faviconPath) {
|
|
2245
|
+
favicon["customfavIcon"] = this.b64(this.options.faviconPath, true);
|
|
2246
|
+
}
|
|
2247
|
+
return {
|
|
2248
|
+
customSiteTitle: this.options.title,
|
|
2249
|
+
...favicon,
|
|
2250
|
+
customCss: this.customCSS(),
|
|
2251
|
+
swaggerOptions: {
|
|
2252
|
+
persistAuthorization: this.options.persistAuthorization
|
|
2253
|
+
}
|
|
2254
|
+
};
|
|
2255
|
+
}
|
|
2256
|
+
b64(file, img = false) {
|
|
2257
|
+
const filePath = path.join(this.options.assetsPath || "", file);
|
|
2258
|
+
const b64 = readFileSync(filePath, {
|
|
2259
|
+
encoding: "base64"
|
|
2260
|
+
});
|
|
2261
|
+
return img ? "data:image/png;base64," + b64 : b64;
|
|
2262
|
+
}
|
|
2263
|
+
}
|
|
2264
|
+
|
|
2265
|
+
class SwaggerBuilder {
|
|
2266
|
+
constructor(app, options) {
|
|
2267
|
+
this.app = app;
|
|
2268
|
+
this.options = options;
|
|
2269
|
+
}
|
|
2270
|
+
createDocument() {
|
|
2271
|
+
const description = this.options.path ? this.options.description + "" + `<br><br><a href="${this.options.openApiJsonPath}">OpenAPI JSON Spec</a> | ` + `<a href="${this.options.openApiYamlPath}">OpenAPI YAML Spec</a>` : this.options.description;
|
|
2272
|
+
const config = (new DocumentBuilder).setTitle(this.options.title).setDescription(description).setVersion(this.options.version || "0.0.1").addBearerAuth(this.options.auth || SWAGGER_UI_CONSTANTS.auth).build();
|
|
2273
|
+
return SwaggerModule.createDocument(this.app, config, {
|
|
2274
|
+
extraModels: this.options.extraModels || []
|
|
2275
|
+
});
|
|
2276
|
+
}
|
|
2277
|
+
registerOpenApiRoute(path, contentType, bodyFactory) {
|
|
2278
|
+
if (!path) return;
|
|
2279
|
+
const httpAdapter = this.app.getHttpAdapter();
|
|
2280
|
+
path = path.startsWith("/") ? path : `/${path}`;
|
|
2281
|
+
httpAdapter.get(path, (_req, res) => {
|
|
2282
|
+
httpAdapter.reply(res, bodyFactory(), 200, {
|
|
2283
|
+
"Content-Type": contentType
|
|
2284
|
+
});
|
|
2285
|
+
});
|
|
2286
|
+
}
|
|
2287
|
+
setupSwagger() {
|
|
2288
|
+
const document = this.createDocument();
|
|
2289
|
+
const swaggerUI = new SwaggerCustomUI({
|
|
2290
|
+
title: this.options.title,
|
|
2291
|
+
path: this.options.path || SWAGGER_UI_CONSTANTS.path,
|
|
2292
|
+
persistAuthorization: this.options.persistAuthorization ?? true,
|
|
2293
|
+
assetsPath: this.options.assetsPath,
|
|
2294
|
+
faviconPath: this.options.faviconFilePath,
|
|
2295
|
+
topbarIconPath: this.options.topbarIconFilePath,
|
|
2296
|
+
topbarBgColor: this.options.topbarBgColor
|
|
2297
|
+
});
|
|
2298
|
+
SwaggerModule.setup(this.options.path || SWAGGER_UI_CONSTANTS.path, this.app, document, {
|
|
2299
|
+
...swaggerUI.getCustomOptions(),
|
|
2300
|
+
jsonDocumentUrl: this.options.openApiJsonPath ? `${this.options.openApiJsonPath}` : undefined,
|
|
2301
|
+
yamlDocumentUrl: this.options.openApiYamlPath ? `${this.options.openApiYamlPath}` : undefined
|
|
2302
|
+
});
|
|
2303
|
+
this.registerOpenApiRoute(this.options.openApiJsonPath, "application/json", () => document);
|
|
2304
|
+
this.registerOpenApiRoute(this.options.openApiYamlPath, "application/x-yaml", () => YAML.stringify(document));
|
|
2305
|
+
}
|
|
2306
|
+
}
|
|
2307
|
+
|
|
2308
|
+
class NestBootstraper {
|
|
2309
|
+
static get logger() {
|
|
2310
|
+
if (!this._logger) {
|
|
2311
|
+
this._logger = new Logger("NestBootstrap");
|
|
2312
|
+
}
|
|
2313
|
+
return this._logger;
|
|
2314
|
+
}
|
|
2315
|
+
static initialize(app) {
|
|
2316
|
+
this.app = app;
|
|
2317
|
+
return this;
|
|
2318
|
+
}
|
|
2319
|
+
static enableLogger(customLogger) {
|
|
2320
|
+
this._logger = customLogger || new Logger("NestBootstrap");
|
|
2321
|
+
this.app.useLogger(this._logger);
|
|
2322
|
+
return this;
|
|
2323
|
+
}
|
|
2324
|
+
static enableCors(origins = [], allowMethods = [ "GET", "POST", "PUT", "DELETE" ]) {
|
|
2325
|
+
const allowedOrigins = origins === "*" ? "*" : origins.map(o => o.trim().toLowerCase());
|
|
2326
|
+
const corsOptions = {
|
|
2327
|
+
origin: (origin, callback) => {
|
|
2328
|
+
if (!origin) return callback(null, true);
|
|
2329
|
+
if (allowedOrigins === "*" || Array.isArray(allowedOrigins) && allowedOrigins.includes(origin.toLowerCase())) {
|
|
2330
|
+
return callback(null, true);
|
|
2331
|
+
}
|
|
2332
|
+
callback(new CorsError(`Origin ${origin} not allowed`));
|
|
2333
|
+
},
|
|
2334
|
+
credentials: true,
|
|
2335
|
+
methods: allowMethods.join(",")
|
|
2336
|
+
};
|
|
2337
|
+
this.app.enableCors(corsOptions);
|
|
2338
|
+
return this;
|
|
2339
|
+
}
|
|
2340
|
+
static useHelmet(options) {
|
|
2341
|
+
try {
|
|
2342
|
+
const helmet = require("helmet");
|
|
2343
|
+
this.app.use(helmet(options));
|
|
2344
|
+
this.logger.log("Helmet middleware enabled successfully.");
|
|
2345
|
+
} catch (e) {
|
|
2346
|
+
this.logger.warn("Helmet not installed. Skipping middleware.");
|
|
2347
|
+
}
|
|
2348
|
+
return this;
|
|
2349
|
+
}
|
|
2350
|
+
static setupSwagger(options) {
|
|
2351
|
+
const swagger = new SwaggerBuilder(this.app, {
|
|
2352
|
+
title: options.title,
|
|
2353
|
+
description: options.description,
|
|
2354
|
+
version: options.version,
|
|
2355
|
+
path: options.path || "api",
|
|
2356
|
+
persistAuthorization: options.persistAuthorization ?? true,
|
|
2357
|
+
assetsPath: options.assetsPath,
|
|
2358
|
+
faviconFilePath: options.faviconPath,
|
|
2359
|
+
topbarIconFilePath: options.topbarIconPath,
|
|
2360
|
+
topbarBgColor: options.topbarBgColor,
|
|
2361
|
+
openApiJsonPath: options.openApiJsonPath,
|
|
2362
|
+
openApiYamlPath: options.openApiYamlPath
|
|
2363
|
+
});
|
|
2364
|
+
swagger.setupSwagger();
|
|
2365
|
+
return this;
|
|
2366
|
+
}
|
|
2367
|
+
static useGlobalPipes(...pipes) {
|
|
2368
|
+
if (pipes.length > 0) this.app.useGlobalPipes(...pipes);
|
|
2369
|
+
return this;
|
|
2370
|
+
}
|
|
2371
|
+
static useGlobalFilters(...filters) {
|
|
2372
|
+
this.app.useGlobalFilters(...filters.length > 0 ? filters : [ new DecafExceptionFilter ]);
|
|
2373
|
+
return this;
|
|
2374
|
+
}
|
|
2375
|
+
static useGlobalInterceptors(...interceptors) {
|
|
2376
|
+
if (interceptors.length > 0) this.app.useGlobalInterceptors(...interceptors);
|
|
2377
|
+
return this;
|
|
2378
|
+
}
|
|
2379
|
+
static async start(port = Number(process.env.PORT) || 3e3, host = undefined, log = true) {
|
|
2380
|
+
this.app.listen(port, host).then(async () => {
|
|
2381
|
+
if (log) {
|
|
2382
|
+
const url = await this.app.getUrl();
|
|
2383
|
+
this.logger.log(`🚀 Application is running at: ${url}`);
|
|
2384
|
+
}
|
|
2385
|
+
});
|
|
2386
|
+
}
|
|
2387
|
+
}
|
|
2388
|
+
|
|
2389
|
+
const pendingProviders = new Map;
|
|
2390
|
+
|
|
2391
|
+
function getRegisteredDecafProviders() {
|
|
2392
|
+
return Array.from(pendingProviders.values());
|
|
2393
|
+
}
|
|
2394
|
+
|
|
2395
|
+
function registerProvider(token, useFactory) {
|
|
2396
|
+
if (pendingProviders.has(token)) return;
|
|
2397
|
+
pendingProviders.set(token, {
|
|
2398
|
+
provide: token,
|
|
2399
|
+
useFactory: useFactory
|
|
2400
|
+
});
|
|
2401
|
+
}
|
|
2402
|
+
|
|
2403
|
+
function isModelConstructor(value) {
|
|
2404
|
+
return typeof value === "function" && value.prototype instanceof Model;
|
|
2405
|
+
}
|
|
2406
|
+
|
|
2407
|
+
function modelServiceToken(model) {
|
|
2408
|
+
return `${model.name}Service`;
|
|
2409
|
+
}
|
|
2410
|
+
|
|
2411
|
+
function modelRepositoryToken(model, flavour) {
|
|
2412
|
+
return flavour ? `${model.name}Repository@${flavour}` : `${model.name}Repository`;
|
|
2413
|
+
}
|
|
2414
|
+
|
|
2415
|
+
function Service(key) {
|
|
2416
|
+
return (target, propertyKey, parameterIndex) => {
|
|
2417
|
+
let resolved = key;
|
|
2418
|
+
if (typeof resolved === "undefined") {
|
|
2419
|
+
const paramTypes = Reflect.getMetadata("design:paramtypes", target) || [];
|
|
2420
|
+
resolved = paramTypes[parameterIndex];
|
|
2421
|
+
}
|
|
2422
|
+
if (typeof resolved === "undefined") {
|
|
2423
|
+
throw new Error(`@Service() could not determine an injection type for parameter ${parameterIndex} of ${target.name}. Provide an explicit argument, e.g. @Service(SomeService).`);
|
|
2424
|
+
}
|
|
2425
|
+
const asModel = typeof resolved !== "string" && isModelConstructor(resolved);
|
|
2426
|
+
const token = asModel ? modelServiceToken(resolved) : injectableServiceKey(resolved);
|
|
2427
|
+
registerProvider(token, () => {
|
|
2428
|
+
if (asModel) return ModelService.forModel(resolved);
|
|
2429
|
+
return typeof resolved === "string" ? Service$1.get(resolved) : Service$1.get(resolved);
|
|
2430
|
+
});
|
|
2431
|
+
return Inject(token)(target, propertyKey, parameterIndex);
|
|
2432
|
+
};
|
|
2433
|
+
}
|
|
2434
|
+
|
|
2435
|
+
function Repository(model, flavour) {
|
|
2436
|
+
return (target, propertyKey, parameterIndex) => {
|
|
2437
|
+
const token = modelRepositoryToken(model, flavour);
|
|
2438
|
+
registerProvider(token, () => Repository$1.forModel(model, flavour));
|
|
2439
|
+
return Inject(token)(target, propertyKey, parameterIndex);
|
|
2440
|
+
};
|
|
2441
|
+
}
|
|
2442
|
+
|
|
2443
|
+
var DecafCoreModule_1;
|
|
2444
|
+
|
|
2445
|
+
let DecafCoreModule = class DecafCoreModule {
|
|
2446
|
+
static {
|
|
2447
|
+
DecafCoreModule_1 = this;
|
|
2448
|
+
}
|
|
2449
|
+
static get persistence() {
|
|
2450
|
+
if (!this._persistence) throw new InternalError("Persistence service not initialized");
|
|
2451
|
+
return this._persistence;
|
|
2452
|
+
}
|
|
2453
|
+
static get log() {
|
|
2454
|
+
if (!this._logger) this._logger = Logging.for(DecafCoreModule_1);
|
|
2455
|
+
return this._logger;
|
|
2456
|
+
}
|
|
2457
|
+
constructor(options, moduleRef) {
|
|
2458
|
+
this.options = options;
|
|
2459
|
+
this.moduleRef = moduleRef;
|
|
2460
|
+
}
|
|
2461
|
+
static forRoot(options) {
|
|
2462
|
+
const log = this.log.for(this.forRoot);
|
|
2463
|
+
const decafProviders = getRegisteredDecafProviders();
|
|
2464
|
+
return {
|
|
2465
|
+
module: DecafCoreModule_1,
|
|
2466
|
+
providers: [ {
|
|
2467
|
+
provide: DECAF_MODULE_OPTIONS,
|
|
2468
|
+
useValue: options
|
|
2469
|
+
}, {
|
|
2470
|
+
provide: DECAF_ADAPTER_ID,
|
|
2471
|
+
useValue: this.persistence?.client
|
|
2472
|
+
}, {
|
|
2473
|
+
provide: DECAF_HANDLERS,
|
|
2474
|
+
useFactory: () => options.handlers?.map(H => {
|
|
2475
|
+
log.info(`Registered request handler: ${H.name}`);
|
|
2476
|
+
return new H;
|
|
2477
|
+
}) ?? []
|
|
2478
|
+
}, DecafRequestContext, DecafHandlerExecutor, {
|
|
2479
|
+
provide: APP_INTERCEPTOR,
|
|
2480
|
+
useClass: DecafRequestHandlerInterceptor
|
|
2481
|
+
}, ...decafProviders ],
|
|
2482
|
+
exports: [ DECAF_MODULE_OPTIONS, DECAF_ADAPTER_ID, DECAF_HANDLERS, DecafRequestContext, DecafHandlerExecutor, ...decafProviders.map(provider => provider.provide) ]
|
|
2483
|
+
};
|
|
2484
|
+
}
|
|
2485
|
+
static async bootPersistence(options) {
|
|
2486
|
+
const log = this.log.for(this.bootPersistence);
|
|
2487
|
+
if (!this._persistence) {
|
|
2488
|
+
const trimmed = options.conf.map(([contr, cfg, ...args]) => {
|
|
2489
|
+
const possible = args.pop();
|
|
2490
|
+
if (!possible) return [ contr, cfg ];
|
|
2491
|
+
return [ contr, cfg, ...args ];
|
|
2492
|
+
});
|
|
2493
|
+
this._persistence = new PersistenceService;
|
|
2494
|
+
await this._persistence.boot(trimmed);
|
|
2495
|
+
const clients = this._persistence.client;
|
|
2496
|
+
for (let i = 0; i < clients.length; i++) {
|
|
2497
|
+
const c = options.conf[i];
|
|
2498
|
+
const possibleTransf = c.slice(2, c.length);
|
|
2499
|
+
let transformer = possibleTransf.pop();
|
|
2500
|
+
if (!transformer || !transformer.from) {
|
|
2501
|
+
const contr = Adapter.transformerFor(clients[i].flavour);
|
|
2502
|
+
if (!contr) throw new InternalError(`No transformer found for flavour ${clients[i].flavour}. you should either @requestToContextTransformer or provide a transformer in the config`);
|
|
2503
|
+
try {
|
|
2504
|
+
transformer = contr.from ? contr : new contr;
|
|
2505
|
+
} catch (e) {
|
|
2506
|
+
throw new InternalError(`Failed to boot transformer for ${clients[i].flavour}: ${e}`);
|
|
2507
|
+
}
|
|
2508
|
+
}
|
|
2509
|
+
requestToContextTransformer(clients[i].flavour)(transformer);
|
|
2510
|
+
}
|
|
2511
|
+
log.info("persistence layer created successfully!");
|
|
2512
|
+
if (options.initialization) {
|
|
2513
|
+
try {
|
|
2514
|
+
await options.initialization();
|
|
2515
|
+
} catch (e) {
|
|
2516
|
+
throw new InternalError(`Failed to initialize application: ${e}`);
|
|
2517
|
+
}
|
|
2518
|
+
}
|
|
2519
|
+
}
|
|
2520
|
+
return this.persistence.client;
|
|
2521
|
+
}
|
|
2522
|
+
async onApplicationShutdown() {
|
|
2523
|
+
const log = DecafCoreModule_1.log.for(this.onApplicationShutdown);
|
|
2524
|
+
const adapters = this.moduleRef.get(DECAF_ADAPTER_ID);
|
|
2525
|
+
for (const adapter of adapters) try {
|
|
2526
|
+
if (adapter) {
|
|
2527
|
+
log.info(`Shutting down ${adapter.toString()}`);
|
|
2528
|
+
await adapter.shutdown();
|
|
2529
|
+
}
|
|
2530
|
+
} catch (e) {
|
|
2531
|
+
log.error(`Failed to shutdown application`, e);
|
|
2532
|
+
}
|
|
2533
|
+
try {
|
|
2534
|
+
await Service$1.shutdown();
|
|
2535
|
+
} catch (e) {
|
|
2536
|
+
log.error(`Failed to shutdown services`, e);
|
|
2537
|
+
}
|
|
2538
|
+
}
|
|
2539
|
+
};
|
|
2540
|
+
|
|
2541
|
+
DecafCoreModule = DecafCoreModule_1 = __decorate([ Global(), Module({}), __param(0, Inject(DECAF_MODULE_OPTIONS)), __metadata("design:paramtypes", [ Object, ModuleRef ]) ], DecafCoreModule);
|
|
2542
|
+
|
|
2543
|
+
const LISTENING_ADAPTERS_FLAVOURS = Symbol("LISTENING_ADAPTERS_FLAVOURS");
|
|
2544
|
+
|
|
2545
|
+
function normalizeEventResponse(args) {
|
|
2546
|
+
const [modelConstr, operation, id, payload] = args;
|
|
2547
|
+
const modelName = modelConstr?.name ?? modelConstr;
|
|
2548
|
+
const serializedPayload = Array.isArray(payload) ? payload.map(e => {
|
|
2549
|
+
try {
|
|
2550
|
+
if (typeof e.serialize === "function") return e.serialize();
|
|
2551
|
+
console.warn(`Payload item for ${modelName} does not have serialize method and is an ${typeof e}, attempting to stringify directly. Item: ${e}`);
|
|
2552
|
+
return typeof e === "string" ? e : JSON.stringify(e);
|
|
2553
|
+
} catch (err) {
|
|
2554
|
+
console.warn(`Failed to serialize payload for ${modelName}: ${err}`);
|
|
2555
|
+
return undefined;
|
|
2556
|
+
}
|
|
2557
|
+
}) : payload && typeof payload.serialize === "function" ? payload.serialize() : typeof payload === "string" ? payload : JSON.stringify(payload);
|
|
2558
|
+
console.debug(`Normalized event response for model ${modelName}, operation ${operation}, id ${id}:`, serializedPayload);
|
|
2559
|
+
return [ modelName, operation, id, serializedPayload ];
|
|
2560
|
+
}
|
|
2561
|
+
|
|
2562
|
+
var EventsController_1;
|
|
2563
|
+
|
|
2564
|
+
let EventsController = EventsController_1 = class EventsController extends DecafController {
|
|
2565
|
+
constructor(clientContext, flavours) {
|
|
2566
|
+
super(clientContext, EventsController_1.name);
|
|
2567
|
+
this.adapters = flavours.map(flavour => Adapter.get(flavour));
|
|
2568
|
+
}
|
|
2569
|
+
listen() {
|
|
2570
|
+
const logger = Logging.for(EventsController_1.name);
|
|
2571
|
+
const events$ = new Observable(observer => {
|
|
2572
|
+
const observerId = `B-${Math.random().toString(36).slice(2, 8)}`.toUpperCase();
|
|
2573
|
+
logger.info(`Creating SSE observer: ${observerId} for client ${this.clientContext.uuid}`);
|
|
2574
|
+
const cb = new class {
|
|
2575
|
+
constructor() {
|
|
2576
|
+
this.observerId = observerId;
|
|
2577
|
+
}
|
|
2578
|
+
refresh(...args) {
|
|
2579
|
+
logger.debug(`SSE observer ${this.observerId} received refresh event`);
|
|
2580
|
+
return Promise.resolve().then(() => {
|
|
2581
|
+
const data = normalizeEventResponse(args);
|
|
2582
|
+
observer.next({
|
|
2583
|
+
type: "message",
|
|
2584
|
+
data: data
|
|
2585
|
+
});
|
|
2586
|
+
logger.debug(`SSE observer ${this.observerId} event pushed to client`);
|
|
2587
|
+
});
|
|
2588
|
+
}
|
|
2589
|
+
};
|
|
2590
|
+
logger.verbose(`Registering observer ${observerId} across ${this.adapters.length} adapter(s)`);
|
|
2591
|
+
for (const adapter of this.adapters) {
|
|
2592
|
+
const adapterName = adapter?.constructor?.name ?? "UnknownAdapter";
|
|
2593
|
+
try {
|
|
2594
|
+
logger.debug(`Registering observer ${observerId} in adapter ${adapterName}`);
|
|
2595
|
+
adapter.observe(cb);
|
|
2596
|
+
} catch (e) {
|
|
2597
|
+
logger.debug(`Failed to register observer ${observerId} in adapter ${adapterName}: ${e?.message || e}`);
|
|
2598
|
+
logger.error(e);
|
|
2599
|
+
}
|
|
2600
|
+
}
|
|
2601
|
+
return () => {
|
|
2602
|
+
logger.debug(`Cleaning up SSE observer ${observerId}`);
|
|
2603
|
+
for (const adapter of this.adapters) {
|
|
2604
|
+
const adapterName = adapter?.constructor?.name ?? "UnknownAdapter";
|
|
2605
|
+
try {
|
|
2606
|
+
logger.debug(`Unregistering observer ${observerId} from adapter ${adapterName}`);
|
|
2607
|
+
adapter.unObserve(cb);
|
|
2608
|
+
} catch (e) {
|
|
2609
|
+
logger.debug(`Failed during cleanup of observer ${observerId} in adapter ${adapterName}: ${e?.message || e}`);
|
|
2610
|
+
logger.error(e);
|
|
2611
|
+
}
|
|
2612
|
+
}
|
|
2613
|
+
};
|
|
2614
|
+
});
|
|
2615
|
+
const HEARTBEAT_INTERVAL_MS = 15e3;
|
|
2616
|
+
const heartbeat$ = interval(HEARTBEAT_INTERVAL_MS).pipe(tap$1(() => {
|
|
2617
|
+
logger.debug("Sending heartbeat");
|
|
2618
|
+
}), map(() => ({
|
|
2619
|
+
type: "heartbeat",
|
|
2620
|
+
data: {
|
|
2621
|
+
ts: (new Date).toISOString()
|
|
2622
|
+
}
|
|
2623
|
+
})));
|
|
2624
|
+
return merge(events$, heartbeat$);
|
|
2625
|
+
}
|
|
2626
|
+
listenForModel(model) {
|
|
2627
|
+
const logger = Logging.for(EventsController_1.name);
|
|
2628
|
+
return new Observable(observer => {
|
|
2629
|
+
const cb = new class {
|
|
2630
|
+
refresh(...args) {
|
|
2631
|
+
return Promise.resolve().then(() => {
|
|
2632
|
+
observer.next({
|
|
2633
|
+
data: args
|
|
2634
|
+
});
|
|
2635
|
+
});
|
|
2636
|
+
}
|
|
2637
|
+
};
|
|
2638
|
+
try {
|
|
2639
|
+
for (const adapter of this.adapters) {
|
|
2640
|
+
adapter.observe(cb);
|
|
2641
|
+
}
|
|
2642
|
+
} catch (e) {
|
|
2643
|
+
observer.error(`Failed to observe event: ${e.message || e}`);
|
|
2644
|
+
}
|
|
2645
|
+
return () => {
|
|
2646
|
+
try {
|
|
2647
|
+
for (const adapter of this.adapters) {
|
|
2648
|
+
adapter.unObserve(cb);
|
|
2649
|
+
}
|
|
2650
|
+
} catch (e) {
|
|
2651
|
+
logger.error(e);
|
|
2652
|
+
}
|
|
2653
|
+
};
|
|
2654
|
+
});
|
|
2655
|
+
}
|
|
2656
|
+
};
|
|
2657
|
+
|
|
2658
|
+
__decorate([ Sse(), __metadata("design:type", Function), __metadata("design:paramtypes", []), __metadata("design:returntype", Observable) ], EventsController.prototype, "listen", null);
|
|
2659
|
+
|
|
2660
|
+
__decorate([ Sse("/:model"), __param(0, Query("model")), __metadata("design:type", Function), __metadata("design:paramtypes", [ String ]), __metadata("design:returntype", Observable) ], EventsController.prototype, "listenForModel", null);
|
|
2661
|
+
|
|
2662
|
+
EventsController = EventsController_1 = __decorate([ Controller(), __param(1, Inject(LISTENING_ADAPTERS_FLAVOURS)), __metadata("design:paramtypes", [ DecafRequestContext, Array ]) ], EventsController);
|
|
2663
|
+
|
|
2664
|
+
var DecafStreamModule_1;
|
|
2665
|
+
|
|
2666
|
+
let DecafStreamModule = DecafStreamModule_1 = class DecafStreamModule {
|
|
2667
|
+
static forFlavours(flavours, path = "events") {
|
|
2668
|
+
return {
|
|
2669
|
+
module: DecafStreamModule_1,
|
|
2670
|
+
controllers: [ EventsController ],
|
|
2671
|
+
imports: [ RouterModule.register([ {
|
|
2672
|
+
path: path.replace(/^\//, ""),
|
|
2673
|
+
module: DecafStreamModule_1
|
|
2674
|
+
} ]) ],
|
|
2675
|
+
providers: [ DecafRequestContext, {
|
|
2676
|
+
provide: LISTENING_ADAPTERS_FLAVOURS,
|
|
2677
|
+
useValue: flavours ?? []
|
|
2678
|
+
} ]
|
|
2679
|
+
};
|
|
2680
|
+
}
|
|
2681
|
+
};
|
|
2682
|
+
|
|
2683
|
+
DecafStreamModule = DecafStreamModule_1 = __decorate([ Module({}) ], DecafStreamModule);
|
|
2684
|
+
|
|
2685
|
+
var DecafModule_1;
|
|
2686
|
+
|
|
2687
|
+
let DecafModule = DecafModule_1 = class DecafModule {
|
|
2688
|
+
static async forRootAsync(options) {
|
|
2689
|
+
const {autoControllers: autoControllers, autoServices: autoServices} = options;
|
|
2690
|
+
const adapters = await DecafCoreModule.bootPersistence(options);
|
|
2691
|
+
const flavours = adapters.map(adapter => adapter.flavour);
|
|
2692
|
+
const imports = [ DecafCoreModule.forRoot(options) ];
|
|
2693
|
+
if (autoControllers) {
|
|
2694
|
+
flavours.forEach(flavour => {
|
|
2695
|
+
imports.push(getModuleFor(flavour).forRoot(flavour, {
|
|
2696
|
+
autoServices: autoServices,
|
|
2697
|
+
controllerExposure: options.controllerExposure,
|
|
2698
|
+
controllerConfig: options.controllerConfig
|
|
2699
|
+
}));
|
|
2700
|
+
});
|
|
2701
|
+
}
|
|
2702
|
+
if (options.observerOptions?.enableObserverEvents) {
|
|
2703
|
+
imports.push(DecafStreamModule.forFlavours(options.observerOptions.observerFlavours || flavours));
|
|
2704
|
+
}
|
|
2705
|
+
return {
|
|
2706
|
+
module: DecafModule_1,
|
|
2707
|
+
imports: imports
|
|
2708
|
+
};
|
|
2709
|
+
}
|
|
2710
|
+
};
|
|
2711
|
+
|
|
2712
|
+
DecafModule = DecafModule_1 = __decorate([ Module({}) ], DecafModule);
|
|
2713
|
+
|
|
2714
|
+
async function runMigrations() {}
|
|
2715
|
+
|
|
2716
|
+
function repoForModel(model) {
|
|
2717
|
+
const m = Model.get(model);
|
|
2718
|
+
if (!m) throw new InternalError(`Failed to find repository for ${model}`);
|
|
2719
|
+
const repo = Repository$1.forModel(m);
|
|
2720
|
+
return repo;
|
|
2721
|
+
}
|
|
2722
|
+
|
|
2723
|
+
const VERSION = "0.11.2";
|
|
2724
|
+
|
|
2725
|
+
const COMMIT = "0a94615";
|
|
2726
|
+
|
|
2727
|
+
const FULL_VERSION = "0.11.2-0a94615";
|
|
2728
|
+
|
|
2729
|
+
const PACKAGE_NAME = "@decaf-ts/for-nest";
|
|
2730
|
+
|
|
2731
|
+
Metadata.allowReregistration(true);
|
|
2732
|
+
|
|
2733
|
+
Metadata.registerLibrary(PACKAGE_NAME, VERSION);
|
|
2734
|
+
|
|
2735
|
+
Metadata.allowReregistration(false);
|
|
2736
|
+
|
|
2737
|
+
export { AUTH_HANDLER, AUTH_META_KEY, ApiOperationFromModel, ApiParamsFromModel, ApiProperty, Auth, AuthInterceptor, BulkApiOperationFromModel, COMMIT, CorsError, DECAF_ADAPTER_ID, DECAF_ADAPTER_OPTIONS, DECAF_CONTEXT_KEY, DECAF_CONTROLLER_CONFIG, DECAF_EXPOSE, DECAF_HANDLERS, DECAF_MODULE_OPTIONS, DECAF_ROUTE, DECAF_TASK_SERVICE_ID, DECORATORS, DECORATORS_PREFIX, DecafAuthHandler, DecafAuthModule, DecafBody, DecafController, DecafCoreModule, DecafExceptionFilter, DecafHandlerExecutor, DecafModelController, DecafModule, DecafParams, DecafRequestContext, DecafRequestHandlerInterceptor, DecafResponseInterceptor, DecafRoleAuthHandler, DecafStreamModule, DtoFor, EventsController, FULL_VERSION, FromModelController, IS_PUBLIC_KEY, NestBootstraper, PACKAGE_NAME, Public, REQUIRED_ROLES_KEY, Repository, RequireRoles, Service, SwaggerBuilder, UseDecafFilter, UseDecafHeaders, VERSION, applyApiDecorators, controllerConfig, createApiPropertyDecorator, createPropertyDecorator, createRouteHandler, defineRouteMethod, expose, getApiDecorators, getEnumType, getEnumValues, getModuleFor, getRegisteredDecafProviders, isUndefined, repoForModel, resolvePersistenceMethod, runMigrations };
|
|
2738
|
+
//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9yLW5lc3QuanMiLCJzb3VyY2VzIjpbIi4uL3NyYy9vdmVycmlkZXMvY29uc3RhbnRzLnRzIiwiLi4vc3JjL292ZXJyaWRlcy9oZWxwZXJzLnRzIiwiLi4vc3JjL292ZXJyaWRlcy9kZWNvcmF0aW9uLnRzIiwiLi4vc3JjL2RlY2FmLW1vZGVsL2RlY29yYXRvcnMvQXBpT3BlcmF0aW9uRnJvbU1vZGVsLnRzIiwiLi4vc3JjL2RlY2FmLW1vZGVsL2RlY29yYXRvcnMvQXBpUGFyYW1zRnJvbU1vZGVsLnRzIiwiLi4vc3JjL2RlY2FmLW1vZGVsL2RlY29yYXRvcnMvRGVjYWZCb2R5LnRzIiwiLi4vc3JjL2RlY2FmLW1vZGVsL2RlY29yYXRvcnMvRGVjYWZQYXJhbXMudHMiLCIuLi9zcmMvY29uc3RhbnRzLnRzIiwiLi4vc3JjL2RlY2FmLW1vZGVsL2RlY29yYXRvcnMvZXhwb3NlLnRzIiwiLi4vc3JjL2RlY2FmLW1vZGVsL2RlY29yYXRvcnMvY29udHJvbGxlci1jb25maWcudHMiLCIuLi9zcmMvYXV0aC9jb25zdGFudHMudHMiLCIuLi9zcmMvcmVxdWVzdC9EZWNhZlJlcXVlc3RDb250ZXh0LnRzIiwiLi4vc3JjL2F1dGgvQXV0aEludGVyY2VwdG9yLnRzIiwiLi4vc3JjL2F1dGgvRGVjYWZBdXRoSGFuZGxlci50cyIsIi4uL3NyYy9hdXRoL2RlY29yYXRvcnMudHMiLCIuLi9zcmMvcmVxdWVzdC9EZWNhZkhhbmRsZXJFeGVjdXRvci50cyIsIi4uL3NyYy9yZXF1ZXN0L0RlY2FmUmVzcG9uc2VJbnRlcmNlcHRvci50cyIsIi4uL3NyYy9vdmVycmlkZXMvb3ZlcnJpZGVzLnRzIiwiLi4vc3JjL292ZXJyaWRlcy9Nb2RlbEJ1aWxkZXJFeHRlbnNpb25zLnRzIiwiLi4vc3JjL2ludGVyY2VwdG9ycy9EZWNhZlJlcXVlc3RIYW5kbGVySW50ZXJjZXB0b3IudHMiLCIuLi9zcmMvYXV0aC9EZWNhZkF1dGhNb2R1bGUudHMiLCIuLi9zcmMvZGVjYWYtbW9kZWwvZGVjb3JhdG9ycy91dGlscy50cyIsIi4uL3NyYy9kZWNhZi1tb2RlbC91dGlscy50cyIsIi4uL3NyYy9jb250cm9sbGVycy50cyIsIi4uL3NyYy9mYWN0b3J5L29wZW5hcGkvRHRvQnVpbGRlci50cyIsIi4uL3NyYy9kZWNhZi1tb2RlbC9Gcm9tTW9kZWxDb250cm9sbGVyLnRzIiwiLi4vc3JjL2RlY2FmLW1vZGVsL0RlY2FmTW9kZWxNb2R1bGUudHMiLCIuLi9zcmMvZGVjb3JhdGlvbi50cyIsIi4uL3NyYy9mYWN0b3J5L2Vycm9ycy9jb3JzLnRzIiwiLi4vc3JjL2ZhY3RvcnkvZXJyb3JzL3Rocm90dGxpbmcudHMiLCIuLi9zcmMvZmFjdG9yeS9leGNlcHRpb25zL0RlY2FmRXJyb3JGaWx0ZXIudHMiLCIuLi9zcmMvZmFjdG9yeS9leGNlcHRpb25zL2RlY29yYXRvcnMudHMiLCIuLi9zcmMvZmFjdG9yeS9vcGVuYXBpL2NvbnN0YW50cy50cyIsIi4uL3NyYy9mYWN0b3J5L29wZW5hcGkvU3dhZ2dlckN1c3RvbVVJLnRzIiwiLi4vc3JjL2ZhY3Rvcnkvb3BlbmFwaS9Td2FnZ2VyQnVpbGRlci50cyIsIi4uL3NyYy9mYWN0b3J5L05lc3RCb290c3RyYXBlci50cyIsIi4uL3NyYy9kZWNvcmF0b3JzLnRzIiwiLi4vc3JjL2NvcmUtbW9kdWxlLnRzIiwiLi4vc3JjL2V2ZW50cy1tb2R1bGUvY29uc3RhbnQudHMiLCIuLi9zcmMvZXZlbnRzLW1vZHVsZS91dGlscy50cyIsIi4uL3NyYy9ldmVudHMtbW9kdWxlL0V2ZW50c0NvbnRyb2xsZXIudHMiLCIuLi9zcmMvZXZlbnRzLW1vZHVsZS9EZWNhZlN0cmVhbU1vZHVsZS50cyIsIi4uL3NyYy9tb2R1bGUudHMiLCIuLi9zcmMvdXRpbHMudHMiLCIuLi9zcmMvaW5kZXgudHMiXSwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IGNvbnN0IERFQ09SQVRPUlNfUFJFRklYID0gXCJzd2FnZ2VyXCI7XG5leHBvcnQgY29uc3QgREVDT1JBVE9SUyA9IHtcbiAgQVBJX09QRVJBVElPTjogYCR7REVDT1JBVE9SU19QUkVGSVh9L2FwaU9wZXJhdGlvbmAsXG4gIEFQSV9SRVNQT05TRTogYCR7REVDT1JBVE9SU19QUkVGSVh9L2FwaVJlc3BvbnNlYCxcbiAgQVBJX1BST0RVQ0VTOiBgJHtERUNPUkFUT1JTX1BSRUZJWH0vYXBpUHJvZHVjZXNgLFxuICBBUElfQ09OU1VNRVM6IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlDb25zdW1lc2AsXG4gIEFQSV9UQUdTOiBgJHtERUNPUkFUT1JTX1BSRUZJWH0vYXBpVXNlVGFnc2AsXG4gIEFQSV9DQUxMQkFDS1M6IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlDYWxsYmFja3NgLFxuICBBUElfUEFSQU1FVEVSUzogYCR7REVDT1JBVE9SU19QUkVGSVh9L2FwaVBhcmFtZXRlcnNgLFxuICBBUElfSEVBREVSUzogYCR7REVDT1JBVE9SU19QUkVGSVh9L2FwaUhlYWRlcnNgLFxuICBBUElfTU9ERUxfUFJPUEVSVElFUzogYCR7REVDT1JBVE9SU19QUkVGSVh9L2FwaU1vZGVsUHJvcGVydGllc2AsXG4gIEFQSV9NT0RFTF9QUk9QRVJUSUVTX0FSUkFZOiBgJHtERUNPUkFUT1JTX1BSRUZJWH0vYXBpTW9kZWxQcm9wZXJ0aWVzQXJyYXlgLFxuICBBUElfU0VDVVJJVFk6IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlTZWN1cml0eWAsXG4gIEFQSV9FWENMVURFX0VORFBPSU5UOiBgJHtERUNPUkFUT1JTX1BSRUZJWH0vYXBpRXhjbHVkZUVuZHBvaW50YCxcbiAgQVBJX0VYQ0xVREVfQ09OVFJPTExFUjogYCR7REVDT1JBVE9SU19QUkVGSVh9L2FwaUV4Y2x1ZGVDb250cm9sbGVyYCxcbiAgQVBJX0VYVFJBX01PREVMUzogYCR7REVDT1JBVE9SU19QUkVGSVh9L2FwaUV4dHJhTW9kZWxzYCxcbiAgQVBJX0VYVEVOU0lPTjogYCR7REVDT1JBVE9SU19QUkVGSVh9L2FwaUV4dGVuc2lvbmAsXG4gIEFQSV9TQ0hFTUE6IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlTY2hlbWFgLFxuICBBUElfREVGQVVMVF9HRVRURVI6IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlEZWZhdWx0R2V0dGVyYCxcbiAgQVBJX0xJTks6IGAke0RFQ09SQVRPUlNfUFJFRklYfS9hcGlMaW5rYCxcbn07XG4iLCJpbXBvcnQgeyBpc0FycmF5LCBpc1VuZGVmaW5lZCwgbmVnYXRlLCBwaWNrQnkgfSBmcm9tIFwibG9kYXNoXCI7XG5pbXBvcnQgeyBERUNPUkFUT1JTIH0gZnJvbSBcIi4vY29uc3RhbnRzXCI7XG5cbmV4cG9ydCBjb25zdCBpc0NvbnN0cnVjdG9yID0gKHZhbDogYW55KTogYm9vbGVhbiA9PiB2YWwgPT09IFwiY29uc3RydWN0b3JcIjtcbmV4cG9ydCBjb25zdCBNRVRBREFUQV9GQUNUT1JZX05BTUUgPSBcIl9PUEVOQVBJX01FVEFEQVRBX0ZBQ1RPUllcIjtcblxuZXhwb3J0IGNvbnN0IE1FVEhPRF9NRVRBREFUQSA9IFwibWV0aG9kXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVNZXRob2REZWNvcmF0b3I8VCA9IGFueT4oXG4gIG1ldGFrZXk6IHN0cmluZyxcbiAgbWV0YWRhdGE6IFQsXG4gIHsgb3ZlcnJpZGVFeGlzdGluZyB9ID0geyBvdmVycmlkZUV4aXN0aW5nOiB0cnVlIH1cbik6IE1ldGhvZERlY29yYXRvciB7XG4gIHJldHVybiAoXG4gICAgdGFyZ2V0OiBvYmplY3QsXG4gICAga2V5OiBzdHJpbmcgfCBzeW1ib2wsXG4gICAgZGVzY3JpcHRvcjogUHJvcGVydHlEZXNjcmlwdG9yXG4gICkgPT4ge1xuICAgIGlmICh0eXBlb2YgbWV0YWRhdGEgPT09IFwib2JqZWN0XCIpIHtcbiAgICAgIGNvbnN0IHByZXZWYWx1ZSA9IFJlZmxlY3QuZ2V0TWV0YWRhdGEobWV0YWtleSwgZGVzY3JpcHRvci52YWx1ZSk7XG4gICAgICBpZiAocHJldlZhbHVlICYmICFvdmVycmlkZUV4aXN0aW5nKSB7XG4gICAgICAgIHJldHVybiBkZXNjcmlwdG9yO1xuICAgICAgfVxuICAgICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShcbiAgICAgICAgbWV0YWtleSxcbiAgICAgICAgeyAuLi5wcmV2VmFsdWUsIC4uLm1ldGFkYXRhIH0sXG4gICAgICAgIGRlc2NyaXB0b3IudmFsdWVcbiAgICAgICk7XG4gICAgICByZXR1cm4gZGVzY3JpcHRvcjtcbiAgICB9XG4gICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShtZXRha2V5LCBtZXRhZGF0YSwgZGVzY3JpcHRvci52YWx1ZSk7XG4gICAgcmV0dXJuIGRlc2NyaXB0b3I7XG4gIH07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVDbGFzc0RlY29yYXRvcjxUIGV4dGVuZHMgQXJyYXk8YW55PiA9IGFueT4oXG4gIG1ldGFrZXk6IHN0cmluZyxcbiAgbWV0YWRhdGE6IFQgPSBbXSBhcyBhbnlcbik6IENsYXNzRGVjb3JhdG9yIHtcbiAgcmV0dXJuICh0YXJnZXQpID0+IHtcbiAgICBjb25zdCBwcmV2VmFsdWUgPSBSZWZsZWN0LmdldE1ldGFkYXRhKG1ldGFrZXksIHRhcmdldCkgfHwgW107XG4gICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShtZXRha2V5LCBbLi4ucHJldlZhbHVlLCAuLi5tZXRhZGF0YV0sIHRhcmdldCk7XG4gICAgcmV0dXJuIHRhcmdldDtcbiAgfTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZVByb3BlcnR5RGVjb3JhdG9yPFQgZXh0ZW5kcyBSZWNvcmQ8c3RyaW5nLCBhbnk+ID0gYW55PihcbiAgbWV0YWtleTogc3RyaW5nLFxuICBtZXRhZGF0YTogVCxcbiAgb3ZlcnJpZGVFeGlzdGluZyA9IHRydWVcbik6IFByb3BlcnR5RGVjb3JhdG9yIHtcbiAgcmV0dXJuICgodGFyZ2V0OiBvYmplY3QsIHByb3BlcnR5S2V5OiBzdHJpbmcpID0+IHtcbiAgICBjb25zdCBwcm9wZXJ0aWVzID1cbiAgICAgIFJlZmxlY3QuZ2V0TWV0YWRhdGEoREVDT1JBVE9SUy5BUElfTU9ERUxfUFJPUEVSVElFU19BUlJBWSwgdGFyZ2V0KSB8fCBbXTtcblxuICAgIGNvbnN0IGtleSA9IGA6JHtwcm9wZXJ0eUtleX1gO1xuICAgIGlmICghcHJvcGVydGllcy5pbmNsdWRlcyhrZXkpKSB7XG4gICAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKFxuICAgICAgICBERUNPUkFUT1JTLkFQSV9NT0RFTF9QUk9QRVJUSUVTX0FSUkFZLFxuICAgICAgICBbLi4ucHJvcGVydGllcywgYDoke3Byb3BlcnR5S2V5fWBdLFxuICAgICAgICB0YXJnZXRcbiAgICAgICk7XG4gICAgfVxuICAgIGNvbnN0IGV4aXN0aW5nTWV0YWRhdGEgPSBSZWZsZWN0LmdldE1ldGFkYXRhKG1ldGFrZXksIHRhcmdldCwgcHJvcGVydHlLZXkpO1xuICAgIGlmIChleGlzdGluZ01ldGFkYXRhKSB7XG4gICAgICBjb25zdCBuZXdNZXRhZGF0YSA9IHBpY2tCeShtZXRhZGF0YSwgbmVnYXRlKGlzVW5kZWZpbmVkKSk7XG4gICAgICBjb25zdCBtZXRhZGF0YVRvU2F2ZSA9IG92ZXJyaWRlRXhpc3RpbmdcbiAgICAgICAgPyB7XG4gICAgICAgICAgICAuLi5leGlzdGluZ01ldGFkYXRhLFxuICAgICAgICAgICAgLi4ubmV3TWV0YWRhdGEsXG4gICAgICAgICAgfVxuICAgICAgICA6IHtcbiAgICAgICAgICAgIC4uLm5ld01ldGFkYXRhLFxuICAgICAgICAgICAgLi4uZXhpc3RpbmdNZXRhZGF0YSxcbiAgICAgICAgICB9O1xuXG4gICAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKG1ldGFrZXksIG1ldGFkYXRhVG9TYXZlLCB0YXJnZXQsIHByb3BlcnR5S2V5KTtcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc3QgdHlwZSA9XG4gICAgICAgICh0YXJnZXQ/LmNvbnN0cnVjdG9yIGFzIGFueSk/LltNRVRBREFUQV9GQUNUT1JZX05BTUUgYXMgYW55XT8uKClbXG4gICAgICAgICAgcHJvcGVydHlLZXlcbiAgICAgICAgXT8udHlwZSA/PyBSZWZsZWN0LmdldE1ldGFkYXRhKFwiZGVzaWduOnR5cGVcIiwgdGFyZ2V0LCBwcm9wZXJ0eUtleSk7XG5cbiAgICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEoXG4gICAgICAgIG1ldGFrZXksXG4gICAgICAgIHtcbiAgICAgICAgICB0eXBlLFxuICAgICAgICAgIC4uLnBpY2tCeShtZXRhZGF0YSwgbmVnYXRlKGlzVW5kZWZpbmVkKSksXG4gICAgICAgIH0sXG4gICAgICAgIHRhcmdldCxcbiAgICAgICAgcHJvcGVydHlLZXlcbiAgICAgICk7XG4gICAgfVxuICB9KSBhcyBQcm9wZXJ0eURlY29yYXRvcjtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZU1peGVkRGVjb3JhdG9yPFQgPSBhbnk+KFxuICBtZXRha2V5OiBzdHJpbmcsXG4gIG1ldGFkYXRhOiBUXG4pOiBNZXRob2REZWNvcmF0b3IgJiBDbGFzc0RlY29yYXRvciB7XG4gIHJldHVybiAoXG4gICAgdGFyZ2V0OiBvYmplY3QsXG4gICAga2V5Pzogc3RyaW5nIHwgc3ltYm9sLFxuICAgIGRlc2NyaXB0b3I/OiBUeXBlZFByb3BlcnR5RGVzY3JpcHRvcjxhbnk+XG4gICk6IGFueSA9PiB7XG4gICAgaWYgKGRlc2NyaXB0b3IpIHtcbiAgICAgIGxldCBtZXRhZGF0YXM6IGFueTtcbiAgICAgIGlmIChBcnJheS5pc0FycmF5KG1ldGFkYXRhKSkge1xuICAgICAgICBjb25zdCBwcmV2aW91c01ldGFkYXRhID1cbiAgICAgICAgICBSZWZsZWN0LmdldE1ldGFkYXRhKG1ldGFrZXksIGRlc2NyaXB0b3IudmFsdWUpIHx8IFtdO1xuICAgICAgICBtZXRhZGF0YXMgPSBbLi4ucHJldmlvdXNNZXRhZGF0YSwgLi4ubWV0YWRhdGFdO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgY29uc3QgcHJldmlvdXNNZXRhZGF0YSA9XG4gICAgICAgICAgUmVmbGVjdC5nZXRNZXRhZGF0YShtZXRha2V5LCBkZXNjcmlwdG9yLnZhbHVlKSB8fCB7fTtcbiAgICAgICAgbWV0YWRhdGFzID0geyAuLi5wcmV2aW91c01ldGFkYXRhLCAuLi5tZXRhZGF0YSB9O1xuICAgICAgfVxuICAgICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShtZXRha2V5LCBtZXRhZGF0YXMsIGRlc2NyaXB0b3IudmFsdWUpO1xuICAgICAgcmV0dXJuIGRlc2NyaXB0b3I7XG4gICAgfVxuXG4gICAgbGV0IG1ldGFkYXRhczogYW55O1xuICAgIGlmIChBcnJheS5pc0FycmF5KG1ldGFkYXRhKSkge1xuICAgICAgY29uc3QgcHJldmlvdXNNZXRhZGF0YSA9IFJlZmxlY3QuZ2V0TWV0YWRhdGEobWV0YWtleSwgdGFyZ2V0KSB8fCBbXTtcbiAgICAgIG1ldGFkYXRhcyA9IFsuLi5wcmV2aW91c01ldGFkYXRhLCAuLi5tZXRhZGF0YV07XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IHByZXZpb3VzTWV0YWRhdGEgPSBSZWZsZWN0LmdldE1ldGFkYXRhKG1ldGFrZXksIHRhcmdldCkgfHwge307XG4gICAgICBtZXRhZGF0YXMgPSBPYmplY3QuYXNzaWduKE9iamVjdC5hc3NpZ24oe30sIHByZXZpb3VzTWV0YWRhdGEpLCBtZXRhZGF0YSk7XG4gICAgfVxuICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEobWV0YWtleSwgbWV0YWRhdGFzLCB0YXJnZXQpO1xuICAgIHJldHVybiB0YXJnZXQ7XG4gIH07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVQYXJhbURlY29yYXRvcjxUIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgYW55PiA9IGFueT4oXG4gIG1ldGFkYXRhOiBULFxuICBpbml0aWFsOiBQYXJ0aWFsPFQ+XG4pOiBNZXRob2REZWNvcmF0b3IgJiBDbGFzc0RlY29yYXRvciB7XG4gIHJldHVybiAoXG4gICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnNhZmUtZnVuY3Rpb24tdHlwZVxuICAgIHRhcmdldDogb2JqZWN0IHwgRnVuY3Rpb24sXG4gICAga2V5Pzogc3RyaW5nIHwgc3ltYm9sLFxuICAgIGRlc2NyaXB0b3I/OiBUeXBlZFByb3BlcnR5RGVzY3JpcHRvcjxhbnk+XG4gICk6IGFueSA9PiB7XG4gICAgY29uc3QgcGFyYW1PcHRpb25zID0ge1xuICAgICAgLi4uaW5pdGlhbCxcbiAgICAgIC4uLnBpY2tCeShtZXRhZGF0YSwgbmVnYXRlKGlzVW5kZWZpbmVkKSksXG4gICAgfTtcblxuICAgIGlmIChkZXNjcmlwdG9yKSB7XG4gICAgICBjb25zdCBwYXJhbWV0ZXJzID1cbiAgICAgICAgUmVmbGVjdC5nZXRNZXRhZGF0YShERUNPUkFUT1JTLkFQSV9QQVJBTUVURVJTLCBkZXNjcmlwdG9yLnZhbHVlKSB8fCBbXTtcbiAgICAgIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEoXG4gICAgICAgIERFQ09SQVRPUlMuQVBJX1BBUkFNRVRFUlMsXG4gICAgICAgIFsuLi5wYXJhbWV0ZXJzLCBwYXJhbU9wdGlvbnNdLFxuICAgICAgICBkZXNjcmlwdG9yLnZhbHVlXG4gICAgICApO1xuICAgICAgcmV0dXJuIGRlc2NyaXB0b3I7XG4gICAgfVxuXG4gICAgaWYgKHR5cGVvZiB0YXJnZXQgPT09IFwib2JqZWN0XCIpIHtcbiAgICAgIHJldHVybiB0YXJnZXQ7XG4gICAgfVxuXG4gICAgY29uc3QgcHJvcGVydHlLZXlzID0gT2JqZWN0LmdldE93blByb3BlcnR5TmFtZXModGFyZ2V0LnByb3RvdHlwZSk7XG5cbiAgICBmb3IgKGNvbnN0IHByb3BlcnR5S2V5IG9mIHByb3BlcnR5S2V5cykge1xuICAgICAgaWYgKGlzQ29uc3RydWN0b3IocHJvcGVydHlLZXkpKSB7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBtZXRob2REZXNjcmlwdG9yID0gT2JqZWN0LmdldE93blByb3BlcnR5RGVzY3JpcHRvcihcbiAgICAgICAgdGFyZ2V0LnByb3RvdHlwZSxcbiAgICAgICAgcHJvcGVydHlLZXlcbiAgICAgICk7XG5cbiAgICAgIGlmICghbWV0aG9kRGVzY3JpcHRvcikge1xuICAgICAgICBjb250aW51ZTtcbiAgICAgIH1cblxuICAgICAgY29uc3QgaXNBcGlNZXRob2QgPSBSZWZsZWN0Lmhhc01ldGFkYXRhKFxuICAgICAgICBNRVRIT0RfTUVUQURBVEEsXG4gICAgICAgIG1ldGhvZERlc2NyaXB0b3IudmFsdWVcbiAgICAgICk7XG5cbiAgICAgIGlmICghaXNBcGlNZXRob2QpIHtcbiAgICAgICAgY29udGludWU7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IHBhcmFtZXRlcnMgPVxuICAgICAgICBSZWZsZWN0LmdldE1ldGFkYXRhKFxuICAgICAgICAgIERFQ09SQVRPUlMuQVBJX1BBUkFNRVRFUlMsXG4gICAgICAgICAgbWV0aG9kRGVzY3JpcHRvci52YWx1ZVxuICAgICAgICApIHx8IFtdO1xuICAgICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShcbiAgICAgICAgREVDT1JBVE9SUy5BUElfUEFSQU1FVEVSUyxcbiAgICAgICAgWy4uLnBhcmFtZXRlcnMsIHBhcmFtT3B0aW9uc10sXG4gICAgICAgIG1ldGhvZERlc2NyaXB0b3IudmFsdWVcbiAgICAgICk7XG4gICAgfVxuICB9O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0VHlwZUlzQXJyYXlUdXBsZShcbiAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnNhZmUtZnVuY3Rpb24tdHlwZVxuICBpbnB1dDogRnVuY3Rpb24gfCBbRnVuY3Rpb25dIHwgdW5kZWZpbmVkIHwgc3RyaW5nIHwgUmVjb3JkPHN0cmluZywgYW55PixcbiAgaXNBcnJheUZsYWc6IGJvb2xlYW5cbiAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnNhZmUtZnVuY3Rpb24tdHlwZVxuKTogW0Z1bmN0aW9uIHwgdW5kZWZpbmVkLCBib29sZWFuXSB7XG4gIGlmICghaW5wdXQpIHtcbiAgICByZXR1cm4gW2lucHV0IGFzIHVuZGVmaW5lZCwgaXNBcnJheUZsYWddO1xuICB9XG4gIGlmIChpc0FycmF5RmxhZykge1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW5zYWZlLWZ1bmN0aW9uLXR5cGVcbiAgICByZXR1cm4gW2lucHV0IGFzIEZ1bmN0aW9uLCBpc0FycmF5RmxhZ107XG4gIH1cbiAgY29uc3QgaXNJbnB1dEFycmF5ID0gaXNBcnJheShpbnB1dCk7XG4gIGNvbnN0IHR5cGUgPSBpc0lucHV0QXJyYXkgPyBpbnB1dFswXSA6IGlucHV0O1xuICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVuc2FmZS1mdW5jdGlvbi10eXBlXG4gIHJldHVybiBbdHlwZSBhcyBGdW5jdGlvbiwgaXNJbnB1dEFycmF5XTtcbn1cbiIsImltcG9ydCB7IFR5cGUgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcbmltcG9ydCB7IGlzU3RyaW5nLCBuZWdhdGUsIHBpY2tCeSB9IGZyb20gXCJsb2Rhc2hcIjtcbmltcG9ydCB7IERFQ09SQVRPUlMgfSBmcm9tIFwiLi9jb25zdGFudHNcIjtcbmltcG9ydCB7IGdldFR5cGVJc0FycmF5VHVwbGUsIE1FVEFEQVRBX0ZBQ1RPUllfTkFNRSB9IGZyb20gXCIuL2hlbHBlcnNcIjtcbmltcG9ydCB0eXBlIHtcbiAgRW51bUFsbG93ZWRUeXBlcyxcbiAgRW51bVNjaGVtYUF0dHJpYnV0ZXMsXG4gIFNjaGVtYU9iamVjdE1ldGFkYXRhLFxuICBTd2FnZ2VyRW51bVR5cGUsXG59IGZyb20gXCIuLi9zd2FnZ2VyLXR5cGVzXCI7XG5cbmV4cG9ydCBjb25zdCBpc1VuZGVmaW5lZCA9IChvYmo6IGFueSk6IG9iaiBpcyB1bmRlZmluZWQgPT5cbiAgdHlwZW9mIG9iaiA9PT0gXCJ1bmRlZmluZWRcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIGdldEVudW1WYWx1ZXMoXG4gIGVudW1UeXBlOiBTd2FnZ2VyRW51bVR5cGUgfCAoKCkgPT4gU3dhZ2dlckVudW1UeXBlKVxuKTogc3RyaW5nW10gfCBudW1iZXJbXSB7XG4gIGlmICh0eXBlb2YgZW51bVR5cGUgPT09IFwiZnVuY3Rpb25cIikge1xuICAgIHJldHVybiBnZXRFbnVtVmFsdWVzKGVudW1UeXBlKCkpO1xuICB9XG5cbiAgaWYgKEFycmF5LmlzQXJyYXkoZW51bVR5cGUpKSB7XG4gICAgcmV0dXJuIGVudW1UeXBlIGFzIHN0cmluZ1tdO1xuICB9XG4gIGlmICh0eXBlb2YgZW51bVR5cGUgIT09IFwib2JqZWN0XCIpIHtcbiAgICByZXR1cm4gW107XG4gIH1cbiAgLy8gRW51bXMgd2l0aCBudW1lcmljIHZhbHVlc1xuICAvLyAgIGVudW0gU2l6ZSB7XG4gIC8vICAgICBTTUFMTCA9IDEsXG4gIC8vICAgICBCSUcgPSAyXG4gIC8vICAgfVxuICAvLyBhcmUgdHJhbnNwaWxlZCB0byBpbmNsdWRlIGEgcmV2ZXJzZSBtYXBwaW5nXG4gIC8vICAgY29uc3QgU2l6ZSA9IHtcbiAgLy8gICAgIFwiMVwiOiBcIlNNQUxMXCIsXG4gIC8vICAgICBcIjJcIjogXCJCSUdcIixcbiAgLy8gICAgIFwiU01BTExcIjogMSxcbiAgLy8gICAgIFwiQklHXCI6IDIsXG4gIC8vICAgfVxuICBjb25zdCBudW1lcmljVmFsdWVzID0gT2JqZWN0LnZhbHVlcyhlbnVtVHlwZSlcbiAgICAuZmlsdGVyKCh2YWx1ZSkgPT4gdHlwZW9mIHZhbHVlID09PSBcIm51bWJlclwiKVxuICAgIC5tYXAoKHZhbHVlOiBhbnkpID0+IHZhbHVlLnRvU3RyaW5nKCkpO1xuXG4gIHJldHVybiBPYmplY3Qua2V5cyhlbnVtVHlwZSlcbiAgICAuZmlsdGVyKChrZXkpID0+ICFudW1lcmljVmFsdWVzLmluY2x1ZGVzKGtleSkpXG4gICAgLm1hcCgoa2V5KSA9PiBlbnVtVHlwZVtrZXkgYXMgYW55XSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRFbnVtVHlwZSh2YWx1ZXM6IChzdHJpbmcgfCBudW1iZXIpW10pOiBcInN0cmluZ1wiIHwgXCJudW1iZXJcIiB7XG4gIGNvbnN0IGhhc1N0cmluZyA9IHZhbHVlcy5maWx0ZXIoaXNTdHJpbmcpLmxlbmd0aCA+IDA7XG4gIHJldHVybiBoYXNTdHJpbmcgPyBcInN0cmluZ1wiIDogXCJudW1iZXJcIjtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZVByb3BlcnR5RGVjb3JhdG9yPFQgZXh0ZW5kcyBSZWNvcmQ8c3RyaW5nLCBhbnk+ID0gYW55PihcbiAgbWV0YWtleTogc3RyaW5nLFxuICBtZXRhZGF0YTogVCxcbiAgb3ZlcnJpZGVFeGlzdGluZyA9IHRydWVcbik6IFByb3BlcnR5RGVjb3JhdG9yIHtcbiAgcmV0dXJuICgodGFyZ2V0OiBvYmplY3QsIHByb3BlcnR5S2V5OiBzdHJpbmcpID0+IHtcbiAgICBjb25zdCBwcm9wZXJ0aWVzID1cbiAgICAgIFJlZmxlY3QuZ2V0TWV0YWRhdGEoREVDT1JBVE9SUy5BUElfTU9ERUxfUFJPUEVSVElFU19BUlJBWSwgdGFyZ2V0KSB8fCBbXTtcblxuICAgIGNvbnN0IGtleSA9IGA6JHtwcm9wZXJ0eUtleX1gO1xuICAgIGlmICghcHJvcGVydGllcy5pbmNsdWRlcyhrZXkpKSB7XG4gICAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKFxuICAgICAgICBERUNPUkFUT1JTLkFQSV9NT0RFTF9QUk9QRVJUSUVTX0FSUkFZLFxuICAgICAgICBbLi4ucHJvcGVydGllcywgYDoke3Byb3BlcnR5S2V5fWBdLFxuICAgICAgICB0YXJnZXRcbiAgICAgICk7XG4gICAgfVxuICAgIGNvbnN0IGV4aXN0aW5nTWV0YWRhdGEgPSBSZWZsZWN0LmdldE1ldGFkYXRhKG1ldGFrZXksIHRhcmdldCwgcHJvcGVydHlLZXkpO1xuICAgIGlmIChleGlzdGluZ01ldGFkYXRhKSB7XG4gICAgICBjb25zdCBuZXdNZXRhZGF0YSA9IHBpY2tCeShtZXRhZGF0YSwgbmVnYXRlKGlzVW5kZWZpbmVkKSk7XG4gICAgICBjb25zdCBtZXRhZGF0YVRvU2F2ZSA9IG92ZXJyaWRlRXhpc3RpbmdcbiAgICAgICAgPyB7XG4gICAgICAgICAgICAuLi5leGlzdGluZ01ldGFkYXRhLFxuICAgICAgICAgICAgLi4ubmV3TWV0YWRhdGEsXG4gICAgICAgICAgfVxuICAgICAgICA6IHtcbiAgICAgICAgICAgIC4uLm5ld01ldGFkYXRhLFxuICAgICAgICAgICAgLi4uZXhpc3RpbmdNZXRhZGF0YSxcbiAgICAgICAgICB9O1xuXG4gICAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKG1ldGFrZXksIG1ldGFkYXRhVG9TYXZlLCB0YXJnZXQsIHByb3BlcnR5S2V5KTtcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc3QgdHlwZSA9XG4gICAgICAgIC8vIEB0cy1leHBlY3QtZXJyb3IgbmVzdCBqcyBjb2RlXG4gICAgICAgIHRhcmdldD8uY29uc3RydWN0b3I/LltNRVRBREFUQV9GQUNUT1JZX05BTUVdPy4oKVtwcm9wZXJ0eUtleV0/LnR5cGUgPz9cbiAgICAgICAgUmVmbGVjdC5nZXRNZXRhZGF0YShcImRlc2lnbjp0eXBlXCIsIHRhcmdldCwgcHJvcGVydHlLZXkpO1xuXG4gICAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKFxuICAgICAgICBtZXRha2V5LFxuICAgICAgICB7XG4gICAgICAgICAgdHlwZSxcbiAgICAgICAgICByZXF1aXJlZDogZmFsc2UsIC8vIERlZmF1bHQgdG8gb3B0aW9uYWwgdW5sZXNzIEByZXF1aXJlZCgpIGlzIHVzZWRcbiAgICAgICAgICAuLi5waWNrQnkobWV0YWRhdGEsIG5lZ2F0ZShpc1VuZGVmaW5lZCkpLFxuICAgICAgICB9LFxuICAgICAgICB0YXJnZXQsXG4gICAgICAgIHByb3BlcnR5S2V5XG4gICAgICApO1xuICAgIH1cbiAgfSkgYXMgUHJvcGVydHlEZWNvcmF0b3I7XG59XG5cbmV4cG9ydCB0eXBlIEFwaVByb3BlcnR5Q29tbW9uT3B0aW9ucyA9IFNjaGVtYU9iamVjdE1ldGFkYXRhICYge1xuICBcIngtZW51bU5hbWVzXCI/OiBzdHJpbmdbXTtcbiAgLyoqXG4gICAqIExhenkgZnVuY3Rpb24gcmV0dXJuaW5nIHRoZSB0eXBlIGZvciB3aGljaCB0aGUgZGVjb3JhdGVkIHByb3BlcnR5XG4gICAqIGNhbiBiZSB1c2VkIGFzIGFuIGlkXG4gICAqXG4gICAqIFVzZSB0b2dldGhlciB3aXRoIEBBcGlEZWZhdWx0R2V0dGVyIG9uIHRoZSBnZXR0ZXIgcm91dGUgb2YgdGhlIHR5cGVcbiAgICogdG8gZ2VuZXJhdGUgT3BlbkFQSSBsaW5rIG9iamVjdHNcbiAgICpcbiAgICogQHNlZSBbU3dhZ2dlciBsaW5rIG9iamVjdHNdKGh0dHBzOi8vc3dhZ2dlci5pby9kb2NzL3NwZWNpZmljYXRpb24vbGlua3MvKVxuICAgKi9cbiAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnNhZmUtZnVuY3Rpb24tdHlwZVxuICBsaW5rPzogKCkgPT4gVHlwZTx1bmtub3duPiB8IEZ1bmN0aW9uO1xufTtcblxuZXhwb3J0IHR5cGUgQXBpUHJvcGVydHlPcHRpb25zID1cbiAgfCBBcGlQcm9wZXJ0eUNvbW1vbk9wdGlvbnNcbiAgfCAoQXBpUHJvcGVydHlDb21tb25PcHRpb25zICYge1xuICAgICAgZW51bU5hbWU6IHN0cmluZztcbiAgICAgIGVudW1TY2hlbWE/OiBFbnVtU2NoZW1hQXR0cmlidXRlcztcbiAgICB9KTtcblxuY29uc3QgaXNFbnVtQXJyYXkgPSAoXG4gIG9wdHM6IEFwaVByb3BlcnR5T3B0aW9uc1xuKTogb3B0cyBpcyB7XG4gIGlzQXJyYXk6IHRydWU7XG4gIGVudW06IEVudW1BbGxvd2VkVHlwZXM7XG4gIHR5cGU6IGFueTtcbiAgaXRlbXM6IGFueTtcbn0gPT4gKG9wdHMuaXNBcnJheSAmJiBcImVudW1cIiBpbiBvcHRzICYmIG9wdHMuZW51bSAhPT0gdW5kZWZpbmVkKSBhcyBib29sZWFuO1xuXG4vKipcbiAqIEBwdWJsaWNBcGlcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIEFwaVByb3BlcnR5KFxuICBvcHRpb25zOiBBcGlQcm9wZXJ0eU9wdGlvbnMgPSB7fVxuKTogUHJvcGVydHlEZWNvcmF0b3Ige1xuICByZXR1cm4gY3JlYXRlQXBpUHJvcGVydHlEZWNvcmF0b3Iob3B0aW9ucyk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVBcGlQcm9wZXJ0eURlY29yYXRvcihcbiAgb3B0aW9uczogQXBpUHJvcGVydHlPcHRpb25zID0ge30sXG4gIG92ZXJyaWRlRXhpc3RpbmcgPSB0cnVlXG4pOiBQcm9wZXJ0eURlY29yYXRvciB7XG4gIGNvbnN0IFt0eXBlLCBpc0FycmF5XSA9IGdldFR5cGVJc0FycmF5VHVwbGUoXG4gICAgb3B0aW9ucy50eXBlLFxuICAgIG9wdGlvbnMuaXNBcnJheSBhcyBhbnlcbiAgKTtcbiAgb3B0aW9ucyA9IHtcbiAgICAuLi5vcHRpb25zLFxuICAgIHR5cGUsXG4gICAgaXNBcnJheSxcbiAgfSBhcyBBcGlQcm9wZXJ0eU9wdGlvbnM7XG5cbiAgaWYgKGlzRW51bUFycmF5KG9wdGlvbnMpKSB7XG4gICAgb3B0aW9ucy50eXBlID0gXCJhcnJheVwiO1xuXG4gICAgY29uc3QgZW51bVZhbHVlcyA9IGdldEVudW1WYWx1ZXMob3B0aW9ucy5lbnVtKTtcbiAgICBvcHRpb25zLml0ZW1zID0ge1xuICAgICAgdHlwZTogZ2V0RW51bVR5cGUoZW51bVZhbHVlcyksXG4gICAgICBlbnVtOiBlbnVtVmFsdWVzLFxuICAgIH07XG4gICAgLy8gQHRzLWV4cGVjdC1lcnJvciBuZXN0IGpzIGNvZGVcbiAgICBkZWxldGUgb3B0aW9ucy5lbnVtO1xuICB9IGVsc2UgaWYgKFwiZW51bVwiIGluIG9wdGlvbnMgJiYgb3B0aW9ucy5lbnVtICE9PSB1bmRlZmluZWQpIHtcbiAgICBjb25zdCBlbnVtVmFsdWVzID0gZ2V0RW51bVZhbHVlcyhvcHRpb25zLmVudW0pO1xuXG4gICAgb3B0aW9ucy5lbnVtID0gZW51bVZhbHVlcztcbiAgICBvcHRpb25zLnR5cGUgPSBnZXRFbnVtVHlwZShlbnVtVmFsdWVzKTtcbiAgfVxuXG4gIGlmIChBcnJheS5pc0FycmF5KG9wdGlvbnMudHlwZSkpIHtcbiAgICBvcHRpb25zLnR5cGUgPSBcImFycmF5XCI7XG4gICAgb3B0aW9ucy5pdGVtcyA9IHtcbiAgICAgIHR5cGU6IFwiYXJyYXlcIixcbiAgICAgIGl0ZW1zOiB7XG4gICAgICAgIHR5cGU6IG9wdGlvbnMudHlwZVswXSxcbiAgICAgIH0sXG4gICAgfTtcbiAgfVxuXG4gIHJldHVybiBjcmVhdGVQcm9wZXJ0eURlY29yYXRvcihcbiAgICBERUNPUkFUT1JTLkFQSV9NT0RFTF9QUk9QRVJUSUVTLFxuICAgIG9wdGlvbnMsXG4gICAgb3ZlcnJpZGVFeGlzdGluZ1xuICApO1xufVxuIiwiaW1wb3J0IHsgYXBwbHksIE1ldGFkYXRhIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0aW9uXCI7XG5pbXBvcnQgeyBBcGlFeGNsdWRlRW5kcG9pbnQgfSBmcm9tIFwiQG5lc3Rqcy9zd2FnZ2VyXCI7XG5pbXBvcnQge1xuICBCdWxrQ3J1ZE9wZXJhdGlvbktleXMsXG4gIE9wZXJhdGlvbktleXMsXG4gIENydWRPcGVyYXRpb25zLFxufSBmcm9tIFwiQGRlY2FmLXRzL2RiLWRlY29yYXRvcnNcIjtcbmltcG9ydCB7IE1vZGVsQ29uc3RydWN0b3IgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5pbXBvcnQgeyBEZWxldGUsIEdldCwgUGF0Y2gsIFBvc3QsIFB1dCB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuXG5pbXBvcnQgeyBpc09wZXJhdGlvbkJsb2NrZWQgYXMgY29yZUlzT3BlcmF0aW9uQmxvY2tlZCB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzIH0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBIdHRwVmVyYnMgfSBmcm9tIFwiLi90eXBlc1wiO1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBEZXRlcm1pbmVzIGlmIGEgZ2l2ZW4gQ1JVRCBvcGVyYXRpb24gaXMgYmxvY2tlZCBmb3IgYSBzcGVjaWZpYyBtb2RlbCBjb25zdHJ1Y3Rvci5cbiAqIEBzdW1tYXJ5IFJldHJpZXZlcyB0aGUgb3BlcmF0aW9uLWJsb2NraW5nIGhhbmRsZXIgbWV0YWRhdGEgc3RvcmVkIG9uIHRoZSBwcm92aWRlZCBtb2RlbCBjb25zdHJ1Y3RvciAodW5kZXIgYE9wZXJhdGlvbktleXMuUkVGTEVDVCArIE9wZXJhdGlvbktleXMuQkxPQ0tgKSwgZXhlY3V0ZXMgaXQgKGlmIHByZXNlbnQpIHdpdGggaXRzIHBlcnNpc3RlZCBhcmd1bWVudHMgcGx1cyB0aGUgcmVxdWVzdGVkIG9wZXJhdGlvbiwgYW5kIHJldHVybnMgd2hldGhlciB0aGUgb3BlcmF0aW9uIGlzIGJsb2NrZWQuIElmIG5vIGhhbmRsZXIgZXhpc3RzLCB0aGUgb3BlcmF0aW9uIGlzIGNvbnNpZGVyZWQgYWxsb3dlZCAocmV0dXJucyBgZmFsc2VgKS5cbiAqIEBwYXJhbSB7TW9kZWxDb25zdHJ1Y3Rvcjxhbnk+fSBNb2RlbENvbnN0cnVjdG9yIC0gVGhlIHRhcmdldCBtb2RlbCBjb25zdHJ1Y3RvciB3aG9zZSBtZXRhZGF0YSBtYXkgaW5jbHVkZSBhIGJsb2NraW5nIGhhbmRsZXIuXG4gKiBAcGFyYW0ge0NydWRPcGVyYXRpb25zfSBvcCAtIFRoZSBDUlVEIG9wZXJhdGlvbiB0byBldmFsdWF0ZSAoZS5nLiwgYE9wZXJhdGlvbktleXMuQ1JFQVRFYCwgYE9wZXJhdGlvbktleXMuUkVBRGAsIGBPcGVyYXRpb25LZXlzLlVQREFURWAsIGBPcGVyYXRpb25LZXlzLkRFTEVURWApLlxuICogQHJldHVybiB7Ym9vbGVhbn0gYHRydWVgIHdoZW4gdGhlIG9wZXJhdGlvbiBpcyBleHBsaWNpdGx5IGJsb2NrZWQgYnkgdGhlIG1vZGVsJ3MgaGFuZGxlcjsgb3RoZXJ3aXNlIGBmYWxzZWAuXG4gKiBAZnVuY3Rpb24gaXNPcGVyYXRpb25CbG9ja2VkXG4gKi9cbi8qKlxuICogQGRlc2NyaXB0aW9uIENvbmRpdGlvbmFsbHkgYXBwbGllcyBhbiBIVFRQIG1ldGhvZCBkZWNvcmF0b3IgZm9yIGEgZ2l2ZW4gbW9kZWwgYW5kIHZlcmIsIGhpZGluZyB0aGUgZW5kcG9pbnQgaW4gU3dhZ2dlciAoYW5kIG5vdCByZWdpc3RlcmluZyB0aGUgcm91dGUpIHdoZW4gdGhlIG1vZGVsIGJsb2NrcyB0aGF0IENSVUQgb3BlcmF0aW9uLlxuICogQHN1bW1hcnkgTWFwcyBhbiBIVFRQIHZlcmIgdG8gaXRzIGNvcnJlc3BvbmRpbmcgYENydWRPcGVyYXRpb25zYCBrZXkgYW5kIE5lc3QgSFRUUCBkZWNvcmF0b3IgKGBAR2V0YCwgYEBQb3N0YCwgZXRjLikuIEl0IGNoZWNrcyBgaXNPcGVyYXRpb25CbG9ja2VkKE1vZGVsQ29uc3RydWN0b3IsIGNydWRPcClgIGFuZCwgaWYgYmxvY2tlZCwgYXBwbGllcyBvbmx5IGBAQXBpRXhjbHVkZUVuZHBvaW50KClgIChTd2FnZ2VyLWhpZGRlbiwgbm8gTmVzdCByb3V0ZSkuIElmIHBlcm1pdHRlZCwgaXQgYXBwbGllcyB0aGUgYXBwcm9wcmlhdGUgSFRUUCBkZWNvcmF0b3Igd2l0aCB0aGUgb3B0aW9uYWwgYHBhdGhgLlxuICogQHBhcmFtIHtNb2RlbENvbnN0cnVjdG9yPGFueT59IE1vZGVsQ29uc3RydWN0b3IgLSBUaGUgbW9kZWwgY29uc3RydWN0b3IgdXNlZCB0byByZXNvbHZlIG9wZXJhdGlvbi1ibG9ja2luZyBydWxlcy5cbiAqIEBwYXJhbSB7SHR0cFZlcmJzfSB2ZXJiIC0gVGhlIEhUVFAgdmVyYiB0byBtYXAgKGUuZy4sIGBcIkdFVFwiIHwgXCJQT1NUXCIgfCBcIlBVVFwiIHwgXCJQQVRDSFwiIHwgXCJERUxFVEVcImApLlxuICogQHBhcmFtIHtzdHJpbmd9IFtwYXRoXSAtIE9wdGlvbmFsIHJvdXRlIHBhdGggcGFzc2VkIHRocm91Z2ggdG8gdGhlIGNvcnJlc3BvbmRpbmcgTmVzdCBIVFRQIG1ldGhvZCBkZWNvcmF0b3IuXG4gKiBAcmV0dXJuIHtNZXRob2REZWNvcmF0b3J9IEEgbWV0aG9kIGRlY29yYXRvciB0aGF0IGVpdGhlciBleGNsdWRlcyB0aGUgZW5kcG9pbnQgZnJvbSBTd2FnZ2VyIChhbmQgcm91dGUgcmVnaXN0cmF0aW9uKSBvciBhcHBsaWVzIHRoZSBjb3JyZWN0IEhUVFAgZGVjb3JhdG9yLlxuICovXG5leHBvcnQgZnVuY3Rpb24gQXBpT3BlcmF0aW9uRnJvbU1vZGVsKFxuICBNb2RlbENvbnN0cnVjdG9yOiBNb2RlbENvbnN0cnVjdG9yPGFueT4sXG4gIHZlcmI6IEh0dHBWZXJicyxcbiAgcGF0aD86IHN0cmluZ1xuKTogTWV0aG9kRGVjb3JhdG9yIHtcbiAgY29uc3QgaHR0cFRvQ3J1ZDogUmVjb3JkPFxuICAgIEh0dHBWZXJicyxcbiAgICBbQ3J1ZE9wZXJhdGlvbnMsIChwYXRoPzogc3RyaW5nKSA9PiBNZXRob2REZWNvcmF0b3JdXG4gID4gPSB7XG4gICAgR0VUOiBbT3BlcmF0aW9uS2V5cy5SRUFELCBHZXRdLFxuICAgIFBPU1Q6IFtPcGVyYXRpb25LZXlzLkNSRUFURSwgUG9zdF0sXG4gICAgUFVUOiBbT3BlcmF0aW9uS2V5cy5VUERBVEUsIFB1dF0sXG4gICAgUEFUQ0g6IFtPcGVyYXRpb25LZXlzLlVQREFURSwgUGF0Y2hdLFxuICAgIERFTEVURTogW09wZXJhdGlvbktleXMuREVMRVRFLCBEZWxldGVdLFxuICB9O1xuXG4gIGNvbnN0IFtjcnVkT3AsIEh0dHBNZXRob2REZWNvcmF0b3JdID0gaHR0cFRvQ3J1ZFt2ZXJiXTtcbiAgY29uc3QgdGFyZ2V0ID0gcmVzb2x2ZUJsb2NrVGFyZ2V0KHZlcmIsIHBhdGgpO1xuICByZXR1cm4gdGFyZ2V0XG4gICAgPyBjb3JlSXNPcGVyYXRpb25CbG9ja2VkKE1vZGVsQ29uc3RydWN0b3IsIHRhcmdldC5raW5kIGFzIGFueSwgdGFyZ2V0LnZhbHVlKVxuICAgICAgPyBhcHBseShBcGlFeGNsdWRlRW5kcG9pbnQoKSlcbiAgICAgIDogYXBwbHkoSHR0cE1ldGhvZERlY29yYXRvcihwYXRoKSlcbiAgICA6IGNvcmVJc09wZXJhdGlvbkJsb2NrZWQoTW9kZWxDb25zdHJ1Y3RvciwgY3J1ZE9wKVxuICAgID8gYXBwbHkoQXBpRXhjbHVkZUVuZHBvaW50KCkpXG4gICAgOiBhcHBseShIdHRwTWV0aG9kRGVjb3JhdG9yKHBhdGgpKTtcbn1cbi8qKlxuICogQGRlc2NyaXB0aW9uIENvbmRpdGlvbmFsbHkgYXBwbGllcyBhbiBIVFRQIG1ldGhvZCBkZWNvcmF0b3IgZm9yIGEgZ2l2ZW4gbW9kZWwgYW5kIHZlcmIsIGhpZGluZyB0aGUgZW5kcG9pbnQgaW4gU3dhZ2dlciAoYW5kIG5vdCByZWdpc3RlcmluZyB0aGUgcm91dGUpIHdoZW4gdGhlIG1vZGVsIGJsb2NrcyB0aGF0IENSVUQgb3BlcmF0aW9uLlxuICogQHN1bW1hcnkgTWFwcyBhbiBIVFRQIHZlcmIgdG8gaXRzIGNvcnJlc3BvbmRpbmcgYENydWRPcGVyYXRpb25zYCBrZXkgYW5kIE5lc3QgSFRUUCBkZWNvcmF0b3IgKGBAR2V0YCwgYEBQb3N0YCwgZXRjLikuIEl0IGNoZWNrcyBgaXNPcGVyYXRpb25CbG9ja2VkKE1vZGVsQ29uc3RydWN0b3IsIGNydWRPcClgIGFuZCwgaWYgYmxvY2tlZCwgYXBwbGllcyBvbmx5IGBAQXBpRXhjbHVkZUVuZHBvaW50KClgIChTd2FnZ2VyLWhpZGRlbiwgbm8gTmVzdCByb3V0ZSkuIElmIHBlcm1pdHRlZCwgaXQgYXBwbGllcyB0aGUgYXBwcm9wcmlhdGUgSFRUUCBkZWNvcmF0b3Igd2l0aCB0aGUgb3B0aW9uYWwgYHBhdGhgLlxuICogQHBhcmFtIHtNb2RlbENvbnN0cnVjdG9yPGFueT59IE1vZGVsQ29uc3RydWN0b3IgLSBUaGUgbW9kZWwgY29uc3RydWN0b3IgdXNlZCB0byByZXNvbHZlIG9wZXJhdGlvbi1ibG9ja2luZyBydWxlcy5cbiAqIEBwYXJhbSB7SHR0cFZlcmJzfSB2ZXJiIC0gVGhlIEhUVFAgdmVyYiB0byBtYXAgKGUuZy4sIGBcIkdFVFwiIHwgXCJQT1NUXCIgfCBcIlBVVFwiIHwgXCJQQVRDSFwiIHwgXCJERUxFVEVcImApLlxuICogQHBhcmFtIHtzdHJpbmd9IFtwYXRoXSAtIE9wdGlvbmFsIHJvdXRlIHBhdGggcGFzc2VkIHRocm91Z2ggdG8gdGhlIGNvcnJlc3BvbmRpbmcgTmVzdCBIVFRQIG1ldGhvZCBkZWNvcmF0b3IuXG4gKiBAcmV0dXJuIHtNZXRob2REZWNvcmF0b3J9IEEgbWV0aG9kIGRlY29yYXRvciB0aGF0IGVpdGhlciBleGNsdWRlcyB0aGUgZW5kcG9pbnQgZnJvbSBTd2FnZ2VyIChhbmQgcm91dGUgcmVnaXN0cmF0aW9uKSBvciBhcHBsaWVzIHRoZSBjb3JyZWN0IEhUVFAgZGVjb3JhdG9yLlxuICovXG5leHBvcnQgZnVuY3Rpb24gQnVsa0FwaU9wZXJhdGlvbkZyb21Nb2RlbChcbiAgTW9kZWxDb25zdHJ1Y3RvcjogTW9kZWxDb25zdHJ1Y3Rvcjxhbnk+LFxuICB2ZXJiOiBIdHRwVmVyYnMsXG4gIHBhdGg/OiBzdHJpbmdcbik6IE1ldGhvZERlY29yYXRvciB7XG4gIGNvbnN0IGh0dHBUb0NydWQ6IFJlY29yZDxcbiAgICBIdHRwVmVyYnMsXG4gICAgW0J1bGtDcnVkT3BlcmF0aW9uS2V5cywgKHBhdGg/OiBzdHJpbmcpID0+IE1ldGhvZERlY29yYXRvcl1cbiAgPiA9IHtcbiAgICBHRVQ6IFtCdWxrQ3J1ZE9wZXJhdGlvbktleXMuUkVBRF9BTEwsIEdldF0sXG4gICAgUE9TVDogW0J1bGtDcnVkT3BlcmF0aW9uS2V5cy5DUkVBVEVfQUxMLCBQb3N0XSxcbiAgICBQVVQ6IFtCdWxrQ3J1ZE9wZXJhdGlvbktleXMuVVBEQVRFX0FMTCwgUHV0XSxcbiAgICBQQVRDSDogW0J1bGtDcnVkT3BlcmF0aW9uS2V5cy5VUERBVEVfQUxMLCBQYXRjaF0sXG4gICAgREVMRVRFOiBbQnVsa0NydWRPcGVyYXRpb25LZXlzLkRFTEVURV9BTEwsIERlbGV0ZV0sXG4gIH07XG5cbiAgY29uc3QgW2NydWRPcCwgSHR0cE1ldGhvZERlY29yYXRvcl0gPSBodHRwVG9DcnVkW3ZlcmJdO1xuICBjb25zdCB0YXJnZXQgPSBwYXRoID8gcmVzb2x2ZUJsb2NrVGFyZ2V0KHZlcmIsIHBhdGgpIDogdW5kZWZpbmVkO1xuICByZXR1cm4gdGFyZ2V0XG4gICAgPyBjb3JlSXNPcGVyYXRpb25CbG9ja2VkKE1vZGVsQ29uc3RydWN0b3IsIHRhcmdldC5raW5kIGFzIGFueSwgdGFyZ2V0LnZhbHVlKVxuICAgICAgPyBhcHBseShBcGlFeGNsdWRlRW5kcG9pbnQoKSlcbiAgICAgIDogYXBwbHkoSHR0cE1ldGhvZERlY29yYXRvcihwYXRoKSlcbiAgICA6IGNvcmVJc09wZXJhdGlvbkJsb2NrZWQoTW9kZWxDb25zdHJ1Y3RvciwgXCJidWxrXCIgYXMgYW55LCBjcnVkT3AgYXMgYW55KVxuICAgID8gYXBwbHkoQXBpRXhjbHVkZUVuZHBvaW50KCkpXG4gICAgOiBhcHBseShIdHRwTWV0aG9kRGVjb3JhdG9yKHBhdGgpKTtcbn1cbmZ1bmN0aW9uIHJlc29sdmVCbG9ja1RhcmdldChcbiAgdmVyYjogSHR0cFZlcmJzLFxuICBwYXRoPzogc3RyaW5nXG4pOiB7IGtpbmQ6IFwiY3J1ZFwiIHwgXCJzdGF0ZW1lbnRcIiB8IFwicXVlcnlcIiB8IFwiYnVsa1wiOyB2YWx1ZTogc3RyaW5nIH0gfCB1bmRlZmluZWQge1xuICBpZiAoIXBhdGgpIHJldHVybiB1bmRlZmluZWQ7XG5cbiAgY29uc3Qgbm9ybWFsaXplZCA9IHBhdGgucmVwbGFjZSgvXlxcLyt8XFwvKyQvZywgXCJcIik7XG4gIGNvbnN0IHN0YXRlbWVudFRhcmdldHM6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4gPSB7XG4gICAgXCJsaXN0QnkvOmtleVwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuTElTVF9CWSxcbiAgICBcInBhZ2luYXRlQnkvOmtleS86cGFnZVwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuUEFHRV9CWSxcbiAgICBcImZpbmQvOnZhbHVlXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5GSU5ELFxuICAgIFwicGFnZS86dmFsdWVcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLlBBR0UsXG4gICAgXCJmaW5kT25lQnkvOmtleS86dmFsdWVcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkZJTkRfT05FX0JZLFxuICAgIFwiZmluZEJ5LzprZXkvOnZhbHVlXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5GSU5EX0JZLFxuICAgIFwic3RhdGVtZW50LzptZXRob2QvKmFyZ3NcIjogXCJzdGF0ZW1lbnRcIixcbiAgICBcImNvdW50T2YvOmZpZWxkXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5DT1VOVF9PRixcbiAgICBcIm1heE9mLzpmaWVsZFwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuTUFYX09GLFxuICAgIFwibWluT2YvOmZpZWxkXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5NSU5fT0YsXG4gICAgXCJhdmdPZi86ZmllbGRcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkFWR19PRixcbiAgICBcInN1bU9mLzpmaWVsZFwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuU1VNX09GLFxuICAgIFwiZGlzdGluY3RPZi86ZmllbGRcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkRJU1RJTkNUX09GLFxuICAgIFwiZ3JvdXBPZi86ZmllbGRcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkdST1VQX09GLFxuICB9O1xuXG4gIGlmIChub3JtYWxpemVkLnN0YXJ0c1dpdGgoXCJxdWVyeS9cIikpIHtcbiAgICByZXR1cm4geyBraW5kOiBcInF1ZXJ5XCIsIHZhbHVlOiBub3JtYWxpemVkLnJlcGxhY2UoL15xdWVyeVxcLy8sIFwiXCIpIH07XG4gIH1cblxuICBjb25zdCBzdGF0ZW1lbnRWYWx1ZSA9IHN0YXRlbWVudFRhcmdldHNbbm9ybWFsaXplZF07XG4gIGlmIChzdGF0ZW1lbnRWYWx1ZSkge1xuICAgIHJldHVybiB7IGtpbmQ6IFwic3RhdGVtZW50XCIsIHZhbHVlOiBzdGF0ZW1lbnRWYWx1ZSB9O1xuICB9XG5cbiAgaWYgKHZlcmIgPT09IFwiR0VUXCIgJiYgbm9ybWFsaXplZCA9PT0gXCJcIikge1xuICAgIHJldHVybiB7IGtpbmQ6IFwiY3J1ZFwiLCB2YWx1ZTogT3BlcmF0aW9uS2V5cy5SRUFEIH07XG4gIH1cblxuICByZXR1cm4gdW5kZWZpbmVkO1xufVxuIiwiaW1wb3J0IHsgYXBwbHlEZWNvcmF0b3JzIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBBcGlQYXJhbSBhcyBBcGlQYXJhbURlYyB9IGZyb20gXCJAbmVzdGpzL3N3YWdnZXJcIjtcbmltcG9ydCB0eXBlIHsgRGVjYWZBcGlQcm9wZXJ0eSB9IGZyb20gXCIuL3R5cGVzXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEFwcGxpZXMgYSBzZXQgb2YgU3dhZ2dlciBgQEFwaVBhcmFtYCBkZWNvcmF0b3JzIGdlbmVyYXRlZCBmcm9tIGEgdHlwZWQgc3BlY2lmaWNhdGlvbiBhcnJheS5cbiAqIEBzdW1tYXJ5IFRyYW5zZm9ybXMgZWFjaCBlbnRyeSBvZiB0aGUgcHJvdmlkZWQgYEFwaVBhcmFtW11gIGludG8gYSBjb3JyZXNwb25kaW5nIGBAQXBpUGFyYW0oKWAgZGVjb3JhdG9yIChkZWZhdWx0aW5nIGBkZXNjcmlwdGlvbmAsIGByZXF1aXJlZGAsIGFuZCBgdHlwZWAgd2hlbiBvbWl0dGVkKSBhbmQgY29tcG9zZXMgdGhlbSBpbnRvIGEgc2luZ2xlIGRlY29yYXRvciB2aWEgTmVzdCdzIGBhcHBseURlY29yYXRvcnNgLiBVc2VmdWwgZm9yIHN5bmNocm9uaXppbmcgcnVudGltZSByb3V0ZSBwYXJhbXMgd2l0aCBPcGVuQVBJIGRvY3VtZW50YXRpb24gaW4gYSBjb25jaXNlLCBkZWNsYXJhdGl2ZSB3YXkuXG4gKiBAcGFyYW0ge0FwaVBhcmFtW119IFtwcm9wcz1bXV0gQXJyYXkgZGVzY3JpYmluZyBwYXRoIHBhcmFtZXRlcnMgdG8gYmUgZG9jdW1lbnRlZC5cbiAqIEBwYXJhbSB7c3RyaW5nfSBwcm9wc1tdLm5hbWUgVGhlIHBhcmFtZXRlcidzIG5hbWUgYXMgaXQgYXBwZWFycyBpbiB0aGUgcm91dGUgdGVtcGxhdGUgKGUuZy4sIGA6aWRgKS5cbiAqIEBwYXJhbSB7c3RyaW5nfSBbcHJvcHNbXS5kZXNjcmlwdGlvbl0gSHVtYW4tcmVhZGFibGUgZXhwbGFuYXRpb24gb2YgdGhlIHBhcmFtZXRlcjsgZGVmYXVsdHMgdG8gYFwiUGF0aCBwYXJhbWV0ZXI6IDxuYW1lPlwiYC5cbiAqIEBwYXJhbSB7Ym9vbGVhbn0gW3Byb3BzW10ucmVxdWlyZWQ9dHJ1ZV0gV2hldGhlciB0aGUgcGFyYW1ldGVyIGlzIHJlcXVpcmVkOyBkZWZhdWx0cyB0byBgdHJ1ZWAuXG4gKiBAcGFyYW0ge0NvbnN0cnVjdG9yPGFueT59IFtwcm9wc1tdLnR5cGU9U3RyaW5nXSBDb25zdHJ1Y3Rvci90eXBlIHVzZWQgYnkgU3dhZ2dlciB0byBpbmZlciBzY2hlbWEgKGUuZy4sIGBTdHJpbmdgLCBgTnVtYmVyYCwgYFVVSURgLCBjdXN0b20gY2xhc3MpLlxuICogQHJldHVybiB7YW55fSBBIGNvbXBvc2VkIGRlY29yYXRvciBhcHBseWluZyBhbGwgZ2VuZXJhdGVkIGBAQXBpUGFyYW1gIGRlY29yYXRvcnMgdG8gdGhlIHRhcmdldCBtZXRob2Qgb3IgY29udHJvbGxlci5cbiAqIEBmdW5jdGlvbiBBcGlQYXJhbXNGcm9tTW9kZWxcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIEFwaVBhcmFtc0Zyb21Nb2RlbChcbiAgcHJvcHM6IERlY2FmQXBpUHJvcGVydHlbXSA9IFtdXG4pOiBNZXRob2REZWNvcmF0b3IgJiBDbGFzc0RlY29yYXRvciB7XG4gIGNvbnN0IGRlY29yYXRvcnMgPSBwcm9wcy5tYXAoKHApID0+XG4gICAgQXBpUGFyYW1EZWMoe1xuICAgICAgbmFtZTogcC5uYW1lLFxuICAgICAgZGVzY3JpcHRpb246IHAuZGVzY3JpcHRpb24gPz8gYFBhdGggcGFyYW1ldGVyOiAke3AubmFtZX1gLFxuICAgICAgcmVxdWlyZWQ6IHAucmVxdWlyZWQgPz8gdHJ1ZSxcbiAgICAgIC8vIHNjaGVtYTogeyB0eXBlOiAnc3RyaW5nJyB9LFxuICAgICAgdHlwZTogcC50eXBlID8/IFN0cmluZyxcbiAgICB9KVxuICApO1xuICByZXR1cm4gYXBwbHlEZWNvcmF0b3JzKC4uLmRlY29yYXRvcnMpO1xufVxuIiwiaW1wb3J0IHtcbiAgY3JlYXRlUGFyYW1EZWNvcmF0b3IsXG4gIEV4ZWN1dGlvbkNvbnRleHQsXG4gIEludGVybmFsU2VydmVyRXJyb3JFeGNlcHRpb24sXG59IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBDdXN0b20gZGVjb3JhdG9yIHRoYXQgZXh0cmFjdHMgdGhlIHJlcXVlc3QgYm9keSBhbmQgaW5zdGFudGlhdGVzIGl0IHVzaW5nIHRoZSBNb2RlbCBjb25zdHJ1Y3RvclxuICogZm91bmQgb24gdGhlIGNvbnRyb2xsZXIuIEhhbmRsZXMgYm90aCBzaW5nbGUgb2JqZWN0cyBhbmQgYXJyYXlzLlxuICovXG5leHBvcnQgY29uc3QgRGVjYWZCb2R5ID0gY3JlYXRlUGFyYW1EZWNvcmF0b3IoXG4gIChkYXRhOiB1bmtub3duLCBjdHg6IEV4ZWN1dGlvbkNvbnRleHQpID0+IHtcbiAgICBjb25zdCByZXF1ZXN0ID0gY3R4LnN3aXRjaFRvSHR0cCgpLmdldFJlcXVlc3QoKTtcbiAgICBjb25zdCBib2R5ID0gcmVxdWVzdC5ib2R5O1xuXG4gICAgLy8gVGhlIER5bmFtaWNNb2RlbENvbnRyb2xsZXIgd2lsbCBoYXZlICdNb2RlbENvbnN0cicgYXZhaWxhYmxlXG4gICAgY29uc3QgY29udHJvbGxlciA9IGN0eC5nZXRDbGFzcygpO1xuICAgIC8vIFdlIGFjY2VzcyB0aGUgc3RhdGljIHByb3BlcnR5IHdlIHdpbGwgYWRkIHRvIHRoZSBEeW5hbWljTW9kZWxDb250cm9sbGVyXG4gICAgY29uc3QgTW9kZWxDb25zdHIgPSAoY29udHJvbGxlciBhcyBhbnkpLmNsYXNzO1xuXG4gICAgaWYgKCFNb2RlbENvbnN0cikge1xuICAgICAgdGhyb3cgbmV3IEludGVybmFsU2VydmVyRXJyb3JFeGNlcHRpb24oXG4gICAgICAgIGBNb2RlbENvbnN0cnVjdG9yIG5vdCBmb3VuZCBvbiBjb250cm9sbGVyICR7Y29udHJvbGxlci5uYW1lfS4gRW5zdXJlIHRoZSBjb250cm9sbGVyIHdhcyBjcmVhdGVkIHZpYSBGcm9tTW9kZWxDb250cm9sbGVyLmBcbiAgICAgICk7XG4gICAgfVxuXG4gICAgaWYgKCFib2R5KSByZXR1cm4gYm9keTtcblxuICAgIGlmIChBcnJheS5pc0FycmF5KGJvZHkpKSB7XG4gICAgICByZXR1cm4gYm9keS5tYXAoKGl0ZW0pID0+IG5ldyBNb2RlbENvbnN0cihpdGVtKSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIG5ldyBNb2RlbENvbnN0cihib2R5KTtcbiAgfVxuKTtcbiIsImltcG9ydCB7IGNyZWF0ZVBhcmFtRGVjb3JhdG9yLCBFeGVjdXRpb25Db250ZXh0IH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBEZWNhZlBhcmFtUHJvcHMsIERlY2FmQXBpUHJvcGVydHkgfSBmcm9tIFwiLi90eXBlc1wiO1xuLyoqXG4gKiBAZGVzY3JpcHRpb24gQ3JlYXRlcyBhIGN1c3RvbSBOZXN0SlMgcGFyYW1ldGVyIGRlY29yYXRvciB0aGF0IGV4dHJhY3RzIGFuZCByZXR1cm5zIHJvdXRlIHBhcmFtZXRlcnMgb3JkZXJlZCBieSBhIHNwZWNpZmljIGtleSBzZXF1ZW5jZS5cbiAqIEBzdW1tYXJ5IFRoZSBgT3JkZXJlZFBhcmFtc2AgZGVjb3JhdG9yIHJlYWRzIHRoZSBpbmNvbWluZyBIVFRQIHJlcXVlc3QncyBgcGFyYW1zYCBvYmplY3QsIG9wdGlvbmFsbHkgb3JkZXJzIGl0IGFjY29yZGluZyB0byBhIHByb3ZpZGVkIGxpc3Qgb2YgcGFyYW1ldGVyIG5hbWVzLCBhbmQgcmV0dXJucyBhbiBvYmplY3QgY29udGFpbmluZyB0aGUgb3JpZ2luYWwgcGFyYW1zLCB0aGUgb3JkZXJlZCB2YWx1ZXMsIGFuZCB0aGUgYXBwbGllZCBvcmRlciBsaXN0LiBUaGlzIGVuc3VyZXMgZGV0ZXJtaW5pc3RpYyBwYXJhbWV0ZXIgYWNjZXNzIGZvciByb3V0ZXMgd2l0aCBtdWx0aXBsZSBwYXRoIHBhcmFtZXRlcnMuXG4gKiBAdGVtcGxhdGUge3N0cmluZ1tdfSBUT3JkZXIgLSBSZXByZXNlbnRzIHRoZSBleHBlY3RlZCBzZXF1ZW5jZSBvZiBwYXJhbWV0ZXIgbmFtZXMuXG4gKiBAcGFyYW0ge1RPcmRlciB8IHVuZGVmaW5lZH0gb3JkZXIgQW4gb3B0aW9uYWwgYXJyYXkgc3BlY2lmeWluZyB0aGUgcGFyYW1ldGVyIG5hbWVzJyBkZXNpcmVkIG9yZGVyLiBJZiBub3QgcHJvdmlkZWQsIHRoZSBkZWNvcmF0b3Igd2lsbCBwcmVzZXJ2ZSB0aGUgb3JkZXIgb2YgdGhlIGtleXMgYXMgdGhleSBhcHBlYXIgaW4gdGhlIG9yaWdpbmFsIGByZXEucGFyYW1zYCBvYmplY3QuXG4gKiBAcGFyYW0ge0V4ZWN1dGlvbkNvbnRleHR9IGN0eCBUaGUgTmVzdEpTIGV4ZWN1dGlvbiBjb250ZXh0LCB1c2VkIHRvIHJldHJpZXZlIHRoZSBjdXJyZW50IEhUVFAgcmVxdWVzdCBvYmplY3QuXG4gKiBAcmV0dXJuIHtEZWNhZlBhcmFtUHJvcHN9IEFuIG9iamVjdCBjb250YWluaW5nOlxuICogLSBgb3JpZ2luYWxgOiB0aGUgcmF3IHBhcmFtZXRlciBtYXAgZnJvbSBgcmVxLnBhcmFtc2A7XG4gKiAtIGBvcmRlcmVkYDogYW4gYXJyYXkgb2YgcGFyYW1ldGVyIHZhbHVlcyBpbiB0aGUgcmVxdWVzdGVkIG9yZGVyO1xuICogLSBgb3JkZXJgOiB0aGUgZWZmZWN0aXZlIG9yZGVyIGxpc3QgdXNlZC5cbiAqIEBmdW5jdGlvbiBPcmRlcmVkUGFyYW1zXG4gKi9cbmNvbnN0IE9yZGVyZWRQYXJhbXMgPSBjcmVhdGVQYXJhbURlY29yYXRvcihcbiAgKG9yZGVyOiBzdHJpbmdbXSB8IHVuZGVmaW5lZCwgY3R4OiBFeGVjdXRpb25Db250ZXh0KTogRGVjYWZQYXJhbVByb3BzID0+IHtcbiAgICBjb25zdCByZXEgPSBjdHguc3dpdGNoVG9IdHRwKCkuZ2V0UmVxdWVzdCgpO1xuICAgIGNvbnN0IG9yaWdpbmFsID0gKHJlcT8ucGFyYW1zID8/IHt9KSBhcyBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xuICAgIGNvbnN0IG9yZGVyTGlzdCA9IG9yZGVyID8/IE9iamVjdC5rZXlzKG9yaWdpbmFsKTtcbiAgICBjb25zdCBvcmRlcmVkID0gb3JkZXJMaXN0Lm1hcCgoaykgPT4gb3JpZ2luYWxba10pO1xuICAgIHJldHVybiB7IHJhdzogb3JpZ2luYWwsIHZhbHVlc0luT3JkZXI6IG9yZGVyZWQsIGtleXNJbk9yZGVyOiBvcmRlckxpc3QgfTtcbiAgfVxuKTtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gQSBoaWdoZXItbGV2ZWwgZGVjb3JhdG9yIGZhY3RvcnkgdGhhdCBsZXZlcmFnZXMgYE9yZGVyZWRQYXJhbXNgIHRvIGV4dHJhY3Qgcm91dGUgcGFyYW1ldGVycyBpbiBhIHNwZWNpZmljIG9yZGVyIGRlcml2ZWQgZnJvbSBhIGxpc3Qgb2YgYEFwaVBhcmFtYCBkZWZpbml0aW9ucy5cbiAqIEBzdW1tYXJ5IGBEZWNhZlBhcmFtc2AgY29tcHV0ZXMgdGhlIG9yZGVyIG9mIHJvdXRlIHBhcmFtZXRlcnMgYmFzZWQgb24gdGhlIHByb3ZpZGVkIGBBcGlQYXJhbVtdYCBzcGVjaWZpY2F0aW9uICh1c2luZyBlYWNoIGVsZW1lbnTigJlzIGBuYW1lYCksIHRoZW4gYXBwbGllcyBgT3JkZXJlZFBhcmFtcyhvcmRlcilgIGFzIGEgcGFyYW1ldGVyIGRlY29yYXRvci4gVGhpcyBlbmFibGVzIHBhcmFtZXRlci1sZXZlbCBiaW5kaW5nIHRoYXQgcmVtYWlucyBjb25zaXN0ZW50IHdpdGggdGhlIGRvY3VtZW50ZWQgQVBJIHBhcmFtZXRlciBtZXRhZGF0YS5cbiAqIEBwYXJhbSB7QXBpUGFyYW1bXX0gW3Byb3BzPVtdXSBBcnJheSBvZiBgQXBpUGFyYW1gIGRlZmluaXRpb25zIHdob3NlIGBuYW1lYCBmaWVsZHMgZGV0ZXJtaW5lIHRoZSBwYXJhbWV0ZXIgZXh0cmFjdGlvbiBvcmRlci5cbiAqIEByZXR1cm4ge1BhcmFtZXRlckRlY29yYXRvcn0gQSBOZXN0SlMgcGFyYW1ldGVyIGRlY29yYXRvciB0aGF0IGluamVjdHMgYW4gb3JkZXJlZCBsaXN0IG9mIHBhcmFtZXRlcnMgYW5kIG1ldGFkYXRhIGludG8gdGhlIGNvbnRyb2xsZXIgbWV0aG9kIGFyZ3VtZW50LlxuICovXG5leHBvcnQgZnVuY3Rpb24gRGVjYWZQYXJhbXMoXG4gIHByb3BzOiBEZWNhZkFwaVByb3BlcnR5W10gPSBbXVxuKTogUGFyYW1ldGVyRGVjb3JhdG9yIHtcbiAgY29uc3Qgb3JkZXIgPSBwcm9wcy5tYXAoKHApID0+IHAubmFtZSk7XG4gIHJldHVybiBPcmRlcmVkUGFyYW1zKG9yZGVyKTtcbn1cblxuZXhwb3J0IGNvbnN0IERlY2FmUXVlcnkgPSBjcmVhdGVQYXJhbURlY29yYXRvcihcbiAgKF86IHVua25vd24sIGN0eDogRXhlY3V0aW9uQ29udGV4dCkgPT4ge1xuICAgIGNvbnN0IHJlcSA9IGN0eC5zd2l0Y2hUb0h0dHAoKS5nZXRSZXF1ZXN0KCk7XG4gICAgY29uc3QgcXVlcnkgPSByZXEucXVlcnkgPz8ge307XG5cbiAgICBjb25zdCBwYXJzZWQ6IGFueSA9IHsgLi4ucXVlcnkgfTtcblxuICAgIC8vIFBhcnNlIGxpbWl0ICYgb2Zmc2V0XG4gICAgaWYgKHBhcnNlZC5saW1pdCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICBjb25zdCBuID0gTnVtYmVyKHBhcnNlZC5saW1pdCk7XG4gICAgICBpZiAoIU51bWJlci5pc05hTihuKSkgcGFyc2VkLmxpbWl0ID0gbjtcbiAgICB9XG5cbiAgICBpZiAocGFyc2VkLm9mZnNldCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICBjb25zdCBuID0gTnVtYmVyKHBhcnNlZC5vZmZzZXQpO1xuICAgICAgaWYgKCFOdW1iZXIuaXNOYU4obikpIHBhcnNlZC5vZmZzZXQgPSBuO1xuICAgIH1cblxuICAgIC8vIFBhcnNlIGJvb2ttYXJrIG9ubHkgaWYgbnVtZXJpY1xuICAgIGlmIChwYXJzZWQuYm9va21hcmsgIT09IHVuZGVmaW5lZCkge1xuICAgICAgY29uc3QgbiA9IE51bWJlcihwYXJzZWQuYm9va21hcmspO1xuICAgICAgcGFyc2VkLmJvb2ttYXJrID0gTnVtYmVyLmlzTmFOKG4pID8gcGFyc2VkLmJvb2ttYXJrIDogbjtcbiAgICB9XG5cbiAgICByZXR1cm4gcGFyc2VkO1xuICB9XG4pO1xuIiwiaW1wb3J0IHsgQWRhcHRlckZsYWdzLCBDb250ZXh0IH0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBMb2dnZXIgfSBmcm9tIFwiQGRlY2FmLXRzL2xvZ2dpbmdcIjtcblxuZXhwb3J0IGNvbnN0IERFQ0FGX01PRFVMRV9PUFRJT05TID0gXCJEZWNhZk1vZHVsZU9wdGlvbnNcIjtcbmV4cG9ydCBjb25zdCBERUNBRl9BREFQVEVSX0lEID0gXCJEZWNhZkFkYXB0ZXJcIjtcbmV4cG9ydCBjb25zdCBERUNBRl9UQVNLX1NFUlZJQ0VfSUQgPSBcIkRlY2FmVGFza1NlcnZpY2VcIjtcblxuZXhwb3J0IGNvbnN0IERFQ0FGX1JPVVRFID0gXCJEZWNhZlJvdXRlXCI7XG5leHBvcnQgY29uc3QgREVDQUZfSEFORExFUlMgPSBTeW1ib2woXCJEZWNhZkhhbmRsZXJzXCIpO1xuZXhwb3J0IGNvbnN0IERFQ0FGX0FEQVBURVJfT1BUSU9OUyA9IFN5bWJvbChcIkRlY2FmQWRhcHRlckZvck9wdGlvbnNcIik7XG5leHBvcnQgY29uc3QgREVDQUZfRVhQT1NFID0gXCJEZWNhZkV4cG9zZVwiO1xuZXhwb3J0IGNvbnN0IERFQ0FGX0NPTlRST0xMRVJfQ09ORklHID0gXCJEZWNhZkNvbnRyb2xsZXJDb25maWdcIjtcblxuZXhwb3J0IGNvbnN0IERFQ0FGX0NPTlRFWFRfS0VZID0gU3ltYm9sKFwiZGVjYWY6Y29udGV4dFwiKTtcblxuZXhwb3J0IHR5cGUgRGVjYWZTZXJ2ZXJGbGFnczxMT0cgZXh0ZW5kcyBMb2dnZXIgPSBMb2dnZXI+ID1cbiAgQWRhcHRlckZsYWdzPExPRz4gJiB7XG4gICAgaGVhZGVyczogUmVjb3JkPHN0cmluZywgYW55PjtcbiAgICBvdmVycmlkZXM6IFJlY29yZDxzdHJpbmcsIGFueT47XG4gIH07XG5cbmV4cG9ydCB0eXBlIERlY2FmU2VydmVyQ3R4ID0gQ29udGV4dDxEZWNhZlNlcnZlckZsYWdzPjtcbiIsImltcG9ydCB7IE1ldGFkYXRhIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0aW9uXCI7XG5cbmltcG9ydCB7IERFQ0FGX0VYUE9TRSB9IGZyb20gXCIuLi8uLi9jb25zdGFudHNcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIGV4cG9zZSguLi5mbGF2b3Vyczogc3RyaW5nW10pIHtcbiAgcmV0dXJuIGZ1bmN0aW9uIGV4cG9zZSh0YXJnZXQ6IGFueSkge1xuICAgIE1ldGFkYXRhLnNldChcbiAgICAgIHRhcmdldCxcbiAgICAgIERFQ0FGX0VYUE9TRSxcbiAgICAgIChmbGF2b3Vycy5sZW5ndGggPyBmbGF2b3VycyA6IHRydWUpIGFzIGFueVxuICAgICk7XG4gICAgcmV0dXJuIHRhcmdldDtcbiAgfTtcbn1cbiIsImltcG9ydCB7IE1ldGFkYXRhIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0aW9uXCI7XG5pbXBvcnQgdHlwZSB7IE1vZGVsQ29udHJvbGxlckZhY3RvcnlDb25maWcgfSBmcm9tIFwiQGRlY2FmLXRzL2Zvci1odHRwL3NlcnZlclwiO1xuXG5pbXBvcnQgeyBERUNBRl9DT05UUk9MTEVSX0NPTkZJRyB9IGZyb20gXCIuLi8uLi9jb25zdGFudHNcIjtcblxuLyoqXG4gKiBDbGFzcyBkZWNvcmF0b3IgdGhhdCBhdHRhY2hlcyBhIHtAbGluayBNb2RlbENvbnRyb2xsZXJGYWN0b3J5Q29uZmlnfSB0byBhIE1vZGVsLFxuICogc28gYEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlKClgIGNhbiBwYXNzIGl0IHRvIGBNb2RlbENvbnRyb2xsZXJGYWN0b3J5LmNyZWF0ZSgpYC5cbiAqXG4gKiBUaGUgcGVyLW1vZGVsIGNvbmZpZyBjYW4gYmUgb3ZlcnJpZGRlbiBieSB0aGUgbW9kdWxlLWxldmVsIGBjb250cm9sbGVyQ29uZmlnYFxuICogb3B0aW9uIGluIGBEZWNhZk1vZHVsZU9wdGlvbnNgLlxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0c1xuICogQGNvbnRyb2xsZXJDb25maWcoeyBhbGxvd0dyb3VwaW5nUXVlcmllczogdHJ1ZSwgYWxsb3dCdWxrU3RhdGVtZW50OiB7IGNyZWF0ZTogdHJ1ZSwgcmVhZDogdHJ1ZSwgdXBkYXRlOiBmYWxzZSwgZGVsZXRlOiB0cnVlIH0gfSlcbiAqIEBtb2RlbCgpXG4gKiBjbGFzcyBPcmRlciBleHRlbmRzIE1vZGVsPE9yZGVyPiB7IC4uLiB9XG4gKiBgYGBcbiAqXG4gKiBAcGFyYW0gY29uZmlnIC0gRmFjdG9yeSBjb25maWd1cmF0aW9uIGtub2JzIChhbGxvd1N0YXRlbWVudGxlc3NRdWVyeSwgYWxsb3dHcm91cGluZ1F1ZXJpZXMsIGFsbG93QnVsa1N0YXRlbWVudCkuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjb250cm9sbGVyQ29uZmlnKGNvbmZpZzogTW9kZWxDb250cm9sbGVyRmFjdG9yeUNvbmZpZykge1xuICByZXR1cm4gZnVuY3Rpb24gY29udHJvbGxlckNvbmZpZ0RlY29yYXRvcih0YXJnZXQ6IGFueSkge1xuICAgIE1ldGFkYXRhLnNldCh0YXJnZXQsIERFQ0FGX0NPTlRST0xMRVJfQ09ORklHLCBjb25maWcgYXMgYW55KTtcbiAgICByZXR1cm4gdGFyZ2V0O1xuICB9O1xufVxuIiwiZXhwb3J0IGNvbnN0IEFVVEhfSEFORExFUiA9IFN5bWJvbChcIkFVVEhfSEFORExFUlwiKTtcbmV4cG9ydCBjb25zdCBBVVRIX01FVEFfS0VZID0gXCJhdXRoOm1ldGFcIjtcbmV4cG9ydCBjb25zdCBJU19QVUJMSUNfS0VZID0gXCJpc1B1YmxpY1wiO1xuZXhwb3J0IGNvbnN0IFJFUVVJUkVEX1JPTEVTX0tFWSA9IFwicmVxdWlyZWRSb2xlc1wiO1xuIiwiaW1wb3J0IHsgSW5qZWN0LCBJbmplY3RhYmxlLCBTY29wZSB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgUkVRVUVTVCB9IGZyb20gXCJAbmVzdGpzL2NvcmVcIjtcbmltcG9ydCB7IFVVSUQgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IFJlcXVlc3RDb250ZXh0IH0gZnJvbSBcIkBkZWNhZi10cy9mb3ItaHR0cC9zZXJ2ZXJcIjtcbmltcG9ydCB7IHR5cGUgUmVxdWVzdCB9IGZyb20gXCJleHByZXNzXCI7XG5cbmltcG9ydCB7IERlY2FmU2VydmVyQ3R4IH0gZnJvbSBcIi4uL2NvbnN0YW50c1wiO1xuXG5ASW5qZWN0YWJsZSh7IHNjb3BlOiBTY29wZS5SRVFVRVNUIH0pXG5leHBvcnQgY2xhc3MgRGVjYWZSZXF1ZXN0Q29udGV4dDxDIGV4dGVuZHMgRGVjYWZTZXJ2ZXJDdHggPSBEZWNhZlNlcnZlckN0eD4gZXh0ZW5kcyBSZXF1ZXN0Q29udGV4dDxSZXF1ZXN0PiB7XG4gIHV1aWQgPSBVVUlELmluc3RhbmNlLmdlbmVyYXRlKCk7XG4gIG92ZXJyaWRlIHJlYWRvbmx5IHJlcXVlc3Q6IFJlcXVlc3Q7XG5cbiAgY29uc3RydWN0b3IoQEluamVjdChSRVFVRVNUKSBwcml2YXRlIHJlYWRvbmx5IHJlcTogUmVxdWVzdCkge1xuICAgIHN1cGVyKFxuICAgICAge1xuICAgICAgICBoZWFkZXJzT2Y6IChyZXF1ZXN0OiBSZXF1ZXN0KSA9PiAocmVxdWVzdCBhcyBhbnkpPy5oZWFkZXJzIHx8IHVuZGVmaW5lZCxcbiAgICAgIH0gYXMgYW55LFxuICAgICAgcmVxXG4gICAgKTtcbiAgICB0aGlzLnJlcXVlc3QgPSByZXE7XG4gIH1cblxuICBwdXQocmVjb3JkOiBSZWNvcmQ8YW55LCBhbnk+KSB7XG4gICAgbGV0IG92ZXJyaWRlczogYW55O1xuICAgIHRyeSB7XG4gICAgICBvdmVycmlkZXMgPSB0aGlzLmdldChcIm92ZXJyaWRlc1wiKTtcbiAgICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW51c2VkLXZhcnNcbiAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICBvdmVycmlkZXMgPSB7fTtcbiAgICB9XG5cbiAgICB0aGlzLmFjY3VtdWxhdGUoe1xuICAgICAgb3ZlcnJpZGVzOiBPYmplY3QuYXNzaWduKG92ZXJyaWRlcywgcmVjb3JkKSxcbiAgICB9KTtcbiAgfVxufVxuIiwiaW1wb3J0IHtcbiAgSW5qZWN0YWJsZSxcbiAgTmVzdEludGVyY2VwdG9yLFxuICBFeGVjdXRpb25Db250ZXh0LFxuICBDYWxsSGFuZGxlcixcbiAgSW5qZWN0LFxuICBPcHRpb25hbCxcbiAgU2NvcGUsXG59IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgUmVmbGVjdG9yIH0gZnJvbSBcIkBuZXN0anMvY29yZVwiO1xuaW1wb3J0IHsgT2JzZXJ2YWJsZSB9IGZyb20gXCJyeGpzXCI7XG5cbmltcG9ydCB7IENvbnN0cnVjdG9yIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0aW9uXCI7XG5pbXBvcnQgeyBMb2dnaW5nIH0gZnJvbSBcIkBkZWNhZi10cy9sb2dnaW5nXCI7XG5pbXBvcnQgeyBBZGFwdGVyIH0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBSZXF1ZXN0VG9Db250ZXh0VHJhbnNmb3JtZXIgfSBmcm9tIFwiQGRlY2FmLXRzL2Zvci1odHRwL3NlcnZlclwiO1xuXG5pbXBvcnQgeyBBVVRIX0hBTkRMRVIsIEFVVEhfTUVUQV9LRVksIElTX1BVQkxJQ19LRVksIFJFUVVJUkVEX1JPTEVTX0tFWSB9IGZyb20gXCIuL2NvbnN0YW50c1wiO1xuaW1wb3J0IHR5cGUgeyBBdXRoSGFuZGxlciB9IGZyb20gXCIuLi90eXBlc1wiO1xuaW1wb3J0IHsgRGVjYWZSZXF1ZXN0Q29udGV4dCB9IGZyb20gXCIuLi9yZXF1ZXN0L0RlY2FmUmVxdWVzdENvbnRleHRcIjtcblxuQEluamVjdGFibGUoeyBzY29wZTogU2NvcGUuUkVRVUVTVCB9KVxuZXhwb3J0IGNsYXNzIEF1dGhJbnRlcmNlcHRvciBpbXBsZW1lbnRzIE5lc3RJbnRlcmNlcHRvciB7XG4gIGNvbnN0cnVjdG9yKFxuICAgIHByaXZhdGUgcmVhZG9ubHkgcmVmbGVjdG9yOiBSZWZsZWN0b3IsXG4gICAgcHJvdGVjdGVkIHJlYWRvbmx5IHJlcXVlc3RDb250ZXh0OiBEZWNhZlJlcXVlc3RDb250ZXh0LFxuICAgIEBPcHRpb25hbCgpIEBJbmplY3QoQVVUSF9IQU5ETEVSKSBwcml2YXRlIHJlYWRvbmx5IGF1dGhIYW5kbGVyPzogQXV0aEhhbmRsZXJcbiAgKSB7fVxuXG4gIGFzeW5jIGludGVyY2VwdChcbiAgICBjdHg6IEV4ZWN1dGlvbkNvbnRleHQsXG4gICAgbmV4dDogQ2FsbEhhbmRsZXJcbiAgKTogUHJvbWlzZTxPYnNlcnZhYmxlPGFueT4+IHtcbiAgICBjb25zdCBsb2cgPSBMb2dnaW5nLmZvcih0aGlzIGFzIGFueSkuZm9yKHRoaXMuaW50ZXJjZXB0KTtcblxuICAgIGNvbnN0IGlzUHVibGljID0gdGhpcy5yZWZsZWN0b3IuZ2V0QWxsQW5kT3ZlcnJpZGU8Ym9vbGVhbj4oSVNfUFVCTElDX0tFWSwgW1xuICAgICAgY3R4LmdldEhhbmRsZXIoKSxcbiAgICAgIGN0eC5nZXRDbGFzcygpLFxuICAgIF0pO1xuXG4gICAgY29uc3QgbW9kZWxOYW1lID1cbiAgICAgIHRoaXMucmVmbGVjdG9yLmdldDxzdHJpbmcgfCBDb25zdHJ1Y3Rvcj4oXG4gICAgICAgIEFVVEhfTUVUQV9LRVksXG4gICAgICAgIGN0eC5nZXRIYW5kbGVyKClcbiAgICAgICkgPz9cbiAgICAgIHRoaXMucmVmbGVjdG9yLmdldDxzdHJpbmcgfCBDb25zdHJ1Y3Rvcj4oQVVUSF9NRVRBX0tFWSwgY3R4LmdldENsYXNzKCkpO1xuXG4gICAgY29uc3QgcmVxdWlyZWRSb2xlcyA9IHRoaXMucmVmbGVjdG9yLmdldEFsbEFuZE92ZXJyaWRlPHN0cmluZ1tdPihcbiAgICAgIFJFUVVJUkVEX1JPTEVTX0tFWSxcbiAgICAgIFtjdHguZ2V0SGFuZGxlcigpLCBjdHguZ2V0Q2xhc3MoKV1cbiAgICApO1xuXG4gICAgbG9nLnZlcmJvc2UoYEludGVyY2VwdGVkIHJlcXVlc3Qke21vZGVsTmFtZSA/IGAgZm9yICR7bW9kZWxOYW1lfWAgOiBcIlwifWApO1xuXG4gICAgaWYgKGlzUHVibGljKSB7XG4gICAgICBsb2cuZGVidWcoYFB1YmxpYyByb3V0ZSDigJQgc2tpcHBpbmcgYXV0aGApO1xuICAgIH0gZWxzZSBpZiAodGhpcy5hdXRoSGFuZGxlcikge1xuICAgICAgYXdhaXQgdGhpcy5hdXRoSGFuZGxlci5hdXRob3JpemUoXG4gICAgICAgIGN0eCxcbiAgICAgICAgbW9kZWxOYW1lIGFzIHN0cmluZyB8IENvbnN0cnVjdG9yLFxuICAgICAgICByZXF1aXJlZFJvbGVzLFxuICAgICAgICB0aGlzLnJlcXVlc3RDb250ZXh0XG4gICAgICApO1xuICAgIH0gZWxzZSB7XG4gICAgICBsb2cuZGVidWcoYE5vIGF1dGggaGFuZGxlciByZWdpc3RlcmVkYCk7XG4gICAgfVxuXG4gICAgLy8gVHJhbnNmb3JtZXJzIHJ1biBBRlRFUiBhdXRoIHNvIHRoZXkgY2FuIHJlYWQgYXV0aC1wb3B1bGF0ZWQgZmllbGRzXG4gICAgLy8gKGUuZy4gYHVzZXJgKSBmcm9tIHRoZSBjb250ZXh0IGFuZCBtYXAgdGhlbSB0byBhZGFwdGVyLXNwZWNpZmljIGtleXNcbiAgICAvLyAoZS5nLiBgVVVJRGAgZm9yIFJhbUFkYXB0ZXIncyBAY3JlYXRlZEJ5L0B1cGRhdGVkQnkgaGFuZGxlcnMpLlxuICAgIGF3YWl0IHRoaXMuYXBwbHlUcmFuc2Zvcm1lcnMoKTtcblxuICAgIGNvbnN0IHVzZXIgPSB0aGlzLnJlcXVlc3RDb250ZXh0LmdldE9yVW5kZWZpbmVkKFwidXNlclwiIGFzIGFueSk7XG4gICAgY29uc3Qgb3JnYW5pemF0aW9uID0gdGhpcy5yZXF1ZXN0Q29udGV4dC5nZXRPclVuZGVmaW5lZChcIm9yZ2FuaXphdGlvblwiIGFzIGFueSk7XG4gICAgaWYgKHVzZXIgfHwgb3JnYW5pemF0aW9uKSB7XG4gICAgICBjb25zdCBjdXJyZW50TG9nID0gdGhpcy5yZXF1ZXN0Q29udGV4dC5nZXQoXCJsb2dnZXJcIiBhcyBhbnkpO1xuICAgICAgY29uc3QgY2hpbGRMb2cgPSBjdXJyZW50TG9nLmZvcih7IHVzZXIsIG9yZ2FuaXphdGlvbiB9KTtcbiAgICAgIHRoaXMucmVxdWVzdENvbnRleHQuYWNjdW11bGF0ZSh7IGxvZ2dlcjogY2hpbGRMb2cgfSBhcyBhbnkpO1xuICAgIH1cblxuICAgIHJldHVybiBuZXh0LmhhbmRsZSgpO1xuICB9XG5cbiAgcHJvdGVjdGVkIGFzeW5jIGFwcGx5VHJhbnNmb3JtZXJzKCk6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IGZsYXZvdXJzID0gQWRhcHRlci5mbGF2b3Vyc1RvVHJhbnNmb3JtKCk7XG4gICAgaWYgKCFmbGF2b3VycykgcmV0dXJuO1xuXG4gICAgZm9yIChjb25zdCBmbGF2b3VyIG9mIGZsYXZvdXJzKSB7XG4gICAgICBjb25zdCB0cmFuc2Zvcm1lciA9IEFkYXB0ZXIudHJhbnNmb3JtZXJGb3IoXG4gICAgICAgIGZsYXZvdXJcbiAgICAgICkgYXMgUmVxdWVzdFRvQ29udGV4dFRyYW5zZm9ybWVyPGFueT47XG4gICAgICBpZiAoIXRyYW5zZm9ybWVyKSBjb250aW51ZTtcbiAgICAgIGNvbnN0IGZyb20gPSBhd2FpdCB0cmFuc2Zvcm1lci5mcm9tKHRoaXMucmVxdWVzdENvbnRleHQpO1xuICAgICAgaWYgKGZyb20pIHRoaXMucmVxdWVzdENvbnRleHQuYWNjdW11bGF0ZShmcm9tKTtcbiAgICB9XG4gIH1cbn1cbiIsImltcG9ydCB7IEV4ZWN1dGlvbkNvbnRleHQgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcbmltcG9ydCB7IEF1dGhvcml6YXRpb25FcnJvciB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgQXV0aEhhbmRsZXIsIEF1dGhEYXRhIH0gZnJvbSBcIkBkZWNhZi10cy9mb3ItaHR0cC9zZXJ2ZXJcIjtcbmltcG9ydCB7IERlY2FmUmVxdWVzdENvbnRleHQgfSBmcm9tIFwiLi4vcmVxdWVzdC9EZWNhZlJlcXVlc3RDb250ZXh0XCI7XG5cbi8qKlxuICogU2ltcGxlIGF1dGggaGFuZGxlciB0aGF0IHJlYWRzIGEgcm9sZSBzdHJpbmcgZnJvbSB0aGUgYEF1dGhvcml6YXRpb246IEJlYXJlciA8cm9sZT5gIGhlYWRlci5cbiAqXG4gKiBPbmx5IG92ZXJyaWRlcyB7QGxpbmsgQXV0aEhhbmRsZXIuZXh0cmFjdEZyb21BdXRofSB0byByZXR1cm4gdGhlIGJlYXJlciB0b2tlbiBhc1xuICogYm90aCB0aGUgdXNlciBpZGVudGlmaWVyIGFuZCB0aGUgc2luZ2xlIHJvbGUuIFRoZSBiYXNlIGNsYXNzIGBiaW5kVG9Db250ZXh0YFxuICogKGBjdHguYWNjdW11bGF0ZShkYXRhKWApIGlzIHN1ZmZpY2llbnQg4oCUIHRoZSB0cmFuc2Zvcm1lci9hZGFwdGVyIGlzIHJlc3BvbnNpYmxlXG4gKiBmb3IgbWFwcGluZyBjb250ZXh0IGZpZWxkcyB0byBhZGFwdGVyLXNwZWNpZmljIGtleXMgbGlrZSBgVVVJRGAuXG4gKi9cbmV4cG9ydCBjbGFzcyBEZWNhZkF1dGhIYW5kbGVyIGV4dGVuZHMgQXV0aEhhbmRsZXI8XG4gIEV4ZWN1dGlvbkNvbnRleHQsXG4gIERlY2FmUmVxdWVzdENvbnRleHQsXG4gIEF1dGhEYXRhXG4+IHtcbiAgcHJvdGVjdGVkIHBhcnNlUmVxdWVzdChyZXE6IGFueSk6IHN0cmluZyB7XG4gICAgY29uc3QgdXNlclJvbGUgPSByZXEuaGVhZGVycy5hdXRob3JpemF0aW9uPy5zcGxpdChcIiBcIilbMV0gYXMgc3RyaW5nO1xuICAgIHJldHVybiB1c2VyUm9sZTtcbiAgfVxuXG4gIHByb3RlY3RlZCBleHRyYWN0RnJvbUF1dGgoY3R4OiBFeGVjdXRpb25Db250ZXh0KTogQXV0aERhdGEge1xuICAgIGNvbnN0IHJlcSA9IGN0eC5zd2l0Y2hUb0h0dHAoKS5nZXRSZXF1ZXN0KCk7XG4gICAgY29uc3QgdXNlclJvbGUgPSB0aGlzLnBhcnNlUmVxdWVzdChyZXEpO1xuICAgIGlmICghdXNlclJvbGUpIHRocm93IG5ldyBBdXRob3JpemF0aW9uRXJyb3IoXCJVbmF1dGhlbnRpY2F0ZWRcIik7XG4gICAgcmV0dXJuIHsgdXNlcjogdXNlclJvbGUsIHJvbGVzOiBbdXNlclJvbGVdIH07XG4gIH1cbn1cblxuLyoqXG4gKiBBbGlhcyBmb3Ige0BsaW5rIERlY2FmQXV0aEhhbmRsZXJ9IGtlcHQgZm9yIGJhY2t3YXJkIGNvbXBhdGliaWxpdHkuXG4gKiBUaGUgXCJyb2xlXCIgdmFyaWFudCBwcmV2aW91c2x5IHJldHVybmVkIHJvbGVzIGZyb20gYGF1dGhvcml6ZWA7IHdpdGggdGhlIGJhc2VcbiAqIGNsYXNzIG5vdyBvcmNoZXN0cmF0aW5nIHJvbGUgY2hlY2tzIGludGVybmFsbHksIHRoZSB0d28gY2xhc3NlcyBhcmUgZXF1aXZhbGVudC5cbiAqL1xuZXhwb3J0IGNsYXNzIERlY2FmUm9sZUF1dGhIYW5kbGVyIGV4dGVuZHMgRGVjYWZBdXRoSGFuZGxlciB7fVxuIiwiaW1wb3J0IHsgYXBwbHlEZWNvcmF0b3JzLCBTZXRNZXRhZGF0YSwgVXNlSW50ZXJjZXB0b3JzIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBBcGlCZWFyZXJBdXRoLCBBcGlTZWN1cml0eSB9IGZyb20gXCJAbmVzdGpzL3N3YWdnZXJcIjtcbmltcG9ydCB7IENvbnN0cnVjdG9yIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0aW9uXCI7XG5cbmltcG9ydCB7IEFVVEhfTUVUQV9LRVksIElTX1BVQkxJQ19LRVksIFJFUVVJUkVEX1JPTEVTX0tFWSB9IGZyb20gXCIuL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgQXV0aEludGVyY2VwdG9yIH0gZnJvbSBcIi4vQXV0aEludGVyY2VwdG9yXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiBBdXRoKG1vZGVsPzogc3RyaW5nIHwgQ29uc3RydWN0b3IpIHtcbiAgY29uc3QgcmVzb3VyY2UgPSBtb2RlbFxuICAgID8gdHlwZW9mIG1vZGVsID09PSBcInN0cmluZ1wiXG4gICAgICA/IG1vZGVsXG4gICAgICA6IG1vZGVsLm5hbWVcbiAgICA6IHVuZGVmaW5lZDtcbiAgY29uc3QgZGVjcyA9IFtBcGlCZWFyZXJBdXRoKCksIFVzZUludGVyY2VwdG9ycyhBdXRoSW50ZXJjZXB0b3IpXTtcbiAgaWYgKHJlc291cmNlKSBkZWNzLnB1c2goU2V0TWV0YWRhdGEoQVVUSF9NRVRBX0tFWSwgcmVzb3VyY2UpKTtcbiAgcmV0dXJuIGFwcGx5RGVjb3JhdG9ycyguLi5kZWNzKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIFB1YmxpYygpIHtcbiAgcmV0dXJuIGFwcGx5RGVjb3JhdG9ycyhTZXRNZXRhZGF0YShJU19QVUJMSUNfS0VZLCB0cnVlKSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBSZXF1aXJlUm9sZXMoLi4ucm9sZXM6IHN0cmluZ1tdKSB7XG4gIHJldHVybiBhcHBseURlY29yYXRvcnMoXG4gICAgQXBpU2VjdXJpdHkoXCJiZWFyZXJcIiksXG4gICAgU2V0TWV0YWRhdGEoUkVRVUlSRURfUk9MRVNfS0VZLCByb2xlcyksXG4gICAgVXNlSW50ZXJjZXB0b3JzKEF1dGhJbnRlcmNlcHRvcilcbiAgKTtcbn1cbiIsImltcG9ydCB7IEluamVjdCwgSW5qZWN0YWJsZSwgU2NvcGUgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcbmltcG9ydCB7IERlY2FmUmVxdWVzdENvbnRleHQgfSBmcm9tIFwiLi9EZWNhZlJlcXVlc3RDb250ZXh0XCI7XG5pbXBvcnQgeyB0eXBlIERlY2FmUmVxdWVzdEhhbmRsZXIgfSBmcm9tIFwiLi4vdHlwZXNcIjtcbmltcG9ydCB7IERFQ0FGX0hBTkRMRVJTIH0gZnJvbSBcIi4uL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgTG9nZ2luZyB9IGZyb20gXCJAZGVjYWYtdHMvbG9nZ2luZ1wiO1xuXG4vKipcbiAqIEBkZXNjcmlwdGlvblxuICogRXhlY3V0ZXMgYWxsIHJlZ2lzdGVyZWQge0BsaW5rIERlY2FmUmVxdWVzdEhhbmRsZXJ9IGluc3RhbmNlcyBmb3IgdGhlIGN1cnJlbnQgcmVxdWVzdCxcbiAqIHByb3ZpZGluZyB0aGVtIHdpdGggYSBzaGFyZWQge0BsaW5rIERlY2FmUmVxdWVzdENvbnRleHR9LlxuICpcbiAqIEBzdW1tYXJ5XG4gKiBUaGUge0BsaW5rIERlY2FmSGFuZGxlckV4ZWN1dG9yfSBjbGFzcyBpcyByZXNwb25zaWJsZSBmb3Igb3JjaGVzdHJhdGluZyBhbmQgZXhlY3V0aW5nXG4gKiBhIHNlcXVlbmNlIG9mIHJlcXVlc3QgaGFuZGxlcnMuIEVhY2ggaGFuZGxlciByZWNlaXZlcyB0aGUgc2FtZSByZXF1ZXN0LXNjb3BlZCBjb250ZXh0LFxuICogYWxsb3dpbmcgY29vcmRpbmF0ZWQgcHJvY2Vzc2luZyBzdWNoIGFzIGF1dGhlbnRpY2F0aW9uLCBtZXRhZGF0YSBleHRyYWN0aW9uLCBhdWRpdGluZyxcbiAqIGFuZCBjdXN0b20gcGlwZWxpbmUgYmVoYXZpb3IuIEhhbmRsZXJzIGFyZSBpbmplY3RlZCB2aWEgdGhlIHtAbGluayBERUNBRl9IQU5ETEVSU30gdG9rZW4sXG4gKiBlbnN1cmluZyBleHRlbnNpYmlsaXR5IGFuZCBsb29zZSBjb3VwbGluZy5cbiAqXG4gKiBAY2xhc3MgRGVjYWZIYW5kbGVyRXhlY3V0b3JcbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHNcbiAqIC8vIEV4YW1wbGUgaGFuZGxlcjpcbiAqIGNsYXNzIEF1dGhIYW5kbGVyIGltcGxlbWVudHMgRGVjYWZSZXF1ZXN0SGFuZGxlciB7XG4gKiAgIGFzeW5jIGhhbmRsZShjb250ZXh0OiBEZWNhZlJlcXVlc3RDb250ZXh0LCByZXE6IFJlcXVlc3QpIHtcbiAqICAgICBjb25zdCB0b2tlbiA9IHJlcS5oZWFkZXJzW1wiYXV0aG9yaXphdGlvblwiXTtcbiAqICAgICBjb25zdCByZXN1bHQgPSBNeVNlcnZpY2UuZG9Tb21ldGhpbmcodG9rZW4pO1xuICogICAgIGNvbnRleHQuc2V0KFwibXkta2V5XCIsIHJlc3VsdCk7XG4gKiAgIH1cbiAqIH1cbiAqXG4gKiAvLyBFeGVjdXRvciB1c2FnZSBpbiBhIHJlcXVlc3Q6XG4gKiBhd2FpdCBleGVjdXRvci5leGVjKHJlcXVlc3QpO1xuICogLy8gQWxsIGhhbmRsZXJzIHdpbGwgcnVuIGluIHNlcXVlbmNlXG4gKiBgYGBcbiAqXG4gKiBAbWVybWFpZFxuICogc2VxdWVuY2VEaWFncmFtXG4gKiAgICAgcGFydGljaXBhbnQgQ2xpZW50XG4gKiAgICAgcGFydGljaXBhbnQgRXhlY3V0b3JcbiAqICAgICBwYXJ0aWNpcGFudCBIYW5kbGVyQVxuICogICAgIHBhcnRpY2lwYW50IEhhbmRsZXJCXG4gKlxuICogICAgIENsaWVudC0+PkV4ZWN1dG9yOiBleGVjKHJlcSlcbiAqICAgICBFeGVjdXRvci0+PkhhbmRsZXJBOiBoYW5kbGUoY29udGV4dCwgcmVxKVxuICogICAgIEhhbmRsZXJBLS0+PkV4ZWN1dG9yOiBjb21wbGV0ZWRcbiAqICAgICBFeGVjdXRvci0+PkhhbmRsZXJCOiBoYW5kbGUoY29udGV4dCwgcmVxKVxuICogICAgIEhhbmRsZXJCLS0+PkV4ZWN1dG9yOiBjb21wbGV0ZWRcbiAqICAgICBFeGVjdXRvci0tPj5DbGllbnQ6IHByb2Nlc3NpbmcgZmluaXNoZWRcbiAqL1xuQEluamVjdGFibGUoeyBzY29wZTogU2NvcGUuUkVRVUVTVCB9KVxuZXhwb3J0IGNsYXNzIERlY2FmSGFuZGxlckV4ZWN1dG9yIHtcbiAgY29uc3RydWN0b3IoXG4gICAgQEluamVjdChERUNBRl9IQU5ETEVSUykgcHJpdmF0ZSByZWFkb25seSBoYW5kbGVyczogRGVjYWZSZXF1ZXN0SGFuZGxlcltdLFxuICAgIHByaXZhdGUgcmVhZG9ubHkgY29udGV4dDogRGVjYWZSZXF1ZXN0Q29udGV4dFxuICApIHt9XG5cbiAgYXN5bmMgZXhlYyhyZXE6IFJlcXVlc3QsIHJlczogUmVzcG9uc2UpIHtcbiAgICBjb25zdCBsb2cgPSBMb2dnaW5nLmZvcihEZWNhZkhhbmRsZXJFeGVjdXRvci5uYW1lKS5mb3IodGhpcy5leGVjKTtcbiAgICBsb2cuZGVidWcoXG4gICAgICBgQ09OVEVYVCAke3RoaXMuY29udGV4dC51dWlkfSBydW5uaW5nICR7dGhpcy5oYW5kbGVycy5sZW5ndGh9IGhhbmRsZXJzIGZvciByZXF1ZXN0ICR7cmVxLm1ldGhvZH0gJHtyZXEudXJsfWBcbiAgICApO1xuICAgIGZvciAoY29uc3QgaGFuZGxlciBvZiB0aGlzLmhhbmRsZXJzKSB7XG4gICAgICBhd2FpdCBoYW5kbGVyLmhhbmRsZSh0aGlzLmNvbnRleHQsIHJlcSwgcmVzKTtcbiAgICB9XG4gIH1cbn1cbiIsImltcG9ydCB7XG4gIEluamVjdGFibGUsXG4gIE5lc3RJbnRlcmNlcHRvcixcbiAgRXhlY3V0aW9uQ29udGV4dCxcbiAgQ2FsbEhhbmRsZXIsXG59IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgT2JzZXJ2YWJsZSwgdGFwIH0gZnJvbSBcInJ4anNcIjtcbmltcG9ydCB7IERlY2FmUmVxdWVzdENvbnRleHQgfSBmcm9tIFwiLi9EZWNhZlJlcXVlc3RDb250ZXh0XCI7XG5cbkBJbmplY3RhYmxlKClcbmV4cG9ydCBjbGFzcyBEZWNhZlJlc3BvbnNlSW50ZXJjZXB0b3IgaW1wbGVtZW50cyBOZXN0SW50ZXJjZXB0b3Ige1xuICBjb25zdHJ1Y3Rvcihwcm90ZWN0ZWQgY3R4OiBEZWNhZlJlcXVlc3RDb250ZXh0KSB7fVxuXG4gIGludGVyY2VwdChjb250ZXh0OiBFeGVjdXRpb25Db250ZXh0LCBuZXh0OiBDYWxsSGFuZGxlcik6IE9ic2VydmFibGU8YW55PiB7XG4gICAgbGV0IHJlc3BvbnNlID0gY29udGV4dC5zd2l0Y2hUb0h0dHAoKS5nZXRSZXNwb25zZSgpO1xuXG4gICAgcmV0dXJuIG5leHQuaGFuZGxlKCkucGlwZShcbiAgICAgIHRhcCgoZGF0YSkgPT4ge1xuICAgICAgICByZXNwb25zZSA9IHRoaXMuY3R4LnRvUmVzcG9uc2UocmVzcG9uc2UpO1xuICAgICAgfSlcbiAgICApO1xuICB9XG59XG4iLCJpbXBvcnQgeyBDb25zdHJ1Y3RvciwgTWV0YWRhdGEgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRpb25cIjtcbmltcG9ydCB7IFJlcXVlc3RUb0NvbnRleHRUcmFuc2Zvcm1lciB9IGZyb20gXCJAZGVjYWYtdHMvZm9yLWh0dHAvc2VydmVyXCI7XG5pbXBvcnQgeyBBZGFwdGVyLCBDb250ZXh0T2YsIENvbnRleHQgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcblxuKEFkYXB0ZXIgYXMgYW55KS50cmFuc2Zvcm1lckZvciA9IGZ1bmN0aW9uIHRvQ29udGV4dEZsYWdzPFxuICBBIGV4dGVuZHMgQWRhcHRlcjxhbnksIGFueSwgYW55LCBhbnk+LFxuPihhZGFwdGVyOiBBIHwgc3RyaW5nKTogQ29uc3RydWN0b3I8UmVxdWVzdFRvQ29udGV4dFRyYW5zZm9ybWVyPENvbnRleHRPZjxBPj4+IHtcbiAgY29uc3QgYWxpYXMgPVxuICAgIHR5cGVvZiBhZGFwdGVyID09PSBcInN0cmluZ1wiID8gYWRhcHRlciA6IChhZGFwdGVyLmFsaWFzIGFzIHN0cmluZyk7XG4gIHJldHVybiBNZXRhZGF0YVtcImlubmVyR2V0XCJdKFN5bWJvbC5mb3IoXCJ0cmFuc2Zvcm1lcnNcIiksIGFsaWFzKTtcbn0uYmluZChBZGFwdGVyKTtcblxuKEFkYXB0ZXIgYXMgYW55KS5mbGF2b3Vyc1RvVHJhbnNmb3JtID0gZnVuY3Rpb24gcmVxdWVzdFRyYW5zZm9ybWVycygpOlxuICB8IHN0cmluZ1tdXG4gIHwgdW5kZWZpbmVkIHtcbiAgY29uc3QgbWV0YSA9IE1ldGFkYXRhW1wiaW5uZXJHZXRcIl0oU3ltYm9sLmZvcihcInRyYW5zZm9ybWVyc1wiKSk7XG4gIGlmICghbWV0YSkgcmV0dXJuIHVuZGVmaW5lZDtcbiAgcmV0dXJuIE9iamVjdC5rZXlzKG1ldGEpO1xufS5iaW5kKEFkYXB0ZXIpO1xuXG4oQ29udGV4dCBhcyBhbnkpLnByb3RvdHlwZS50b1Jlc3BvbnNlID0gZnVuY3Rpb24gdG9SZXNwb25zZTxcbiAgUkVTIGV4dGVuZHMgeyBoZWFkZXI6IGFueSB9LFxuPih0aGlzOiBDb250ZXh0LCByZXM6IFJFUyk6IFJFUyB7XG4gIGNvbnN0IHBlbmRpbmcgPSB0aGlzLnBlbmRpbmcoKTtcbiAgaWYgKHBlbmRpbmcpIHJlcy5oZWFkZXIoXCJ4LXBlbmRpbmctdGFza1wiLCBKU09OLnN0cmluZ2lmeShwZW5kaW5nKSk7XG4gIHJldHVybiByZXM7XG59O1xuIiwiaW1wb3J0IFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5pbXBvcnQgeyBDb25zdHJ1Y3RvciB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdGlvblwiO1xuaW1wb3J0IHsgTW9kZWxCdWlsZGVyIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvblwiO1xuaW1wb3J0IHsgQXV0aCB9IGZyb20gXCIuLi9kZWNhZi1tb2RlbC9kZWNvcmF0b3JzL2RlY29yYXRvcnNcIjtcblxuZGVjbGFyZSBtb2R1bGUgXCJAZGVjYWYtdHMvZGVjb3JhdG9yLXZhbGlkYXRpb25cIiB7XG4gIGV4cG9ydCBpbnRlcmZhY2UgTW9kZWxCdWlsZGVyPE0+IHtcbiAgICBBdXRoKG1vZGVsOiBzdHJpbmcgfCBDb25zdHJ1Y3Rvcik6IE1vZGVsQnVpbGRlcjxNPjtcbiAgICBkZWNvcmF0ZUNsYXNzKGRlY29yYXRvcjogQ2xhc3NEZWNvcmF0b3IpOiBNb2RlbEJ1aWxkZXI8TT47XG4gIH1cbn1cblxuY29uc3QgcHJvdG90eXBlID0gTW9kZWxCdWlsZGVyLnByb3RvdHlwZSBhcyBNb2RlbEJ1aWxkZXI8YW55PiAmIHtcbiAgQXV0aDogKG1vZGVsOiBzdHJpbmcgfCBDb25zdHJ1Y3RvcikgPT4gTW9kZWxCdWlsZGVyPGFueT47XG59O1xuXG5pZiAoIXByb3RvdHlwZS5kZWNvcmF0ZUNsYXNzKSB7XG4gIHByb3RvdHlwZS5kZWNvcmF0ZUNsYXNzID0gZnVuY3Rpb24gKGRlY29yYXRvcjogQ2xhc3NEZWNvcmF0b3IpIHtcbiAgICBpZiAoISh0aGlzIGFzIGFueSkuX2NsYXNzRGVjb3JhdG9ycykge1xuICAgICAgKHRoaXMgYXMgYW55KS5fY2xhc3NEZWNvcmF0b3JzID0gW107XG4gICAgfVxuICAgICh0aGlzIGFzIGFueSkuX2NsYXNzRGVjb3JhdG9ycy5wdXNoKGRlY29yYXRvcik7XG4gICAgcmV0dXJuIHRoaXM7XG4gIH07XG59XG5cbnByb3RvdHlwZS5BdXRoID0gZnVuY3Rpb24gKG1vZGVsOiBzdHJpbmcgfCBDb25zdHJ1Y3Rvcikge1xuICByZXR1cm4gdGhpcy5kZWNvcmF0ZUNsYXNzKEF1dGgobW9kZWwpKTtcbn07XG5cbmlmICghKHByb3RvdHlwZSBhcyBhbnkpLl9faGFzQ2xhc3NEZWNvcmF0b3JTdXBwb3J0KSB7XG4gIGNvbnN0IG9yaWdpbmFsQnVpbGQgPSBwcm90b3R5cGUuYnVpbGQ7XG4gIHByb3RvdHlwZS5idWlsZCA9IGZ1bmN0aW9uICgpIHtcbiAgICBsZXQgcmVzdWx0ID0gb3JpZ2luYWxCdWlsZC5jYWxsKHRoaXMpO1xuICAgIGNvbnN0IGRlY29yYXRvcnMgPSAodGhpcyBhcyBhbnkpLl9jbGFzc0RlY29yYXRvcnM7XG4gICAgaWYgKGRlY29yYXRvcnM/Lmxlbmd0aCkge1xuICAgICAgZm9yIChjb25zdCBkZWNvcmF0b3Igb2YgZGVjb3JhdG9ycykge1xuICAgICAgICBjb25zdCBkZWNvcmF0ZWQgPSBkZWNvcmF0b3IocmVzdWx0IGFzIGFueSkgYXMgdHlwZW9mIHJlc3VsdCB8IHZvaWQ7XG4gICAgICAgIGlmIChkZWNvcmF0ZWQpIHtcbiAgICAgICAgICByZXN1bHQgPSBkZWNvcmF0ZWQ7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG4gICAgcmV0dXJuIHJlc3VsdDtcbiAgfTtcbiAgKHByb3RvdHlwZSBhcyBhbnkpLl9faGFzQ2xhc3NEZWNvcmF0b3JTdXBwb3J0ID0gdHJ1ZTtcbn1cbiIsImltcG9ydCB7XG4gIENhbGxIYW5kbGVyLFxuICBFeGVjdXRpb25Db250ZXh0LFxuICBJbmplY3RhYmxlLFxuICBOZXN0SW50ZXJjZXB0b3IsXG4gIFNjb3BlLFxufSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcbmltcG9ydCB7IERlY2FmSGFuZGxlckV4ZWN1dG9yLCBEZWNhZlJlcXVlc3RDb250ZXh0IH0gZnJvbSBcIi4uL3JlcXVlc3RcIjtcbmltcG9ydCB7IERlZmF1bHRBZGFwdGVyRmxhZ3MgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7XG4gIERFQ0FGX0FEQVBURVJfT1BUSU9OUyxcbiAgRGVjYWZTZXJ2ZXJGbGFncyxcbn0gZnJvbSBcIi4uL2NvbnN0YW50c1wiO1xuaW1wb3J0IFwiLi4vb3ZlcnJpZGVzXCI7XG5pbXBvcnQgeyBMb2dnaW5nIH0gZnJvbSBcIkBkZWNhZi10cy9sb2dnaW5nXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uXG4gKiBJbnRlcmNlcHRvciByZXNwb25zaWJsZSBmb3IgZXhlY3V0aW5nIGFsbCByZWdpc3RlcmVkIERlY2FmIHJlcXVlc3QgaGFuZGxlcnNcbiAqIGJlZm9yZSB0aGUgY29udHJvbGxlciBtZXRob2QgaXMgaW52b2tlZC5cbiAqXG4gKiBAc3VtbWFyeVxuICogVGhlIHtAbGluayBEZWNhZlJlcXVlc3RIYW5kbGVySW50ZXJjZXB0b3J9IGludGVncmF0ZXMgdGhlIERlY2FmIHJlcXVlc3QtaGFuZGxpbmcgcGlwZWxpbmVcbiAqIGludG8gTmVzdEpTJyBpbnRlcmNlcHRvciBtZWNoYW5pc20uIEJlZm9yZSBwYXNzaW5nIGV4ZWN1dGlvbiB0byB0aGUgbmV4dCBoYW5kbGVyIGluIHRoZVxuICogTmVzdEpTIGNoYWluLCBpdCBkZWxlZ2F0ZXMgcmVxdWVzdCBwcm9jZXNzaW5nIHRvIHRoZSB7QGxpbmsgRGVjYWZIYW5kbGVyRXhlY3V0b3J9LCB3aGljaFxuICogc2VxdWVudGlhbGx5IHJ1bnMgYWxsIHJlZ2lzdGVyZWQge0BsaW5rIERlY2FmUmVxdWVzdEhhbmRsZXJ9IGluc3RhbmNlcy4gVGhpcyBhbGxvd3NcbiAqIGJlaGF2aW9ycyBzdWNoIGFzIGF1dGhlbnRpY2F0aW9uLCBsb2dnaW5nLCB0ZW5hbnQgcmVzb2x1dGlvbiwgb3IgbWV0YWRhdGEgZW5yaWNobWVudFxuICogdG8gb2NjdXIgcHJpb3IgdG8gY29udHJvbGxlciBleGVjdXRpb24uXG4gKlxuICogQGNsYXNzIERlY2FmUmVxdWVzdEhhbmRsZXJJbnRlcmNlcHRvclxuICpcbiAqIEBleGFtcGxlXG4gKiBgYGB0c1xuICogLy8gQXBwbHkgZ2xvYmFsbHk6XG4gKiBhcHAudXNlR2xvYmFsSW50ZXJjZXB0b3JzKG5ldyBEZWNhZlJlcXVlc3RIYW5kbGVySW50ZXJjZXB0b3IoZXhlY3V0b3IpKTtcbiAqXG4gKiAvLyBPciBpbiBhIG1vZHVsZTpcbiAqIEBNb2R1bGUoe1xuICogICBwcm92aWRlcnM6IFtcbiAqICAgICBEZWNhZkhhbmRsZXJFeGVjdXRvcixcbiAqICAgICB7XG4gKiAgICAgICBwcm92aWRlOiBBUFBfSU5URVJDRVBUT1IsXG4gKiAgICAgICB1c2VDbGFzczogRGVjYWZSZXF1ZXN0SGFuZGxlckludGVyY2VwdG9yLFxuICogICAgIH0sXG4gKiAgIF0sXG4gKiB9KVxuICogZXhwb3J0IGNsYXNzIEFwcE1vZHVsZSB7fVxuICogYGBgXG4gKlxuICogQG1lcm1haWRcbiAqIHNlcXVlbmNlRGlhZ3JhbVxuICogICAgIHBhcnRpY2lwYW50IENsaWVudFxuICogICAgIHBhcnRpY2lwYW50IEludGVyY2VwdG9yXG4gKiAgICAgcGFydGljaXBhbnQgRXhlY3V0b3JcbiAqICAgICBwYXJ0aWNpcGFudCBDb250cm9sbGVyXG4gKlxuICogICAgIENsaWVudC0+PkludGVyY2VwdG9yOiBIVFRQIFJlcXVlc3RcbiAqICAgICBJbnRlcmNlcHRvci0+PkV4ZWN1dG9yOiBleGVjKHJlcXVlc3QpXG4gKiAgICAgRXhlY3V0b3ItLT4+SW50ZXJjZXB0b3I6IGhhbmRsZXJzIGNvbXBsZXRlZFxuICogICAgIEludGVyY2VwdG9yLT4+Q29udHJvbGxlcjogbmV4dC5oYW5kbGUoKVxuICogICAgIENvbnRyb2xsZXItLT4+Q2xpZW50OiBSZXNwb25zZVxuICovXG5ASW5qZWN0YWJsZSh7IHNjb3BlOiBTY29wZS5SRVFVRVNUIH0pXG5leHBvcnQgY2xhc3MgRGVjYWZSZXF1ZXN0SGFuZGxlckludGVyY2VwdG9yIGltcGxlbWVudHMgTmVzdEludGVyY2VwdG9yIHtcbiAgY29uc3RydWN0b3IoXG4gICAgcHJvdGVjdGVkIHJlYWRvbmx5IHJlcXVlc3RDb250ZXh0OiBEZWNhZlJlcXVlc3RDb250ZXh0LFxuICAgIHByb3RlY3RlZCByZWFkb25seSBleGVjdXRvcjogRGVjYWZIYW5kbGVyRXhlY3V0b3JcbiAgKSB7fVxuXG4gIHByb3RlY3RlZCBhc3luYyBjb250ZXh0dWFsaXplKHJlcTogYW55KTogUHJvbWlzZTx2b2lkPiB7XG4gICAgY29uc3QgaGVhZGVycyA9IHJlcS5oZWFkZXJzO1xuICAgIGNvbnN0IGZsYWdzOiBEZWNhZlNlcnZlckZsYWdzID0ge1xuICAgICAgaGVhZGVyczogaGVhZGVycyxcbiAgICAgIG92ZXJyaWRlczoge30sXG4gICAgfSBhcyBhbnk7XG4gICAgT2JqZWN0LmFzc2lnbihmbGFncy5vdmVycmlkZXMsIHJlcT8uW0RFQ0FGX0FEQVBURVJfT1BUSU9OU10gPz8ge30pO1xuXG4gICAgY29uc3QgaXAgPSBleHRyYWN0SXAocmVxKTtcbiAgICBjb25zdCBsb2dnZXIgPSBMb2dnaW5nLmdldCgpLmZvcih7IGlwIH0pO1xuXG4gICAgdGhpcy5yZXF1ZXN0Q29udGV4dC5hY2N1bXVsYXRlKFxuICAgICAgT2JqZWN0LmFzc2lnbihcbiAgICAgICAge30sXG4gICAgICAgIERlZmF1bHRBZGFwdGVyRmxhZ3MsXG4gICAgICAgIHtcbiAgICAgICAgICBsb2dnZXIsXG4gICAgICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLFxuICAgICAgICAgIG9wZXJhdGlvbjogYCR7cmVxLm1ldGhvZH0gJHtyZXEudXJsfWAsXG4gICAgICAgIH0sXG4gICAgICAgIGZsYWdzXG4gICAgICApXG4gICAgKTtcbiAgfVxuXG4gIGFzeW5jIGludGVyY2VwdChjb250ZXh0OiBFeGVjdXRpb25Db250ZXh0LCBuZXh0OiBDYWxsSGFuZGxlcikge1xuICAgIGNvbnN0IHJlcSA9IGNvbnRleHQuc3dpdGNoVG9IdHRwKCkuZ2V0UmVxdWVzdCgpO1xuICAgIGNvbnN0IHJlcyA9IGNvbnRleHQuc3dpdGNoVG9IdHRwKCkuZ2V0UmVzcG9uc2UoKTtcbiAgICBjb25zdCBsb2cgPSBMb2dnaW5nLmZvcihEZWNhZlJlcXVlc3RIYW5kbGVySW50ZXJjZXB0b3IpLmZvcih0aGlzLmludGVyY2VwdCk7XG4gICAgbG9nLmRlYnVnKFxuICAgICAgYENPTlRFWFQgJHt0aGlzLnJlcXVlc3RDb250ZXh0LnV1aWR9IC0gcmVxdWVzdDogJHtyZXEubWV0aG9kfSAke3JlcS51cmx9YFxuICAgICk7XG4gICAgYXdhaXQgdGhpcy5jb250ZXh0dWFsaXplKHJlcSk7XG4gICAgbG9nLmRlYnVnKFxuICAgICAgYENPTlRFWFQgJHt0aGlzLnJlcXVlc3RDb250ZXh0LnV1aWR9IGNvbnRleHR1YWxpemVkIC0gcmVxdWVzdDogJHtyZXEubWV0aG9kfSAke3JlcS51cmx9YFxuICAgICk7XG5cbiAgICBhd2FpdCB0aGlzLmV4ZWN1dG9yLmV4ZWMocmVxLCByZXMpO1xuICAgIGxvZy5kZWJ1ZyhcbiAgICAgIGBDT05URVhUICR7dGhpcy5yZXF1ZXN0Q29udGV4dC51dWlkfSBleGVjdXRvcnMgZmluaXNoZWQgLSByZXF1ZXN0OiAke3JlcS5tZXRob2R9ICR7cmVxLnVybH1gXG4gICAgKTtcbiAgICByZXR1cm4gbmV4dC5oYW5kbGUoKTtcbiAgfVxufVxuXG5mdW5jdGlvbiBleHRyYWN0SXAocmVxOiBhbnkpOiBzdHJpbmcgfCB1bmRlZmluZWQge1xuICBjb25zdCBoZWFkZXJzID0gcmVxLmhlYWRlcnM7XG4gIGZ1bmN0aW9uIHBhcnNlSXBIZWFkZXIodmFsdWU/OiBzdHJpbmcgfCBzdHJpbmdbXSk6IHN0cmluZyB8IHVuZGVmaW5lZCB7XG4gICAgaWYgKCF2YWx1ZSkgcmV0dXJuIHVuZGVmaW5lZDtcbiAgICBjb25zdCBjYW5kaWRhdGUgPSBBcnJheS5pc0FycmF5KHZhbHVlKSA/IHZhbHVlWzBdIDogdmFsdWU7XG4gICAgcmV0dXJuIGNhbmRpZGF0ZVxuICAgICAgLnNwbGl0KFwiLFwiKVxuICAgICAgLm1hcCgoc2VnbWVudCkgPT4gc2VnbWVudC50cmltKCkpXG4gICAgICAuZmlsdGVyKEJvb2xlYW4pWzBdO1xuICB9XG4gIHJldHVybiAoXG4gICAgcGFyc2VJcEhlYWRlcihoZWFkZXJzPy5bXCJ4LWZvcndhcmRlZC1mb3JcIl0pID8/XG4gICAgcGFyc2VJcEhlYWRlcihoZWFkZXJzPy5bXCJ4LXJlYWwtaXBcIl0pID8/XG4gICAgcGFyc2VJcEhlYWRlcihoZWFkZXJzPy5bXCJYLUZvcndhcmRlZC1Gb3JcIl0pID8/XG4gICAgcGFyc2VJcEhlYWRlcihoZWFkZXJzPy5bXCJYLVJlYWwtSVBcIl0pID8/XG4gICAgcmVxLmlwXG4gICk7XG59XG4iLCJpbXBvcnQgeyBEeW5hbWljTW9kdWxlLCBNb2R1bGUsIFR5cGUgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcbmltcG9ydCB7IEFQUF9JTlRFUkNFUFRPUiB9IGZyb20gXCJAbmVzdGpzL2NvcmVcIjtcblxuaW1wb3J0IHsgQXV0aEludGVyY2VwdG9yIH0gZnJvbSBcIi4vQXV0aEludGVyY2VwdG9yXCI7XG5pbXBvcnQgeyBBVVRIX0hBTkRMRVIgfSBmcm9tIFwiLi9jb25zdGFudHNcIjtcbmltcG9ydCB7IERlY2FmUmVxdWVzdEhhbmRsZXJJbnRlcmNlcHRvciB9IGZyb20gXCIuLi9pbnRlcmNlcHRvcnMvRGVjYWZSZXF1ZXN0SGFuZGxlckludGVyY2VwdG9yXCI7XG5pbXBvcnQgeyBBdXRoSGFuZGxlciB9IGZyb20gXCIuLi90eXBlc1wiO1xuXG5leHBvcnQgdHlwZSBEZWNhZkF1dGhNb2R1bGVPcHRpb25zID0ge1xuICBnbG9iYWw/OiBib29sZWFuO1xuICBoYW5kbGVyPzogVHlwZTxBdXRoSGFuZGxlcj47XG59O1xuXG5ATW9kdWxlKHt9KVxuZXhwb3J0IGNsYXNzIERlY2FmQXV0aE1vZHVsZSB7XG4gIHN0YXRpYyBmb3JSb290KFxuICAgIG9wdGlvbnM6IERlY2FmQXV0aE1vZHVsZU9wdGlvbnMgPSB7fVxuICApOiBEeW5hbWljTW9kdWxlIHtcbiAgICBjb25zdCBwcm92aWRlcnM6IER5bmFtaWNNb2R1bGVbXCJwcm92aWRlcnNcIl0gPSBbXG4gICAgICBBdXRoSW50ZXJjZXB0b3IsXG4gICAgICBEZWNhZlJlcXVlc3RIYW5kbGVySW50ZXJjZXB0b3IsXG4gICAgICB7XG4gICAgICAgIHByb3ZpZGU6IEFQUF9JTlRFUkNFUFRPUixcbiAgICAgICAgdXNlQ2xhc3M6IERlY2FmUmVxdWVzdEhhbmRsZXJJbnRlcmNlcHRvcixcbiAgICAgIH0sXG4gICAgXTtcblxuICAgIGlmIChvcHRpb25zLmhhbmRsZXIpIHtcbiAgICAgIHByb3ZpZGVycy5wdXNoKG9wdGlvbnMuaGFuZGxlcik7XG4gICAgICBwcm92aWRlcnMucHVzaCh7XG4gICAgICAgIHByb3ZpZGU6IEFVVEhfSEFORExFUixcbiAgICAgICAgdXNlQ2xhc3M6IG9wdGlvbnMuaGFuZGxlcixcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIGlmIChvcHRpb25zLmdsb2JhbCkge1xuICAgICAgKHByb3ZpZGVycyBhcyBhbnlbXSkucHVzaCh7XG4gICAgICAgIHByb3ZpZGU6IEFQUF9JTlRFUkNFUFRPUixcbiAgICAgICAgdXNlRXhpc3Rpbmc6IEF1dGhJbnRlcmNlcHRvcixcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIHJldHVybiB7XG4gICAgICBtb2R1bGU6IERlY2FmQXV0aE1vZHVsZSxcbiAgICAgIGdsb2JhbDogb3B0aW9ucy5nbG9iYWwgPz8gZmFsc2UsXG4gICAgICBwcm92aWRlcnMsXG4gICAgICBleHBvcnRzOiBbQXV0aEludGVyY2VwdG9yLCBBVVRIX0hBTkRMRVJdLFxuICAgIH07XG4gIH1cbn1cbiIsImltcG9ydCB7IERlbGV0ZSwgR2V0LCBQYXRjaCwgUG9zdCwgUHV0IH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBIdHRwVmVyYnMgfSBmcm9tIFwiLi90eXBlc1wiO1xuXG4vKipcbiAqIE1hcHMgYW4gSFRUUCB2ZXJiIHRvIGl0cyBjb3JyZXNwb25kaW5nIE5lc3RKUyBtZXRob2QgZGVjb3JhdG9yLlxuICpcbiAqIEBwYXJhbSB2ZXJiIC0gSFRUUCB2ZXJiIHRvIGJlIGNvbnZlcnRlZCAoZS5nLiBHRVQsIFBPU1QsIFBVVCwgUEFUQ0gsIERFTEVURSkuXG4gKiBAcmV0dXJucyBBIE5lc3RKUyBtZXRob2QgZGVjb3JhdG9yIG1hdGNoaW5nIHRoZSBwcm92aWRlZCBIVFRQIHZlcmIuXG4gKlxuICogQHRocm93cyB7RXJyb3J9IElmIHRoZSBwcm92aWRlZCBIVFRQIHZlcmIgaXMgbm90IHN1cHBvcnRlZCBvciBub3QgbWFwcGVkXG4gKiB0byBhIE5lc3RKUyBkZWNvcmF0b3IuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBIdHRwVmVyYlRvRGVjb3JhdG9yKFxuICB2ZXJiOiBIdHRwVmVyYnNcbik6IChwYXRoPzogc3RyaW5nKSA9PiBNZXRob2REZWNvcmF0b3Ige1xuICBjb25zdCBodHRwVG9DcnVkOiBSZWNvcmQ8SHR0cFZlcmJzLCAocGF0aD86IHN0cmluZykgPT4gTWV0aG9kRGVjb3JhdG9yPiA9IHtcbiAgICBHRVQ6IEdldCxcbiAgICBQT1NUOiBQb3N0LFxuICAgIFBVVDogUHV0LFxuICAgIFBBVENIOiBQYXRjaCxcbiAgICBERUxFVEU6IERlbGV0ZSxcbiAgfTtcblxuICBjb25zdCBkZWNvcmF0b3IgPSBodHRwVG9DcnVkW3ZlcmJdO1xuXG4gIGlmICghZGVjb3JhdG9yKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgYFVuc3VwcG9ydGVkIEhUVFAgdmVyYiBcIiR7dmVyYn1cIi4gTm8gTmVzdEpTIGRlY29yYXRvciBtYXBwaW5nIHdhcyBmb3VuZC5gXG4gICAgKTtcbiAgfVxuXG4gIHJldHVybiBkZWNvcmF0b3I7XG59XG4iLCJpbXBvcnQgeyBRdWVyeSB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgTG9nZ2VyIH0gZnJvbSBcIkBkZWNhZi10cy9sb2dnaW5nXCI7XG5pbXBvcnQgeyB0eXBlIERlY29yYXRvckJ1bmRsZSB9IGZyb20gXCIuL3R5cGVzXCI7XG5pbXBvcnQge1xuICBBcGlOb0NvbnRlbnRSZXNwb25zZSxcbiAgQXBpT2tSZXNwb25zZSxcbiAgQXBpT3BlcmF0aW9uLFxuICBBcGlQYXJhbSxcbiAgQXBpUXVlcnksXG59IGZyb20gXCJAbmVzdGpzL3N3YWdnZXJcIjtcbmltcG9ydCB7XG4gIERlY2FmQXBpUHJvcGVydHksXG4gIHR5cGUgRGVjYWZQYXJhbVByb3BzLFxuICBEZWNhZlBhcmFtcyxcbn0gZnJvbSBcIi4vZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHR5cGUgeyBIdHRwVmVyYnMgfSBmcm9tIFwiLi9kZWNvcmF0b3JzL3R5cGVzXCI7XG5pbXBvcnQge1xuICBEaXJlY3Rpb25MaW1pdE9mZnNldCxcbiAgTW9kZWxTZXJ2aWNlLFxuICBPcmRlckRpcmVjdGlvbixcbiAgUmVwbyxcbn0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBNb2RlbCB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdG9yLXZhbGlkYXRpb25cIjtcbmltcG9ydCB7IEh0dHBWZXJiVG9EZWNvcmF0b3IgfSBmcm9tIFwiLi9kZWNvcmF0b3JzL3V0aWxzXCI7XG5pbXBvcnQgeyBEZWNhZk1vZGVsQ29udHJvbGxlciB9IGZyb20gXCIuLi9jb250cm9sbGVyc1wiO1xuXG5jb25zdCBleHRyYWN0UGF0aFBhcmFtcyA9IChyb3V0ZVBhdGg6IHN0cmluZyk6IHN0cmluZ1tdID0+IHtcbiAgcmV0dXJuIHJvdXRlUGF0aFxuICAgIC5zcGxpdChcIi9cIilcbiAgICAuZmlsdGVyKChwKSA9PiBwLnN0YXJ0c1dpdGgoXCI6XCIpKVxuICAgIC5tYXAoKHApID0+IHAuc2xpY2UoMSkpO1xufTtcblxuY29uc3QgYXBpUGFyYW1TcGVjID0gKG5hbWU6IHN0cmluZyk6IERlY2FmQXBpUHJvcGVydHkgPT4gKHtcbiAgbmFtZSxcbiAgZGVzY3JpcHRpb246IGAke25hbWV9IHBhcmFtZXRlciBmb3IgdGhlIHF1ZXJ5YCxcbiAgcmVxdWlyZWQ6IHRydWUsXG4gIHR5cGU6IFN0cmluZyxcbn0pO1xuXG5leHBvcnQgZnVuY3Rpb24gZ2V0QXBpRGVjb3JhdG9ycyhcbiAgbWV0aG9kTmFtZTogc3RyaW5nLFxuICByb3V0ZVBhdGg6IHN0cmluZyxcbiAgaHR0cFZlcmI6IEh0dHBWZXJicyxcbiAgaW5jbHVkZVF1ZXJ5UGFyYW1zOiBib29sZWFuID0gZmFsc2Vcbik6IERlY29yYXRvckJ1bmRsZSB7XG4gIGNvbnN0IE5lc3RIdHRwUm91dGVEZWMgPSBIdHRwVmVyYlRvRGVjb3JhdG9yKGh0dHBWZXJiKTtcbiAgY29uc3QgYXBpUGF0aFBhcmFtcyA9IGV4dHJhY3RQYXRoUGFyYW1zKHJvdXRlUGF0aCkubWFwKGFwaVBhcmFtU3BlYyk7XG5cbiAgY29uc3Qgc3dhZ2dlclF1ZXJ5UGFyYW1zID0gW107XG4gIGlmIChodHRwVmVyYiA9PT0gXCJHRVRcIiAmJiBpbmNsdWRlUXVlcnlQYXJhbXMpIHtcbiAgICBzd2FnZ2VyUXVlcnlQYXJhbXMucHVzaChcbiAgICAgIEFwaVF1ZXJ5KHtcbiAgICAgICAgbmFtZTogXCJkaXJlY3Rpb25cIixcbiAgICAgICAgcmVxdWlyZWQ6IGZhbHNlLFxuICAgICAgICBlbnVtOiBPcmRlckRpcmVjdGlvbixcbiAgICAgICAgZGVzY3JpcHRpb246IFwidGhlIHNvcnQgb3JkZXIgd2hlbiBhcHBsaWNhYmxlXCIsXG4gICAgICB9KSxcbiAgICAgIEFwaVF1ZXJ5KHtcbiAgICAgICAgbmFtZTogXCJsaW1pdFwiLFxuICAgICAgICByZXF1aXJlZDogZmFsc2UsXG4gICAgICAgIGRlc2NyaXB0aW9uOiBcImxpbWl0IG9yIHBhZ2Ugc2l6ZSB3aGVuIGFwcGxpY2FibGVcIixcbiAgICAgIH0pLFxuICAgICAgQXBpUXVlcnkoe1xuICAgICAgICBuYW1lOiBcIm9mZnNldFwiLFxuICAgICAgICByZXF1aXJlZDogZmFsc2UsXG4gICAgICAgIGRlc2NyaXB0aW9uOiBcIm9mZnNldCBvciBib29rbWFyayB3aGVuIGFwcGxpY2FibGVcIixcbiAgICAgIH0pXG4gICAgKTtcbiAgfVxuXG4gIHJldHVybiB7XG4gICAgbWV0aG9kOiBbXG4gICAgICBOZXN0SHR0cFJvdXRlRGVjKHJvdXRlUGF0aCksXG4gICAgICAuLi5hcGlQYXRoUGFyYW1zLm1hcChBcGlQYXJhbSksXG4gICAgICAuLi5zd2FnZ2VyUXVlcnlQYXJhbXMsXG4gICAgICBBcGlPcGVyYXRpb24oe1xuICAgICAgICBzdW1tYXJ5OiBgUmV0cmlldmUgcmVjb3JkcyB1c2luZyBhY2NvcmRpbmcgdG8gXCIke21ldGhvZE5hbWV9XCIuYCxcbiAgICAgIH0pLFxuICAgICAgQXBpT2tSZXNwb25zZSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBgUmVzdWx0IHN1Y2Nlc3NmdWxseSByZXRyaWV2ZWQuYCxcbiAgICAgIH0pLFxuICAgICAgQXBpTm9Db250ZW50UmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYE5vIGNvbnRlbnQgcmV0dXJuZWQgYnkgdGhlIG1ldGhvZC5gLFxuICAgICAgfSksXG4gICAgXSxcbiAgICBwYXJhbXM6IFtEZWNhZlBhcmFtcyhhcGlQYXRoUGFyYW1zKSwgUXVlcnkoKV0sIC8vICwgRGVjYWZCb2R5KCldLFxuICB9O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gYXBwbHlBcGlEZWNvcmF0b3JzKFxuICB0YXJnZXQ6IGFueSxcbiAgbWV0aG9kTmFtZTogc3RyaW5nLFxuICBkZXNjcmlwdG9yOiBQcm9wZXJ0eURlc2NyaXB0b3IsXG4gIGRlY29yYXRvcnM6IERlY29yYXRvckJ1bmRsZVxuKSB7XG4gIGNvbnN0IHByb3RvID0gdGFyZ2V0Py5wcm90b3R5cGUgPz8gdGFyZ2V0O1xuICBkZWNvcmF0b3JzLm1ldGhvZC5mb3JFYWNoKChkKSA9PiBkKHByb3RvLCBtZXRob2ROYW1lLCBkZXNjcmlwdG9yKSk7XG4gIGRlY29yYXRvcnMucGFyYW1zPy5mb3JFYWNoKChkLCBpbmRleCkgPT4gZChwcm90bywgbWV0aG9kTmFtZSwgaW5kZXgpKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIHJlc29sdmVQZXJzaXN0ZW5jZU1ldGhvZDxUIGV4dGVuZHMgTW9kZWw8Ym9vbGVhbj4+KFxuICBwZXJzaXN0ZW5jZTogUmVwbzxUPiB8IE1vZGVsU2VydmljZTxUPixcbiAgbWV0aG9kTmFtZTogc3RyaW5nLFxuICAuLi5hcmdzOiBhbnlbXVxuKTogYW55IHtcbiAgaWYgKHBlcnNpc3RlbmNlIGluc3RhbmNlb2YgTW9kZWxTZXJ2aWNlKSB7XG4gICAgcmV0dXJuIHR5cGVvZiAocGVyc2lzdGVuY2UgYXMgYW55KVttZXRob2ROYW1lXSA9PT0gXCJmdW5jdGlvblwiXG4gICAgICA/IChwZXJzaXN0ZW5jZSBhcyBhbnkpW21ldGhvZE5hbWVdKC4uLmFyZ3MpXG4gICAgICA6IHBlcnNpc3RlbmNlLnN0YXRlbWVudChtZXRob2ROYW1lLCAuLi5hcmdzKTtcbiAgfVxuXG4gIGlmICh0eXBlb2YgKHBlcnNpc3RlbmNlIGFzIGFueSlbbWV0aG9kTmFtZV0gPT09IFwiZnVuY3Rpb25cIilcbiAgICByZXR1cm4gKHBlcnNpc3RlbmNlIGFzIGFueSlbbWV0aG9kTmFtZV0oLi4uYXJncyk7XG5cbiAgdGhyb3cgbmV3IEVycm9yKFxuICAgIGBQZXJzaXN0ZW5jZSBtZXRob2QgXCIke21ldGhvZE5hbWV9XCIgbm90IGZvdW5kIG9uICR7cGVyc2lzdGVuY2U/LmNvbnN0cnVjdG9yPy5uYW1lfWBcbiAgKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGNyZWF0ZVJvdXRlSGFuZGxlcjxUPihtZXRob2ROYW1lOiBzdHJpbmcpIHtcbiAgcmV0dXJuIGFzeW5jIGZ1bmN0aW9uIChcbiAgICB0aGlzOiBEZWNhZk1vZGVsQ29udHJvbGxlcjxhbnk+LFxuICAgIHBhdGhQYXJhbXM6IERlY2FmUGFyYW1Qcm9wcyxcbiAgICBxdWVyeVBhcmFtczogRGlyZWN0aW9uTGltaXRPZmZzZXRcbiAgKTogUHJvbWlzZTxUPiB7XG4gICAgY29uc3QgbG9nOiBMb2dnZXIgPSB0aGlzLmxvZy5mb3IobWV0aG9kTmFtZSk7XG5cbiAgICB0cnkge1xuICAgICAgbG9nLmRlYnVnKFxuICAgICAgICBgSW52b2tpbmcgcGVyc2lzdGVuY2UgbWV0aG9kIFwiJHttZXRob2ROYW1lfVwiIGdpdmVuIHBhcmFtZXRlcnM6ICR7SlNPTi5zdHJpbmdpZnkocGF0aFBhcmFtcy52YWx1ZXNJbk9yZGVyKX1gXG4gICAgICApO1xuICAgICAgY29uc3QgeyBkaXJlY3Rpb24sIGxpbWl0LCBvZmZzZXQgfSA9IHF1ZXJ5UGFyYW1zO1xuICAgICAgcmV0dXJuIGF3YWl0IHJlc29sdmVQZXJzaXN0ZW5jZU1ldGhvZChcbiAgICAgICAgdGhpcy5wZXJzaXN0ZW5jZSgpLFxuICAgICAgICBtZXRob2ROYW1lLFxuICAgICAgICAuLi5wYXRoUGFyYW1zLnZhbHVlc0luT3JkZXIsXG4gICAgICAgIGRpcmVjdGlvbixcbiAgICAgICAgbGltaXQsXG4gICAgICAgIG9mZnNldFxuICAgICAgKTtcbiAgICB9IGNhdGNoIChlOiBhbnkpIHtcbiAgICAgIGxvZy5lcnJvcihgQ3VzdG9tIHF1ZXJ5IFwiJHttZXRob2ROYW1lfVwiIGZhaWxlZGAsIGUpO1xuICAgICAgdGhyb3cgZTtcbiAgICB9XG4gIH07XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBkZWZpbmVSb3V0ZU1ldGhvZChcbiAgQ29udHJvbGxlckNsYXNzOiBuZXcgKC4uLmFyZ3M6IGFueVtdKSA9PiBhbnksXG4gIG1ldGhvZE5hbWU6IHN0cmluZyxcbiAgaGFuZGxlcjogKC4uLmFyZ3M6IGFueVtdKSA9PiBhbnlcbik6IFByb3BlcnR5RGVzY3JpcHRvciB8IHVuZGVmaW5lZCB7XG4gIE9iamVjdC5kZWZpbmVQcm9wZXJ0eShcbiAgICBDb250cm9sbGVyQ2xhc3MucHJvdG90eXBlIHx8IENvbnRyb2xsZXJDbGFzcyxcbiAgICBtZXRob2ROYW1lLFxuICAgIHtcbiAgICAgIHZhbHVlOiBoYW5kbGVyLFxuICAgICAgd3JpdGFibGU6IGZhbHNlLFxuICAgIH1cbiAgKTtcblxuICByZXR1cm4gT2JqZWN0LmdldE93blByb3BlcnR5RGVzY3JpcHRvcihcbiAgICBDb250cm9sbGVyQ2xhc3MucHJvdG90eXBlIHx8IENvbnRyb2xsZXJDbGFzcyxcbiAgICBtZXRob2ROYW1lXG4gICk7XG59XG4iLCJpbXBvcnQgeyB0eXBlIFJlcXVlc3QgfSBmcm9tIFwiZXhwcmVzc1wiO1xuaW1wb3J0IHsgRGVjYWZDb250cm9sbGVyIGFzIEh0dHBEZWNhZkNvbnRyb2xsZXIgfSBmcm9tIFwiQGRlY2FmLXRzL2Zvci1odHRwL3NlcnZlclwiO1xuaW1wb3J0IHtcbiAgQ29udGV4dCxcbiAgTW9kZWxTZXJ2aWNlLFxuICBSZXBvLFxuICBSZXBvc2l0b3J5LFxuICBTZXJ2aWNlLFxufSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IE1vZGVsLCB0eXBlIE1vZGVsQ29uc3RydWN0b3IgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5cbmltcG9ydCB7IERFQ0FGX0FEQVBURVJfT1BUSU9OUywgRGVjYWZTZXJ2ZXJDdHggfSBmcm9tIFwiLi9jb25zdGFudHNcIjtcbmltcG9ydCB7IERlY2FmUmVxdWVzdENvbnRleHQgfSBmcm9tIFwiLi9yZXF1ZXN0L0RlY2FmUmVxdWVzdENvbnRleHRcIjtcblxuZXhwb3J0IGFic3RyYWN0IGNsYXNzIERlY2FmQ29udHJvbGxlcjxcbiAgQ09OVEVYVCBleHRlbmRzIERlY2FmU2VydmVyQ3R4ID0gRGVjYWZTZXJ2ZXJDdHgsXG4+IGV4dGVuZHMgSHR0cERlY2FmQ29udHJvbGxlcjxSZXF1ZXN0LCBhbnksIERlY2FmUmVxdWVzdENvbnRleHQ+IHtcbiAgcHJvdGVjdGVkIGNvbnN0cnVjdG9yKFxuICAgIHByb3RlY3RlZCByZWFkb25seSBjbGllbnRDb250ZXh0OiBEZWNhZlJlcXVlc3RDb250ZXh0LFxuICAgIF9uYW1lOiBzdHJpbmdcbiAgKSB7XG4gICAgc3VwZXIoY2xpZW50Q29udGV4dCk7XG4gIH1cbn1cblxuZXhwb3J0IGFic3RyYWN0IGNsYXNzIERlY2FmTW9kZWxDb250cm9sbGVyPFxuICBNIGV4dGVuZHMgTW9kZWw8Ym9vbGVhbj4sXG4gIEMgZXh0ZW5kcyBEZWNhZlNlcnZlckN0eCA9IERlY2FmU2VydmVyQ3R4LFxuPiBleHRlbmRzIERlY2FmQ29udHJvbGxlcjxDPiB7XG4gIHByaXZhdGUgX3BlcnNpc3RlbmNlPzogUmVwbzxNPiB8IE1vZGVsU2VydmljZTxNPjtcblxuICBwcm90ZWN0ZWQgY29uc3RydWN0b3IoXG4gICAgcHJvdGVjdGVkIG92ZXJyaWRlIHJlYWRvbmx5IGNsaWVudENvbnRleHQ6IERlY2FmUmVxdWVzdENvbnRleHQsXG4gICAgX25hbWU6IHN0cmluZ1xuICApIHtcbiAgICBzdXBlcihjbGllbnRDb250ZXh0LCBfbmFtZSk7XG4gIH1cblxuICBhYnN0cmFjdCBnZXQgY2xhc3MoKTogTW9kZWxDb25zdHJ1Y3RvcjxNPjtcblxuICBwZXJzaXN0ZW5jZShjdHg/OiBDb250ZXh0PGFueT4pOiBSZXBvPE0+IHwgTW9kZWxTZXJ2aWNlPE0+IHtcbiAgICBpZiAoIXRoaXMuX3BlcnNpc3RlbmNlKVxuICAgICAgdHJ5IHtcbiAgICAgICAgdGhpcy5fcGVyc2lzdGVuY2UgPSBTZXJ2aWNlLmdldDxNb2RlbFNlcnZpY2U8TT4+KHRoaXMuY2xhc3MpO1xuICAgICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVudXNlZC12YXJzXG4gICAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgdGhpcy5fcGVyc2lzdGVuY2UgPSBNb2RlbFNlcnZpY2UuZ2V0U2VydmljZShcbiAgICAgICAgICAgIHRoaXMuY2xhc3NcbiAgICAgICAgICApIGFzIE1vZGVsU2VydmljZTxNPjtcbiAgICAgICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVudXNlZC12YXJzXG4gICAgICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgICAgICB0aGlzLl9wZXJzaXN0ZW5jZSA9IFJlcG9zaXRvcnkuZm9yTW9kZWwodGhpcy5jbGFzcykgYXMgUmVwbzxNPjtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgY29uc3QgY2VydHMgPSAodGhpcy5jbGllbnRDb250ZXh0LnJlcXVlc3QgYXMgYW55KVtERUNBRl9BREFQVEVSX09QVElPTlNdIHx8IHt9O1xuICAgIGlmIChjdHgpIHtcbiAgICAgIHRoaXMuY2xpZW50Q29udGV4dC5wdXQoY2VydHMpO1xuICAgIH1cblxuICAgIHJldHVybiBjdHhcbiAgICAgID8gdGhpcy5fcGVyc2lzdGVuY2UgaW5zdGFuY2VvZiBSZXBvc2l0b3J5XG4gICAgICAgID8gdGhpcy5fcGVyc2lzdGVuY2Uub3ZlcnJpZGUoY2VydHMpXG4gICAgICAgIDogdGhpcy5fcGVyc2lzdGVuY2UuZm9yKGNlcnRzKVxuICAgICAgOiB0aGlzLl9wZXJzaXN0ZW5jZTtcbiAgfVxufVxuIiwiaW1wb3J0IHtcbiAgQnVsa0NydWRPcGVyYXRpb25LZXlzLFxuICBJbnRlcm5hbEVycm9yLFxuICBPcGVyYXRpb25LZXlzLFxufSBmcm9tIFwiQGRlY2FmLXRzL2RiLWRlY29yYXRvcnNcIjtcbmltcG9ydCB7IENvbnN0cnVjdG9yLCBNZXRhZGF0YSB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdGlvblwiO1xuaW1wb3J0IHsgQXBpUHJvcGVydHkgfSBmcm9tIFwiLi4vLi4vb3ZlcnJpZGVzL2RlY29yYXRpb25cIjtcbmltcG9ydCB7IE1vZGVsLCBWYWxpZGF0aW9uS2V5cyB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdG9yLXZhbGlkYXRpb25cIjtcbmltcG9ydCB7IFRyYW5zYWN0aW9uT3BlcmF0aW9uS2V5cyB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgdG9QYXNjYWxDYXNlIH0gZnJvbSBcIkBkZWNhZi10cy9sb2dnaW5nXCI7XG5pbXBvcnQgeyBERUNPUkFUT1JTIH0gZnJvbSBcIi4uLy4uL292ZXJyaWRlcy9jb25zdGFudHNcIjtcblxuY29uc3QgZHRvQ2FjaGUgPSBuZXcgTWFwPFxuICBPcGVyYXRpb25LZXlzLFxuICBXZWFrTWFwPENvbnN0cnVjdG9yPGFueT4sIENvbnN0cnVjdG9yPGFueT4+XG4+KCk7XG5cbi8qKlxuICogQnVpbGRzIGEgTmVzdC9Td2FnZ2VyIERUTyBjbGFzcyBmb3IgdGhlIGdpdmVuIG1vZGVsIGFuZCBDUlVEIG9wZXJhdGlvbi5cbiAqXG4gKiBSdWxlczpcbiAqICAtIE9ubHkgQ1JFQVRFIGFuZCBVUERBVEUgKGFuZCB0aGVpciBidWxrIHZhcmlhbnRzKSBwcm9kdWNlIGEgRFRPO1xuICogICAgYWxsIG90aGVyIG9wZXJhdGlvbnMgcmV0dXJuIHRoZSBvcmlnaW5hbCBtb2RlbCBjbGFzcyB1bmNoYW5nZWQuXG4gKiAgLSBAZ2VuZXJhdGVkKCkgcHJvcGVydGllcyAoY3JlYXRlZEF0LCB1cGRhdGVkQXQsIGNyZWF0ZWRCeSwgdXBkYXRlZEJ5LFxuICogICAgdXVpZCwgdmVyc2lvbiwgQGNvbXBvc2VkIHBrcywg4oCmKSBhcmUgKipuZXZlcioqIGV4cG9zZWQgaW4gYW55IERUTy5cbiAqICAtIFRoZSBAcGsoKSBwcm9wZXJ0eTpcbiAqICAgICAg4oCiIFVQREFURSDigJMgYWx3YXlzIGluY2x1ZGVkLlxuICogICAgICDigKIgQ1JFQVRFICDigJMgaW5jbHVkZWQgb25seSB3aGVuIHRoZSBwayBpcyBOT1QgYXV0by1nZW5lcmF0ZWRcbiAqICAgICAgICAgICAgICAgICAgKGNoZWNrZWQgdmlhIE1vZGVsLnBrUHJvcHMoKS5nZW5lcmF0ZWQgQU5EIE1vZGVsLmdlbmVyYXRlZCgpKS5cbiAqICAtIFJlbGF0aW9uIHByb3BlcnRpZXMgKEBvbmVUb09uZSwgQG9uZVRvTWFueSwg4oCmKTpcbiAqICAgICAg4oCiIENSRUFURSAg4oCTIG5lc3RlZCBhcyBEdG9Gb3IoQ1JFQVRFLCBSZWxhdGVkTW9kZWwpLiBXaGVuIHRoZSByZWxhdGlvblxuICogICAgICAgICAgICAgICAgICBjcmVhdGVzIGEgY2lyY3VsYXIgcmVmZXJlbmNlIChtb2RlbCBBIOKGkiBtb2RlbCBCIOKGkiBtb2RlbCBBKSxcbiAqICAgICAgICAgICAgICAgICAgdGhlIGJhY2stcmVmZXJlbmNlIHVzZXMgdGhlIHJlbGF0ZWQgbW9kZWwncyBQSyB0eXBlXG4gKiAgICAgICAgICAgICAgICAgIChzdHJpbmcvaW50ZWdlcikgaW5zdGVhZCBvZiB0aGUgZnVsbCBEVE8gdG8gYnJlYWsgdGhlIGN5Y2xlLlxuICogICAgICDigKIgVVBEQVRFICDigJMgdW5pb24gb2YgRHRvRm9yKFVQREFURSwgUmVsYXRlZE1vZGVsKSAqKm9yKiogdGhlXG4gKiAgICAgICAgICAgICAgICAgIHJlbGF0ZWQgbW9kZWwncyBwcmltYXJ5LWtleSB0eXBlIChzdHJpbmcgLyBpbnRlZ2VyKSxcbiAqICAgICAgICAgICAgICAgICAgZXhwcmVzc2VkIGFzIGEgU3dhZ2dlciBvbmVPZi4gQ2lyY3VsYXIgYmFjay1yZWZlcmVuY2VzXG4gKiAgICAgICAgICAgICAgICAgIG9taXQgdGhlIERUTyAkcmVmIGFuZCBvbmx5IGluY2x1ZGUgdGhlIFBLIHR5cGUuXG4gKlxuICogTWV0YWRhdGEucHJvcGVydGllcygpIG5vdyByZXR1cm5zIEFMTCBwcm9wZXJ0aWVzIGFjcm9zcyB0aGUgcHJvdG90eXBlIGNoYWluLFxuICogc28gRFRPIGluaGVyaXRhbmNlIGlzIG5vIGxvbmdlciBuZWVkZWQ7IGV2ZXJ5IERUTyBpcyBhIGZsYXQgY2xhc3MuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBEdG9Gb3I8TSBleHRlbmRzIE1vZGVsPihcbiAgb3A6IE9wZXJhdGlvbktleXMsXG4gIG1vZGVsOiBDb25zdHJ1Y3RvcjxNPixcbiAgc3RhY2s6IFNldDxDb25zdHJ1Y3Rvcjxhbnk+PiA9IG5ldyBTZXQoKVxuKTogQ29uc3RydWN0b3I8YW55PiB7XG4gIGlmICghVHJhbnNhY3Rpb25PcGVyYXRpb25LZXlzLmluY2x1ZGVzKG9wKSkge1xuICAgIHJldHVybiBtb2RlbDtcbiAgfVxuXG4gIGNvbnN0IGNhY2hlID0gZ2V0RHRvQ2FjaGUob3ApO1xuICBjb25zdCBjYWNoZWQgPSBjYWNoZS5nZXQobW9kZWwpO1xuICBpZiAoY2FjaGVkKSByZXR1cm4gY2FjaGVkO1xuXG4gIGNvbnN0IGlzVXBkYXRlT3AgPSBbXG4gICAgT3BlcmF0aW9uS2V5cy5VUERBVEUsXG4gICAgQnVsa0NydWRPcGVyYXRpb25LZXlzLlVQREFURV9BTEwsXG4gIF0uaW5jbHVkZXMob3AgYXMgYW55KTtcblxuICBjbGFzcyBEeW5hbWljRFRPIHt9XG4gIGNhY2hlLnNldChtb2RlbCwgRHluYW1pY0RUTyk7XG5cbiAgT2JqZWN0LmRlZmluZVByb3BlcnR5KER5bmFtaWNEVE8sIFwibmFtZVwiLCB7XG4gICAgdmFsdWU6IGAke3RvUGFzY2FsQ2FzZShtb2RlbC5uYW1lKX0ke3RvUGFzY2FsQ2FzZShvcCl9RFRPYCxcbiAgfSk7XG5cbiAgY29uc3QgcGtQcm9wID0gKCgpID0+IHtcbiAgICB0cnkge1xuICAgICAgcmV0dXJuIE1vZGVsLnBrKG1vZGVsKTtcbiAgICB9IGNhdGNoIHtcbiAgICAgIHJldHVybiB1bmRlZmluZWQ7XG4gICAgfVxuICB9KSgpO1xuXG4gIGNvbnN0IHBrUHJvcHNNZXRhZGF0YSA9IHBrUHJvcCA/IE1vZGVsLnBrUHJvcHMobW9kZWwgYXMgYW55KSA6IHVuZGVmaW5lZDtcbiAgY29uc3QgcGtEZXNpZ25UeXBlID0gcGtQcm9wXG4gICAgPyBSZWZsZWN0LmdldE1ldGFkYXRhKFwiZGVzaWduOnR5cGVcIiwgbW9kZWwucHJvdG90eXBlLCBwa1Byb3AgYXMgc3RyaW5nKVxuICAgIDogdW5kZWZpbmVkO1xuICBjb25zdCBwa1R5cGVJc051bWVyaWMgPVxuICAgIHBrRGVzaWduVHlwZSA9PT0gTnVtYmVyIHx8IHBrRGVzaWduVHlwZSA9PT0gQmlnSW50O1xuICBjb25zdCBwa0lzR2VuZXJhdGVkID1cbiAgICAhIXBrUHJvcHNNZXRhZGF0YT8uZ2VuZXJhdGVkIHx8XG4gICAgKHBrUHJvcFxuICAgICAgPyBNb2RlbC5nZW5lcmF0ZWRCeVNlcXVlbmNlKG1vZGVsIGFzIGFueSwgcGtQcm9wIGFzIGFueSkgfHxcbiAgICAgICAgaXNQcm9wZXJ0eUdlbmVyYXRlZEFjcm9zc0luaGVyaXRhbmNlKG1vZGVsLCBwa1Byb3AgYXMgc3RyaW5nKSB8fFxuICAgICAgICBwa1R5cGVJc051bWVyaWNcbiAgICAgIDogZmFsc2UpO1xuXG4gIGNvbnN0IGFsbFByb3BzID0gQXJyYXkuZnJvbShuZXcgU2V0KE1ldGFkYXRhLnByb3BlcnRpZXMobW9kZWwpIHx8IFtdKSk7XG4gIGNvbnN0IHJlbGF0aW9ucyA9IG5ldyBTZXQ8c3RyaW5nPigoTW9kZWwucmVsYXRpb25zKG1vZGVsKSBhcyBzdHJpbmdbXSkgfHwgW10pO1xuICBjb25zdCBzY2FsYXJQcm9wczogc3RyaW5nW10gPSBbXTtcblxuICBmb3IgKGNvbnN0IHByb3Agb2YgYWxsUHJvcHMpIHtcbiAgICBpZiAoIXByb3ApIGNvbnRpbnVlO1xuICAgIGlmIChyZWxhdGlvbnMuaGFzKHByb3ApKSBjb250aW51ZTtcblxuICAgIGlmIChwcm9wID09PSBwa1Byb3AgJiYgIWlzVXBkYXRlT3AgJiYgcGtJc0dlbmVyYXRlZCkgY29udGludWU7XG4gICAgaWYgKHByb3AgIT09IHBrUHJvcCAmJiBpc1Byb3BlcnR5R2VuZXJhdGVkQWNyb3NzSW5oZXJpdGFuY2UobW9kZWwsIHByb3ApKVxuICAgICAgY29udGludWU7XG5cbiAgICBzY2FsYXJQcm9wcy5wdXNoKHByb3ApO1xuICB9XG5cbiAgZm9yIChjb25zdCBwcm9wIG9mIHNjYWxhclByb3BzKSB7XG4gICAgY29uc3QgdmFsaWRhdGlvbiA9IGdldFZhbGlkYXRpb25BY3Jvc3NJbmhlcml0YW5jZShtb2RlbCwgcHJvcCk7XG4gICAgY29uc3QgaXNSZXF1aXJlZCA9ICEhdmFsaWRhdGlvbj8uW1ZhbGlkYXRpb25LZXlzLlJFUVVJUkVEXTtcbiAgICBjb25zdCB0eXBlSGludCA9XG4gICAgICBnZXRUeXBlQWNyb3NzSW5oZXJpdGFuY2UobW9kZWwsIHByb3ApID8/XG4gICAgICBSZWZsZWN0LmdldE1ldGFkYXRhKFwiZGVzaWduOnR5cGVcIiwgbW9kZWwucHJvdG90eXBlLCBwcm9wKTtcblxuICAgIGNvbnN0IGFwaU9wdGlvbnM6IFBhcmFtZXRlcnM8dHlwZW9mIEFwaVByb3BlcnR5PlswXSA9IHtcbiAgICAgIHJlcXVpcmVkOiBpc1JlcXVpcmVkLFxuICAgIH07XG4gICAgaWYgKHR5cGVIaW50KSBhcGlPcHRpb25zLnR5cGUgPSB0eXBlSGludDtcblxuICAgIEFwaVByb3BlcnR5KGFwaU9wdGlvbnMpKER5bmFtaWNEVE8ucHJvdG90eXBlLCBwcm9wKTtcblxuICAgIGNvbnN0IGRlc2lnblR5cGUgPVxuICAgICAgUmVmbGVjdC5nZXRNZXRhZGF0YShcImRlc2lnbjp0eXBlXCIsIG1vZGVsLnByb3RvdHlwZSwgcHJvcCkgPz8gdHlwZUhpbnQ7XG4gICAgaWYgKHR5cGVvZiBkZXNpZ25UeXBlICE9PSBcInVuZGVmaW5lZFwiKSB7XG4gICAgICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKFxuICAgICAgICBcImRlc2lnbjp0eXBlXCIsXG4gICAgICAgIGRlc2lnblR5cGUsXG4gICAgICAgIER5bmFtaWNEVE8ucHJvdG90eXBlLFxuICAgICAgICBwcm9wXG4gICAgICApO1xuICAgIH1cblxuICAgIE9iamVjdC5kZWZpbmVQcm9wZXJ0eShEeW5hbWljRFRPLnByb3RvdHlwZSwgcHJvcCwge1xuICAgICAgdmFsdWU6IHVuZGVmaW5lZCxcbiAgICAgIHdyaXRhYmxlOiB0cnVlLFxuICAgICAgZW51bWVyYWJsZTogdHJ1ZSxcbiAgICAgIGNvbmZpZ3VyYWJsZTogdHJ1ZSxcbiAgICB9KTtcbiAgfVxuXG4gIGZvciAoY29uc3QgcmVsYXRpb24gb2YgcmVsYXRpb25zKSB7XG4gICAgY29uc3QgcmVsYXRpb25NZXRhID0gTWV0YWRhdGEucmVsYXRpb25zKG1vZGVsLCByZWxhdGlvbiBhcyBhbnkpO1xuICAgIGlmICghcmVsYXRpb25NZXRhKSB7XG4gICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihgTWV0YWRhdGEgZm9yIHJlbGF0aW9uICR7cmVsYXRpb259IG5vdCBmb3VuZGApO1xuICAgIH1cblxuICAgIGxldCByZWxhdGlvblR5cGU6IENvbnN0cnVjdG9yPGFueT4gfCB1bmRlZmluZWQgPVxuICAgICAgcmVsYXRpb25NZXRhLmNsYXNzIGFzIENvbnN0cnVjdG9yPGFueT47XG4gICAgaWYgKHR5cGVvZiByZWxhdGlvblR5cGUgPT09IFwiZnVuY3Rpb25cIiAmJiAhcmVsYXRpb25UeXBlLm5hbWUpIHtcbiAgICAgIHJlbGF0aW9uVHlwZSA9IChyZWxhdGlvblR5cGUgYXMgYW55KSgpO1xuICAgIH1cbiAgICBpZiAoIXJlbGF0aW9uVHlwZSB8fCB0eXBlb2YgcmVsYXRpb25UeXBlICE9PSBcImZ1bmN0aW9uXCIpIHtcbiAgICAgIHRocm93IG5ldyBJbnRlcm5hbEVycm9yKGBUeXBlIGZvciByZWxhdGlvbiAke3JlbGF0aW9ufSBub3QgZm91bmRgKTtcbiAgICB9XG4gICAgaWYgKCFNb2RlbC5nZXQocmVsYXRpb25UeXBlLm5hbWUpKSB7XG4gICAgICBjb250aW51ZTtcbiAgICB9XG5cbiAgICBjb25zdCBtZXRhID0gTWV0YWRhdGEudmFsaWRhdGlvbkZvcihtb2RlbCwgcmVsYXRpb24gYXMgYW55KTtcbiAgICBjb25zdCBpc0FycmF5ID0gISEobWV0YSBhcyBhbnkpPy5bVmFsaWRhdGlvbktleXMuTElTVF07XG4gICAgY29uc3QgaXNSZXF1aXJlZCA9ICEhKG1ldGEgYXMgYW55KT8uW1ZhbGlkYXRpb25LZXlzLlJFUVVJUkVEXTtcblxuICAgIC8vIERldGVjdCBjaXJjdWxhciByZWZlcmVuY2U6IGlmIHRoZSByZWxhdGVkIG1vZGVsIGlzIGFscmVhZHkgaW4gdGhlXG4gICAgLy8gcmVjdXJzaW9uIHN0YWNrLCB1c2UgdGhlIFBLIHR5cGUgaW5zdGVhZCBvZiB0aGUgZnVsbCBEVE8gdG8gYnJlYWtcbiAgICAvLyB0aGUgY3ljbGUuIFRoaXMgcHJldmVudHMgQG5lc3Rqcy9zd2FnZ2VyJ3MgU2NoZW1hT2JqZWN0RmFjdG9yeSBmcm9tXG4gICAgLy8gaW5maW5pdGVseSByZWN1cnNpbmcgd2hlbiBleHBsb3JpbmcgdGhlIHNjaGVtYXMuXG4gICAgY29uc3QgaXNDaXJjdWxhciA9IHN0YWNrLmhhcyhyZWxhdGlvblR5cGUpO1xuXG4gICAgaWYgKGlzQ2lyY3VsYXIpIHtcbiAgICAgIGNvbnN0IHBrVHlwZU5hbWUgPSBnZXRQa09wZW5BcGlUeXBlKHJlbGF0aW9uVHlwZSk7XG4gICAgICBhZGRSZWxhdGlvblBrUmVmKER5bmFtaWNEVE8sIHJlbGF0aW9uLCBwa1R5cGVOYW1lLCBpc0FycmF5LCBpc1JlcXVpcmVkKTtcbiAgICAgIGNvbnRpbnVlO1xuICAgIH1cblxuICAgIGNvbnN0IHJlbGF0aW9uRHRvID0gRHRvRm9yKG9wLCByZWxhdGlvblR5cGUsIG5ldyBTZXQoc3RhY2spLmFkZChtb2RlbCkpO1xuXG4gICAgaWYgKGlzVXBkYXRlT3ApIHtcbiAgICAgIGFkZFJlbGF0aW9uVXBkYXRlKFxuICAgICAgICBEeW5hbWljRFRPLFxuICAgICAgICByZWxhdGlvbixcbiAgICAgICAgcmVsYXRpb25UeXBlLFxuICAgICAgICByZWxhdGlvbkR0byxcbiAgICAgICAgaXNBcnJheSxcbiAgICAgICAgaXNSZXF1aXJlZFxuICAgICAgKTtcbiAgICB9IGVsc2Uge1xuICAgICAgYWRkUmVsYXRpb24oRHluYW1pY0RUTywgcmVsYXRpb24sIHJlbGF0aW9uRHRvLCBpc0FycmF5LCBpc1JlcXVpcmVkKTtcbiAgICB9XG4gIH1cblxuICByZXR1cm4gRHluYW1pY0RUTztcbn1cblxuZnVuY3Rpb24gYWRkUmVsYXRpb24oXG4gIER0b0NsYXNzOiBhbnksXG4gIHJlbGF0aW9uOiBzdHJpbmcsXG4gIHJlbGF0aW9uRHRvOiBhbnksXG4gIGlzQXJyYXk6IGJvb2xlYW4sXG4gIGlzUmVxdWlyZWQ6IGJvb2xlYW5cbik6IHZvaWQge1xuICBjb25zdCBhcGlPcHRpb25zOiBQYXJhbWV0ZXJzPHR5cGVvZiBBcGlQcm9wZXJ0eT5bMF0gPSB7XG4gICAgdHlwZTogcmVsYXRpb25EdG8sXG4gICAgcmVxdWlyZWQ6IGlzUmVxdWlyZWQsXG4gICAgaXNBcnJheSxcbiAgfTtcbiAgQXBpUHJvcGVydHkoYXBpT3B0aW9ucykoRHRvQ2xhc3MucHJvdG90eXBlLCByZWxhdGlvbik7XG4gIFJlZmxlY3QuZGVmaW5lTWV0YWRhdGEoXG4gICAgXCJkZXNpZ246dHlwZVwiLFxuICAgIGlzQXJyYXkgPyBBcnJheSA6IHJlbGF0aW9uRHRvLFxuICAgIER0b0NsYXNzLnByb3RvdHlwZSxcbiAgICByZWxhdGlvblxuICApO1xuICBPYmplY3QuZGVmaW5lUHJvcGVydHkoRHRvQ2xhc3MucHJvdG90eXBlLCByZWxhdGlvbiwge1xuICAgIHZhbHVlOiB1bmRlZmluZWQsXG4gICAgd3JpdGFibGU6IHRydWUsXG4gICAgZW51bWVyYWJsZTogdHJ1ZSxcbiAgICBjb25maWd1cmFibGU6IHRydWUsXG4gIH0pO1xufVxuXG4vKipcbiAqIEFkZHMgYSByZWxhdGlvbiBwcm9wZXJ0eSB0aGF0IHVzZXMgdGhlIFBLIHR5cGUgKHN0cmluZy9pbnRlZ2VyKSBpbnN0ZWFkIG9mXG4gKiBhIGZ1bGwgRFRPIGNsYXNzLiBVc2VkIHRvIGJyZWFrIGNpcmN1bGFyIHJlZmVyZW5jZXMgKGUuZy4gQ3VzdG9tZXIg4oaSIE9yZGVyIOKGkiBDdXN0b21lcikuXG4gKi9cbmZ1bmN0aW9uIGFkZFJlbGF0aW9uUGtSZWYoXG4gIER0b0NsYXNzOiBhbnksXG4gIHJlbGF0aW9uOiBzdHJpbmcsXG4gIHBrVHlwZU5hbWU6IHN0cmluZyxcbiAgaXNBcnJheTogYm9vbGVhbixcbiAgaXNSZXF1aXJlZDogYm9vbGVhblxuKTogdm9pZCB7XG4gIGNvbnN0IGFwaU9wdGlvbnM6IFBhcmFtZXRlcnM8dHlwZW9mIEFwaVByb3BlcnR5PlswXSA9IGlzQXJyYXlcbiAgICA/ICh7IHR5cGU6IFwiYXJyYXlcIiwgaXRlbXM6IHsgdHlwZTogcGtUeXBlTmFtZSB9LCByZXF1aXJlZDogaXNSZXF1aXJlZCB9IGFzIGFueSlcbiAgICA6ICh7IHR5cGU6IHBrVHlwZU5hbWUgPT09IFwiaW50ZWdlclwiID8gTnVtYmVyIDogU3RyaW5nLCByZXF1aXJlZDogaXNSZXF1aXJlZCB9IGFzIGFueSk7XG4gIEFwaVByb3BlcnR5KGFwaU9wdGlvbnMpKER0b0NsYXNzLnByb3RvdHlwZSwgcmVsYXRpb24pO1xuICBSZWZsZWN0LmRlZmluZU1ldGFkYXRhKFxuICAgIFwiZGVzaWduOnR5cGVcIixcbiAgICBpc0FycmF5ID8gQXJyYXkgOiAocGtUeXBlTmFtZSA9PT0gXCJpbnRlZ2VyXCIgPyBOdW1iZXIgOiBTdHJpbmcpLFxuICAgIER0b0NsYXNzLnByb3RvdHlwZSxcbiAgICByZWxhdGlvblxuICApO1xuICBPYmplY3QuZGVmaW5lUHJvcGVydHkoRHRvQ2xhc3MucHJvdG90eXBlLCByZWxhdGlvbiwge1xuICAgIHZhbHVlOiB1bmRlZmluZWQsXG4gICAgd3JpdGFibGU6IHRydWUsXG4gICAgZW51bWVyYWJsZTogdHJ1ZSxcbiAgICBjb25maWd1cmFibGU6IHRydWUsXG4gIH0pO1xufVxuXG5mdW5jdGlvbiBhZGRSZWxhdGlvblVwZGF0ZShcbiAgRHRvQ2xhc3M6IGFueSxcbiAgcmVsYXRpb246IHN0cmluZyxcbiAgcmVsYXRpb25UeXBlOiBDb25zdHJ1Y3Rvcjxhbnk+LFxuICByZWxhdGlvbkR0bzogYW55LFxuICBpc0FycmF5OiBib29sZWFuLFxuICBpc1JlcXVpcmVkOiBib29sZWFuXG4pOiB2b2lkIHtcbiAgY29uc3QgZXh0cmFNb2RlbHMgPVxuICAgIFJlZmxlY3QuZ2V0TWV0YWRhdGEoREVDT1JBVE9SUy5BUElfRVhUUkFfTU9ERUxTLCBEdG9DbGFzcykgfHwgW107XG4gIGlmICghZXh0cmFNb2RlbHMuaW5jbHVkZXMocmVsYXRpb25EdG8pKSB7XG4gICAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShcbiAgICAgIERFQ09SQVRPUlMuQVBJX0VYVFJBX01PREVMUyxcbiAgICAgIFsuLi5leHRyYU1vZGVscywgcmVsYXRpb25EdG9dLFxuICAgICAgRHRvQ2xhc3NcbiAgICApO1xuICB9XG5cbiAgY29uc3QgZHRvUmVmID0gYCMvY29tcG9uZW50cy9zY2hlbWFzLyR7cmVsYXRpb25EdG8ubmFtZX1gO1xuICBjb25zdCBwa1R5cGVOYW1lID0gZ2V0UGtPcGVuQXBpVHlwZShyZWxhdGlvblR5cGUpO1xuICBjb25zdCBvbmVPZkl0ZW1zID0gW3sgJHJlZjogZHRvUmVmIH0sIHsgdHlwZTogcGtUeXBlTmFtZSB9XTtcblxuICBjb25zdCBhcGlPcHRpb25zOiBQYXJhbWV0ZXJzPHR5cGVvZiBBcGlQcm9wZXJ0eT5bMF0gPSBpc0FycmF5XG4gICAgPyAoeyB0eXBlOiBcImFycmF5XCIsIHJlcXVpcmVkOiBpc1JlcXVpcmVkLCBvbmVPZjogb25lT2ZJdGVtcyB9IGFzIGFueSlcbiAgICA6ICh7IHR5cGU6IE9iamVjdCwgcmVxdWlyZWQ6IGlzUmVxdWlyZWQsIG9uZU9mOiBvbmVPZkl0ZW1zIH0gYXMgYW55KTtcblxuICBBcGlQcm9wZXJ0eShhcGlPcHRpb25zKShEdG9DbGFzcy5wcm90b3R5cGUsIHJlbGF0aW9uKTtcbiAgUmVmbGVjdC5kZWZpbmVNZXRhZGF0YShcbiAgICBcImRlc2lnbjp0eXBlXCIsXG4gICAgaXNBcnJheSA/IEFycmF5IDogT2JqZWN0LFxuICAgIER0b0NsYXNzLnByb3RvdHlwZSxcbiAgICByZWxhdGlvblxuICApO1xuICBPYmplY3QuZGVmaW5lUHJvcGVydHkoRHRvQ2xhc3MucHJvdG90eXBlLCByZWxhdGlvbiwge1xuICAgIHZhbHVlOiB1bmRlZmluZWQsXG4gICAgd3JpdGFibGU6IHRydWUsXG4gICAgZW51bWVyYWJsZTogdHJ1ZSxcbiAgICBjb25maWd1cmFibGU6IHRydWUsXG4gIH0pO1xufVxuXG5mdW5jdGlvbiBnZXRQa09wZW5BcGlUeXBlKHJlbGF0aW9uVHlwZTogQ29uc3RydWN0b3I8YW55Pik6IHN0cmluZyB7XG4gIHRyeSB7XG4gICAgY29uc3QgcGtQcm9wc01ldGFkYXRhID0gTW9kZWwucGtQcm9wcyhyZWxhdGlvblR5cGUgYXMgYW55KTtcbiAgICBjb25zdCBwa1R5cGUgPSBwa1Byb3BzTWV0YWRhdGE/LnR5cGU7XG4gICAgaWYgKHBrVHlwZSA9PT0gTnVtYmVyIHx8IHBrVHlwZSA9PT0gQmlnSW50KSByZXR1cm4gXCJpbnRlZ2VyXCI7XG4gICAgcmV0dXJuIFwic3RyaW5nXCI7XG4gIH0gY2F0Y2gge1xuICAgIHJldHVybiBcInN0cmluZ1wiO1xuICB9XG59XG5cbmZ1bmN0aW9uIGlzUHJvcGVydHlHZW5lcmF0ZWRBY3Jvc3NJbmhlcml0YW5jZShcbiAgbW9kZWw6IENvbnN0cnVjdG9yPGFueT4sXG4gIHByb3A6IHN0cmluZ1xuKTogYm9vbGVhbiB7XG4gIGxldCBjdXJyZW50OiBhbnkgPSBtb2RlbDtcbiAgd2hpbGUgKGN1cnJlbnQgJiYgY3VycmVudCAhPT0gT2JqZWN0ICYmIGN1cnJlbnQgIT09IEZ1bmN0aW9uKSB7XG4gICAgaWYgKE1vZGVsLmdlbmVyYXRlZChjdXJyZW50LCBwcm9wIGFzIGFueSkpIHJldHVybiB0cnVlO1xuICAgIGN1cnJlbnQgPSBPYmplY3QuZ2V0UHJvdG90eXBlT2YoY3VycmVudCk7XG4gIH1cbiAgcmV0dXJuIGZhbHNlO1xufVxuXG5mdW5jdGlvbiBnZXRWYWxpZGF0aW9uQWNyb3NzSW5oZXJpdGFuY2UoXG4gIG1vZGVsOiBDb25zdHJ1Y3Rvcjxhbnk+LFxuICBwcm9wOiBzdHJpbmdcbik6IFJlY29yZDxzdHJpbmcsIGFueT4gfCB1bmRlZmluZWQge1xuICBsZXQgY3VycmVudDogYW55ID0gbW9kZWw7XG4gIHdoaWxlIChjdXJyZW50ICYmIGN1cnJlbnQgIT09IE9iamVjdCAmJiBjdXJyZW50ICE9PSBGdW5jdGlvbikge1xuICAgIGNvbnN0IHZhbGlkYXRpb24gPSBNZXRhZGF0YS52YWxpZGF0aW9uRm9yKGN1cnJlbnQsIHByb3AgYXMgYW55KTtcbiAgICBpZiAodmFsaWRhdGlvbikgcmV0dXJuIHZhbGlkYXRpb24gYXMgYW55O1xuICAgIGN1cnJlbnQgPSBPYmplY3QuZ2V0UHJvdG90eXBlT2YoY3VycmVudCk7XG4gIH1cbiAgcmV0dXJuIHVuZGVmaW5lZDtcbn1cblxuZnVuY3Rpb24gZ2V0VHlwZUFjcm9zc0luaGVyaXRhbmNlKG1vZGVsOiBDb25zdHJ1Y3Rvcjxhbnk+LCBwcm9wOiBzdHJpbmcpOiBhbnkge1xuICBsZXQgY3VycmVudDogYW55ID0gbW9kZWw7XG4gIHdoaWxlIChjdXJyZW50ICYmIGN1cnJlbnQgIT09IE9iamVjdCAmJiBjdXJyZW50ICE9PSBGdW5jdGlvbikge1xuICAgIGNvbnN0IHR5cGUgPSBNZXRhZGF0YS50eXBlKGN1cnJlbnQsIHByb3ApO1xuICAgIGlmICh0eXBlKSByZXR1cm4gdHlwZTtcbiAgICBjdXJyZW50ID0gT2JqZWN0LmdldFByb3RvdHlwZU9mKGN1cnJlbnQpO1xuICB9XG4gIHJldHVybiB1bmRlZmluZWQ7XG59XG5cbmZ1bmN0aW9uIGdldER0b0NhY2hlKFxuICBvcDogT3BlcmF0aW9uS2V5c1xuKTogV2Vha01hcDxDb25zdHJ1Y3Rvcjxhbnk+LCBDb25zdHJ1Y3Rvcjxhbnk+PiB7XG4gIGlmICghZHRvQ2FjaGUuaGFzKG9wKSkge1xuICAgIGR0b0NhY2hlLnNldChvcCwgbmV3IFdlYWtNYXAoKSk7XG4gIH1cbiAgcmV0dXJuIGR0b0NhY2hlLmdldChvcCkhO1xufVxuIiwiaW1wb3J0IHsgQ29udHJvbGxlciwgUGFyYW0sIFF1ZXJ5LCBSZXNwb25zZSB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHtcbiAgQXBpQmFkUmVxdWVzdFJlc3BvbnNlLFxuICBBcGlCb2R5LFxuICBBcGlDcmVhdGVkUmVzcG9uc2UsXG4gIEFwaUV4dHJhTW9kZWxzLFxuICBBcGlOb0NvbnRlbnRSZXNwb25zZSxcbiAgQXBpTm90Rm91bmRSZXNwb25zZSxcbiAgQXBpT2tSZXNwb25zZSxcbiAgQXBpT3BlcmF0aW9uLFxuICBBcGlQYXJhbSxcbiAgQXBpUXVlcnksXG4gIEFwaVRhZ3MsXG4gIEFwaVVucHJvY2Vzc2FibGVFbnRpdHlSZXNwb25zZSxcbiAgZ2V0U2NoZW1hUGF0aCxcbn0gZnJvbSBcIkBuZXN0anMvc3dhZ2dlclwiO1xuaW1wb3J0IHtcbiAgdHlwZSBEaXJlY3Rpb25MaW1pdE9mZnNldCxcbiAgTW9kZWxTZXJ2aWNlLFxuICBPcmRlckRpcmVjdGlvbixcbiAgUGVyc2lzdGVuY2VLZXlzLFxuICBQcmVwYXJlZFN0YXRlbWVudEtleXMsXG4gIHR5cGUgUmVwbyxcbiAgUmVwb3NpdG9yeSxcbiAgU2VydmljZSxcbn0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBNb2RlbCwgTW9kZWxDb25zdHJ1Y3RvciB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdG9yLXZhbGlkYXRpb25cIjtcbmltcG9ydCB7IExvZ2dpbmcsIHRvS2ViYWJDYXNlIH0gZnJvbSBcIkBkZWNhZi10cy9sb2dnaW5nXCI7XG5pbXBvcnQge1xuICBCdWxrQ3J1ZE9wZXJhdGlvbktleXMsXG4gIERCS2V5cyxcbiAgT3BlcmF0aW9uS2V5cyxcbiAgVmFsaWRhdGlvbkVycm9yLFxufSBmcm9tIFwiQGRlY2FmLXRzL2RiLWRlY29yYXRvcnNcIjtcbmltcG9ydCB7IENvbnN0cnVjdG9yLCBNZXRhZGF0YSB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdGlvblwiO1xuaW1wb3J0IHtcbiAgQXBpT3BlcmF0aW9uRnJvbU1vZGVsLFxuICBBcGlQYXJhbXNGcm9tTW9kZWwsXG4gIHR5cGUgRGVjYWZBcGlQcm9wZXJ0eSxcbiAgRGVjYWZCb2R5LFxuICBEZWNhZlBhcmFtcyxcbiAgRGVjYWZRdWVyeSxcbn0gZnJvbSBcIi4vZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHsgSHR0cFZlcmJUb0RlY29yYXRvciB9IGZyb20gXCIuL2RlY29yYXRvcnMvdXRpbHNcIjtcbmltcG9ydCB0eXBlIHsgSHR0cFZlcmJzIH0gZnJvbSBcIi4vZGVjb3JhdG9ycy90eXBlc1wiO1xuaW1wb3J0IHsgRGVjYWZSZXF1ZXN0Q29udGV4dCB9IGZyb20gXCIuLi9yZXF1ZXN0XCI7XG5pbXBvcnQgeyBERUNBRl9DT05UUk9MTEVSX0NPTkZJRywgREVDQUZfUk9VVEUgfSBmcm9tIFwiLi4vY29uc3RhbnRzXCI7XG5pbXBvcnQgeyBBdXRoIH0gZnJvbSBcIi4vZGVjb3JhdG9ycy9kZWNvcmF0b3JzXCI7XG5pbXBvcnQgeyBDb250cm9sbGVyQ29uc3RydWN0b3IgfSBmcm9tIFwiLi90eXBlc1wiO1xuaW1wb3J0IHsgRGVjYWZNb2RlbENvbnRyb2xsZXIgfSBmcm9tIFwiLi4vY29udHJvbGxlcnNcIjtcbmltcG9ydCB7IER0b0ZvciB9IGZyb20gXCIuLi9mYWN0b3J5L29wZW5hcGkvRHRvQnVpbGRlclwiO1xuaW1wb3J0IFwiLi4vb3ZlcnJpZGVzXCI7XG5pbXBvcnQge1xuICBNb2RlbENvbnRyb2xsZXJGYWN0b3J5LFxuICB0eXBlIE1vZGVsQ29udHJvbGxlckZhY3RvcnlDb25maWcsXG4gIHR5cGUgU2VydmVyUm91dGUsXG59IGZyb20gXCJAZGVjYWYtdHMvZm9yLWh0dHAvc2VydmVyXCI7XG5cbmV4cG9ydCBjbGFzcyBGcm9tTW9kZWxDb250cm9sbGVyIHtcbiAgcHJpdmF0ZSBzdGF0aWMgcmVhZG9ubHkgbG9nID0gTG9nZ2luZy5mb3IoRnJvbU1vZGVsQ29udHJvbGxlci5uYW1lKTtcblxuICBzdGF0aWMgZ2V0UGVyc2lzdGVuY2U8VCBleHRlbmRzIE1vZGVsPGJvb2xlYW4+PihcbiAgICBNb2RlbENsYXp6OiBNb2RlbENvbnN0cnVjdG9yPFQ+XG4gICk6IFJlcG88VD4gfCBNb2RlbFNlcnZpY2U8VD4ge1xuICAgIHRyeSB7XG4gICAgICByZXR1cm4gU2VydmljZS5nZXQ8TW9kZWxTZXJ2aWNlPFQ+PihNb2RlbENsYXp6KTtcbiAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICB0cnkge1xuICAgICAgICByZXR1cm4gTW9kZWxTZXJ2aWNlLmdldFNlcnZpY2UoTW9kZWxDbGF6eikgYXMgTW9kZWxTZXJ2aWNlPFQ+O1xuICAgICAgfSBjYXRjaCAoZTI6IHVua25vd24pIHtcbiAgICAgICAgcmV0dXJuIFJlcG9zaXRvcnkuZm9yTW9kZWwoTW9kZWxDbGF6eikgYXMgUmVwbzxUPjtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICBzdGF0aWMgY3JlYXRlUXVlcnlSb3V0ZXNGcm9tUmVwb3NpdG9yeTxUIGV4dGVuZHMgTW9kZWw8Ym9vbGVhbj4+KFxuICAgIHBlcnNpc3RlbmNlOiBSZXBvPFQ+IHwgTW9kZWxTZXJ2aWNlPFQ+LFxuICAgIHByZWZpeDogc3RyaW5nID0gUGVyc2lzdGVuY2VLZXlzLlFVRVJZXG4gICk6IENvbnRyb2xsZXJDb25zdHJ1Y3RvcjxUPiB7XG4gICAgY29uc3QgcmVwbzogUmVwbzxUPiA9XG4gICAgICBwZXJzaXN0ZW5jZSBpbnN0YW5jZW9mIE1vZGVsU2VydmljZSA/IHBlcnNpc3RlbmNlLnJlcG8gOiBwZXJzaXN0ZW5jZTtcbiAgICBjb25zdCBNb2RlbENvbnN0cjogQ29uc3RydWN0b3IgPSByZXBvLmNsYXNzO1xuICAgIGNvbnN0IHF1ZXJ5TWV0aG9kczogUmVjb3JkPHN0cmluZywgeyBmaWVsZHM/OiBzdHJpbmdbXSB8IHVuZGVmaW5lZCB9PiA9XG4gICAgICBNZXRhZGF0YS5nZXQoXG4gICAgICAgIHJlcG8uY29uc3RydWN0b3IgYXMgQ29uc3RydWN0b3IsXG4gICAgICAgIE1ldGFkYXRhLmtleShQZXJzaXN0ZW5jZUtleXMuUVVFUlkpXG4gICAgICApID8/IHt9O1xuXG4gICAgY29uc3Qgcm91dGVNZXRob2RzOiBSZWNvcmQ8c3RyaW5nLCBhbnk+ID1cbiAgICAgIE1ldGFkYXRhLmdldChcbiAgICAgICAgcGVyc2lzdGVuY2UuY29uc3RydWN0b3IgYXMgQ29uc3RydWN0b3IsXG4gICAgICAgIE1ldGFkYXRhLmtleShERUNBRl9ST1VURSlcbiAgICAgICkgPz8ge307XG5cbiAgICBjbGFzcyBRdWVyeUNvbnRyb2xsZXIgZXh0ZW5kcyBEZWNhZk1vZGVsQ29udHJvbGxlcjxUPiB7XG4gICAgICBvdmVycmlkZSBnZXQgY2xhc3MoKTogTW9kZWxDb25zdHJ1Y3RvcjxUPiB7XG4gICAgICAgIHRocm93IG5ldyBFcnJvcihcIk1ldGhvZCBub3QgaW1wbGVtZW50ZWQuXCIpO1xuICAgICAgfVxuICAgICAgY29uc3RydWN0b3IoY2xpZW50Q29udGV4dDogRGVjYWZSZXF1ZXN0Q29udGV4dCwgbmFtZTogc3RyaW5nKSB7XG4gICAgICAgIHN1cGVyKGNsaWVudENvbnRleHQsIG5hbWUpO1xuICAgICAgfVxuICAgIH1cblxuICAgIGZvciAoY29uc3QgW21ldGhvZE5hbWUsIHBhcmFtc10gb2YgT2JqZWN0LmVudHJpZXMocm91dGVNZXRob2RzKSkge1xuICAgICAgY29uc3Qgcm91dGVQYXRoID0gW3BhcmFtcy5wYXRoLnJlcGxhY2UoL15cXC8rfFxcLyskL2csIFwiXCIpXVxuICAgICAgICAuZmlsdGVyKChzZWdtZW50OiBzdHJpbmcpID0+IHNlZ21lbnQgJiYgc2VnbWVudC50cmltKCkpXG4gICAgICAgIC5qb2luKFwiL1wiKTtcblxuICAgICAgY29uc3QgaGFuZGxlciA9IEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlQ29tcGxleFF1ZXJ5SGFuZGxlcihcbiAgICAgICAgbWV0aG9kTmFtZVxuICAgICAgKSBhcyBhbnk7XG4gICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmRlZmluZU1ldGhvZChcbiAgICAgICAgUXVlcnlDb250cm9sbGVyLFxuICAgICAgICBtZXRob2ROYW1lLFxuICAgICAgICBoYW5kbGVyXG4gICAgICApO1xuXG4gICAgICBjb25zdCBodHRwRGVjb3JhdG9yID0gSHR0cFZlcmJUb0RlY29yYXRvcihwYXJhbXMuaHR0cE1ldGhvZCBhcyBIdHRwVmVyYnMpKHJvdXRlUGF0aCB8fCB1bmRlZmluZWQpO1xuICAgICAgY29uc3QgZGVjb3JhdG9ycyA9IEZyb21Nb2RlbENvbnRyb2xsZXIuZ2V0UXVlcnlEZWNvcmF0b3JzKFxuICAgICAgICBtZXRob2ROYW1lLFxuICAgICAgICByb3V0ZVBhdGgsXG4gICAgICAgIHBhcmFtcy5odHRwTWV0aG9kXG4gICAgICApO1xuICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5hcHBseURlY29yYXRvcnMoXG4gICAgICAgIFF1ZXJ5Q29udHJvbGxlcixcbiAgICAgICAgbWV0aG9kTmFtZSxcbiAgICAgICAgW2h0dHBEZWNvcmF0b3IsIC4uLmRlY29yYXRvcnNdXG4gICAgICApO1xuICAgIH1cblxuICAgIGZvciAoY29uc3QgW21ldGhvZE5hbWUsIG9ialZhbHVlc10gb2YgT2JqZWN0LmVudHJpZXMocXVlcnlNZXRob2RzKSkge1xuICAgICAgY29uc3QgZmllbGRzID0gb2JqVmFsdWVzLmZpZWxkcyA/PyBbXTtcbiAgICAgIGNvbnN0IHJvdXRlUGF0aCA9IFtwcmVmaXgsIG1ldGhvZE5hbWUsIC4uLmZpZWxkcy5tYXAoKGYpID0+IGA6JHtmfWApXVxuICAgICAgICAuZmlsdGVyKChzZWdtZW50KSA9PiBzZWdtZW50ICYmIHNlZ21lbnQudHJpbSgpKVxuICAgICAgICAuam9pbihcIi9cIik7XG5cbiAgICAgIGNvbnN0IGhhbmRsZXIgPSBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZUNvbXBsZXhRdWVyeUhhbmRsZXIoXG4gICAgICAgIG1ldGhvZE5hbWVcbiAgICAgICkgYXMgYW55O1xuICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5kZWZpbmVNZXRob2QoXG4gICAgICAgIFF1ZXJ5Q29udHJvbGxlcixcbiAgICAgICAgbWV0aG9kTmFtZSxcbiAgICAgICAgaGFuZGxlclxuICAgICAgKTtcblxuICAgICAgY29uc3QgaHR0cERlY29yYXRvciA9IEh0dHBWZXJiVG9EZWNvcmF0b3IoXCJHRVRcIiBhcyBIdHRwVmVyYnMpKHJvdXRlUGF0aCB8fCB1bmRlZmluZWQpO1xuICAgICAgY29uc3QgZGVjb3JhdG9ycyA9IEZyb21Nb2RlbENvbnRyb2xsZXIuZ2V0UXVlcnlEZWNvcmF0b3JzKFxuICAgICAgICBtZXRob2ROYW1lLFxuICAgICAgICByb3V0ZVBhdGgsXG4gICAgICAgIFwiR0VUXCIsXG4gICAgICAgIHRydWVcbiAgICAgICk7XG4gICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmFwcGx5RGVjb3JhdG9ycyhcbiAgICAgICAgUXVlcnlDb250cm9sbGVyLFxuICAgICAgICBtZXRob2ROYW1lLFxuICAgICAgICBbaHR0cERlY29yYXRvciwgLi4uZGVjb3JhdG9yc11cbiAgICAgICk7XG4gICAgfVxuXG4gICAgcmV0dXJuIFF1ZXJ5Q29udHJvbGxlcjtcbiAgfVxuXG4gIHN0YXRpYyBjcmVhdGU8VCBleHRlbmRzIE1vZGVsPGFueT4+KFxuICAgIE1vZGVsQ29uc3RyOiBNb2RlbENvbnN0cnVjdG9yPFQ+LFxuICAgIG1vZHVsZUNvbmZpZ092ZXJyaWRlcz86IFJlY29yZDxzdHJpbmcsIE1vZGVsQ29udHJvbGxlckZhY3RvcnlDb25maWc+XG4gICk6IENvbnRyb2xsZXJDb25zdHJ1Y3RvcjxUPiB7XG4gICAgY29uc3QgbG9nID0gRnJvbU1vZGVsQ29udHJvbGxlci5sb2cuZm9yKEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlKTtcbiAgICBjb25zdCB0YWJsZU5hbWUgPSBNb2RlbC50YWJsZU5hbWUoTW9kZWxDb25zdHIpO1xuICAgIGNvbnN0IHJvdXRlUGF0aCA9IHRvS2ViYWJDYXNlKHRhYmxlTmFtZSk7XG4gICAgY29uc3QgbW9kZWxDbGF6ek5hbWUgPSBNb2RlbENvbnN0ci5uYW1lO1xuICAgIGNvbnN0IHBlcnNpc3RlbmNlID0gRnJvbU1vZGVsQ29udHJvbGxlci5nZXRQZXJzaXN0ZW5jZShNb2RlbENvbnN0cik7XG5cbiAgICAvLyBXaGVuIHBlcnNpc3RlbmNlIGlzIGEgTW9kZWxTZXJ2aWNlLCB0aGUgQHF1ZXJ5L0Byb3V0ZSBtZXRhZGF0YSBsaXZlcyBvblxuICAgIC8vIHRoZSB1bmRlcmx5aW5nIHJlcG9zaXRvcnkgY2xhc3MgKGN1c3RvbSByZXBvKSwgbm90IG9uIE1vZGVsU2VydmljZSBpdHNlbGYuXG4gICAgLy8gUGFzcyB0aGUgcmVwbyB0byB0aGUgZmFjdG9yeSBzbyBhZGRDb21wbGV4UXVlcmllcygpIGNhbiBkaXNjb3ZlciB0aGVtLlxuICAgIGNvbnN0IGZhY3RvcnlQZXJzaXN0ZW5jZSA9XG4gICAgICBwZXJzaXN0ZW5jZSBpbnN0YW5jZW9mIE1vZGVsU2VydmljZSA/IHBlcnNpc3RlbmNlLnJlcG8gOiBwZXJzaXN0ZW5jZTtcblxuICAgIGNvbnN0IGRlY29yYXRvckNvbmZpZyA9IE1ldGFkYXRhLmdldChcbiAgICAgIE1vZGVsQ29uc3RyLFxuICAgICAgTWV0YWRhdGEua2V5KERFQ0FGX0NPTlRST0xMRVJfQ09ORklHKVxuICAgICkgYXMgTW9kZWxDb250cm9sbGVyRmFjdG9yeUNvbmZpZyB8IHVuZGVmaW5lZDtcbiAgICBjb25zdCBtb2R1bGVPdmVycmlkZSA9IG1vZHVsZUNvbmZpZ092ZXJyaWRlcz8uW01vZGVsQ29uc3RyLm5hbWVdO1xuICAgIGNvbnN0IG1lcmdlZENvbmZpZzogTW9kZWxDb250cm9sbGVyRmFjdG9yeUNvbmZpZyA9IHtcbiAgICAgIC4uLihkZWNvcmF0b3JDb25maWcgfHwge30pLFxuICAgICAgLi4uKG1vZHVsZU92ZXJyaWRlIHx8IHt9KSxcbiAgICB9O1xuXG4gICAgY29uc3QgRmFjdG9yeUNvbnRyb2xsZXIgPSBNb2RlbENvbnRyb2xsZXJGYWN0b3J5LmNyZWF0ZTxUPihcbiAgICAgIE1vZGVsQ29uc3RyLFxuICAgICAgZmFjdG9yeVBlcnNpc3RlbmNlLFxuICAgICAgbWVyZ2VkQ29uZmlnXG4gICAgKTtcbiAgICBjb25zdCBmYWN0b3J5Um91dGVzID0gKEZhY3RvcnlDb250cm9sbGVyIGFzIGFueSkuX19yb3V0ZXNfXyBhc1xuICAgICAgfCBTZXJ2ZXJSb3V0ZVtdXG4gICAgICB8IHVuZGVmaW5lZDtcblxuICAgIGNvbnN0IHsgZ2V0UEssIGFwaVByb3BlcnRpZXMsIHBhdGg6IHBrUGF0aCB9ID1cbiAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuZ2V0Um91dGVQYXJhbWV0ZXJzRnJvbU1vZGVsKE1vZGVsQ29uc3RyKTtcblxuICAgIGxvZy5kZWJ1ZyhcbiAgICAgIGBDcmVhdGluZyBjb250cm9sbGVyIGZvciBtb2RlbDogJHttb2RlbENsYXp6TmFtZX0gd2l0aCAke2ZhY3RvcnlSb3V0ZXM/Lmxlbmd0aCA/PyAwfSBmYWN0b3J5IHJvdXRlc2BcbiAgICApO1xuXG4gICAgQENvbnRyb2xsZXIocm91dGVQYXRoKVxuICAgIEBBcGlUYWdzKG1vZGVsQ2xhenpOYW1lKVxuICAgIEBBcGlFeHRyYU1vZGVscyhNb2RlbENvbnN0cilcbiAgICBAQXV0aChNb2RlbENvbnN0cilcbiAgICBjbGFzcyBEeW5hbWljTW9kZWxDb250cm9sbGVyIGV4dGVuZHMgRGVjYWZNb2RlbENvbnRyb2xsZXI8VD4ge1xuICAgICAgcHJpdmF0ZSByZWFkb25seSBwazogc3RyaW5nID0gTW9kZWwucGsoTW9kZWxDb25zdHIpIGFzIHN0cmluZztcblxuICAgICAgcHJvdGVjdGVkIHN0YXRpYyBnZXQgY2xhc3MoKSB7XG4gICAgICAgIHJldHVybiBNb2RlbENvbnN0cjtcbiAgICAgIH1cblxuICAgICAgb3ZlcnJpZGUgZ2V0IGNsYXNzKCk6IE1vZGVsQ29uc3RydWN0b3I8VD4ge1xuICAgICAgICByZXR1cm4gTW9kZWxDb25zdHI7XG4gICAgICB9XG5cbiAgICAgIGNvbnN0cnVjdG9yKGNsaWVudENvbnRleHQ6IERlY2FmUmVxdWVzdENvbnRleHQpIHtcbiAgICAgICAgc3VwZXIoY2xpZW50Q29udGV4dCwgRHluYW1pY01vZGVsQ29udHJvbGxlci5uYW1lKTtcbiAgICAgICAgbG9nLmluZm8oXG4gICAgICAgICAgYFJlZ2lzdGVyaW5nIGR5bmFtaWMgY29udHJvbGxlciBmb3IgbW9kZWw6ICR7dGhpcy5jbGFzcy5uYW1lfSByb3V0ZTogLyR7cm91dGVQYXRofWBcbiAgICAgICAgKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICBpZiAoZmFjdG9yeVJvdXRlcykge1xuICAgICAgY29uc3Qgc29ydGVkUm91dGVzID0gWy4uLmZhY3RvcnlSb3V0ZXNdLnNvcnQoKGEsIGIpID0+IHtcbiAgICAgICAgY29uc3QgYVNlZ21lbnRzID0gYS5wYXRoLnNwbGl0KFwiL1wiKS5maWx0ZXIoQm9vbGVhbik7XG4gICAgICAgIGNvbnN0IGJTZWdtZW50cyA9IGIucGF0aC5zcGxpdChcIi9cIikuZmlsdGVyKEJvb2xlYW4pO1xuICAgICAgICBjb25zdCBhUGFyYW1Db3VudCA9IGFTZWdtZW50cy5maWx0ZXIoKHMpID0+IHMuc3RhcnRzV2l0aChcIjpcIikpLmxlbmd0aDtcbiAgICAgICAgY29uc3QgYlBhcmFtQ291bnQgPSBiU2VnbWVudHMuZmlsdGVyKChzKSA9PiBzLnN0YXJ0c1dpdGgoXCI6XCIpKS5sZW5ndGg7XG4gICAgICAgIGNvbnN0IGFMaXRlcmFsQ291bnQgPSBhU2VnbWVudHMubGVuZ3RoIC0gYVBhcmFtQ291bnQ7XG4gICAgICAgIGNvbnN0IGJMaXRlcmFsQ291bnQgPSBiU2VnbWVudHMubGVuZ3RoIC0gYlBhcmFtQ291bnQ7XG4gICAgICAgIGlmIChhTGl0ZXJhbENvdW50ICE9PSBiTGl0ZXJhbENvdW50KVxuICAgICAgICAgIHJldHVybiBiTGl0ZXJhbENvdW50IC0gYUxpdGVyYWxDb3VudDtcbiAgICAgICAgaWYgKGFQYXJhbUNvdW50ICE9PSBiUGFyYW1Db3VudClcbiAgICAgICAgICByZXR1cm4gYVBhcmFtQ291bnQgLSBiUGFyYW1Db3VudDtcbiAgICAgICAgcmV0dXJuIDA7XG4gICAgICB9KTtcbiAgICAgIGZvciAoY29uc3Qgcm91dGUgb2Ygc29ydGVkUm91dGVzKSB7XG4gICAgICAgIGNvbnN0IHJlZ2lzdHJhdGlvbiA9IEZyb21Nb2RlbENvbnRyb2xsZXIubWF0Y2hSb3V0ZShcbiAgICAgICAgICByb3V0ZSxcbiAgICAgICAgICBwa1BhdGgsXG4gICAgICAgICAgYXBpUHJvcGVydGllcyxcbiAgICAgICAgICBnZXRQSyxcbiAgICAgICAgICBNb2RlbENvbnN0cixcbiAgICAgICAgICBtb2RlbENsYXp6TmFtZSxcbiAgICAgICAgICBmYWN0b3J5UGVyc2lzdGVuY2VcbiAgICAgICAgKTtcbiAgICAgICAgaWYgKCFyZWdpc3RyYXRpb24pIGNvbnRpbnVlO1xuXG4gICAgICAgIGNvbnN0IHsgbWV0aG9kTmFtZSwgaGFuZGxlciwgZGVjb3JhdG9ycywgcGFyYW1EZWNvcmF0b3JzIH0gPVxuICAgICAgICAgIHJlZ2lzdHJhdGlvbjtcblxuICAgICAgICBjb25zdCBkZXNjcmlwdG9yID0gRnJvbU1vZGVsQ29udHJvbGxlci5kZWZpbmVNZXRob2QoXG4gICAgICAgICAgRHluYW1pY01vZGVsQ29udHJvbGxlcixcbiAgICAgICAgICBtZXRob2ROYW1lLFxuICAgICAgICAgIGhhbmRsZXJcbiAgICAgICAgKTtcblxuICAgICAgICBpZiAoZGVzY3JpcHRvcikge1xuICAgICAgICAgIGNvbnN0IGh0dHBEZWNvcmF0b3IgPSBIdHRwVmVyYlRvRGVjb3JhdG9yKHJvdXRlLm1ldGhvZCBhcyBIdHRwVmVyYnMpKFxuICAgICAgICAgICAgcm91dGUucGF0aC5yZXBsYWNlKC9eXFwvK3xcXC8rJC9nLCBcIlwiKSB8fCB1bmRlZmluZWRcbiAgICAgICAgICApO1xuICAgICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuYXBwbHlEZWNvcmF0b3JzKFxuICAgICAgICAgICAgRHluYW1pY01vZGVsQ29udHJvbGxlcixcbiAgICAgICAgICAgIG1ldGhvZE5hbWUsXG4gICAgICAgICAgICBbaHR0cERlY29yYXRvciwgLi4uZGVjb3JhdG9yc10sXG4gICAgICAgICAgICBwYXJhbURlY29yYXRvcnNcbiAgICAgICAgICApO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuXG4gICAgcmV0dXJuIER5bmFtaWNNb2RlbENvbnRyb2xsZXIgYXMgYW55O1xuICB9XG5cbiAgc3RhdGljIGdldFJvdXRlUGFyYW1ldGVyc0Zyb21Nb2RlbDxUIGV4dGVuZHMgTW9kZWw8YW55Pj4oXG4gICAgTW9kZWxDbGF6ejogTW9kZWxDb25zdHJ1Y3RvcjxUPlxuICApOiB7XG4gICAgcGF0aDogc3RyaW5nO1xuICAgIGRlc2NyaXB0aW9uOiBzdHJpbmc7XG4gICAgYXBpUHJvcGVydGllczogRGVjYWZBcGlQcm9wZXJ0eVtdO1xuICAgIGdldFBLOiAoLi4ucGFyYW1zOiBBcnJheTxzdHJpbmcgfCBudW1iZXI+KSA9PiBzdHJpbmc7XG4gIH0ge1xuICAgIGNvbnN0IHBrID0gTW9kZWwucGsoTW9kZWxDbGF6eikgYXMga2V5b2YgTW9kZWw8YW55PjtcbiAgICBjb25zdCBjb21wb3NlZCA9IE1ldGFkYXRhLmdldChcbiAgICAgIE1vZGVsQ2xhenosXG4gICAgICBNZXRhZGF0YS5rZXkoREJLZXlzLkNPTVBPU0VELCBwaylcbiAgICApO1xuICAgIGNvbnN0IGNvbXBvc2VkS2V5cyA9IGNvbXBvc2VkPy5hcmdzID8/IFtdO1xuXG4gICAgY29uc3QgdW5pcXVlS2V5cyA9XG4gICAgICBBcnJheS5pc0FycmF5KGNvbXBvc2VkS2V5cykgJiYgY29tcG9zZWRLZXlzLmxlbmd0aCA+IDBcbiAgICAgICAgPyBBcnJheS5mcm9tKG5ldyBTZXQoWy4uLmNvbXBvc2VkS2V5c10pKVxuICAgICAgICA6IEFycmF5LmZyb20obmV3IFNldChbcGtdKSk7XG5cbiAgICBjb25zdCBkZXNjcmlwdGlvbiA9IE1ldGFkYXRhLmRlc2NyaXB0aW9uKE1vZGVsQ2xhenopID8/IFwiXCI7XG4gICAgY29uc3QgcGF0aCA9IGA6JHt1bmlxdWVLZXlzLmpvaW4oXCIvOlwiKX1gO1xuICAgIGNvbnN0IGFwaVByb3BlcnRpZXM6IERlY2FmQXBpUHJvcGVydHlbXSA9IHVuaXF1ZUtleXMubWFwKChrZXkpID0+IHtcbiAgICAgIHJldHVybiB7XG4gICAgICAgIG5hbWU6IGtleSxcbiAgICAgICAgZGVzY3JpcHRpb246IE1ldGFkYXRhLmRlc2NyaXB0aW9uKE1vZGVsQ2xhenosIGtleSksXG4gICAgICAgIHJlcXVpcmVkOiB0cnVlLFxuICAgICAgICB0eXBlOiBTdHJpbmcsXG4gICAgICB9O1xuICAgIH0pO1xuXG4gICAgcmV0dXJuIHtcbiAgICAgIHBhdGgsXG4gICAgICBkZXNjcmlwdGlvbixcbiAgICAgIGFwaVByb3BlcnRpZXMsXG4gICAgICBnZXRQSzogKC4uLnBhcmFtczogQXJyYXk8c3RyaW5nIHwgbnVtYmVyPikgPT5cbiAgICAgICAgY29tcG9zZWQ/LnNlcGFyYXRvciA/IHBhcmFtcy5qb2luKGNvbXBvc2VkLnNlcGFyYXRvcikgOiBwYXJhbXMuam9pbihcIlwiKSxcbiAgICB9O1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgZGVmaW5lTWV0aG9kKFxuICAgIHRhcmdldDogYW55LFxuICAgIG1ldGhvZE5hbWU6IHN0cmluZyxcbiAgICBoYW5kbGVyOiAoLi4uYXJnczogYW55W10pID0+IGFueVxuICApOiBQcm9wZXJ0eURlc2NyaXB0b3IgfCB1bmRlZmluZWQge1xuICAgIE9iamVjdC5kZWZpbmVQcm9wZXJ0eShcbiAgICAgIHRhcmdldC5wcm90b3R5cGUgfHwgdGFyZ2V0LFxuICAgICAgbWV0aG9kTmFtZSxcbiAgICAgIHtcbiAgICAgICAgdmFsdWU6IGhhbmRsZXIsXG4gICAgICAgIHdyaXRhYmxlOiBmYWxzZSxcbiAgICAgICAgY29uZmlndXJhYmxlOiB0cnVlLFxuICAgICAgICBlbnVtZXJhYmxlOiBmYWxzZSxcbiAgICAgIH1cbiAgICApO1xuXG4gICAgcmV0dXJuIE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3IoXG4gICAgICB0YXJnZXQucHJvdG90eXBlIHx8IHRhcmdldCxcbiAgICAgIG1ldGhvZE5hbWVcbiAgICApO1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgYXBwbHlEZWNvcmF0b3JzKFxuICAgIHRhcmdldDogYW55LFxuICAgIG1ldGhvZE5hbWU6IHN0cmluZyxcbiAgICBtZXRob2REZWNvcmF0b3JzOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+LFxuICAgIHBhcmFtRGVjb3JhdG9yczogQXJyYXk8eyBkZWNvcmF0b3I6IFBhcmFtZXRlckRlY29yYXRvcjsgaW5kZXg6IG51bWJlciB9PiA9IFtdXG4gICkge1xuICAgIGNvbnN0IHByb3RvID0gdGFyZ2V0Py5wcm90b3R5cGUgPz8gdGFyZ2V0O1xuICAgIGNvbnN0IGRlc2NyaXB0b3IgPSBPYmplY3QuZ2V0T3duUHJvcGVydHlEZXNjcmlwdG9yKHByb3RvLCBtZXRob2ROYW1lKTtcbiAgICBtZXRob2REZWNvcmF0b3JzLmZvckVhY2goKGQpID0+IGQocHJvdG8sIG1ldGhvZE5hbWUsIGRlc2NyaXB0b3IpKTtcbiAgICBwYXJhbURlY29yYXRvcnMuZm9yRWFjaCgoeyBkZWNvcmF0b3IsIGluZGV4IH0pID0+XG4gICAgICBkZWNvcmF0b3IocHJvdG8sIG1ldGhvZE5hbWUsIGluZGV4KVxuICAgICk7XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBtYXRjaFJvdXRlKFxuICAgIHJvdXRlOiBTZXJ2ZXJSb3V0ZSxcbiAgICBwa1BhdGg6IHN0cmluZyxcbiAgICBhcGlQcm9wZXJ0aWVzOiBEZWNhZkFwaVByb3BlcnR5W10sXG4gICAgZ2V0UEs6ICguLi5wOiBBcnJheTxzdHJpbmcgfCBudW1iZXI+KSA9PiBzdHJpbmcsXG4gICAgTW9kZWxDb25zdHI6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nLFxuICAgIHBlcnNpc3RlbmNlPzogYW55XG4gICk6IHtcbiAgICBtZXRob2ROYW1lOiBzdHJpbmc7XG4gICAgaGFuZGxlcjogKC4uLmFyZ3M6IGFueVtdKSA9PiBhbnk7XG4gICAgZGVjb3JhdG9yczogQXJyYXk8KHRhcmdldDogYW55LCBrZXk6IHN0cmluZywgZGVzYzogYW55KSA9PiB2b2lkPjtcbiAgICBwYXJhbURlY29yYXRvcnM6IEFycmF5PHsgZGVjb3JhdG9yOiBQYXJhbWV0ZXJEZWNvcmF0b3I7IGluZGV4OiBudW1iZXIgfT47XG4gIH0gfCB1bmRlZmluZWQge1xuICAgIGNvbnN0IHsgbWV0aG9kLCBwYXRoIH0gPSByb3V0ZTtcbiAgICBjb25zdCBub3JtYWxpemVkUGF0aCA9IHBhdGgucmVwbGFjZSgvXlxcLyt8XFwvKyQvZywgXCJcIik7XG5cbiAgICBpZiAobWV0aG9kID09PSBcIlBPU1RcIiAmJiBub3JtYWxpemVkUGF0aCA9PT0gXCJcIikge1xuICAgICAgcmV0dXJuIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlUmVnaXN0cmF0aW9uKFxuICAgICAgICBcImNyZWF0ZVwiLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZUNyZWF0ZUhhbmRsZXIoTW9kZWxDb25zdHIsIG1vZGVsQ2xhenpOYW1lKSxcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVDcmVhdGVEZWNvcmF0b3JzKE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSksXG4gICAgICAgIFt7IGRlY29yYXRvcjogRGVjYWZCb2R5KCkgYXMgYW55LCBpbmRleDogMCB9LCB7IGRlY29yYXRvcjogUmVzcG9uc2UoeyBwYXNzdGhyb3VnaDogdHJ1ZSB9KSBhcyBhbnksIGluZGV4OiAxIH1dXG4gICAgICApO1xuICAgIH1cblxuICAgIGlmIChtZXRob2QgPT09IFwiUE9TVFwiICYmIG5vcm1hbGl6ZWRQYXRoID09PSBcImJ1bGtcIikge1xuICAgICAgcmV0dXJuIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlUmVnaXN0cmF0aW9uKFxuICAgICAgICBcImNyZWF0ZUFsbFwiLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZUJ1bGtDcmVhdGVIYW5kbGVyKE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSksXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuYnVsa0NyZWF0ZURlY29yYXRvcnMoTW9kZWxDb25zdHIsIG1vZGVsQ2xhenpOYW1lKSxcbiAgICAgICAgW3sgZGVjb3JhdG9yOiBEZWNhZkJvZHkoKSBhcyBhbnksIGluZGV4OiAwIH0sIHsgZGVjb3JhdG9yOiBSZXNwb25zZSh7IHBhc3N0aHJvdWdoOiB0cnVlIH0pIGFzIGFueSwgaW5kZXg6IDEgfV1cbiAgICAgICk7XG4gICAgfVxuXG4gICAgaWYgKG1ldGhvZCA9PT0gXCJHRVRcIiAmJiBub3JtYWxpemVkUGF0aCA9PT0gXCJidWxrXCIpIHtcbiAgICAgIHJldHVybiBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZVJlZ2lzdHJhdGlvbihcbiAgICAgICAgXCJyZWFkQWxsXCIsXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlQnVsa1JlYWRIYW5kbGVyKG1vZGVsQ2xhenpOYW1lKSxcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5idWxrUmVhZERlY29yYXRvcnMoTW9kZWxDb25zdHIsIG1vZGVsQ2xhenpOYW1lKSxcbiAgICAgICAgW3sgZGVjb3JhdG9yOiBRdWVyeShcImlkc1wiKSBhcyBhbnksIGluZGV4OiAwIH1dXG4gICAgICApO1xuICAgIH1cblxuICAgIGlmIChtZXRob2QgPT09IFwiUFVUXCIgJiYgbm9ybWFsaXplZFBhdGggPT09IFwiYnVsa1wiKSB7XG4gICAgICByZXR1cm4gRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVSZWdpc3RyYXRpb24oXG4gICAgICAgIFwidXBkYXRlQWxsXCIsXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlQnVsa1VwZGF0ZUhhbmRsZXIobW9kZWxDbGF6ek5hbWUpLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmJ1bGtVcGRhdGVEZWNvcmF0b3JzKE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSwgYXBpUHJvcGVydGllcyksXG4gICAgICAgIFt7IGRlY29yYXRvcjogRGVjYWZCb2R5KCkgYXMgYW55LCBpbmRleDogMCB9LCB7IGRlY29yYXRvcjogUmVzcG9uc2UoeyBwYXNzdGhyb3VnaDogdHJ1ZSB9KSBhcyBhbnksIGluZGV4OiAxIH1dXG4gICAgICApO1xuICAgIH1cblxuICAgIGlmIChtZXRob2QgPT09IFwiREVMRVRFXCIgJiYgbm9ybWFsaXplZFBhdGggPT09IFwiYnVsa1wiKSB7XG4gICAgICByZXR1cm4gRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVSZWdpc3RyYXRpb24oXG4gICAgICAgIFwiZGVsZXRlQWxsXCIsXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlQnVsa0RlbGV0ZUhhbmRsZXIobW9kZWxDbGF6ek5hbWUpLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmJ1bGtEZWxldGVEZWNvcmF0b3JzKE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSwgYXBpUHJvcGVydGllcyksXG4gICAgICAgIFt7IGRlY29yYXRvcjogUXVlcnkoXCJpZHNcIikgYXMgYW55LCBpbmRleDogMCB9LCB7IGRlY29yYXRvcjogUmVzcG9uc2UoeyBwYXNzdGhyb3VnaDogdHJ1ZSB9KSBhcyBhbnksIGluZGV4OiAxIH1dXG4gICAgICApO1xuICAgIH1cblxuICAgIGlmIChtZXRob2QgPT09IFwiR0VUXCIgJiYgbm9ybWFsaXplZFBhdGggPT09IHBrUGF0aCkge1xuICAgICAgcmV0dXJuIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlUmVnaXN0cmF0aW9uKFxuICAgICAgICBcInJlYWRcIixcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVSZWFkSGFuZGxlcihnZXRQSywgbW9kZWxDbGF6ek5hbWUpLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLnJlYWREZWNvcmF0b3JzKE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSwgYXBpUHJvcGVydGllcywgcGtQYXRoKSxcbiAgICAgICAgW3sgZGVjb3JhdG9yOiBEZWNhZlBhcmFtcyhhcGlQcm9wZXJ0aWVzKSBhcyBhbnksIGluZGV4OiAwIH1dXG4gICAgICApO1xuICAgIH1cblxuICAgIGlmIChtZXRob2QgPT09IFwiUFVUXCIgJiYgbm9ybWFsaXplZFBhdGggPT09IHBrUGF0aCkge1xuICAgICAgcmV0dXJuIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlUmVnaXN0cmF0aW9uKFxuICAgICAgICBcInVwZGF0ZVwiLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZVVwZGF0ZUhhbmRsZXIoZ2V0UEssIE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSksXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIudXBkYXRlRGVjb3JhdG9ycyhNb2RlbENvbnN0ciwgbW9kZWxDbGF6ek5hbWUsIGFwaVByb3BlcnRpZXMsIHBrUGF0aCksXG4gICAgICAgIFtcbiAgICAgICAgICB7IGRlY29yYXRvcjogRGVjYWZQYXJhbXMoYXBpUHJvcGVydGllcykgYXMgYW55LCBpbmRleDogMCB9LFxuICAgICAgICAgIHsgZGVjb3JhdG9yOiBEZWNhZkJvZHkoKSBhcyBhbnksIGluZGV4OiAxIH0sXG4gICAgICAgICAgeyBkZWNvcmF0b3I6IFJlc3BvbnNlKHsgcGFzc3Rocm91Z2g6IHRydWUgfSkgYXMgYW55LCBpbmRleDogMiB9LFxuICAgICAgICBdXG4gICAgICApO1xuICAgIH1cblxuICAgIGlmIChtZXRob2QgPT09IFwiREVMRVRFXCIgJiYgbm9ybWFsaXplZFBhdGggPT09IHBrUGF0aCkge1xuICAgICAgcmV0dXJuIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlUmVnaXN0cmF0aW9uKFxuICAgICAgICBcImRlbGV0ZVwiLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZURlbGV0ZUhhbmRsZXIoZ2V0UEssIG1vZGVsQ2xhenpOYW1lKSxcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5kZWxldGVEZWNvcmF0b3JzKE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSwgYXBpUHJvcGVydGllcywgcGtQYXRoKSxcbiAgICAgICAgW1xuICAgICAgICAgIHsgZGVjb3JhdG9yOiBEZWNhZlBhcmFtcyhhcGlQcm9wZXJ0aWVzKSBhcyBhbnksIGluZGV4OiAwIH0sXG4gICAgICAgICAgeyBkZWNvcmF0b3I6IFJlc3BvbnNlKHsgcGFzc3Rocm91Z2g6IHRydWUgfSkgYXMgYW55LCBpbmRleDogMSB9LFxuICAgICAgICBdXG4gICAgICApO1xuICAgIH1cblxuICAgIC8vIENvbXBvc2VkIFBLIGZhbGxiYWNrIHJvdXRlcyAoZmlsdGVyRW1wdHkpIOKAlCBwYXRoIGRpZmZlcnMgZnJvbSBwa1BhdGhcbiAgICBjb25zdCBmYWxsYmFja1NlZ21lbnRzID0gbm9ybWFsaXplZFBhdGguc3BsaXQoXCIvXCIpLmZpbHRlcihCb29sZWFuKTtcbiAgICBjb25zdCBpc0FsbFBhcmFtcyA9IGZhbGxiYWNrU2VnbWVudHMubGVuZ3RoID4gMCAmJiBmYWxsYmFja1NlZ21lbnRzLmV2ZXJ5KChzKSA9PiBzLnN0YXJ0c1dpdGgoXCI6XCIpKTtcbiAgICBpZiAoaXNBbGxQYXJhbXMgJiYgbm9ybWFsaXplZFBhdGggIT09IHBrUGF0aCkge1xuICAgICAgY29uc3QgZmFsbGJhY2tBcGlQcm9wcyA9IGZhbGxiYWNrU2VnbWVudHMubWFwKChzKSA9PiBzLnNsaWNlKDEpKS5tYXAoKG5hbWUpID0+ICh7XG4gICAgICAgIG5hbWUsXG4gICAgICAgIGRlc2NyaXB0aW9uOiBgJHtuYW1lfSBwYXJhbWV0ZXJgLFxuICAgICAgICByZXF1aXJlZDogdHJ1ZSxcbiAgICAgICAgdHlwZTogU3RyaW5nLFxuICAgICAgfSkpO1xuICAgICAgY29uc3Qgc3VmZml4ID0gZmFsbGJhY2tTZWdtZW50cy5tYXAoKHMpID0+IHMuc2xpY2UoMSkpLmpvaW4oXCJBbmRcIik7XG4gICAgICBpZiAobWV0aG9kID09PSBcIkdFVFwiKSB7XG4gICAgICAgIHJldHVybiBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZVJlZ2lzdHJhdGlvbihcbiAgICAgICAgICBgcmVhZEJ5JHtzdWZmaXh9YCxcbiAgICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZVJlYWRIYW5kbGVyKGdldFBLLCBtb2RlbENsYXp6TmFtZSksXG4gICAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5yZWFkRGVjb3JhdG9ycyhNb2RlbENvbnN0ciwgbW9kZWxDbGF6ek5hbWUsIGZhbGxiYWNrQXBpUHJvcHMsIG5vcm1hbGl6ZWRQYXRoKSxcbiAgICAgICAgICBbeyBkZWNvcmF0b3I6IERlY2FmUGFyYW1zKGZhbGxiYWNrQXBpUHJvcHMpIGFzIGFueSwgaW5kZXg6IDAgfV1cbiAgICAgICAgKTtcbiAgICAgIH1cbiAgICAgIGlmIChtZXRob2QgPT09IFwiUFVUXCIpIHtcbiAgICAgICAgcmV0dXJuIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlUmVnaXN0cmF0aW9uKFxuICAgICAgICAgIGB1cGRhdGVCeSR7c3VmZml4fWAsXG4gICAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVVcGRhdGVIYW5kbGVyKGdldFBLLCBNb2RlbENvbnN0ciwgbW9kZWxDbGF6ek5hbWUpLFxuICAgICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIudXBkYXRlRGVjb3JhdG9ycyhNb2RlbENvbnN0ciwgbW9kZWxDbGF6ek5hbWUsIGZhbGxiYWNrQXBpUHJvcHMsIG5vcm1hbGl6ZWRQYXRoKSxcbiAgICAgICAgICBbXG4gICAgICAgICAgICB7IGRlY29yYXRvcjogRGVjYWZQYXJhbXMoZmFsbGJhY2tBcGlQcm9wcykgYXMgYW55LCBpbmRleDogMCB9LFxuICAgICAgICAgICAgeyBkZWNvcmF0b3I6IERlY2FmQm9keSgpIGFzIGFueSwgaW5kZXg6IDEgfSxcbiAgICAgICAgICAgIHsgZGVjb3JhdG9yOiBSZXNwb25zZSh7IHBhc3N0aHJvdWdoOiB0cnVlIH0pIGFzIGFueSwgaW5kZXg6IDIgfSxcbiAgICAgICAgICBdXG4gICAgICAgICk7XG4gICAgICB9XG4gICAgICBpZiAobWV0aG9kID09PSBcIkRFTEVURVwiKSB7XG4gICAgICAgIHJldHVybiBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZVJlZ2lzdHJhdGlvbihcbiAgICAgICAgICBgZGVsZXRlQnkke3N1ZmZpeH1gLFxuICAgICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlRGVsZXRlSGFuZGxlcihnZXRQSywgbW9kZWxDbGF6ek5hbWUpLFxuICAgICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuZGVsZXRlRGVjb3JhdG9ycyhNb2RlbENvbnN0ciwgbW9kZWxDbGF6ek5hbWUsIGZhbGxiYWNrQXBpUHJvcHMsIG5vcm1hbGl6ZWRQYXRoKSxcbiAgICAgICAgICBbXG4gICAgICAgICAgICB7IGRlY29yYXRvcjogRGVjYWZQYXJhbXMoZmFsbGJhY2tBcGlQcm9wcykgYXMgYW55LCBpbmRleDogMCB9LFxuICAgICAgICAgICAgeyBkZWNvcmF0b3I6IFJlc3BvbnNlKHsgcGFzc3Rocm91Z2g6IHRydWUgfSkgYXMgYW55LCBpbmRleDogMSB9LFxuICAgICAgICAgIF1cbiAgICAgICAgKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICBpZiAobWV0aG9kID09PSBcIkdFVFwiICYmIG5vcm1hbGl6ZWRQYXRoID09PSBcInN0YXRlbWVudC86bWV0aG9kLyphcmdzXCIpIHtcbiAgICAgIHJldHVybiBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZVJlZ2lzdHJhdGlvbihcbiAgICAgICAgXCJzdGF0ZW1lbnRcIixcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVTdGF0ZW1lbnRIYW5kbGVyKG1vZGVsQ2xhenpOYW1lKSxcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5zdGF0ZW1lbnREZWNvcmF0b3JzKE1vZGVsQ29uc3RyLCBtb2RlbENsYXp6TmFtZSksXG4gICAgICAgIFtcbiAgICAgICAgICB7IGRlY29yYXRvcjogUGFyYW0oXCJtZXRob2RcIikgYXMgYW55LCBpbmRleDogMCB9LFxuICAgICAgICAgIHsgZGVjb3JhdG9yOiBQYXJhbShcImFyZ3NcIikgYXMgYW55LCBpbmRleDogMSB9LFxuICAgICAgICAgIHsgZGVjb3JhdG9yOiBEZWNhZlF1ZXJ5KCkgYXMgYW55LCBpbmRleDogMiB9LFxuICAgICAgICBdXG4gICAgICApO1xuICAgIH1cblxuICAgIGNvbnN0IHN0YXRlbWVudFJvdXRlczogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHtcbiAgICAgIFwibGlzdEJ5LzprZXlcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkxJU1RfQlksXG4gICAgICBcInBhZ2luYXRlQnkvOmtleS86cGFnZVwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuUEFHRV9CWSxcbiAgICAgIFwiZmluZC86dmFsdWVcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkZJTkQsXG4gICAgICBcInBhZ2UvOnZhbHVlXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5QQUdFLFxuICAgICAgXCJmaW5kT25lQnkvOmtleS86dmFsdWVcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkZJTkRfT05FX0JZLFxuICAgICAgXCJmaW5kQnkvOmtleS86dmFsdWVcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkZJTkRfQlksXG4gICAgICBcImNvdW50T2YvOmZpZWxkXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5DT1VOVF9PRixcbiAgICAgIFwibWF4T2YvOmZpZWxkXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5NQVhfT0YsXG4gICAgICBcIm1pbk9mLzpmaWVsZFwiOiBQcmVwYXJlZFN0YXRlbWVudEtleXMuTUlOX09GLFxuICAgICAgXCJhdmdPZi86ZmllbGRcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkFWR19PRixcbiAgICAgIFwic3VtT2YvOmZpZWxkXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5TVU1fT0YsXG4gICAgICBcImRpc3RpbmN0T2YvOmZpZWxkXCI6IFByZXBhcmVkU3RhdGVtZW50S2V5cy5ESVNUSU5DVF9PRixcbiAgICAgIFwiZ3JvdXBPZi86ZmllbGRcIjogUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkdST1VQX09GLFxuICAgIH07XG5cbiAgICBjb25zdCBzdGF0ZW1lbnRLZXkgPSBzdGF0ZW1lbnRSb3V0ZXNbbm9ybWFsaXplZFBhdGhdO1xuICAgIGlmIChzdGF0ZW1lbnRLZXkgJiYgbWV0aG9kID09PSBcIkdFVFwiKSB7XG4gICAgICByZXR1cm4gRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVSZWdpc3RyYXRpb24oXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuc3RhdGVtZW50TWV0aG9kTmFtZShub3JtYWxpemVkUGF0aCksXG4gICAgICAgIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlU3RhdGVtZW50U2hvcnRjdXRIYW5kbGVyKHN0YXRlbWVudEtleSwgbW9kZWxDbGF6ek5hbWUpLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLnN0YXRlbWVudFNob3J0Y3V0RGVjb3JhdG9ycyhNb2RlbENvbnN0ciwgbW9kZWxDbGF6ek5hbWUsIG5vcm1hbGl6ZWRQYXRoLCBzdGF0ZW1lbnRLZXkpLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLnN0YXRlbWVudFNob3J0Y3V0UGFyYW1zKG5vcm1hbGl6ZWRQYXRoKVxuICAgICAgKTtcbiAgICB9XG5cbiAgICBpZiAobWV0aG9kID09PSBcIkdFVFwiICYmIG5vcm1hbGl6ZWRQYXRoLnN0YXJ0c1dpdGgoXCJxdWVyeS9cIikpIHtcbiAgICAgIGNvbnN0IHF1ZXJ5TWV0aG9kID0gbm9ybWFsaXplZFBhdGgucmVwbGFjZSgvXnF1ZXJ5XFwvLywgXCJcIikuc3BsaXQoXCIvXCIpWzBdO1xuICAgICAgcmV0dXJuIEZyb21Nb2RlbENvbnRyb2xsZXIuY3JlYXRlUmVnaXN0cmF0aW9uKFxuICAgICAgICBxdWVyeU1ldGhvZCxcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVDb21wbGV4UXVlcnlIYW5kbGVyKHF1ZXJ5TWV0aG9kKSxcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5nZXRRdWVyeURlY29yYXRvcnMocXVlcnlNZXRob2QsIG5vcm1hbGl6ZWRQYXRoLCBcIkdFVFwiLCB0cnVlKSxcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jb21wbGV4UXVlcnlQYXJhbXMobm9ybWFsaXplZFBhdGgpXG4gICAgICApO1xuICAgIH1cblxuICAgIC8vIEZhbGxiYWNrIGZvciBjdXN0b20gQHJvdXRlKCkgcGF0aHMgKGUuZy4gXCJtZXRhZGF0YS9mb3ItcHJvZHVjdC86cHJvZHVjdENvZGVcIilcbiAgICBjb25zdCBwYXRoU2VnbWVudHMgPSBub3JtYWxpemVkUGF0aC5zcGxpdChcIi9cIikuZmlsdGVyKEJvb2xlYW4pO1xuICAgIGNvbnN0IGtub3duUHJlZml4ZXMgPSBuZXcgU2V0PHN0cmluZz4oW1xuICAgICAgXCJsaXN0QnlcIiwgXCJmaW5kQnlcIiwgXCJmaW5kQnlQYWdpbmF0ZVwiLCBcImZpbmRPbmVCeVwiLCBcInBhZ2luYXRlQnlcIixcbiAgICAgIFwiZmluZFwiLCBcInBhZ2VcIiwgXCJjb3VudE9mXCIsIFwibWF4T2ZcIiwgXCJtaW5PZlwiLCBcImF2Z09mXCIsIFwic3VtT2ZcIixcbiAgICAgIFwiZGlzdGluY3RPZlwiLCBcImdyb3VwT2ZcIiwgXCJzdGF0ZW1lbnRcIiwgXCJidWxrXCIsIFwicXVlcnlcIixcbiAgICBdKTtcbiAgICBpZiAoXG4gICAgICBwYXRoU2VnbWVudHMubGVuZ3RoID4gMCAmJlxuICAgICAgIW5vcm1hbGl6ZWRQYXRoLnN0YXJ0c1dpdGgoXCJxdWVyeS9cIikgJiZcbiAgICAgICFrbm93blByZWZpeGVzLmhhcyhwYXRoU2VnbWVudHNbMF0pXG4gICAgKSB7XG4gICAgICAvLyBMb29rIHVwIHRoZSBhY3R1YWwgbWV0aG9kIG5hbWUgZnJvbSBAcm91dGUgbWV0YWRhdGEgYnkgbWF0Y2hpbmcgdGhlIHBhdGguXG4gICAgICAvLyBUaGUgQHJvdXRlIGRlY29yYXRvciBzdG9yZXMgeyBwYXRoLCBodHRwTWV0aG9kLCBoYW5kbGVyIH0ga2V5ZWQgYnkgbWV0aG9kIG5hbWVcbiAgICAgIC8vIG9uIHRoZSByZXBvc2l0b3J5IGNvbnN0cnVjdG9yLlxuICAgICAgY29uc3Qgcm91dGVNZXRhZGF0YTogUmVjb3JkPHN0cmluZywgYW55PiA9XG4gICAgICAgIE1ldGFkYXRhLmdldChcbiAgICAgICAgICAocGVyc2lzdGVuY2UgYXMgYW55KT8uY29uc3RydWN0b3IsXG4gICAgICAgICAgTWV0YWRhdGEua2V5KERFQ0FGX1JPVVRFKVxuICAgICAgICApID8/IHt9O1xuICAgICAgY29uc3QgbWF0Y2hlZEVudHJ5ID0gT2JqZWN0LmVudHJpZXMocm91dGVNZXRhZGF0YSkuZmluZChcbiAgICAgICAgKFssIGluZm9dKSA9PlxuICAgICAgICAgIGluZm8gJiZcbiAgICAgICAgICB0eXBlb2YgaW5mbyA9PT0gXCJvYmplY3RcIiAmJlxuICAgICAgICAgIGluZm8ucGF0aD8ucmVwbGFjZSgvXlxcLyt8XFwvKyQvZywgXCJcIikgPT09IG5vcm1hbGl6ZWRQYXRoXG4gICAgICApO1xuICAgICAgY29uc3QgYWN0dWFsTWV0aG9kTmFtZSA9IG1hdGNoZWRFbnRyeT8uWzBdIHx8IHBhdGhTZWdtZW50c1swXTtcblxuICAgICAgY29uc3QgcGFyYW1TZWdtZW50cyA9IHBhdGhTZWdtZW50cy5maWx0ZXIoKHMpID0+IHMuc3RhcnRzV2l0aChcIjpcIikpO1xuICAgICAgY29uc3QgYXBpUGF0aFBhcmFtcyA9IHBhcmFtU2VnbWVudHMubWFwKChzKSA9PiBzLnNsaWNlKDEpKS5tYXAoKG5hbWUpID0+ICh7XG4gICAgICAgIG5hbWUsXG4gICAgICAgIGRlc2NyaXB0aW9uOiBgJHtuYW1lfSBwYXJhbWV0ZXIgZm9yIHRoZSBxdWVyeWAsXG4gICAgICAgIHJlcXVpcmVkOiB0cnVlLFxuICAgICAgICB0eXBlOiBTdHJpbmcsXG4gICAgICB9KSk7XG5cbiAgICAgIHJldHVybiBGcm9tTW9kZWxDb250cm9sbGVyLmNyZWF0ZVJlZ2lzdHJhdGlvbihcbiAgICAgICAgYWN0dWFsTWV0aG9kTmFtZSxcbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGVDdXN0b21Sb3V0ZUhhbmRsZXIoYWN0dWFsTWV0aG9kTmFtZSksXG4gICAgICAgIFtcbiAgICAgICAgICAuLi5hcGlQYXRoUGFyYW1zLm1hcCgocCkgPT4gQXBpUGFyYW0ocCkpLFxuICAgICAgICAgIEFwaU9wZXJhdGlvbih7IHN1bW1hcnk6IGBSZXRyaWV2ZSByZWNvcmRzIHVzaW5nIFwiJHthY3R1YWxNZXRob2ROYW1lfVwiLmAgfSksXG4gICAgICAgICAgQXBpT2tSZXNwb25zZSh7IGRlc2NyaXB0aW9uOiBcIlJlc3VsdCBzdWNjZXNzZnVsbHkgcmV0cmlldmVkLlwiIH0pLFxuICAgICAgICAgIEFwaU5vQ29udGVudFJlc3BvbnNlKHsgZGVzY3JpcHRpb246IFwiTm8gY29udGVudCByZXR1cm5lZCBieSB0aGUgbWV0aG9kLlwiIH0pLFxuICAgICAgICBdLFxuICAgICAgICBGcm9tTW9kZWxDb250cm9sbGVyLmNvbXBsZXhRdWVyeVBhcmFtcyhub3JtYWxpemVkUGF0aClcbiAgICAgICk7XG4gICAgfVxuXG4gICAgcmV0dXJuIHVuZGVmaW5lZDtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGNyZWF0ZVJlZ2lzdHJhdGlvbihcbiAgICBtZXRob2ROYW1lOiBzdHJpbmcsXG4gICAgaGFuZGxlcjogKC4uLmFyZ3M6IGFueVtdKSA9PiBhbnksXG4gICAgZGVjb3JhdG9yczogQXJyYXk8KHRhcmdldDogYW55LCBrZXk6IHN0cmluZywgZGVzYzogYW55KSA9PiB2b2lkPixcbiAgICBwYXJhbURlY29yYXRvcnM6IEFycmF5PHsgZGVjb3JhdG9yOiBQYXJhbWV0ZXJEZWNvcmF0b3I7IGluZGV4OiBudW1iZXIgfT5cbiAgKSB7XG4gICAgcmV0dXJuIHsgbWV0aG9kTmFtZSwgaGFuZGxlciwgZGVjb3JhdG9ycywgcGFyYW1EZWNvcmF0b3JzIH07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBzdGF0ZW1lbnRNZXRob2ROYW1lKHBhdGg6IHN0cmluZyk6IHN0cmluZyB7XG4gICAgY29uc3QgZmlyc3RTZWdtZW50ID0gcGF0aC5zcGxpdChcIi9cIilbMF07XG4gICAgcmV0dXJuIGZpcnN0U2VnbWVudDtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIHN0YXRlbWVudFNob3J0Y3V0UGFyYW1zKFxuICAgIHBhdGg6IHN0cmluZ1xuICApOiBBcnJheTx7IGRlY29yYXRvcjogUGFyYW1ldGVyRGVjb3JhdG9yOyBpbmRleDogbnVtYmVyIH0+IHtcbiAgICBjb25zdCBzZWdtZW50cyA9IHBhdGguc3BsaXQoXCIvXCIpLmZpbHRlcigocykgPT4gcy5zdGFydHNXaXRoKFwiOlwiKSk7XG4gICAgY29uc3QgcGFyYW1zOiBBcnJheTx7IGRlY29yYXRvcjogUGFyYW1ldGVyRGVjb3JhdG9yOyBpbmRleDogbnVtYmVyIH0+ID0gW107XG4gICAgc2VnbWVudHMuZm9yRWFjaCgoc2VnLCBpKSA9PiB7XG4gICAgICBjb25zdCBuYW1lID0gc2VnLnJlcGxhY2UoXCI6XCIsIFwiXCIpO1xuICAgICAgcGFyYW1zLnB1c2goeyBkZWNvcmF0b3I6IFBhcmFtKG5hbWUpIGFzIGFueSwgaW5kZXg6IGkgfSk7XG4gICAgfSk7XG4gICAgaWYgKFxuICAgICAgcGF0aC5zdGFydHNXaXRoKFwibGlzdEJ5L1wiKSB8fFxuICAgICAgcGF0aC5zdGFydHNXaXRoKFwicGFnaW5hdGVCeS9cIikgfHxcbiAgICAgIHBhdGguc3RhcnRzV2l0aChcImZpbmQvXCIpIHx8XG4gICAgICBwYXRoLnN0YXJ0c1dpdGgoXCJwYWdlL1wiKVxuICAgICkge1xuICAgICAgcGFyYW1zLnB1c2goeyBkZWNvcmF0b3I6IERlY2FmUXVlcnkoKSBhcyBhbnksIGluZGV4OiBzZWdtZW50cy5sZW5ndGggfSk7XG4gICAgfVxuICAgIHJldHVybiBwYXJhbXM7XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBjb21wbGV4UXVlcnlQYXJhbXMoXG4gICAgcGF0aDogc3RyaW5nXG4gICk6IEFycmF5PHsgZGVjb3JhdG9yOiBQYXJhbWV0ZXJEZWNvcmF0b3I7IGluZGV4OiBudW1iZXIgfT4ge1xuICAgIGNvbnN0IHNlZ21lbnRzID0gcGF0aC5zcGxpdChcIi9cIikuZmlsdGVyKChzKSA9PiBzLnN0YXJ0c1dpdGgoXCI6XCIpKTtcbiAgICBjb25zdCBwYXJhbXM6IEFycmF5PHsgZGVjb3JhdG9yOiBQYXJhbWV0ZXJEZWNvcmF0b3I7IGluZGV4OiBudW1iZXIgfT4gPSBbXTtcbiAgICBzZWdtZW50cy5mb3JFYWNoKChzZWcsIGkpID0+IHtcbiAgICAgIGNvbnN0IG5hbWUgPSBzZWcucmVwbGFjZShcIjpcIiwgXCJcIik7XG4gICAgICBwYXJhbXMucHVzaCh7IGRlY29yYXRvcjogUGFyYW0obmFtZSkgYXMgYW55LCBpbmRleDogaSB9KTtcbiAgICB9KTtcbiAgICBpZiAocGF0aC5zdGFydHNXaXRoKFwicXVlcnkvXCIpKSB7XG4gICAgICBwYXJhbXMucHVzaCh7IGRlY29yYXRvcjogRGVjYWZRdWVyeSgpIGFzIGFueSwgaW5kZXg6IHNlZ21lbnRzLmxlbmd0aCB9KTtcbiAgICB9XG4gICAgcmV0dXJuIHBhcmFtcztcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGNyZWF0ZUNyZWF0ZUhhbmRsZXIoXG4gICAgTW9kZWxDb25zdHI6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nXG4gICkge1xuICAgIHJldHVybiBhc3luYyBmdW5jdGlvbiBjcmVhdGUoXG4gICAgICB0aGlzOiBhbnksXG4gICAgICBkYXRhOiBhbnksXG4gICAgICByZXNwPzogYW55XG4gICAgKTogUHJvbWlzZTxNb2RlbDxhbnk+PiB7XG4gICAgICBjb25zdCB7IGN0eCwgbG9nIH0gPSAoXG4gICAgICAgIGF3YWl0IHRoaXMubG9nQ3R4KFtdLCBPcGVyYXRpb25LZXlzLkNSRUFURSwgdHJ1ZSlcbiAgICAgICkuZm9yKGNyZWF0ZSk7XG4gICAgICBsb2cudmVyYm9zZShgY3JlYXRpbmcgbmV3ICR7bW9kZWxDbGF6ek5hbWV9YCk7XG4gICAgICBsZXQgY3JlYXRlZDogTW9kZWw7XG4gICAgICB0cnkge1xuICAgICAgICBjcmVhdGVkID0gYXdhaXQgdGhpcy5wZXJzaXN0ZW5jZShjdHgpLmNyZWF0ZShkYXRhLCBjdHgpO1xuICAgICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgICBsb2cuZXJyb3IoYEZhaWxlZCB0byBjcmVhdGUgbmV3ICR7bW9kZWxDbGF6ek5hbWV9YCwgZSBhcyBFcnJvcik7XG4gICAgICAgIHRocm93IGU7XG4gICAgICB9XG4gICAgICBsb2cuaW5mbyhgY3JlYXRlZCBuZXcgJHttb2RlbENsYXp6TmFtZX0gd2l0aCBpZCAkeyhjcmVhdGVkIGFzIGFueSlbdGhpcy5wa119YCk7XG4gICAgICBpZiAocmVzcCkgY3R4LnRvUmVzcG9uc2UocmVzcCk7XG4gICAgICByZXR1cm4gY3JlYXRlZDtcbiAgICB9O1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgY3JlYXRlQnVsa0NyZWF0ZUhhbmRsZXIoXG4gICAgTW9kZWxDb25zdHI6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nXG4gICkge1xuICAgIHJldHVybiBhc3luYyBmdW5jdGlvbiBjcmVhdGVBbGwoXG4gICAgICB0aGlzOiBhbnksXG4gICAgICBkYXRhOiBhbnlbXSxcbiAgICAgIHJlc3A/OiBhbnlcbiAgICApOiBQcm9taXNlPE1vZGVsW10+IHtcbiAgICAgIGNvbnN0IHsgY3R4LCBsb2cgfSA9IChcbiAgICAgICAgYXdhaXQgdGhpcy5sb2dDdHgoW10sIEJ1bGtDcnVkT3BlcmF0aW9uS2V5cy5DUkVBVEVfQUxMLCB0cnVlKVxuICAgICAgKS5mb3IoY3JlYXRlQWxsKTtcbiAgICAgIGxvZy52ZXJib3NlKGBjcmVhdGluZyBuZXcgJHttb2RlbENsYXp6TmFtZX1gKTtcbiAgICAgIGxldCBjcmVhdGVkOiBhbnlbXTtcbiAgICAgIHRyeSB7XG4gICAgICAgIGNyZWF0ZWQgPSBhd2FpdCB0aGlzLnBlcnNpc3RlbmNlKGN0eCkuY3JlYXRlQWxsKFxuICAgICAgICAgIGRhdGEubWFwKChkKSA9PiBuZXcgTW9kZWxDb25zdHIoZCkpLFxuICAgICAgICAgIGN0eFxuICAgICAgICApO1xuICAgICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgICBsb2cuZXJyb3IoYEZhaWxlZCB0byBjcmVhdGUgbmV3ICR7bW9kZWxDbGF6ek5hbWV9YCwgZSBhcyBFcnJvcik7XG4gICAgICAgIHRocm93IGU7XG4gICAgICB9XG4gICAgICBsb2cuaW5mbyhgY3JlYXRlZCBuZXcgJHttb2RlbENsYXp6TmFtZX0gd2l0aCBpZCAkeyhjcmVhdGVkIGFzIGFueSlbdGhpcy5wa119YCk7XG4gICAgICBpZiAocmVzcCkgY3R4LnRvUmVzcG9uc2UocmVzcCk7XG4gICAgICByZXR1cm4gY3JlYXRlZDtcbiAgICB9O1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgY3JlYXRlQnVsa1JlYWRIYW5kbGVyKG1vZGVsQ2xhenpOYW1lOiBzdHJpbmcpIHtcbiAgICByZXR1cm4gYXN5bmMgZnVuY3Rpb24gcmVhZEFsbChcbiAgICAgIHRoaXM6IGFueSxcbiAgICAgIGlkczogc3RyaW5nW11cbiAgICApOiBQcm9taXNlPE1vZGVsW10+IHtcbiAgICAgIGNvbnN0IHsgY3R4LCBsb2cgfSA9IChcbiAgICAgICAgYXdhaXQgdGhpcy5sb2dDdHgoW10sIEJ1bGtDcnVkT3BlcmF0aW9uS2V5cy5SRUFEX0FMTCwgdHJ1ZSlcbiAgICAgICkuZm9yKHJlYWRBbGwpO1xuICAgICAgY29uc3Qgbm9ybWFsaXplZElkcyA9IEFycmF5LmlzQXJyYXkoaWRzKSA/IGlkcyA6IFtpZHNdO1xuICAgICAgbGV0IHJlYWQ6IE1vZGVsW107XG4gICAgICB0cnkge1xuICAgICAgICBsb2cuZGVidWcoYHJlYWRpbmcgJHtub3JtYWxpemVkSWRzfSAke21vZGVsQ2xhenpOYW1lfWApO1xuICAgICAgICByZWFkID0gYXdhaXQgdGhpcy5wZXJzaXN0ZW5jZShjdHgpLnJlYWRBbGwobm9ybWFsaXplZElkcyBhcyBhbnksIGN0eCk7XG4gICAgICB9IGNhdGNoIChlOiB1bmtub3duKSB7XG4gICAgICAgIGxvZy5lcnJvcihgRmFpbGVkIHRvIHJlYWQgJHttb2RlbENsYXp6TmFtZX1gLCBlIGFzIEVycm9yKTtcbiAgICAgICAgdGhyb3cgZTtcbiAgICAgIH1cbiAgICAgIGxvZy5pbmZvKGByZWFkICR7cmVhZC5sZW5ndGh9ICR7bW9kZWxDbGF6ek5hbWV9YCk7XG4gICAgICByZXR1cm4gcmVhZDtcbiAgICB9O1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgY3JlYXRlQnVsa1VwZGF0ZUhhbmRsZXIobW9kZWxDbGF6ek5hbWU6IHN0cmluZykge1xuICAgIHJldHVybiBhc3luYyBmdW5jdGlvbiB1cGRhdGVBbGwoXG4gICAgICB0aGlzOiBhbnksXG4gICAgICBib2R5OiBhbnlbXSxcbiAgICAgIHJlc3A/OiBhbnlcbiAgICApOiBQcm9taXNlPGFueVtdPiB7XG4gICAgICBjb25zdCB7IGN0eCwgbG9nIH0gPSAoXG4gICAgICAgIGF3YWl0IHRoaXMubG9nQ3R4KFtdLCBCdWxrQ3J1ZE9wZXJhdGlvbktleXMuVVBEQVRFX0FMTCwgdHJ1ZSlcbiAgICAgICkuZm9yKHVwZGF0ZUFsbCk7XG4gICAgICBsZXQgdXBkYXRlZDogYW55W107XG4gICAgICB0cnkge1xuICAgICAgICBsb2cuaW5mbyhgdXBkYXRpbmcgJHtib2R5Lmxlbmd0aH0gJHttb2RlbENsYXp6TmFtZX1gKTtcbiAgICAgICAgdXBkYXRlZCA9IGF3YWl0IHRoaXMucGVyc2lzdGVuY2UoY3R4KS51cGRhdGVBbGwoYm9keSwgY3R4KTtcbiAgICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgICAgbG9nLmVycm9yKGUgYXMgRXJyb3IpO1xuICAgICAgICB0aHJvdyBlO1xuICAgICAgfVxuICAgICAgaWYgKHJlc3ApIGN0eC50b1Jlc3BvbnNlKHJlc3ApO1xuICAgICAgcmV0dXJuIHVwZGF0ZWQ7XG4gICAgfTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGNyZWF0ZUJ1bGtEZWxldGVIYW5kbGVyKG1vZGVsQ2xhenpOYW1lOiBzdHJpbmcpIHtcbiAgICByZXR1cm4gYXN5bmMgZnVuY3Rpb24gZGVsZXRlQWxsKFxuICAgICAgdGhpczogYW55LFxuICAgICAgaWRzOiBzdHJpbmdbXSxcbiAgICAgIHJlc3A/OiBhbnlcbiAgICApOiBQcm9taXNlPE1vZGVsW10+IHtcbiAgICAgIGNvbnN0IHsgY3R4LCBsb2cgfSA9IChcbiAgICAgICAgYXdhaXQgdGhpcy5sb2dDdHgoW10sIEJ1bGtDcnVkT3BlcmF0aW9uS2V5cy5ERUxFVEVfQUxMLCB0cnVlKVxuICAgICAgKS5mb3IoZGVsZXRlQWxsKTtcbiAgICAgIGNvbnN0IG5vcm1hbGl6ZWRJZHMgPSBBcnJheS5pc0FycmF5KGlkcykgPyBpZHMgOiBbaWRzXTtcbiAgICAgIGxldCByZWFkOiBNb2RlbFtdO1xuICAgICAgdHJ5IHtcbiAgICAgICAgbG9nLmRlYnVnKGBkZWxldGluZyAke25vcm1hbGl6ZWRJZHMubGVuZ3RofSAke21vZGVsQ2xhenpOYW1lfWApO1xuICAgICAgICByZWFkID0gYXdhaXQgdGhpcy5wZXJzaXN0ZW5jZShjdHgpLmRlbGV0ZUFsbChub3JtYWxpemVkSWRzLCBjdHgpO1xuICAgICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgICBsb2cuZXJyb3IoYEZhaWxlZCB0byBkZWxldGUgJHttb2RlbENsYXp6TmFtZX1gLCBlIGFzIEVycm9yKTtcbiAgICAgICAgdGhyb3cgZTtcbiAgICAgIH1cbiAgICAgIGxvZy5pbmZvKGBkZWxldGVkICR7cmVhZC5sZW5ndGh9ICR7bW9kZWxDbGF6ek5hbWV9YCk7XG4gICAgICBpZiAocmVzcCkgY3R4LnRvUmVzcG9uc2UocmVzcCk7XG4gICAgICByZXR1cm4gcmVhZDtcbiAgICB9O1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgY3JlYXRlUmVhZEhhbmRsZXIoXG4gICAgZ2V0UEs6ICguLi5wOiBBcnJheTxzdHJpbmcgfCBudW1iZXI+KSA9PiBzdHJpbmcsXG4gICAgbW9kZWxDbGF6ek5hbWU6IHN0cmluZ1xuICApIHtcbiAgICByZXR1cm4gYXN5bmMgZnVuY3Rpb24gcmVhZChcbiAgICAgIHRoaXM6IGFueSxcbiAgICAgIHJvdXRlUGFyYW1zOiBhbnlcbiAgICApOiBQcm9taXNlPE1vZGVsPiB7XG4gICAgICBjb25zdCB7IGN0eCwgbG9nIH0gPSAoXG4gICAgICAgIGF3YWl0IHRoaXMubG9nQ3R4KFtdLCBPcGVyYXRpb25LZXlzLlJFQUQsIHRydWUpXG4gICAgICApLmZvcihyZWFkKTtcbiAgICAgIGNvbnN0IGlkID0gZ2V0UEsoLi4ucm91dGVQYXJhbXMudmFsdWVzSW5PcmRlcik7XG4gICAgICBpZiAodHlwZW9mIGlkID09PSBcInVuZGVmaW5lZFwiKVxuICAgICAgICB0aHJvdyBuZXcgVmFsaWRhdGlvbkVycm9yKGBObyAke3RoaXMucGt9IHByb3ZpZGVkYCk7XG4gICAgICBsZXQgcmVhZFJlc3VsdDogTW9kZWw7XG4gICAgICB0cnkge1xuICAgICAgICBsb2cuZGVidWcoYHJlYWRpbmcgJHttb2RlbENsYXp6TmFtZX0gd2l0aCAke3RoaXMucGt9ICR7aWR9YCk7XG4gICAgICAgIHJlYWRSZXN1bHQgPSBhd2FpdCB0aGlzLnBlcnNpc3RlbmNlKGN0eCkucmVhZChpZCwgY3R4KTtcbiAgICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgICAgbG9nLmVycm9yKGBGYWlsZWQgdG8gcmVhZCAke21vZGVsQ2xhenpOYW1lfSB3aXRoIGlkICR7aWR9YCwgZSBhcyBFcnJvcik7XG4gICAgICAgIHRocm93IGU7XG4gICAgICB9XG4gICAgICBsb2cuaW5mbyhgcmVhZCAke21vZGVsQ2xhenpOYW1lfSB3aXRoIGlkICR7KHJlYWRSZXN1bHQgYXMgYW55KVt0aGlzLnBrXX1gKTtcbiAgICAgIHJldHVybiByZWFkUmVzdWx0O1xuICAgIH07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBjcmVhdGVVcGRhdGVIYW5kbGVyKFxuICAgIGdldFBLOiAoLi4ucDogQXJyYXk8c3RyaW5nIHwgbnVtYmVyPikgPT4gc3RyaW5nLFxuICAgIE1vZGVsQ29uc3RyOiBNb2RlbENvbnN0cnVjdG9yPGFueT4sXG4gICAgbW9kZWxDbGF6ek5hbWU6IHN0cmluZ1xuICApIHtcbiAgICByZXR1cm4gYXN5bmMgZnVuY3Rpb24gdXBkYXRlKFxuICAgICAgdGhpczogYW55LFxuICAgICAgcm91dGVQYXJhbXM6IGFueSxcbiAgICAgIGJvZHk6IGFueSxcbiAgICAgIHJlc3A/OiBhbnlcbiAgICApOiBQcm9taXNlPGFueT4ge1xuICAgICAgY29uc3QgeyBjdHgsIGxvZyB9ID0gKFxuICAgICAgICBhd2FpdCB0aGlzLmxvZ0N0eChbXSwgT3BlcmF0aW9uS2V5cy5VUERBVEUsIHRydWUpXG4gICAgICApLmZvcih1cGRhdGUpO1xuICAgICAgY29uc3QgaWQgPSBnZXRQSyguLi5yb3V0ZVBhcmFtcy52YWx1ZXNJbk9yZGVyKTtcbiAgICAgIGlmICh0eXBlb2YgaWQgPT09IFwidW5kZWZpbmVkXCIpXG4gICAgICAgIHRocm93IG5ldyBWYWxpZGF0aW9uRXJyb3IoYE5vICR7dGhpcy5wa30gcHJvdmlkZWRgKTtcbiAgICAgIGxldCB1cGRhdGVkOiBhbnk7XG4gICAgICB0cnkge1xuICAgICAgICBsb2cuaW5mbyhgdXBkYXRpbmcgJHttb2RlbENsYXp6TmFtZX0gd2l0aCAke3RoaXMucGt9ICR7aWR9YCk7XG4gICAgICAgIGNvbnN0IHBheWxvYWQgPSBKU09OLnBhcnNlKEpTT04uc3RyaW5naWZ5KGJvZHkpKTtcbiAgICAgICAgdXBkYXRlZCA9IGF3YWl0IHRoaXMucGVyc2lzdGVuY2UoY3R4KS51cGRhdGUoXG4gICAgICAgICAgbmV3IE1vZGVsQ29uc3RyKHsgLi4ucGF5bG9hZCwgW3RoaXMucGtdOiBpZCB9KSxcbiAgICAgICAgICBjdHhcbiAgICAgICAgKTtcbiAgICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgICAgbG9nLmVycm9yKGUgYXMgRXJyb3IpO1xuICAgICAgICB0aHJvdyBlO1xuICAgICAgfVxuICAgICAgaWYgKHJlc3ApIGN0eC50b1Jlc3BvbnNlKHJlc3ApO1xuICAgICAgcmV0dXJuIHVwZGF0ZWQ7XG4gICAgfTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGNyZWF0ZURlbGV0ZUhhbmRsZXIoXG4gICAgZ2V0UEs6ICguLi5wOiBBcnJheTxzdHJpbmcgfCBudW1iZXI+KSA9PiBzdHJpbmcsXG4gICAgbW9kZWxDbGF6ek5hbWU6IHN0cmluZ1xuICApIHtcbiAgICByZXR1cm4gYXN5bmMgZnVuY3Rpb24gcmVtb3ZlKFxuICAgICAgdGhpczogYW55LFxuICAgICAgcm91dGVQYXJhbXM6IGFueSxcbiAgICAgIHJlc3A/OiBhbnlcbiAgICApOiBQcm9taXNlPE1vZGVsPiB7XG4gICAgICBjb25zdCB7IGN0eCwgbG9nIH0gPSAoXG4gICAgICAgIGF3YWl0IHRoaXMubG9nQ3R4KFtdLCBPcGVyYXRpb25LZXlzLkRFTEVURSwgdHJ1ZSlcbiAgICAgICkuZm9yKHJlbW92ZSk7XG4gICAgICBjb25zdCBpZCA9IGdldFBLKC4uLnJvdXRlUGFyYW1zLnZhbHVlc0luT3JkZXIpO1xuICAgICAgaWYgKHR5cGVvZiBpZCA9PT0gXCJ1bmRlZmluZWRcIilcbiAgICAgICAgdGhyb3cgbmV3IFZhbGlkYXRpb25FcnJvcihgTm8gJHt0aGlzLnBrfSBwcm92aWRlZGApO1xuICAgICAgbGV0IGRlbDogTW9kZWw7XG4gICAgICB0cnkge1xuICAgICAgICBsb2cuZGVidWcoYGRlbGV0aW5nICR7bW9kZWxDbGF6ek5hbWV9IHdpdGggJHt0aGlzLnBrfSAke2lkfWApO1xuICAgICAgICBkZWwgPSBhd2FpdCB0aGlzLnBlcnNpc3RlbmNlKGN0eCkuZGVsZXRlKGlkLCBjdHgpO1xuICAgICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgICBsb2cuZXJyb3IoYEZhaWxlZCB0byBkZWxldGUgJHttb2RlbENsYXp6TmFtZX0gd2l0aCBpZCAke2lkfWAsIGUgYXMgRXJyb3IpO1xuICAgICAgICB0aHJvdyBlO1xuICAgICAgfVxuICAgICAgbG9nLmluZm8oYGRlbGV0ZWQgJHttb2RlbENsYXp6TmFtZX0gd2l0aCBpZCAke2lkfWApO1xuICAgICAgaWYgKHJlc3ApIGN0eC50b1Jlc3BvbnNlKHJlc3ApO1xuICAgICAgcmV0dXJuIGRlbDtcbiAgICB9O1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgY3JlYXRlU3RhdGVtZW50SGFuZGxlcihtb2RlbENsYXp6TmFtZTogc3RyaW5nKSB7XG4gICAgcmV0dXJuIGFzeW5jIGZ1bmN0aW9uIHN0YXRlbWVudChcbiAgICAgIHRoaXM6IGFueSxcbiAgICAgIG5hbWU6IHN0cmluZyxcbiAgICAgIGFyZ3M6IChzdHJpbmcgfCBudW1iZXIpW10sXG4gICAgICBkZXRhaWxzOiBEaXJlY3Rpb25MaW1pdE9mZnNldFxuICAgICk6IFByb21pc2U8YW55PiB7XG4gICAgICBjb25zdCB7IGN0eCB9ID0gKFxuICAgICAgICBhd2FpdCB0aGlzLmxvZ0N0eChbXSwgUGVyc2lzdGVuY2VLZXlzLlNUQVRFTUVOVCwgdHJ1ZSlcbiAgICAgICkuZm9yKHN0YXRlbWVudCk7XG4gICAgICBjb25zdCB7IGRpcmVjdGlvbiwgb2Zmc2V0LCBsaW1pdCwgYm9va21hcmsgfSA9IGRldGFpbHM7XG4gICAgICBhcmdzID0gYXJncy5tYXAoXG4gICAgICAgIChhKSA9PiAodHlwZW9mIGEgPT09IFwic3RyaW5nXCIgPyBwYXJzZUludChhIGFzIHN0cmluZykgOiBhKSB8fCBhXG4gICAgICApIGFzIGFueVtdO1xuICAgICAgY29uc3QgcGF0aERpcmVjdGlvbiA9IGFyZ3MubGVuZ3RoID4gMSA/IGFyZ3NbMV0gOiB1bmRlZmluZWQ7XG4gICAgICBjb25zdCByZXNvbHZlZERpcmVjdGlvbiA9IChkaXJlY3Rpb24gPz8gcGF0aERpcmVjdGlvbikgYXNcbiAgICAgICAgfCBzdHJpbmdcbiAgICAgICAgfCB1bmRlZmluZWQ7XG4gICAgICBpZiAocmVzb2x2ZWREaXJlY3Rpb24gJiYgYXJncy5sZW5ndGggPiAxKSBhcmdzWzFdID0gcmVzb2x2ZWREaXJlY3Rpb247XG4gICAgICBzd2l0Y2ggKG5hbWUpIHtcbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuRklORDpcbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuRklORF9CWTpcbiAgICAgICAgICBicmVhaztcbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuTElTVF9CWTpcbiAgICAgICAgICBhcmdzLnB1c2goZGlyZWN0aW9uIGFzIHN0cmluZyk7XG4gICAgICAgICAgYnJlYWs7XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLlBBR0U6XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLlBBR0VfQlk6XG4gICAgICAgICAgYXJncyA9IFtcbiAgICAgICAgICAgIGFyZ3NbMF0sXG4gICAgICAgICAgICByZXNvbHZlZERpcmVjdGlvbiBhcyBhbnksXG4gICAgICAgICAgICB7IGxpbWl0LCBvZmZzZXQsIGJvb2ttYXJrIH0sXG4gICAgICAgICAgXTtcbiAgICAgICAgICBicmVhaztcbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuRklORF9PTkVfQlk6XG4gICAgICAgICAgYnJlYWs7XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkNPVU5UX09GOlxuICAgICAgICBjYXNlIFByZXBhcmVkU3RhdGVtZW50S2V5cy5NQVhfT0Y6XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLk1JTl9PRjpcbiAgICAgICAgY2FzZSBQcmVwYXJlZFN0YXRlbWVudEtleXMuQVZHX09GOlxuICAgICAgICBjYXNlIFByZXBhcmVkU3RhdGVtZW50S2V5cy5TVU1fT0Y6XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkRJU1RJTkNUX09GOlxuICAgICAgICBjYXNlIFByZXBhcmVkU3RhdGVtZW50S2V5cy5HUk9VUF9PRjpcbiAgICAgICAgICBicmVhaztcbiAgICAgIH1cbiAgICAgIHJldHVybiB0aGlzLnBlcnNpc3RlbmNlKGN0eCkuc3RhdGVtZW50KG5hbWUsIC4uLmFyZ3MsIGN0eCk7XG4gICAgfTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGNyZWF0ZVN0YXRlbWVudFNob3J0Y3V0SGFuZGxlcihcbiAgICBzdGF0ZW1lbnRLZXk6IHN0cmluZyxcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nXG4gICkge1xuICAgIHJldHVybiBhc3luYyBmdW5jdGlvbiBzdGF0ZW1lbnRTaG9ydGN1dChcbiAgICAgIHRoaXM6IGFueSxcbiAgICAgIC4uLmFyZ3M6IGFueVtdXG4gICAgKTogUHJvbWlzZTxhbnk+IHtcbiAgICAgIGNvbnN0IHsgY3R4IH0gPSAoXG4gICAgICAgIGF3YWl0IHRoaXMubG9nQ3R4KFtdLCBzdGF0ZW1lbnRLZXksIHRydWUpXG4gICAgICApLmZvcihzdGF0ZW1lbnRTaG9ydGN1dCk7XG5cbiAgICAgIHN3aXRjaCAoc3RhdGVtZW50S2V5KSB7XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkxJU1RfQlk6IHtcbiAgICAgICAgICBjb25zdCBba2V5LCBkZXRhaWxzXSA9IGFyZ3M7XG4gICAgICAgICAgcmV0dXJuIHRoaXMucGVyc2lzdGVuY2UoY3R4KS5saXN0QnkoXG4gICAgICAgICAgICBrZXksXG4gICAgICAgICAgICAoZGV0YWlscyBhcyBhbnkpPy5kaXJlY3Rpb24gYXMgT3JkZXJEaXJlY3Rpb24sXG4gICAgICAgICAgICBjdHhcbiAgICAgICAgICApO1xuICAgICAgICB9XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLlBBR0VfQlk6IHtcbiAgICAgICAgICBjb25zdCBba2V5LCBwYWdlLCBkZXRhaWxzXSA9IGFyZ3M7XG4gICAgICAgICAgcmV0dXJuIHRoaXMucGVyc2lzdGVuY2UoY3R4KS5wYWdpbmF0ZUJ5KFxuICAgICAgICAgICAga2V5LFxuICAgICAgICAgICAgKGRldGFpbHMgYXMgYW55KT8uZGlyZWN0aW9uIGFzIE9yZGVyRGlyZWN0aW9uLFxuICAgICAgICAgICAge1xuICAgICAgICAgICAgICBsaW1pdDogKGRldGFpbHMgYXMgYW55KT8ubGltaXQsXG4gICAgICAgICAgICAgIG9mZnNldDogKGRldGFpbHMgYXMgYW55KT8ub2Zmc2V0LFxuICAgICAgICAgICAgICBwYWdlLFxuICAgICAgICAgICAgfSBhcyBhbnksXG4gICAgICAgICAgICBjdHhcbiAgICAgICAgICApO1xuICAgICAgICB9XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkZJTkQ6IHtcbiAgICAgICAgICBjb25zdCBbdmFsdWUsIGRldGFpbHNdID0gYXJncztcbiAgICAgICAgICBjb25zdCBkaXJlY3Rpb24gPVxuICAgICAgICAgICAgKGRldGFpbHMgYXMgYW55KT8uZGlyZWN0aW9uID8/IE9yZGVyRGlyZWN0aW9uLkFTQztcbiAgICAgICAgICBjb25zdCBwZXJzaXN0ZW5jZSA9IHRoaXMucGVyc2lzdGVuY2UoY3R4KTtcbiAgICAgICAgICBpZiAodHlwZW9mIHBlcnNpc3RlbmNlLmZpbmQgPT09IFwiZnVuY3Rpb25cIilcbiAgICAgICAgICAgIHJldHVybiBwZXJzaXN0ZW5jZS5maW5kKHZhbHVlLCBkaXJlY3Rpb24sIGN0eCk7XG4gICAgICAgICAgcmV0dXJuIHBlcnNpc3RlbmNlLnN0YXRlbWVudChQcmVwYXJlZFN0YXRlbWVudEtleXMuRklORCwgdmFsdWUsIGRpcmVjdGlvbiwgY3R4KTtcbiAgICAgICAgfVxuICAgICAgICBjYXNlIFByZXBhcmVkU3RhdGVtZW50S2V5cy5QQUdFOiB7XG4gICAgICAgICAgY29uc3QgW3ZhbHVlLCBkZXRhaWxzXSA9IGFyZ3M7XG4gICAgICAgICAgY29uc3QgcmVmID0ge1xuICAgICAgICAgICAgb2Zmc2V0OiAoZGV0YWlscyBhcyBhbnkpPy5vZmZzZXQgPz8gMSxcbiAgICAgICAgICAgIGxpbWl0OiAoZGV0YWlscyBhcyBhbnkpPy5saW1pdCA/PyAxMCxcbiAgICAgICAgICAgIGJvb2ttYXJrOiAoZGV0YWlscyBhcyBhbnkpPy5ib29rbWFyayxcbiAgICAgICAgICB9O1xuICAgICAgICAgIGNvbnN0IHBlcnNpc3RlbmNlID0gdGhpcy5wZXJzaXN0ZW5jZShjdHgpO1xuICAgICAgICAgIGNvbnN0IGRpcmVjdGlvbiA9IChkZXRhaWxzIGFzIGFueSk/LmRpcmVjdGlvbiA/PyBPcmRlckRpcmVjdGlvbi5BU0M7XG4gICAgICAgICAgaWYgKHR5cGVvZiBwZXJzaXN0ZW5jZS5wYWdlID09PSBcImZ1bmN0aW9uXCIpXG4gICAgICAgICAgICByZXR1cm4gcGVyc2lzdGVuY2UucGFnZSh2YWx1ZSwgZGlyZWN0aW9uLCByZWYsIGN0eCk7XG4gICAgICAgICAgcmV0dXJuIHBlcnNpc3RlbmNlLnN0YXRlbWVudChQcmVwYXJlZFN0YXRlbWVudEtleXMuUEFHRSwgdmFsdWUsIGRpcmVjdGlvbiwgcmVmLCBjdHgpO1xuICAgICAgICB9XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkZJTkRfT05FX0JZOiB7XG4gICAgICAgICAgY29uc3QgW2tleSwgdmFsdWVdID0gYXJncztcbiAgICAgICAgICByZXR1cm4gdGhpcy5wZXJzaXN0ZW5jZShjdHgpLmZpbmRPbmVCeShrZXksIHZhbHVlLCBjdHgpO1xuICAgICAgICB9XG4gICAgICAgIGNhc2UgUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkZJTkRfQlk6IHtcbiAgICAgICAgICBjb25zdCBba2V5LCB2YWx1ZV0gPSBhcmdzO1xuICAgICAgICAgIHJldHVybiB0aGlzLnBlcnNpc3RlbmNlKGN0eClcbiAgICAgICAgICAgIC5mb3IoY3R4LnRvT3ZlcnJpZGVzKCkpXG4gICAgICAgICAgICAuZmluZEJ5KGtleSwgdmFsdWUsIGN0eCk7XG4gICAgICAgIH1cbiAgICAgICAgZGVmYXVsdDpcbiAgICAgICAgICBpZiAoXG4gICAgICAgICAgICBzdGF0ZW1lbnRLZXkgPT09IFByZXBhcmVkU3RhdGVtZW50S2V5cy5DT1VOVF9PRiB8fFxuICAgICAgICAgICAgc3RhdGVtZW50S2V5ID09PSBQcmVwYXJlZFN0YXRlbWVudEtleXMuTUFYX09GIHx8XG4gICAgICAgICAgICBzdGF0ZW1lbnRLZXkgPT09IFByZXBhcmVkU3RhdGVtZW50S2V5cy5NSU5fT0YgfHxcbiAgICAgICAgICAgIHN0YXRlbWVudEtleSA9PT0gUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkFWR19PRiB8fFxuICAgICAgICAgICAgc3RhdGVtZW50S2V5ID09PSBQcmVwYXJlZFN0YXRlbWVudEtleXMuU1VNX09GIHx8XG4gICAgICAgICAgICBzdGF0ZW1lbnRLZXkgPT09IFByZXBhcmVkU3RhdGVtZW50S2V5cy5ESVNUSU5DVF9PRiB8fFxuICAgICAgICAgICAgc3RhdGVtZW50S2V5ID09PSBQcmVwYXJlZFN0YXRlbWVudEtleXMuR1JPVVBfT0ZcbiAgICAgICAgICApIHtcbiAgICAgICAgICAgIGNvbnN0IFtmaWVsZF0gPSBhcmdzO1xuICAgICAgICAgICAgcmV0dXJuIHRoaXMucGVyc2lzdGVuY2UoY3R4KS5zdGF0ZW1lbnQoc3RhdGVtZW50S2V5LCBmaWVsZCwgY3R4KTtcbiAgICAgICAgICB9XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBVbmtub3duIHN0YXRlbWVudDogJHtzdGF0ZW1lbnRLZXl9YCk7XG4gICAgICB9XG4gICAgfTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGNyZWF0ZUNvbXBsZXhRdWVyeUhhbmRsZXIobWV0aG9kTmFtZTogc3RyaW5nKSB7XG4gICAgcmV0dXJuIGFzeW5jIGZ1bmN0aW9uIGNvbXBsZXhRdWVyeShcbiAgICAgIHRoaXM6IGFueSxcbiAgICAgIC4uLmFyZ3M6IGFueVtdXG4gICAgKTogUHJvbWlzZTxhbnk+IHtcbiAgICAgIGNvbnN0IGxvZzogYW55ID0gdGhpcy5sb2c/LmZvcj8uKGNvbXBsZXhRdWVyeSk7XG4gICAgICB0cnkge1xuICAgICAgICBpZiAobG9nKSBsb2cuZGVidWcoYEludm9raW5nIGN1c3RvbSBxdWVyeSBcIiR7bWV0aG9kTmFtZX1cImApO1xuICAgICAgICBjb25zdCB7IGN0eCB9ID0gKFxuICAgICAgICAgIGF3YWl0IHRoaXMubG9nQ3R4KFtdLCBtZXRob2ROYW1lLCB0cnVlKVxuICAgICAgICApLmZvcihjb21wbGV4UXVlcnkpO1xuICAgICAgICBjb25zdCBwZXJzaXN0ZW5jZSA9IHRoaXMucGVyc2lzdGVuY2UoY3R4KTtcbiAgICAgICAgY29uc3Qgc3ByZWFkQXJncyA9IEZyb21Nb2RlbENvbnRyb2xsZXIuZXh0cmFjdFF1ZXJ5QXJncyhhcmdzKTtcbiAgICAgICAgaWYgKHR5cGVvZiBwZXJzaXN0ZW5jZVttZXRob2ROYW1lXSA9PT0gXCJmdW5jdGlvblwiKSB7XG4gICAgICAgICAgcmV0dXJuIHBlcnNpc3RlbmNlW21ldGhvZE5hbWVdKC4uLnNwcmVhZEFyZ3MsIGN0eCk7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKHR5cGVvZiBwZXJzaXN0ZW5jZS5zdGF0ZW1lbnQgPT09IFwiZnVuY3Rpb25cIikge1xuICAgICAgICAgIHJldHVybiBwZXJzaXN0ZW5jZS5zdGF0ZW1lbnQobWV0aG9kTmFtZSwgLi4uc3ByZWFkQXJncywgY3R4KTtcbiAgICAgICAgfVxuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgYFBlcnNpc3RlbmNlIG1ldGhvZCBcIiR7bWV0aG9kTmFtZX1cIiBub3QgZm91bmQgb24gJHtwZXJzaXN0ZW5jZT8uY29uc3RydWN0b3I/Lm5hbWV9YFxuICAgICAgICApO1xuICAgICAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgICAgIGlmIChsb2cpIGxvZy5lcnJvcihgQ3VzdG9tIHF1ZXJ5IFwiJHttZXRob2ROYW1lfVwiIGZhaWxlZGAsIGUpO1xuICAgICAgICB0aHJvdyBlO1xuICAgICAgfVxuICAgIH07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBjcmVhdGVDdXN0b21Sb3V0ZUhhbmRsZXIobWV0aG9kTmFtZTogc3RyaW5nKSB7XG4gICAgcmV0dXJuIGFzeW5jIGZ1bmN0aW9uIGN1c3RvbVJvdXRlKFxuICAgICAgdGhpczogYW55LFxuICAgICAgLi4uYXJnczogYW55W11cbiAgICApOiBQcm9taXNlPGFueT4ge1xuICAgICAgY29uc3QgbG9nOiBhbnkgPSB0aGlzLmxvZz8uZm9yPy4oY3VzdG9tUm91dGUpO1xuICAgICAgY29uc3QgeyBjdHggfSA9IChcbiAgICAgICAgYXdhaXQgdGhpcy5sb2dDdHgoW10sIG1ldGhvZE5hbWUsIHRydWUpXG4gICAgICApLmZvcihjdXN0b21Sb3V0ZSk7XG4gICAgICBjb25zdCBwZXJzaXN0ZW5jZSA9IHRoaXMucGVyc2lzdGVuY2UoY3R4KTtcbiAgICAgIGNvbnN0IHNwcmVhZEFyZ3MgPSBGcm9tTW9kZWxDb250cm9sbGVyLmV4dHJhY3RRdWVyeUFyZ3MoYXJncyk7XG5cbiAgICAgIC8vIFRyeSB0aGUgcGVyc2lzdGVuY2UgZGlyZWN0bHkgKHdvcmtzIHdoZW4gaXQncyBhIGN1c3RvbSBSZXBvc2l0b3J5KVxuICAgICAgaWYgKHR5cGVvZiBwZXJzaXN0ZW5jZVttZXRob2ROYW1lXSA9PT0gXCJmdW5jdGlvblwiKSB7XG4gICAgICAgIHJldHVybiBwZXJzaXN0ZW5jZVttZXRob2ROYW1lXSguLi5zcHJlYWRBcmdzLCBjdHgpO1xuICAgICAgfVxuICAgICAgLy8gV2hlbiBwZXJzaXN0ZW5jZSBpcyBhIE1vZGVsU2VydmljZSwgdGhlIG1ldGhvZCBsaXZlcyBvbiB0aGUgdW5kZXJseWluZyByZXBvXG4gICAgICBpZiAocGVyc2lzdGVuY2U/LnJlcG8gJiYgdHlwZW9mIHBlcnNpc3RlbmNlLnJlcG9bbWV0aG9kTmFtZV0gPT09IFwiZnVuY3Rpb25cIikge1xuICAgICAgICByZXR1cm4gcGVyc2lzdGVuY2UucmVwb1ttZXRob2ROYW1lXSguLi5zcHJlYWRBcmdzLCBjdHgpO1xuICAgICAgfVxuICAgICAgLy8gRmFsbCBiYWNrIHRvIHN0YXRlbWVudCBnYXRld2F5XG4gICAgICBpZiAodHlwZW9mIHBlcnNpc3RlbmNlLnN0YXRlbWVudCA9PT0gXCJmdW5jdGlvblwiKSB7XG4gICAgICAgIHJldHVybiBwZXJzaXN0ZW5jZS5zdGF0ZW1lbnQobWV0aG9kTmFtZSwgLi4uc3ByZWFkQXJncywgY3R4KTtcbiAgICAgIH1cbiAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgYE1ldGhvZCBcIiR7bWV0aG9kTmFtZX1cIiBub3QgZm91bmQgb24gJHtwZXJzaXN0ZW5jZT8uY29uc3RydWN0b3I/Lm5hbWV9IG9yIGl0cyByZXBvYFxuICAgICAgKTtcbiAgICB9O1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgZXh0cmFjdFF1ZXJ5QXJncyhhcmdzOiBhbnlbXSk6IGFueVtdIHtcbiAgICBpZiAoYXJncy5sZW5ndGggPT09IDApIHJldHVybiBhcmdzO1xuICAgIGNvbnN0IGxhc3QgPSBhcmdzW2FyZ3MubGVuZ3RoIC0gMV07XG4gICAgaWYgKFxuICAgICAgbGFzdCAmJlxuICAgICAgdHlwZW9mIGxhc3QgPT09IFwib2JqZWN0XCIgJiZcbiAgICAgICFBcnJheS5pc0FycmF5KGxhc3QpXG4gICAgKSB7XG4gICAgICBjb25zdCBxdWVyeU9iaiA9IGFyZ3MucG9wKCk7XG4gICAgICBjb25zdCBoYXNEaXJlY3Rpb24gPSBxdWVyeU9iai5kaXJlY3Rpb24gIT09IHVuZGVmaW5lZDtcbiAgICAgIGNvbnN0IGhhc0xpbWl0ID0gcXVlcnlPYmoubGltaXQgIT09IHVuZGVmaW5lZDtcbiAgICAgIGNvbnN0IGhhc09mZnNldCA9IHF1ZXJ5T2JqLm9mZnNldCAhPT0gdW5kZWZpbmVkO1xuICAgICAgaWYgKCFoYXNEaXJlY3Rpb24gJiYgIWhhc0xpbWl0ICYmICFoYXNPZmZzZXQpIHJldHVybiBhcmdzO1xuICAgICAgY29uc3QgZXh0cmFzOiBhbnlbXSA9IFtdO1xuICAgICAgaWYgKGhhc0RpcmVjdGlvbikgZXh0cmFzLnB1c2gocXVlcnlPYmouZGlyZWN0aW9uKTtcbiAgICAgIGVsc2UgaWYgKGhhc0xpbWl0IHx8IGhhc09mZnNldCkgZXh0cmFzLnB1c2godW5kZWZpbmVkKTtcbiAgICAgIGlmIChoYXNMaW1pdCkgZXh0cmFzLnB1c2gocXVlcnlPYmoubGltaXQpO1xuICAgICAgaWYgKGhhc09mZnNldCkgZXh0cmFzLnB1c2gocXVlcnlPYmoub2Zmc2V0KTtcbiAgICAgIHJldHVybiBbLi4uYXJncywgLi4uZXh0cmFzXTtcbiAgICB9XG4gICAgcmV0dXJuIGFyZ3M7XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBjcmVhdGVDcmVhdGVEZWNvcmF0b3JzKFxuICAgIE1vZGVsQ29uc3RyOiBNb2RlbENvbnN0cnVjdG9yPGFueT4sXG4gICAgbW9kZWxDbGF6ek5hbWU6IHN0cmluZ1xuICApOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+IHtcbiAgICByZXR1cm4gW1xuICAgICAgQXBpT3BlcmF0aW9uRnJvbU1vZGVsKE1vZGVsQ29uc3RyLCBcIlBPU1RcIiksXG4gICAgICBBcGlPcGVyYXRpb24oeyBzdW1tYXJ5OiBgQ3JlYXRlIGEgbmV3ICR7bW9kZWxDbGF6ek5hbWV9LmAgfSksXG4gICAgICBBcGlCb2R5KHtcbiAgICAgICAgZGVzY3JpcHRpb246IGBQYXlsb2FkIGZvciAke21vZGVsQ2xhenpOYW1lfWAsXG4gICAgICAgIHR5cGU6IER0b0ZvcihPcGVyYXRpb25LZXlzLkNSRUFURSwgTW9kZWxDb25zdHIpLFxuICAgICAgfSksXG4gICAgICBBcGlDcmVhdGVkUmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYCR7bW9kZWxDbGF6ek5hbWV9IGNyZWF0ZWQgc3VjY2Vzc2Z1bGx5LmAsXG4gICAgICAgIHNjaGVtYTogeyAkcmVmOiBnZXRTY2hlbWFQYXRoKE1vZGVsQ29uc3RyKSB9LFxuICAgICAgfSksXG4gICAgICBBcGlCYWRSZXF1ZXN0UmVzcG9uc2UoeyBkZXNjcmlwdGlvbjogXCJQYXlsb2FkIHZhbGlkYXRpb24gZmFpbGVkLlwiIH0pLFxuICAgICAgQXBpVW5wcm9jZXNzYWJsZUVudGl0eVJlc3BvbnNlKHtcbiAgICAgICAgZGVzY3JpcHRpb246IFwiUmVwb3NpdG9yeSByZWplY3RlZCB0aGUgcHJvdmlkZWQgcGF5bG9hZC5cIixcbiAgICAgIH0pLFxuICAgIF07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBidWxrQ3JlYXRlRGVjb3JhdG9ycyhcbiAgICBNb2RlbENvbnN0cjogTW9kZWxDb25zdHJ1Y3Rvcjxhbnk+LFxuICAgIG1vZGVsQ2xhenpOYW1lOiBzdHJpbmdcbiAgKTogQXJyYXk8KHRhcmdldDogYW55LCBrZXk6IHN0cmluZywgZGVzYzogYW55KSA9PiB2b2lkPiB7XG4gICAgcmV0dXJuIFtcbiAgICAgIEFwaU9wZXJhdGlvbkZyb21Nb2RlbChNb2RlbENvbnN0ciwgXCJQT1NUXCIsIFwiYnVsa1wiKSxcbiAgICAgIEFwaU9wZXJhdGlvbih7IHN1bW1hcnk6IGBDcmVhdGUgYSBuZXcgJHttb2RlbENsYXp6TmFtZX0uYCB9KSxcbiAgICAgIEFwaUJvZHkoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYFBheWxvYWQgZm9yICR7bW9kZWxDbGF6ek5hbWV9YCxcbiAgICAgICAgc2NoZW1hOiB7XG4gICAgICAgICAgdHlwZTogXCJhcnJheVwiLFxuICAgICAgICAgIGl0ZW1zOiB7ICRyZWY6IGdldFNjaGVtYVBhdGgoTW9kZWxDb25zdHIpIH0sXG4gICAgICAgIH0sXG4gICAgICB9KSxcbiAgICAgIEFwaUNyZWF0ZWRSZXNwb25zZSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBgJHttb2RlbENsYXp6TmFtZX0gY3JlYXRlZCBzdWNjZXNzZnVsbHkuYCxcbiAgICAgICAgc2NoZW1hOiB7XG4gICAgICAgICAgdHlwZTogXCJhcnJheVwiLFxuICAgICAgICAgIGl0ZW1zOiB7ICRyZWY6IGdldFNjaGVtYVBhdGgoTW9kZWxDb25zdHIpIH0sXG4gICAgICAgIH0sXG4gICAgICB9KSxcbiAgICAgIEFwaUJhZFJlcXVlc3RSZXNwb25zZSh7IGRlc2NyaXB0aW9uOiBcIlBheWxvYWQgdmFsaWRhdGlvbiBmYWlsZWQuXCIgfSksXG4gICAgICBBcGlVbnByb2Nlc3NhYmxlRW50aXR5UmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogXCJSZXBvc2l0b3J5IHJlamVjdGVkIHRoZSBwcm92aWRlZCBwYXlsb2FkLlwiLFxuICAgICAgfSksXG4gICAgXTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGJ1bGtSZWFkRGVjb3JhdG9ycyhcbiAgICBNb2RlbENvbnN0cjogTW9kZWxDb25zdHJ1Y3Rvcjxhbnk+LFxuICAgIG1vZGVsQ2xhenpOYW1lOiBzdHJpbmdcbiAgKTogQXJyYXk8KHRhcmdldDogYW55LCBrZXk6IHN0cmluZywgZGVzYzogYW55KSA9PiB2b2lkPiB7XG4gICAgcmV0dXJuIFtcbiAgICAgIEFwaU9wZXJhdGlvbkZyb21Nb2RlbChNb2RlbENvbnN0ciwgXCJHRVRcIiwgXCJidWxrXCIpLFxuICAgICAgQXBpT3BlcmF0aW9uKHsgc3VtbWFyeTogYFJldHJpZXZlICR7bW9kZWxDbGF6ek5hbWV9IHJlY29yZHMgYnkgaWRzLmAgfSksXG4gICAgICBBcGlRdWVyeSh7IG5hbWU6IFwiaWRzXCIsIHJlcXVpcmVkOiB0cnVlLCB0eXBlOiBcImFycmF5XCIgfSksXG4gICAgICBBcGlPa1Jlc3BvbnNlKHtcbiAgICAgICAgZGVzY3JpcHRpb246IGAke21vZGVsQ2xhenpOYW1lfSByZXRyaWV2ZWQgc3VjY2Vzc2Z1bGx5LmAsXG4gICAgICAgIHNjaGVtYToge1xuICAgICAgICAgIHR5cGU6IFwiYXJyYXlcIixcbiAgICAgICAgICBpdGVtczogeyAkcmVmOiBnZXRTY2hlbWFQYXRoKE1vZGVsQ29uc3RyKSB9LFxuICAgICAgICB9LFxuICAgICAgfSksXG4gICAgICBBcGlOb3RGb3VuZFJlc3BvbnNlKHtcbiAgICAgICAgZGVzY3JpcHRpb246IGBObyAke21vZGVsQ2xhenpOYW1lfSByZWNvcmQgbWF0Y2hlcyB0aGUgcHJvdmlkZWQgaWRlbnRpZmllci5gLFxuICAgICAgfSksXG4gICAgXTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGJ1bGtVcGRhdGVEZWNvcmF0b3JzKFxuICAgIE1vZGVsQ29uc3RyOiBNb2RlbENvbnN0cnVjdG9yPGFueT4sXG4gICAgbW9kZWxDbGF6ek5hbWU6IHN0cmluZyxcbiAgICBhcGlQcm9wZXJ0aWVzOiBEZWNhZkFwaVByb3BlcnR5W11cbiAgKTogQXJyYXk8KHRhcmdldDogYW55LCBrZXk6IHN0cmluZywgZGVzYzogYW55KSA9PiB2b2lkPiB7XG4gICAgcmV0dXJuIFtcbiAgICAgIEFwaU9wZXJhdGlvbkZyb21Nb2RlbChNb2RlbENvbnN0ciwgXCJQVVRcIiwgXCJidWxrXCIpLFxuICAgICAgQXBpUGFyYW1zRnJvbU1vZGVsKGFwaVByb3BlcnRpZXMpLFxuICAgICAgQXBpT3BlcmF0aW9uKHtcbiAgICAgICAgc3VtbWFyeTogYFJlcGxhY2UgZXhpc3RpbmcgJHttb2RlbENsYXp6TmFtZX0gcmVjb3JkcyB3aXRoIG5ldyBwYXlsb2Fkcy5gLFxuICAgICAgfSksXG4gICAgICBBcGlCb2R5KHtcbiAgICAgICAgZGVzY3JpcHRpb246IGBQYXlsb2FkIGZvciByZXBsYWNpbmcgZXhpc3RpbmcgcmVjb3JkcyBvZiAke21vZGVsQ2xhenpOYW1lfWAsXG4gICAgICAgIHNjaGVtYToge1xuICAgICAgICAgIHR5cGU6IFwiYXJyYXlcIixcbiAgICAgICAgICAkcmVmOiBnZXRTY2hlbWFQYXRoKER0b0ZvcihPcGVyYXRpb25LZXlzLlVQREFURSwgTW9kZWxDb25zdHIpKSxcbiAgICAgICAgfSxcbiAgICAgIH0pLFxuICAgICAgQXBpT2tSZXNwb25zZSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBgJHttb2RlbENsYXp6TmFtZX0gdXBkYXRlZCBzdWNjZXNzZnVsbHkuYCxcbiAgICAgICAgc2NoZW1hOiB7XG4gICAgICAgICAgdHlwZTogXCJhcnJheVwiLFxuICAgICAgICAgIGl0ZW1zOiB7ICRyZWY6IGdldFNjaGVtYVBhdGgoTW9kZWxDb25zdHIpIH0sXG4gICAgICAgIH0sXG4gICAgICB9KSxcbiAgICAgIEFwaU5vdEZvdW5kUmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYE5vICR7bW9kZWxDbGF6ek5hbWV9IHJlY29yZCBtYXRjaGVzIHRoZSBwcm92aWRlZCBpZGVudGlmaWVyLmAsXG4gICAgICB9KSxcbiAgICAgIEFwaUJhZFJlcXVlc3RSZXNwb25zZSh7IGRlc2NyaXB0aW9uOiBcIlBheWxvYWQgdmFsaWRhdGlvbiBmYWlsZWQuXCIgfSksXG4gICAgXTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIGJ1bGtEZWxldGVEZWNvcmF0b3JzKFxuICAgIE1vZGVsQ29uc3RyOiBNb2RlbENvbnN0cnVjdG9yPGFueT4sXG4gICAgbW9kZWxDbGF6ek5hbWU6IHN0cmluZyxcbiAgICBhcGlQcm9wZXJ0aWVzOiBEZWNhZkFwaVByb3BlcnR5W11cbiAgKTogQXJyYXk8KHRhcmdldDogYW55LCBrZXk6IHN0cmluZywgZGVzYzogYW55KSA9PiB2b2lkPiB7XG4gICAgcmV0dXJuIFtcbiAgICAgIEFwaU9wZXJhdGlvbkZyb21Nb2RlbChNb2RlbENvbnN0ciwgXCJERUxFVEVcIiwgXCJidWxrXCIpLFxuICAgICAgQXBpUGFyYW1zRnJvbU1vZGVsKGFwaVByb3BlcnRpZXMpLFxuICAgICAgQXBpT3BlcmF0aW9uKHsgc3VtbWFyeTogYERlbGV0ZSAke21vZGVsQ2xhenpOYW1lfSByZWNvcmRzIGJ5IGlkcy5gIH0pLFxuICAgICAgQXBpUXVlcnkoeyBuYW1lOiBcImlkc1wiLCByZXF1aXJlZDogdHJ1ZSwgdHlwZTogXCJhcnJheVwiIH0pLFxuICAgICAgQXBpT2tSZXNwb25zZSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBgJHttb2RlbENsYXp6TmFtZX0gZGVsZXRlZCBzdWNjZXNzZnVsbHkuYCxcbiAgICAgICAgc2NoZW1hOiB7XG4gICAgICAgICAgdHlwZTogXCJhcnJheVwiLFxuICAgICAgICAgIGl0ZW1zOiB7ICRyZWY6IGdldFNjaGVtYVBhdGgoTW9kZWxDb25zdHIpIH0sXG4gICAgICAgIH0sXG4gICAgICB9KSxcbiAgICAgIEFwaU5vdEZvdW5kUmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYE5vICR7bW9kZWxDbGF6ek5hbWV9IHJlY29yZCBtYXRjaGVzIHRoZSBwcm92aWRlZCBpZGVudGlmaWVyLmAsXG4gICAgICB9KSxcbiAgICBdO1xuICB9XG5cbiAgcHJpdmF0ZSBzdGF0aWMgcmVhZERlY29yYXRvcnMoXG4gICAgTW9kZWxDb25zdHI6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nLFxuICAgIGFwaVByb3BlcnRpZXM6IERlY2FmQXBpUHJvcGVydHlbXSxcbiAgICBwa1BhdGg6IHN0cmluZ1xuICApOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+IHtcbiAgICByZXR1cm4gW1xuICAgICAgQXBpT3BlcmF0aW9uRnJvbU1vZGVsKE1vZGVsQ29uc3RyLCBcIkdFVFwiLCBwa1BhdGgpLFxuICAgICAgQXBpUGFyYW1zRnJvbU1vZGVsKGFwaVByb3BlcnRpZXMpLFxuICAgICAgQXBpT3BlcmF0aW9uKHsgc3VtbWFyeTogYFJldHJpZXZlIGEgJHttb2RlbENsYXp6TmFtZX0gcmVjb3JkIGJ5IGlkLmAgfSksXG4gICAgICBBcGlPa1Jlc3BvbnNlKHtcbiAgICAgICAgZGVzY3JpcHRpb246IGAke21vZGVsQ2xhenpOYW1lfSByZXRyaWV2ZWQgc3VjY2Vzc2Z1bGx5LmAsXG4gICAgICAgIHNjaGVtYTogeyAkcmVmOiBnZXRTY2hlbWFQYXRoKE1vZGVsQ29uc3RyKSB9LFxuICAgICAgfSksXG4gICAgICBBcGlOb3RGb3VuZFJlc3BvbnNlKHtcbiAgICAgICAgZGVzY3JpcHRpb246IGBObyAke21vZGVsQ2xhenpOYW1lfSByZWNvcmQgbWF0Y2hlcyB0aGUgcHJvdmlkZWQgaWRlbnRpZmllci5gLFxuICAgICAgfSksXG4gICAgXTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIHVwZGF0ZURlY29yYXRvcnMoXG4gICAgTW9kZWxDb25zdHI6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nLFxuICAgIGFwaVByb3BlcnRpZXM6IERlY2FmQXBpUHJvcGVydHlbXSxcbiAgICBwa1BhdGg6IHN0cmluZ1xuICApOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+IHtcbiAgICByZXR1cm4gW1xuICAgICAgQXBpT3BlcmF0aW9uRnJvbU1vZGVsKE1vZGVsQ29uc3RyLCBcIlBVVFwiLCBwa1BhdGgpLFxuICAgICAgQXBpUGFyYW1zRnJvbU1vZGVsKGFwaVByb3BlcnRpZXMpLFxuICAgICAgQXBpT3BlcmF0aW9uKHtcbiAgICAgICAgc3VtbWFyeTogYFJlcGxhY2UgYW4gZXhpc3RpbmcgJHttb2RlbENsYXp6TmFtZX0gcmVjb3JkIHdpdGggYSBuZXcgcGF5bG9hZC5gLFxuICAgICAgfSksXG4gICAgICBBcGlCb2R5KHtcbiAgICAgICAgZGVzY3JpcHRpb246IGBQYXlsb2FkIGZvciByZXBsYWNpbmcgYW4gZXhpc3RpbmcgcmVjb3JkIG9mICR7bW9kZWxDbGF6ek5hbWV9YCxcbiAgICAgICAgdHlwZTogRHRvRm9yKE9wZXJhdGlvbktleXMuVVBEQVRFLCBNb2RlbENvbnN0ciksXG4gICAgICB9KSxcbiAgICAgIEFwaU9rUmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYCR7bW9kZWxDbGF6ek5hbWV9IHVwZGF0ZWQgc3VjY2Vzc2Z1bGx5LmAsXG4gICAgICAgIHNjaGVtYTogeyAkcmVmOiBnZXRTY2hlbWFQYXRoKE1vZGVsQ29uc3RyKSB9LFxuICAgICAgfSksXG4gICAgICBBcGlOb3RGb3VuZFJlc3BvbnNlKHtcbiAgICAgICAgZGVzY3JpcHRpb246IGBObyAke21vZGVsQ2xhenpOYW1lfSByZWNvcmQgbWF0Y2hlcyB0aGUgcHJvdmlkZWQgaWRlbnRpZmllci5gLFxuICAgICAgfSksXG4gICAgICBBcGlCYWRSZXF1ZXN0UmVzcG9uc2UoeyBkZXNjcmlwdGlvbjogXCJQYXlsb2FkIHZhbGlkYXRpb24gZmFpbGVkLlwiIH0pLFxuICAgIF07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBkZWxldGVEZWNvcmF0b3JzKFxuICAgIE1vZGVsQ29uc3RyOiBNb2RlbENvbnN0cnVjdG9yPGFueT4sXG4gICAgbW9kZWxDbGF6ek5hbWU6IHN0cmluZyxcbiAgICBhcGlQcm9wZXJ0aWVzOiBEZWNhZkFwaVByb3BlcnR5W10sXG4gICAgcGtQYXRoOiBzdHJpbmdcbiAgKTogQXJyYXk8KHRhcmdldDogYW55LCBrZXk6IHN0cmluZywgZGVzYzogYW55KSA9PiB2b2lkPiB7XG4gICAgcmV0dXJuIFtcbiAgICAgIEFwaU9wZXJhdGlvbkZyb21Nb2RlbChNb2RlbENvbnN0ciwgXCJERUxFVEVcIiwgcGtQYXRoKSxcbiAgICAgIEFwaVBhcmFtc0Zyb21Nb2RlbChhcGlQcm9wZXJ0aWVzKSxcbiAgICAgIEFwaU9wZXJhdGlvbih7IHN1bW1hcnk6IGBEZWxldGUgYSAke21vZGVsQ2xhenpOYW1lfSByZWNvcmQgYnkgaWQuYCB9KSxcbiAgICAgIEFwaU9rUmVzcG9uc2Uoe1xuICAgICAgICBkZXNjcmlwdGlvbjogYCR7bW9kZWxDbGF6ek5hbWV9IGRlbGV0ZWQgc3VjY2Vzc2Z1bGx5LmAsXG4gICAgICAgIHNjaGVtYTogeyAkcmVmOiBnZXRTY2hlbWFQYXRoKE1vZGVsQ29uc3RyKSB9LFxuICAgICAgfSksXG4gICAgICBBcGlOb3RGb3VuZFJlc3BvbnNlKHtcbiAgICAgICAgZGVzY3JpcHRpb246IGBObyAke21vZGVsQ2xhenpOYW1lfSByZWNvcmQgbWF0Y2hlcyB0aGUgcHJvdmlkZWQgaWRlbnRpZmllci5gLFxuICAgICAgfSksXG4gICAgXTtcbiAgfVxuXG4gIHByaXZhdGUgc3RhdGljIHN0YXRlbWVudERlY29yYXRvcnMoXG4gICAgTW9kZWxDb25zdHI6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nXG4gICk6IEFycmF5PCh0YXJnZXQ6IGFueSwga2V5OiBzdHJpbmcsIGRlc2M6IGFueSkgPT4gdm9pZD4ge1xuICAgIHJldHVybiBbXG4gICAgICBBcGlPcGVyYXRpb25Gcm9tTW9kZWwoTW9kZWxDb25zdHIsIFwiR0VUXCIsIFwic3RhdGVtZW50LzptZXRob2QvKmFyZ3NcIiksXG4gICAgICBBcGlPcGVyYXRpb24oe1xuICAgICAgICBzdW1tYXJ5OiBgRXhlY3V0ZXMgYSBwcmVwYXJlZCBzdGF0ZW1lbnQgb24gJHttb2RlbENsYXp6TmFtZX0uYCxcbiAgICAgIH0pLFxuICAgICAgQXBpUGFyYW0oeyBuYW1lOiBcIm1ldGhvZFwiLCBkZXNjcmlwdGlvbjogXCJ0aGUgcHJlcGFyZWQgc3RhdGVtZW50IHRvIGV4ZWN1dGVcIiB9KSxcbiAgICAgIEFwaVBhcmFtKHtcbiAgICAgICAgbmFtZTogXCJhcmdzXCIsXG4gICAgICAgIGRlc2NyaXB0aW9uOiBcImNvbmNhdGVuYXRlZCBsaXN0IG9mIGFyZ3VtZW50cyB0aGUgcHJlcGFyZWQgc3RhdGVtZW50IGNhbiBhY2NlcHRcIixcbiAgICAgIH0pLFxuICAgICAgQXBpUXVlcnkoe1xuICAgICAgICBuYW1lOiBcImRpcmVjdGlvblwiLFxuICAgICAgICByZXF1aXJlZDogdHJ1ZSxcbiAgICAgICAgZW51bTogT3JkZXJEaXJlY3Rpb24sXG4gICAgICAgIGRlc2NyaXB0aW9uOiBcInRoZSBzb3J0IG9yZGVyIHdoZW4gYXBwbGljYWJsZVwiLFxuICAgICAgfSksXG4gICAgICBBcGlRdWVyeSh7IG5hbWU6IFwibGltaXRcIiwgcmVxdWlyZWQ6IHRydWUsIGRlc2NyaXB0aW9uOiBcImxpbWl0IG9yIHBhZ2Ugc2l6ZSB3aGVuIGFwcGxpY2FibGVcIiB9KSxcbiAgICAgIEFwaVF1ZXJ5KHsgbmFtZTogXCJvZmZzZXRcIiwgcmVxdWlyZWQ6IHRydWUsIGRlc2NyaXB0aW9uOiBcIm9mZnNldCBvciBib29rbWFyayB3aGVuIGFwcGxpY2FibGVcIiB9KSxcbiAgICAgIEFwaU9rUmVzcG9uc2UoeyBkZXNjcmlwdGlvbjogYCR7bW9kZWxDbGF6ek5hbWV9IGxpc3RlZCBmb3VuZC5gIH0pLFxuICAgICAgQXBpTm90Rm91bmRSZXNwb25zZSh7XG4gICAgICAgIGRlc2NyaXB0aW9uOiBgTm8gJHttb2RlbENsYXp6TmFtZX0gcmVjb3JkIG1hdGNoZXMgdGhlIHByb3ZpZGVkIGlkZW50aWZpZXIuYCxcbiAgICAgIH0pLFxuICAgIF07XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBzdGF0ZW1lbnRTaG9ydGN1dERlY29yYXRvcnMoXG4gICAgTW9kZWxDb25zdHI6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICBtb2RlbENsYXp6TmFtZTogc3RyaW5nLFxuICAgIHBhdGg6IHN0cmluZyxcbiAgICBzdGF0ZW1lbnRLZXk6IHN0cmluZ1xuICApOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+IHtcbiAgICBjb25zdCBiYXNlOiBBcnJheTwodGFyZ2V0OiBhbnksIGtleTogc3RyaW5nLCBkZXNjOiBhbnkpID0+IHZvaWQ+ID0gW1xuICAgICAgQXBpT3BlcmF0aW9uRnJvbU1vZGVsKE1vZGVsQ29uc3RyLCBcIkdFVFwiLCBwYXRoKSxcbiAgICAgIEFwaU9wZXJhdGlvbih7IHN1bW1hcnk6IGBSZXRyaWV2ZSAke21vZGVsQ2xhenpOYW1lfSByZWNvcmRzLmAgfSksXG4gICAgICBBcGlPa1Jlc3BvbnNlKHsgZGVzY3JpcHRpb246IGAke21vZGVsQ2xhenpOYW1lfSByZXRyaWV2ZWQgc3VjY2Vzc2Z1bGx5LmAgfSksXG4gICAgXTtcblxuICAgIGNvbnN0IHNlZ21lbnRzID0gcGF0aC5zcGxpdChcIi9cIikuZmlsdGVyKChzKSA9PiBzLnN0YXJ0c1dpdGgoXCI6XCIpKTtcbiAgICBzZWdtZW50cy5mb3JFYWNoKChzZWcpID0+IHtcbiAgICAgIGNvbnN0IG5hbWUgPSBzZWcucmVwbGFjZShcIjpcIiwgXCJcIik7XG4gICAgICBiYXNlLnB1c2goQXBpUGFyYW0oeyBuYW1lLCBkZXNjcmlwdGlvbjogYFRoZSAke25hbWV9IHBhcmFtZXRlcmAgfSkpO1xuICAgIH0pO1xuXG4gICAgaWYgKFxuICAgICAgcGF0aC5zdGFydHNXaXRoKFwibGlzdEJ5L1wiKSB8fFxuICAgICAgcGF0aC5zdGFydHNXaXRoKFwicGFnaW5hdGVCeS9cIikgfHxcbiAgICAgIHBhdGguc3RhcnRzV2l0aChcImZpbmQvXCIpIHx8XG4gICAgICBwYXRoLnN0YXJ0c1dpdGgoXCJwYWdlL1wiKVxuICAgICkge1xuICAgICAgYmFzZS5wdXNoKFxuICAgICAgICBBcGlRdWVyeSh7XG4gICAgICAgICAgbmFtZTogXCJkaXJlY3Rpb25cIixcbiAgICAgICAgICByZXF1aXJlZDogdHJ1ZSxcbiAgICAgICAgICBlbnVtOiBPcmRlckRpcmVjdGlvbixcbiAgICAgICAgICBkZXNjcmlwdGlvbjogXCJ0aGUgc29ydCBvcmRlclwiLFxuICAgICAgICB9KVxuICAgICAgKTtcbiAgICB9XG4gICAgaWYgKHBhdGguc3RhcnRzV2l0aChcInBhZ2luYXRlQnkvXCIpIHx8IHBhdGguc3RhcnRzV2l0aChcInBhZ2UvXCIpKSB7XG4gICAgICBiYXNlLnB1c2goXG4gICAgICAgIEFwaVF1ZXJ5KHsgbmFtZTogXCJsaW1pdFwiLCByZXF1aXJlZDogZmFsc2UsIGRlc2NyaXB0aW9uOiBcInBhZ2Ugc2l6ZVwiIH0pLFxuICAgICAgICBBcGlRdWVyeSh7IG5hbWU6IFwib2Zmc2V0XCIsIHJlcXVpcmVkOiBmYWxzZSwgZGVzY3JpcHRpb246IFwicGFnZSBudW1iZXJcIiB9KSxcbiAgICAgICAgQXBpUXVlcnkoeyBuYW1lOiBcImJvb2ttYXJrXCIsIHJlcXVpcmVkOiBmYWxzZSwgZGVzY3JpcHRpb246IFwiYm9va21hcmsgZm9yIGN1cnNvciBwYWdpbmF0aW9uXCIgfSlcbiAgICAgICk7XG4gICAgfVxuXG4gICAgaWYgKHBhdGguc3RhcnRzV2l0aChcImZpbmRPbmVCeS9cIikgfHwgcGF0aC5zdGFydHNXaXRoKFwiZmluZEJ5L1wiKSkge1xuICAgICAgYmFzZS5wdXNoKFxuICAgICAgICBBcGlOb3RGb3VuZFJlc3BvbnNlKHtcbiAgICAgICAgICBkZXNjcmlwdGlvbjogYE5vICR7bW9kZWxDbGF6ek5hbWV9IHJlY29yZCBtYXRjaGVzIHRoZSBwcm92aWRlZCBpZGVudGlmaWVyLmAsXG4gICAgICAgIH0pXG4gICAgICApO1xuICAgIH1cblxuICAgIGlmIChcbiAgICAgIHN0YXRlbWVudEtleSA9PT0gUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkNPVU5UX09GIHx8XG4gICAgICBzdGF0ZW1lbnRLZXkgPT09IFByZXBhcmVkU3RhdGVtZW50S2V5cy5BVkdfT0YgfHxcbiAgICAgIHN0YXRlbWVudEtleSA9PT0gUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLlNVTV9PRlxuICAgICkge1xuICAgICAgYmFzZS5wdXNoKEFwaU9rUmVzcG9uc2UoeyBkZXNjcmlwdGlvbjogYFJlc3VsdCBmb3IgJHttb2RlbENsYXp6TmFtZX0uYCwgdHlwZTogTnVtYmVyIH0pKTtcbiAgICB9XG4gICAgaWYgKHN0YXRlbWVudEtleSA9PT0gUHJlcGFyZWRTdGF0ZW1lbnRLZXlzLkRJU1RJTkNUX09GKSB7XG4gICAgICBiYXNlLnB1c2goQXBpT2tSZXNwb25zZSh7IGRlc2NyaXB0aW9uOiBgRGlzdGluY3QgdmFsdWVzIGZvciAke21vZGVsQ2xhenpOYW1lfS5gLCB0eXBlOiBbU3RyaW5nXSB9KSk7XG4gICAgfVxuXG4gICAgcmV0dXJuIGJhc2U7XG4gIH1cblxuICBwcml2YXRlIHN0YXRpYyBnZXRRdWVyeURlY29yYXRvcnMoXG4gICAgbWV0aG9kTmFtZTogc3RyaW5nLFxuICAgIHJvdXRlUGF0aDogc3RyaW5nLFxuICAgIGh0dHBWZXJiOiBzdHJpbmcsXG4gICAgaW5jbHVkZVF1ZXJ5UGFyYW1zOiBib29sZWFuID0gZmFsc2VcbiAgKTogQXJyYXk8KHRhcmdldDogYW55LCBrZXk6IHN0cmluZywgZGVzYzogYW55KSA9PiB2b2lkPiB7XG4gICAgY29uc3QgZXh0cmFjdFBhdGhQYXJhbXMgPSAocDogc3RyaW5nKTogc3RyaW5nW10gPT5cbiAgICAgIHAuc3BsaXQoXCIvXCIpLmZpbHRlcigocykgPT4gcy5zdGFydHNXaXRoKFwiOlwiKSkubWFwKChzKSA9PiBzLnNsaWNlKDEpKTtcblxuICAgIGNvbnN0IGFwaVBhdGhQYXJhbXMgPSBleHRyYWN0UGF0aFBhcmFtcyhyb3V0ZVBhdGgpLm1hcCgobmFtZSkgPT4gKHtcbiAgICAgIG5hbWUsXG4gICAgICBkZXNjcmlwdGlvbjogYCR7bmFtZX0gcGFyYW1ldGVyIGZvciB0aGUgcXVlcnlgLFxuICAgICAgcmVxdWlyZWQ6IHRydWUsXG4gICAgICB0eXBlOiBTdHJpbmcsXG4gICAgfSkpO1xuXG4gICAgY29uc3QgZGVjb3JhdG9yczogQXJyYXk8KHRhcmdldDogYW55LCBrZXk6IHN0cmluZywgZGVzYzogYW55KSA9PiB2b2lkPiA9IFtcbiAgICAgIC4uLmFwaVBhdGhQYXJhbXMubWFwKChwKSA9PiBBcGlQYXJhbShwKSksXG4gICAgICBBcGlPcGVyYXRpb24oeyBzdW1tYXJ5OiBgUmV0cmlldmUgcmVjb3JkcyB1c2luZyBcIiR7bWV0aG9kTmFtZX1cIi5gIH0pLFxuICAgICAgQXBpT2tSZXNwb25zZSh7IGRlc2NyaXB0aW9uOiBcIlJlc3VsdCBzdWNjZXNzZnVsbHkgcmV0cmlldmVkLlwiIH0pLFxuICAgICAgQXBpTm9Db250ZW50UmVzcG9uc2UoeyBkZXNjcmlwdGlvbjogXCJObyBjb250ZW50IHJldHVybmVkIGJ5IHRoZSBtZXRob2QuXCIgfSksXG4gICAgXTtcblxuICAgIGlmIChodHRwVmVyYiA9PT0gXCJHRVRcIiAmJiBpbmNsdWRlUXVlcnlQYXJhbXMpIHtcbiAgICAgIGRlY29yYXRvcnMucHVzaChcbiAgICAgICAgQXBpUXVlcnkoe1xuICAgICAgICAgIG5hbWU6IFwiZGlyZWN0aW9uXCIsXG4gICAgICAgICAgcmVxdWlyZWQ6IGZhbHNlLFxuICAgICAgICAgIGVudW06IE9yZGVyRGlyZWN0aW9uLFxuICAgICAgICAgIGRlc2NyaXB0aW9uOiBcInRoZSBzb3J0IG9yZGVyIHdoZW4gYXBwbGljYWJsZVwiLFxuICAgICAgICB9KSxcbiAgICAgICAgQXBpUXVlcnkoeyBuYW1lOiBcImxpbWl0XCIsIHJlcXVpcmVkOiBmYWxzZSwgZGVzY3JpcHRpb246IFwibGltaXQgb3IgcGFnZSBzaXplXCIgfSksXG4gICAgICAgIEFwaVF1ZXJ5KHsgbmFtZTogXCJvZmZzZXRcIiwgcmVxdWlyZWQ6IGZhbHNlLCBkZXNjcmlwdGlvbjogXCJvZmZzZXQgb3IgYm9va21hcmtcIiB9KVxuICAgICAgKTtcbiAgICB9XG5cbiAgICByZXR1cm4gZGVjb3JhdG9ycztcbiAgfVxufVxuIiwiaW1wb3J0IHsgRHluYW1pY01vZHVsZSwgTW9kdWxlLCBQcm92aWRlciB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgQWRhcHRlciwgTW9kZWxTZXJ2aWNlIH0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBMb2dnaW5nIH0gZnJvbSBcIkBkZWNhZi10cy9sb2dnaW5nXCI7XG5pbXBvcnQgeyBGcm9tTW9kZWxDb250cm9sbGVyIH0gZnJvbSBcIi4vRnJvbU1vZGVsQ29udHJvbGxlclwiO1xuaW1wb3J0IHsgRGVjYWZNb2R1bGVPcHRpb25zIH0gZnJvbSBcIi4uL3R5cGVzXCI7XG5pbXBvcnQgeyBNb2RlbCwgTW9kZWxDb25zdHJ1Y3RvciB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdG9yLXZhbGlkYXRpb25cIjtcbmltcG9ydCB7IE1ldGFkYXRhIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0aW9uXCI7XG5pbXBvcnQgeyBERUNBRl9FWFBPU0UgfSBmcm9tIFwiLi4vY29uc3RhbnRzXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRNb2R1bGVGb3IoZmxhdm91cjogc3RyaW5nKSB7XG4gIEBNb2R1bGUoe30pXG4gIGNsYXNzIERlY2FmTW9kZWxNb2R1bGUge1xuICAgIHN0YXRpYyByZWFkb25seSBsb2cgPSBMb2dnaW5nLmZvcihEZWNhZk1vZGVsTW9kdWxlLm5hbWUpLmZvcihmbGF2b3VyKTtcblxuICAgIHN0YXRpYyBjcmVhdGVNb2RlbFNlcnZpY2VzPFQgZXh0ZW5kcyBNb2RlbDxib29sZWFuPj4oXG4gICAgICBtb2RlbHM6IE1vZGVsQ29uc3RydWN0b3I8VD5bXVxuICAgICk6IFByb3ZpZGVyW10ge1xuICAgICAgcmV0dXJuIG1vZGVscy5tYXAoKG1vZGVsKSA9PiAoe1xuICAgICAgICBwcm92aWRlOiBgJHttb2RlbC5uYW1lfVNlcnZpY2VgLFxuICAgICAgICB1c2VGYWN0b3J5OiAoKSA9PiBNb2RlbFNlcnZpY2UuZm9yTW9kZWwobW9kZWwgYXMgYW55KSxcbiAgICAgIH0pKTtcbiAgICB9XG5cbiAgICBzdGF0aWMgaXNFeHBvc2VkKFxuICAgICAgbW9kZWw6IE1vZGVsQ29uc3RydWN0b3I8YW55PixcbiAgICAgIGV4cG9zdXJlPzogUmVjb3JkPHN0cmluZywgYm9vbGVhbiB8IHN0cmluZ1tdPlxuICAgICk6IGJvb2xlYW4ge1xuICAgICAgY29uc3Qgb3ZlcnJpZGUgPSBleHBvc3VyZT8uW21vZGVsLm5hbWVdO1xuICAgICAgY29uc3QgdmFsdWUgPVxuICAgICAgICB0eXBlb2Ygb3ZlcnJpZGUgIT09IFwidW5kZWZpbmVkXCJcbiAgICAgICAgICA/IG92ZXJyaWRlXG4gICAgICAgICAgOiBNZXRhZGF0YS5nZXQobW9kZWwsIE1ldGFkYXRhLmtleShERUNBRl9FWFBPU0UpKTtcblxuICAgICAgaWYgKHR5cGVvZiB2YWx1ZSA9PT0gXCJ1bmRlZmluZWRcIikgcmV0dXJuIHRydWU7XG4gICAgICBpZiAodmFsdWUgPT09IHRydWUpIHJldHVybiB0cnVlO1xuICAgICAgaWYgKEFycmF5LmlzQXJyYXkodmFsdWUpKSByZXR1cm4gdmFsdWUuaW5jbHVkZXMoZmxhdm91cik7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuXG4gICAgc3RhdGljIGZvclJvb3QoXG4gICAgICBmbGF2b3VyOiBzdHJpbmcsXG4gICAgICBvcHRpb25zOiBQYXJ0aWFsPERlY2FmTW9kdWxlT3B0aW9ucz4gPSB7fVxuICAgICk6IER5bmFtaWNNb2R1bGUge1xuICAgICAgY29uc3QgbG9nID0gdGhpcy5sb2cuZm9yKHRoaXMuZm9yUm9vdCk7XG4gICAgICBsb2cuaW5mbyhgR2VuZXJhdGluZyBjb250cm9sbGVycyBmb3IgZmxhdm91ci4uLmApO1xuXG4gICAgICBjb25zdCB0cmFja2VkTW9kZWxzID0gQWRhcHRlci5tb2RlbHMoZmxhdm91cikuZmlsdGVyKChtb2RlbCkgPT5cbiAgICAgICAgdGhpcy5pc0V4cG9zZWQobW9kZWwsIG9wdGlvbnMuY29udHJvbGxlckV4cG9zdXJlKVxuICAgICAgKTtcblxuICAgICAgbGV0IG1vZGVsU2VydmljZXM6IFByb3ZpZGVyW10gPSBbXTtcbiAgICAgIGlmIChvcHRpb25zLmF1dG9TZXJ2aWNlcykge1xuICAgICAgICBsb2cuaW5mbyhcIkF1dG8tc2VydmljZXMgZW5hYmxlZC4gSW5pdGlhbGl6aW5nIHNlcnZpY2UgZ2VuZXJhdGlvbi5cIik7XG4gICAgICAgIG1vZGVsU2VydmljZXMgPSB0aGlzLmNyZWF0ZU1vZGVsU2VydmljZXModHJhY2tlZE1vZGVscyk7XG4gICAgICAgIGxvZy5pbmZvKFxuICAgICAgICAgIGBBdXRvLXNlcnZpY2VzIGNvbXBsZXRlZC4gJHttb2RlbFNlcnZpY2VzLmxlbmd0aH0gc2VydmljZXMgaW5pdGlhbGl6ZWQuYFxuICAgICAgICApO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBjb250cm9sbGVycyA9IHRyYWNrZWRNb2RlbHMubWFwKChtb2RlbCkgPT5cbiAgICAgICAgRnJvbU1vZGVsQ29udHJvbGxlci5jcmVhdGUobW9kZWwsIG9wdGlvbnMuY29udHJvbGxlckNvbmZpZylcbiAgICAgICk7XG4gICAgICBsb2cuaW5mbyhgR2VuZXJhdGVkICR7Y29udHJvbGxlcnMubGVuZ3RofSBjb250cm9sbGVyc2ApO1xuXG4gICAgICByZXR1cm4ge1xuICAgICAgICBtb2R1bGU6IERlY2FmTW9kZWxNb2R1bGUsXG4gICAgICAgIGNvbnRyb2xsZXJzLFxuICAgICAgICBwcm92aWRlcnM6IFtcbiAgICAgICAgICAuLi5tb2RlbFNlcnZpY2VzLFxuICAgICAgICBdLFxuICAgICAgfTtcbiAgICB9XG4gIH1cbiAgT2JqZWN0LmFzc2lnbihEZWNhZk1vZGVsTW9kdWxlLCBcIm5hbWVcIiwge1xuICAgIHZhbHVlOiBgRGVjYWZNb2R1bGUke2ZsYXZvdXJ9YCxcbiAgfSk7XG4gIHJldHVybiBEZWNhZk1vZGVsTW9kdWxlO1xufVxuIiwiaW1wb3J0IHtcbiAgRGVmYXVsdEluamVjdGFibGVzQ29uZmlnLFxuICBJbmplY3RhYmxlQ29uZmlnLFxuICBJbmplY3RhYmxlc0tleXMsXG4gIEluamVjdE9wdGlvbnMsXG59IGZyb20gXCJAZGVjYWYtdHMvaW5qZWN0YWJsZS1kZWNvcmF0b3JzXCI7XG5pbXBvcnQgeyBJbmplY3QsIEluamVjdGFibGUsIFNjb3BlIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBDb25zdHJ1Y3RvciwgRGVjb3JhdGlvbiwgRGVjb3JhdGlvbktleXMgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRpb25cIjtcbmltcG9ydCB7IFZhbGlkYXRpb25LZXlzIH0gZnJvbSBcIkBkZWNhZi10cy9kZWNvcmF0b3ItdmFsaWRhdGlvblwiO1xuaW1wb3J0IHsgUGVyc2lzdGVuY2VLZXlzIH0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBBcGlQcm9wZXJ0eSB9IGZyb20gXCIuL292ZXJyaWRlcy9kZWNvcmF0aW9uXCI7XG5pbXBvcnQgeyBBdXRoIH0gZnJvbSBcIi4vZGVjYWYtbW9kZWxcIjtcblxuRGVjb3JhdGlvbi5mb3IoSW5qZWN0YWJsZXNLZXlzLklOSkVDVEFCTEUpXG4gIC5leHRlbmQoe1xuICAgIGRlY29yYXRvcjogZnVuY3Rpb24gbmVzdEluamVjdGFibGUoXG4gICAgICBjYXRlZ29yeTogc3RyaW5nIHwgQ29uc3RydWN0b3IsXG4gICAgICBjZmc/OiBJbmplY3RhYmxlQ29uZmlnXG4gICAgKSB7XG4gICAgICBjZmcgPVxuICAgICAgICBjZmcgfHxcbiAgICAgICAgKHR5cGVvZiBjYXRlZ29yeSA9PT0gXCJvYmplY3RcIlxuICAgICAgICAgID8gT2JqZWN0LmFzc2lnbihcbiAgICAgICAgICAgICAgY2F0ZWdvcnkgYXMgUmVjb3JkPGFueSwgYW55PixcbiAgICAgICAgICAgICAgRGVmYXVsdEluamVjdGFibGVzQ29uZmlnXG4gICAgICAgICAgICApXG4gICAgICAgICAgOiBEZWZhdWx0SW5qZWN0YWJsZXNDb25maWcpO1xuICAgICAgcmV0dXJuIEluamVjdGFibGUoe1xuICAgICAgICBzY29wZTogY2ZnLnNpbmdsZXRvbiA/IFNjb3BlLkRFRkFVTFQgOiBTY29wZS5SRVFVRVNULFxuICAgICAgICBkdXJhYmxlOiBjZmcuc2luZ2xldG9uID8gdW5kZWZpbmVkIDogdHJ1ZSxcbiAgICAgIH0pO1xuICAgIH0sXG4gIH0pXG4gIC5hcHBseSgpO1xuXG5EZWNvcmF0aW9uLmZvcihJbmplY3RhYmxlc0tleXMuSU5KRUNUKVxuICAuZXh0ZW5kKHtcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIG5lc3RJbmplY3QoXG4gICAgICBjYXRlZ29yeTogc3ltYm9sIHwgc3RyaW5nIHwgQ29uc3RydWN0b3IsXG4gICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVudXNlZC12YXJzXG4gICAgICBjZmc6IEluamVjdE9wdGlvbnNcbiAgICApIHtcbiAgICAgIHJldHVybiBmdW5jdGlvbiBpbm5lck5lc3RJbmplY3QoXG4gICAgICAgIHRhcmdldDogb2JqZWN0LFxuICAgICAgICBwcm9wZXJ0eUtleT86IGFueSxcbiAgICAgICAgZGVzY3JpcHRvcj86IGFueVxuICAgICAgKSB7XG4gICAgICAgIHJldHVybiBJbmplY3QoY2F0ZWdvcnkgfHwgKHRhcmdldCBhcyBDb25zdHJ1Y3RvcikpKFxuICAgICAgICAgIHRhcmdldCxcbiAgICAgICAgICBwcm9wZXJ0eUtleSxcbiAgICAgICAgICBkZXNjcmlwdG9yXG4gICAgICAgICk7XG4gICAgICB9O1xuICAgIH0sXG4gIH0pXG4gIC5hcHBseSgpO1xuXG5EZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5SRVFVSVJFRClcbiAgLmV4dGVuZChcbiAgICBBcGlQcm9wZXJ0eSh7XG4gICAgICByZXF1aXJlZDogdHJ1ZSxcbiAgICB9KVxuICApXG4gIC5hcHBseSgpO1xuXG5EZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5NQVgpXG4gIC5leHRlbmQoe1xuICAgIGRlY29yYXRvcjogZnVuY3Rpb24gbWF4RGVjKG1heDogbnVtYmVyKSB7XG4gICAgICByZXR1cm4gQXBpUHJvcGVydHkoeyBtYXhpbXVtOiBtYXgsIHJlcXVpcmVkOiBmYWxzZSB9KTtcbiAgICB9LFxuICB9KVxuICAuYXBwbHkoKTtcblxuRGVjb3JhdGlvbi5mb3IoVmFsaWRhdGlvbktleXMuTUlOKVxuICAuZXh0ZW5kKHtcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIG1pbkRlYyhtaW46IG51bWJlcikge1xuICAgICAgcmV0dXJuIEFwaVByb3BlcnR5KHsgbWluaW11bTogbWluLCByZXF1aXJlZDogZmFsc2UgfSk7XG4gICAgfSxcbiAgfSlcbiAgLmFwcGx5KCk7XG5cbkRlY29yYXRpb24uZm9yKFZhbGlkYXRpb25LZXlzLk1BWF9MRU5HVEgpXG4gIC5leHRlbmQoe1xuICAgIGRlY29yYXRvcjogZnVuY3Rpb24gbWF4TGVuZ3RoRGVjKG1heDogbnVtYmVyKSB7XG4gICAgICByZXR1cm4gQXBpUHJvcGVydHkoeyBtYXhMZW5ndGg6IG1heCwgcmVxdWlyZWQ6IGZhbHNlIH0pO1xuICAgIH0sXG4gIH0pXG4gIC5hcHBseSgpO1xuXG5EZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5NSU5fTEVOR1RIKVxuICAuZXh0ZW5kKHtcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIG1pbkxlbmd0aERlYyhtaW46IG51bWJlcikge1xuICAgICAgcmV0dXJuIEFwaVByb3BlcnR5KHsgbWluTGVuZ3RoOiBtaW4sIHJlcXVpcmVkOiBmYWxzZSB9KTtcbiAgICB9LFxuICB9KVxuICAuYXBwbHkoKTtcblxuRGVjb3JhdGlvbi5mb3IoVmFsaWRhdGlvbktleXMuVFlQRSlcbiAgLmV4dGVuZCh7XG4gICAgZGVjb3JhdG9yOiBmdW5jdGlvbiB0eXBlRGVjKFxuICAgICAgdHlwZTpcbiAgICAgICAgfCAoQ29uc3RydWN0b3IgfCAoKCkgPT4gQ29uc3RydWN0b3IpKVtdXG4gICAgICAgIHwgQ29uc3RydWN0b3JcbiAgICAgICAgfCAoKCkgPT4gQ29uc3RydWN0b3IpXG4gICAgKSB7XG4gICAgICByZXR1cm4gKHRhcmdldDogb2JqZWN0LCBwcm9wOiBhbnkpID0+IHtcbiAgICAgICAgdHlwZSA9IEFycmF5LmlzQXJyYXkodHlwZSkgPyB0eXBlWzBdIDogdHlwZTtcbiAgICAgICAgaWYgKHR5cGVvZiB0eXBlID09PSBcImZ1bmN0aW9uXCIgJiYgIXR5cGUubmFtZSlcbiAgICAgICAgICB0eXBlID0gKHR5cGUgYXMgKCkgPT4gQ29uc3RydWN0b3IpKCk7XG4gICAgICAgIHJldHVybiBBcGlQcm9wZXJ0eSh7XG4gICAgICAgICAgdHlwZTogdHlwZSBhcyBhbnksXG4gICAgICAgICAgcmVxdWlyZWQ6IGZhbHNlLFxuICAgICAgICB9KSh0YXJnZXQsIHByb3ApO1xuICAgICAgfTtcbiAgICB9LFxuICB9KVxuICAuYXBwbHkoKTtcbi8vXG4vLyBEZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5MSVNUKVxuLy8gICAuZXh0ZW5kKHtcbi8vICAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIGxpc3REZWMoXG4vLyAgICAgICBjbGF6ejpcbi8vICAgICAgICAgfCBDb25zdHJ1Y3RvclxuLy8gICAgICAgICB8ICgoKSA9PiBDb25zdHJ1Y3Rvcilcbi8vICAgICAgICAgfCAoQ29uc3RydWN0b3IgfCAoKCkgPT4gQ29uc3RydWN0b3IpKVtdLFxuLy8gICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby11bnVzZWQtdmFyc1xuLy8gICAgICAgY29sbGVjdGlvbjogXCJBcnJheVwiIHwgXCJTZXRcIiA9IFwiQXJyYXlcIlxuLy8gICAgICkge1xuLy8gICAgICAgcmV0dXJuICh0YXJnZXQ6IG9iamVjdCwgcHJvcDogYW55KSA9PiB7XG4vLyAgICAgICAgIGNsYXp6ID0gQXJyYXkuaXNBcnJheShjbGF6eikgPyBjbGF6elswXSA6IGNsYXp6O1xuLy8gICAgICAgICBpZiAodHlwZW9mIGNsYXp6ID09PSBcImZ1bmN0aW9uXCIgJiYgIWNsYXp6Lm5hbWUpXG4vLyAgICAgICAgICAgY2xhenogPSAoY2xhenogYXMgKCkgPT4gQ29uc3RydWN0b3IpKCk7XG4vLyAgICAgICAgIHJldHVybiBBcGlQcm9wZXJ0eSh7IHR5cGU6IFtjbGF6eiBhcyBhbnldIH0pKHRhcmdldCwgcHJvcCk7XG4vLyAgICAgICB9O1xuLy8gICAgIH0sXG4vLyAgIH0pXG4vLyAgIC5hcHBseSgpO1xuLy8gLy9cbkRlY29yYXRpb24uZm9yKFZhbGlkYXRpb25LZXlzLkRBVEUpXG4gIC5leHRlbmQoe1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW51c2VkLXZhcnNcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIGRhdGVEZWMoZm9ybWF0OiBzdHJpbmcpIHtcbiAgICAgIHJldHVybiBBcGlQcm9wZXJ0eSh7XG4gICAgICAgIHR5cGU6IFN0cmluZyxcbiAgICAgICAgZm9ybWF0OiBcImRhdGUtdGltZVwiLFxuICAgICAgICByZXF1aXJlZDogZmFsc2UsXG4gICAgICAgIC8vIGV4YW1wbGU6IHBhcnNlRGF0ZShmb3JtYXQsIG5ldyBEYXRlKCkpLFxuICAgICAgfSk7XG4gICAgfSxcbiAgfSlcbiAgLmFwcGx5KCk7XG5cbkRlY29yYXRpb24uZm9yKFZhbGlkYXRpb25LZXlzLkVOVU0pXG4gIC5leHRlbmQoe1xuICAgIGRlY29yYXRvcjogZnVuY3Rpb24gb3B0aW9uRGVjKG9wdGlvbnM6IHN0cmluZ1tdIHwgUmVjb3JkPHN0cmluZywgYW55Pikge1xuICAgICAgY29uc3Qgb3B0cyA9IEFycmF5LmlzQXJyYXkob3B0aW9ucykgPyBvcHRpb25zIDogT2JqZWN0LnZhbHVlcyhvcHRpb25zKTtcbiAgICAgIHJldHVybiBBcGlQcm9wZXJ0eSh7XG4gICAgICAgIGVudW06IG9wdHMsXG4gICAgICAgIHJlcXVpcmVkOiBmYWxzZSxcbiAgICAgIH0pO1xuICAgIH0sXG4gIH0pXG4gIC5hcHBseSgpO1xuXG5EZWNvcmF0aW9uLmZvcihWYWxpZGF0aW9uS2V5cy5QQVRURVJOKVxuICAuZXh0ZW5kKHtcbiAgICBkZWNvcmF0b3I6IGZ1bmN0aW9uIHBhdHRlcm5EZWMocGF0OiBSZWdFeHAgfCBzdHJpbmcpIHtcbiAgICAgIHJldHVybiBBcGlQcm9wZXJ0eSh7XG4gICAgICAgIHBhdHRlcm46IHR5cGVvZiBwYXQgPT09IFwic3RyaW5nXCIgPyBwYXQgOiBwYXQuc291cmNlLFxuICAgICAgICByZXF1aXJlZDogZmFsc2UsXG4gICAgICB9KTtcbiAgICB9LFxuICB9KVxuICAuYXBwbHkoKTtcblxuRGVjb3JhdGlvbi5mb3IoUGVyc2lzdGVuY2VLZXlzLkNPTFVNTilcbiAgLmV4dGVuZCh7XG4gICAgZGVjb3JhdG9yOiBmdW5jdGlvbiBjb2x1bW5EZWMobmFtZTogc3RyaW5nKSB7XG4gICAgICByZXR1cm4gQXBpUHJvcGVydHkoe1xuICAgICAgICBuYW1lOiBuYW1lLFxuICAgICAgICByZXF1aXJlZDogZmFsc2UsXG4gICAgICB9KTtcbiAgICB9LFxuICB9KVxuICAuYXBwbHkoKTtcblxuRGVjb3JhdGlvbi5mb3IoRGVjb3JhdGlvbktleXMuREVTQ1JJUFRJT04pXG4gIC5leHRlbmQoe1xuICAgIGRlY29yYXRvcjogZnVuY3Rpb24gZGVzY3JpcHRpb25EZWMoZGVzY3JpcHRpb246IHN0cmluZykge1xuICAgICAgcmV0dXJuIEFwaVByb3BlcnR5KHtcbiAgICAgICAgZGVzY3JpcHRpb246IGRlc2NyaXB0aW9uLFxuICAgICAgICByZXF1aXJlZDogZmFsc2UsXG4gICAgICB9KTtcbiAgICB9LFxuICB9KVxuICAuYXBwbHkoKTtcblxuRGVjb3JhdGlvbi5mb3IoUGVyc2lzdGVuY2VLZXlzLkFVVEgpLmV4dGVuZCh7IGRlY29yYXRvcjogQXV0aCB9KS5hcHBseSgpO1xuIiwiaW1wb3J0IHsgRm9yYmlkZGVuRXJyb3IgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcblxuZXhwb3J0IGNsYXNzIENvcnNFcnJvciBleHRlbmRzIEZvcmJpZGRlbkVycm9yIHtcbiAgY29uc3RydWN0b3IobXNnOiBzdHJpbmcgfCBFcnJvcikge1xuICAgIHN1cGVyKG1zZywgQ29yc0Vycm9yLm5hbWUpO1xuICB9XG59XG4iLCJpbXBvcnQgeyBCYXNlRXJyb3IgfSBmcm9tIFwiQGRlY2FmLXRzL2RiLWRlY29yYXRvcnNcIjtcblxuZXhwb3J0IGNsYXNzIFRvTWFueVJlcXVlc3RzRXJyb3IgZXh0ZW5kcyBCYXNlRXJyb3Ige1xuICBjb25zdHJ1Y3Rvcihtc2c6IHN0cmluZyB8IEVycm9yICkge1xuICAgIHN1cGVyKFRvTWFueVJlcXVlc3RzRXJyb3IubmFtZSwgbXNnICwgNDI5KTtcbiAgfVxufSIsImltcG9ydCB7XG4gIEFyZ3VtZW50c0hvc3QsXG4gIENhdGNoLFxuICBFeGNlcHRpb25GaWx0ZXIsXG4gIE5vdEFjY2VwdGFibGVFeGNlcHRpb24sXG4gIE5vdEZvdW5kRXhjZXB0aW9uLFxufSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcbmltcG9ydCB7IEJhc2VFcnJvciwgSW50ZXJuYWxFcnJvciB9IGZyb20gXCJAZGVjYWYtdHMvZGItZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHsgTG9nZ2VkRW52aXJvbm1lbnQgfSBmcm9tIFwiQGRlY2FmLXRzL2xvZ2dpbmdcIjtcbmltcG9ydCB7IFVuc3VwcG9ydGVkRXJyb3IgfSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcbmltcG9ydCB7IFRvTWFueVJlcXVlc3RzRXJyb3IgfSBmcm9tIFwiLi4vZXJyb3JzL3Rocm90dGxpbmdcIjtcblxuQENhdGNoKClcbmV4cG9ydCBjbGFzcyBEZWNhZkV4Y2VwdGlvbkZpbHRlciBpbXBsZW1lbnRzIEV4Y2VwdGlvbkZpbHRlciB7XG4gIGNhdGNoKGV4Y2VwdGlvbjogRXJyb3IsIGhvc3Q6IEFyZ3VtZW50c0hvc3QpIHtcbiAgICBjb25zdCBjdHggPSBob3N0LnN3aXRjaFRvSHR0cCgpO1xuICAgIGNvbnN0IHJlc3BvbnNlID0gY3R4LmdldFJlc3BvbnNlKCk7XG4gICAgY29uc3QgcmVxdWVzdCA9IGN0eC5nZXRSZXF1ZXN0KCk7XG5cbiAgICBjb25zdCBpc1Byb2R1Y3Rpb24gPSBMb2dnZWRFbnZpcm9ubWVudC5lbnYgPT09IFwicHJvZHVjdGlvblwiO1xuICAgIGxldCBzdGF0dXNDb2RlO1xuXG4gICAgaWYgKFxuICAgICAgZXhjZXB0aW9uIGluc3RhbmNlb2YgTm90Rm91bmRFeGNlcHRpb24gfHxcbiAgICAgIGV4Y2VwdGlvbiBpbnN0YW5jZW9mIFVuc3VwcG9ydGVkRXJyb3JcbiAgICApIHtcbiAgICAgIGV4Y2VwdGlvbiA9IG5ldyBOb3RBY2NlcHRhYmxlRXhjZXB0aW9uKGV4Y2VwdGlvbi5tZXNzYWdlKTtcbiAgICAgIHN0YXR1c0NvZGUgPSAoZXhjZXB0aW9uIGFzIE5vdEFjY2VwdGFibGVFeGNlcHRpb24pLmdldFN0YXR1cygpO1xuICAgIH0gZWxzZSBpZiAoIShleGNlcHRpb24gaW5zdGFuY2VvZiBCYXNlRXJyb3IpKSB7XG4gICAgICBpZigoZXhjZXB0aW9uIGFzIGFueSkuc3RhdHVzID09PSA0Mjkpe1xuICAgICAgICBleGNlcHRpb24gPSBuZXcgVG9NYW55UmVxdWVzdHNFcnJvcihleGNlcHRpb24ubWVzc2FnZSk7XG4gICAgICB9ZWxzZXtcbiAgICAgICAgZXhjZXB0aW9uID0gbmV3IEludGVybmFsRXJyb3IoZXhjZXB0aW9uLm1lc3NhZ2UpO1xuICAgICAgfVxuICAgIH1cblxuICAgIHJlc3BvbnNlLnN0YXR1cygoZXhjZXB0aW9uIGFzIEJhc2VFcnJvcikuY29kZSB8fCBzdGF0dXNDb2RlKS5qc29uKHtcbiAgICAgIHN0YXR1czogKGV4Y2VwdGlvbiBhcyBCYXNlRXJyb3IpLmNvZGUgfHwgc3RhdHVzQ29kZSxcbiAgICAgIGVycm9yOiBpc1Byb2R1Y3Rpb24gPyBleGNlcHRpb24ubmFtZSA6IGV4Y2VwdGlvbi5tZXNzYWdlLFxuICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKCksXG4gICAgICBwYXRoOiByZXF1ZXN0LnVybCxcbiAgICAgIG1ldGhvZDogcmVxdWVzdC5tZXRob2QsXG4gICAgfSk7XG4gIH1cbn1cbiIsImltcG9ydCB7IFVzZUZpbHRlcnMsIFVzZUludGVyY2VwdG9ycyB9IGZyb20gXCJAbmVzdGpzL2NvbW1vblwiO1xuaW1wb3J0IHsgRGVjYWZFeGNlcHRpb25GaWx0ZXIgfSBmcm9tIFwiLi9EZWNhZkVycm9yRmlsdGVyXCI7XG5pbXBvcnQgeyBEZWNhZlJlc3BvbnNlSW50ZXJjZXB0b3IgfSBmcm9tIFwiLi4vLi4vcmVxdWVzdC9pbmRleFwiO1xuXG5leHBvcnQgZnVuY3Rpb24gVXNlRGVjYWZGaWx0ZXIoKSB7XG4gIHJldHVybiBVc2VGaWx0ZXJzKG5ldyBEZWNhZkV4Y2VwdGlvbkZpbHRlcigpKTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIFVzZURlY2FmSGVhZGVycygpIHtcbiAgcmV0dXJuIFVzZUludGVyY2VwdG9ycyhEZWNhZlJlc3BvbnNlSW50ZXJjZXB0b3IpO1xufVxuIiwiaW1wb3J0IHsgQ29uc3RydWN0b3IgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRpb25cIjtcbmltcG9ydCB7IFNlY3VyaXR5U2NoZW1lT2JqZWN0IH0gZnJvbSBcIi4uLy4uL3N3YWdnZXItdHlwZXNcIjtcblxuZXhwb3J0IGludGVyZmFjZSBTd2FnZ2VyT3B0aW9ucyB7XG4gIHRpdGxlOiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmc7XG4gIHZlcnNpb24/OiBzdHJpbmc7XG4gIGFzc2V0c1BhdGg/OiBzdHJpbmc7XG4gIGZhdmljb25GaWxlUGF0aD86IHN0cmluZztcbiAgdG9wYmFySWNvbkZpbGVQYXRoPzogc3RyaW5nO1xuICBwZXJzaXN0QXV0aG9yaXphdGlvbj86IGJvb2xlYW47XG4gIHBhdGg6IHN0cmluZztcbiAgYXV0aD86IFNlY3VyaXR5U2NoZW1lT2JqZWN0O1xuICB0b3BiYXJCZ0NvbG9yPzogc3RyaW5nO1xuICBleHRyYU1vZGVscz86IENvbnN0cnVjdG9yW107XG5cbiAgLyoqXG4gICAqIFBhdGggdG8gZXhwb3NlIHRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24gaW4gSlNPTiBmb3JtYXQuXG4gICAqIEV4YW1wbGU6IC9hcGkvb3BlbmFwaS5qc29uXG4gICAqL1xuICBvcGVuQXBpSnNvblBhdGg/OiBzdHJpbmc7XG5cbiAgLyoqXG4gICAqIFBhdGggdG8gZXhwb3NlIHRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24gaW4gWUFNTCBmb3JtYXQuXG4gICAqIFJlcXVpcmVzICd5YW1sJyBwYWNrYWdlIGFzIHBlZXIgZGVwZW5kZW5jeS5cbiAgICogRXhhbXBsZTogL2FwaS9vcGVuYXBpLnlhbWxcbiAgICovXG4gIG9wZW5BcGlZYW1sUGF0aD86IHN0cmluZztcbn1cblxuZXhwb3J0IGNvbnN0IFNXQUdHRVJfVUlfQ09OU1RBTlRTOiBTd2FnZ2VyT3B0aW9ucyA9IHtcbiAgdGl0bGU6IFwiU3dhZ2dlciB8IE9wZW5BUEkgU3BlY2lmaWNhdGlvbiAoT0FTKVwiLFxuICBkZXNjcmlwdGlvbjogXCJTdGFuZGFyZGl6ZWQgZm9ybWF0IGZvciBkZXNjcmliaW5nIFJFU1RmdWwgQVBJc1wiLFxuICB2ZXJzaW9uOiBcIjAuMC4xXCIsXG4gIHBhdGg6IFwiZG9jc1wiLFxuICBmYXZpY29uRmlsZVBhdGg6IFwiXCIsXG4gIHRvcGJhckljb25GaWxlUGF0aDogXCJcIixcbiAgYXV0aDoge1xuICAgIHR5cGU6IFwiaHR0cFwiLFxuICAgIHNjaGVtZTogXCJiZWFyZXJcIixcbiAgICBiZWFyZXJGb3JtYXQ6IFwiSldUXCIsXG4gICAgbmFtZTogXCJBdXRob3JpemF0aW9uXCIsXG4gICAgZGVzY3JpcHRpb246IFwiRW50ZXIgSldUIHRva2VuXCIsXG4gICAgaW46IFwiaGVhZGVyXCIsXG4gIH0sXG4gIHBlcnNpc3RBdXRob3JpemF0aW9uOiB0cnVlLFxuICB0b3BiYXJCZ0NvbG9yOiBcIiMwMDAwMDBcIixcbn07XG4iLCJpbXBvcnQgeyBTV0FHR0VSX1VJX0NPTlNUQU5UUyB9IGZyb20gXCIuL2NvbnN0YW50c1wiO1xuaW1wb3J0IHsgcmVhZEZpbGVTeW5jIH0gZnJvbSBcImZzXCI7XG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gXCJwYXRoXCI7XG5cbmV4cG9ydCBpbnRlcmZhY2UgU3dhZ2dlclVJT3B0aW9ucyB7XG4gIHRpdGxlPzogc3RyaW5nO1xuICBwYXRoPzogc3RyaW5nO1xuICBwZXJzaXN0QXV0aG9yaXphdGlvbjogYm9vbGVhbjtcbiAgYXNzZXRzUGF0aD86IHN0cmluZztcbiAgZmF2aWNvblBhdGg/OiBzdHJpbmc7XG4gIHRvcGJhckljb25QYXRoPzogc3RyaW5nO1xuICB0b3BiYXJCZ0NvbG9yPzogc3RyaW5nO1xufVxuXG5leHBvcnQgY2xhc3MgU3dhZ2dlckN1c3RvbVVJIHtcbiAgcmVhZG9ubHkgb3B0aW9uczogU3dhZ2dlclVJT3B0aW9ucztcbiAgLy8gcHJpdmF0ZSByZWFkb25seSBhc3NldHNQYXRoOiBzdHJpbmcgPSBwYXRoLmpvaW4oXG4gIC8vICAgX19kaXJuYW1lLFxuICAvLyAgIFwiLi5cIixcbiAgLy8gICBcIi4uXCIsXG4gIC8vICAgXCJ3b3JrZG9jc1wiLFxuICAvLyAgIFwiYXNzZXRzXCJcbiAgLy8gKTtcblxuICBjb25zdHJ1Y3RvcihvcHRpb25zOiBTd2FnZ2VyVUlPcHRpb25zKSB7XG4gICAgdGhpcy5vcHRpb25zID0ge1xuICAgICAgLi4ub3B0aW9ucyxcbiAgICB9O1xuICB9XG5cbiAgcHJpdmF0ZSBjdXN0b21DU1MoKSB7XG4gICAgbGV0IGNzcyA9IFwiXCI7XG4gICAgaWYgKHRoaXMub3B0aW9ucy50b3BiYXJJY29uUGF0aCkge1xuICAgICAgY29uc3QgaW1nID0gdGhpcy5iNjQodGhpcy5vcHRpb25zLnRvcGJhckljb25QYXRoKTtcbiAgICAgIGNzcyArPSBgLnRvcGJhci13cmFwcGVyIHsgY29udGVudDogdXJsKCdkYXRhOmltYWdlL3BuZztiYXNlNjQsJHtpbWd9Jyk7IHdpZHRoOiAyMDBweDsgaGVpZ2h0OiBhdXRvOyB9XFxuYDtcbiAgICB9XG4gICAgcmV0dXJuIChcbiAgICAgIGNzcyArXG4gICAgICBgXG4gICAgICAgIC50b3BiYXItd3JhcHBlciBzdmcgeyB2aXNpYmlsaXR5OiBoaWRkZW47IH1cbiAgICAgICAgLnN3YWdnZXItdWkgLnRvcGJhciB7IGJhY2tncm91bmQtY29sb3I6ICR7dGhpcy5vcHRpb25zLnRvcGJhckJnQ29sb3IgfHwgU1dBR0dFUl9VSV9DT05TVEFOVFMudG9wYmFyQmdDb2xvcn07IH1cbiAgICAgIGBcbiAgICApO1xuICB9XG5cbiAgZ2V0Q3VzdG9tT3B0aW9ucygpIHtcbiAgICBjb25zdCBmYXZpY29uOiBSZWNvcmQ8c3RyaW5nLCBhbnk+ID0ge307XG4gICAgaWYgKHRoaXMub3B0aW9ucy5mYXZpY29uUGF0aCkge1xuICAgICAgZmF2aWNvbltcImN1c3RvbWZhdkljb25cIl0gPSB0aGlzLmI2NCh0aGlzLm9wdGlvbnMuZmF2aWNvblBhdGgsIHRydWUpO1xuICAgIH1cblxuICAgIHJldHVybiB7XG4gICAgICBjdXN0b21TaXRlVGl0bGU6IHRoaXMub3B0aW9ucy50aXRsZSxcbiAgICAgIC4uLmZhdmljb24sXG4gICAgICBjdXN0b21Dc3M6IHRoaXMuY3VzdG9tQ1NTKCksXG4gICAgICBzd2FnZ2VyT3B0aW9uczoge1xuICAgICAgICBwZXJzaXN0QXV0aG9yaXphdGlvbjogdGhpcy5vcHRpb25zLnBlcnNpc3RBdXRob3JpemF0aW9uLFxuICAgICAgfSxcbiAgICB9O1xuICB9XG5cbiAgYjY0KGZpbGU6IHN0cmluZywgaW1nOiBib29sZWFuID0gZmFsc2UpIHtcbiAgICBjb25zdCBmaWxlUGF0aCA9IHBhdGguam9pbih0aGlzLm9wdGlvbnMuYXNzZXRzUGF0aCB8fCBcIlwiLCBmaWxlKTtcbiAgICBjb25zdCBiNjQgPSByZWFkRmlsZVN5bmMoZmlsZVBhdGgsIHsgZW5jb2Rpbmc6IFwiYmFzZTY0XCIgfSk7XG4gICAgcmV0dXJuIGltZyA/IFwiZGF0YTppbWFnZS9wbmc7YmFzZTY0LFwiICsgYjY0IDogYjY0O1xuICB9XG59XG4iLCJpbXBvcnQgeyBJTmVzdEFwcGxpY2F0aW9uIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBEb2N1bWVudEJ1aWxkZXIsIFN3YWdnZXJNb2R1bGUgfSBmcm9tIFwiQG5lc3Rqcy9zd2FnZ2VyXCI7XG5pbXBvcnQgeyBTd2FnZ2VyQ3VzdG9tVUkgfSBmcm9tIFwiLi9Td2FnZ2VyQ3VzdG9tVUlcIjtcbmltcG9ydCB7IFNXQUdHRVJfVUlfQ09OU1RBTlRTLCBTd2FnZ2VyT3B0aW9ucyB9IGZyb20gXCIuL2NvbnN0YW50c1wiO1xuaW1wb3J0IFlBTUwgZnJvbSBcInlhbWxcIjtcblxuZXhwb3J0IGNsYXNzIFN3YWdnZXJCdWlsZGVyIHtcbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSByZWFkb25seSBhcHA6IElOZXN0QXBwbGljYXRpb24sXG4gICAgcHJpdmF0ZSByZWFkb25seSBvcHRpb25zOiBTd2FnZ2VyT3B0aW9uc1xuICApIHt9XG5cbiAgcHJpdmF0ZSBjcmVhdGVEb2N1bWVudCgpIHtcbiAgICBjb25zdCBkZXNjcmlwdGlvbiA9IHRoaXMub3B0aW9ucy5wYXRoXG4gICAgICA/IHRoaXMub3B0aW9ucy5kZXNjcmlwdGlvbiArXG4gICAgICAgIFwiXCIgK1xuICAgICAgICBgPGJyPjxicj48YSBocmVmPVwiJHt0aGlzLm9wdGlvbnMub3BlbkFwaUpzb25QYXRofVwiPk9wZW5BUEkgSlNPTiBTcGVjPC9hPiB8IGAgK1xuICAgICAgICBgPGEgaHJlZj1cIiR7dGhpcy5vcHRpb25zLm9wZW5BcGlZYW1sUGF0aH1cIj5PcGVuQVBJIFlBTUwgU3BlYzwvYT5gXG4gICAgICA6IHRoaXMub3B0aW9ucy5kZXNjcmlwdGlvbjtcblxuICAgIGNvbnN0IGNvbmZpZyA9IG5ldyBEb2N1bWVudEJ1aWxkZXIoKVxuICAgICAgLnNldFRpdGxlKHRoaXMub3B0aW9ucy50aXRsZSlcbiAgICAgIC5zZXREZXNjcmlwdGlvbihkZXNjcmlwdGlvbilcbiAgICAgIC5zZXRWZXJzaW9uKHRoaXMub3B0aW9ucy52ZXJzaW9uIHx8IFwiMC4wLjFcIilcbiAgICAgIC5hZGRCZWFyZXJBdXRoKHRoaXMub3B0aW9ucy5hdXRoIHx8IFNXQUdHRVJfVUlfQ09OU1RBTlRTLmF1dGgpXG4gICAgICAuYnVpbGQoKTtcblxuICAgIHJldHVybiBTd2FnZ2VyTW9kdWxlLmNyZWF0ZURvY3VtZW50KHRoaXMuYXBwLCBjb25maWcsIHtcbiAgICAgIGV4dHJhTW9kZWxzOiB0aGlzLm9wdGlvbnMuZXh0cmFNb2RlbHMgfHwgW10sXG4gICAgfSk7XG4gIH1cblxuICBwcml2YXRlIHJlZ2lzdGVyT3BlbkFwaVJvdXRlKFxuICAgIHBhdGg6IHN0cmluZyB8IHVuZGVmaW5lZCxcbiAgICBjb250ZW50VHlwZTogXCJhcHBsaWNhdGlvbi9qc29uXCIgfCBcImFwcGxpY2F0aW9uL3gteWFtbFwiLFxuICAgIGJvZHlGYWN0b3J5OiAoKSA9PiB2b2lkXG4gICk6IHZvaWQge1xuICAgIGlmICghcGF0aCkgcmV0dXJuO1xuXG4gICAgY29uc3QgaHR0cEFkYXB0ZXIgPSB0aGlzLmFwcC5nZXRIdHRwQWRhcHRlcigpIGFzIGFueTtcbiAgICBwYXRoID0gcGF0aC5zdGFydHNXaXRoKFwiL1wiKSA/IHBhdGggOiBgLyR7cGF0aH1gO1xuICAgIGh0dHBBZGFwdGVyLmdldChwYXRoLCAoX3JlcTogYW55LCByZXM6IGFueSkgPT4ge1xuICAgICAgaHR0cEFkYXB0ZXIucmVwbHkocmVzLCBib2R5RmFjdG9yeSgpLCAyMDAsIHtcbiAgICAgICAgXCJDb250ZW50LVR5cGVcIjogY29udGVudFR5cGUsXG4gICAgICB9KTtcbiAgICB9KTtcbiAgfVxuXG4gIHB1YmxpYyBzZXR1cFN3YWdnZXIoKSB7XG4gICAgY29uc3QgZG9jdW1lbnQgPSB0aGlzLmNyZWF0ZURvY3VtZW50KCk7XG4gICAgY29uc3Qgc3dhZ2dlclVJID0gbmV3IFN3YWdnZXJDdXN0b21VSSh7XG4gICAgICB0aXRsZTogdGhpcy5vcHRpb25zLnRpdGxlLFxuICAgICAgcGF0aDogdGhpcy5vcHRpb25zLnBhdGggfHwgU1dBR0dFUl9VSV9DT05TVEFOVFMucGF0aCxcbiAgICAgIHBlcnNpc3RBdXRob3JpemF0aW9uOiB0aGlzLm9wdGlvbnMucGVyc2lzdEF1dGhvcml6YXRpb24gPz8gdHJ1ZSxcbiAgICAgIGFzc2V0c1BhdGg6IHRoaXMub3B0aW9ucy5hc3NldHNQYXRoLFxuICAgICAgZmF2aWNvblBhdGg6IHRoaXMub3B0aW9ucy5mYXZpY29uRmlsZVBhdGgsXG4gICAgICB0b3BiYXJJY29uUGF0aDogdGhpcy5vcHRpb25zLnRvcGJhckljb25GaWxlUGF0aCxcbiAgICAgIHRvcGJhckJnQ29sb3I6IHRoaXMub3B0aW9ucy50b3BiYXJCZ0NvbG9yLFxuICAgIH0pO1xuICAgIFN3YWdnZXJNb2R1bGUuc2V0dXAoXG4gICAgICB0aGlzLm9wdGlvbnMucGF0aCB8fCBTV0FHR0VSX1VJX0NPTlNUQU5UUy5wYXRoLFxuICAgICAgdGhpcy5hcHAsXG4gICAgICBkb2N1bWVudCxcbiAgICAgIHtcbiAgICAgICAgLi4uc3dhZ2dlclVJLmdldEN1c3RvbU9wdGlvbnMoKSxcbiAgICAgICAganNvbkRvY3VtZW50VXJsOiB0aGlzLm9wdGlvbnMub3BlbkFwaUpzb25QYXRoXG4gICAgICAgICAgPyBgJHt0aGlzLm9wdGlvbnMub3BlbkFwaUpzb25QYXRofWBcbiAgICAgICAgICA6IHVuZGVmaW5lZCxcbiAgICAgICAgeWFtbERvY3VtZW50VXJsOiB0aGlzLm9wdGlvbnMub3BlbkFwaVlhbWxQYXRoXG4gICAgICAgICAgPyBgJHt0aGlzLm9wdGlvbnMub3BlbkFwaVlhbWxQYXRofWBcbiAgICAgICAgICA6IHVuZGVmaW5lZCxcbiAgICAgIH1cbiAgICApO1xuXG4gICAgdGhpcy5yZWdpc3Rlck9wZW5BcGlSb3V0ZShcbiAgICAgIHRoaXMub3B0aW9ucy5vcGVuQXBpSnNvblBhdGgsXG4gICAgICBcImFwcGxpY2F0aW9uL2pzb25cIixcbiAgICAgICgpID0+IGRvY3VtZW50XG4gICAgKTtcblxuICAgIHRoaXMucmVnaXN0ZXJPcGVuQXBpUm91dGUoXG4gICAgICB0aGlzLm9wdGlvbnMub3BlbkFwaVlhbWxQYXRoLFxuICAgICAgXCJhcHBsaWNhdGlvbi94LXlhbWxcIixcbiAgICAgICgpID0+IFlBTUwuc3RyaW5naWZ5KGRvY3VtZW50KVxuICAgICk7XG4gIH1cblxuICAvLyBwcml2YXRlIGdldFZlcnNpb24oKSB7XG4gIC8vICAgLy8gY29uc3QgcGFja2FnZUpzb24gPSBwYXRoLmpvaW4oX19kaXJuYW1lLCBcIi4uXCIsIFwiLi5cIiwgXCJwYWNrYWdlLmpzb25cIik7XG4gIC8vICAgLy8gY29uc3Qge3ZlcnNpb259ID0gcmVxdWlyZShwYWNrYWdlSnNvbik7XG4gIC8vICAgcmV0dXJuIFwidmVyc2lvblwiO1xuICAvLyB9XG59XG4iLCJpbXBvcnQge1xuICBJTmVzdEFwcGxpY2F0aW9uLFxuICBMb2dnZXIsXG4gIE5lc3RJbnRlcmNlcHRvcixcbiAgUGlwZVRyYW5zZm9ybSxcbn0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBEZWNhZkV4Y2VwdGlvbkZpbHRlciB9IGZyb20gXCIuL2V4Y2VwdGlvbnNcIjtcbmltcG9ydCB7IFN3YWdnZXJCdWlsZGVyIH0gZnJvbSBcIi4vb3BlbmFwaVwiO1xuaW1wb3J0IHsgQ29yc09wdGlvbnMgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb24vaW50ZXJmYWNlcy9leHRlcm5hbC9jb3JzLW9wdGlvbnMuaW50ZXJmYWNlXCI7XG5pbXBvcnQgeyBDb3JzRXJyb3IgfSBmcm9tIFwiLi9lcnJvcnNcIjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb25cbiAqIERlZmluZXMgYWxsIGN1c3RvbWl6YWJsZSBwYXJhbWV0ZXJzIGZvciBTd2FnZ2VyIHNldHVwLlxuICpcbiAqIEBzdW1tYXJ5XG4gKiBUaGlzIGludGVyZmFjZSBhbGxvd3MgZGV2ZWxvcGVycyB0byBjdXN0b21pemUgaG93IFN3YWdnZXIgVUkgaXMgY29uZmlndXJlZFxuICogd2l0aGluIHRoZSBOZXN0SlMgYXBwbGljYXRpb24uIEl0IGluY2x1ZGVzIHBhcmFtZXRlcnMgZm9yIHRpdGxlcywgcGF0aHMsXG4gKiBjb2xvciBzY2hlbWVzLCBhbmQgYXNzZXQgcGF0aHMgdG8gdGFpbG9yIHRoZSBBUEkgZG9jdW1lbnRhdGlvbiBleHBlcmllbmNlLlxuICpcbiAqIEBwYXJhbSB7c3RyaW5nfSB0aXRsZSAtIFRpdGxlIGRpc3BsYXllZCBpbiBTd2FnZ2VyIFVJLlxuICogQHBhcmFtIHtzdHJpbmd9IGRlc2NyaXB0aW9uIC0gRGVzY3JpcHRpb24gc2hvd24gYmVsb3cgdGhlIHRpdGxlLlxuICogQHBhcmFtIHtzdHJpbmd9IHZlcnNpb24gLSBBUEkgdmVyc2lvbiBkaXNwbGF5ZWQgaW4gdGhlIGRvY3VtZW50YXRpb24uXG4gKiBAcGFyYW0ge3N0cmluZ30gW3BhdGhdIC0gT3B0aW9uYWwgcGF0aCB3aGVyZSBTd2FnZ2VyIHdpbGwgYmUgYXZhaWxhYmxlLlxuICogQHBhcmFtIHtib29sZWFufSBbcGVyc2lzdEF1dGhvcml6YXRpb25dIC0gV2hldGhlciBhdXRob3JpemF0aW9uIHRva2VucyBwZXJzaXN0IGFjcm9zcyByZWxvYWRzLlxuICogQHBhcmFtIHtzdHJpbmd9IFthc3NldHNQYXRoXSAtIFBhdGggdG8gY3VzdG9tIGFzc2V0cyBmb3IgU3dhZ2dlciBVSS5cbiAqIEBwYXJhbSB7c3RyaW5nfSBbdG9wYmFyQmdDb2xvcl0gLSBDdXN0b20gYmFja2dyb3VuZCBjb2xvciBmb3IgdGhlIFN3YWdnZXIgdG9wIGJhci5cbiAqIEBwYXJhbSB7c3RyaW5nfSBbdG9wYmFySWNvblBhdGhdIC0gUGF0aCB0byBhIGN1c3RvbSBpY29uIGRpc3BsYXllZCBpbiB0aGUgdG9wIGJhci5cbiAqIEBwYXJhbSB7c3RyaW5nfSBbZmF2aWNvblBhdGhdIC0gUGF0aCB0byBhIGN1c3RvbSBmYXZpY29uLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIFN3YWdnZXJTZXR1cE9wdGlvbnMge1xuICB0aXRsZTogc3RyaW5nO1xuICBkZXNjcmlwdGlvbjogc3RyaW5nO1xuICB2ZXJzaW9uOiBzdHJpbmc7XG4gIHBhdGg/OiBzdHJpbmc7XG4gIHBlcnNpc3RBdXRob3JpemF0aW9uPzogYm9vbGVhbjtcbiAgYXNzZXRzUGF0aD86IHN0cmluZztcbiAgdG9wYmFyQmdDb2xvcj86IHN0cmluZztcbiAgdG9wYmFySWNvblBhdGg/OiBzdHJpbmc7XG4gIGZhdmljb25QYXRoPzogc3RyaW5nO1xuICAvKipcbiAgICogUGF0aCB0byBleHBvc2UgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBpbiBKU09OIGZvcm1hdC5cbiAgICogRXhhbXBsZTogL2FwaS9vcGVuYXBpLmpzb25cbiAgICovXG4gIG9wZW5BcGlKc29uUGF0aD86IHN0cmluZztcbiAgLyoqXG4gICAqIFBhdGggdG8gZXhwb3NlIHRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24gaW4gWUFNTCBmb3JtYXQuXG4gICAqIFJlcXVpcmVzICd5YW1sJyBwYWNrYWdlIGFzIHBlZXIgZGVwZW5kZW5jeS5cbiAgICogRXhhbXBsZTogL2FwaS9vcGVuYXBpLnlhbWxcbiAgICovXG4gIG9wZW5BcGlZYW1sUGF0aD86IHN0cmluZztcbn1cblxuLyoqXG4gKiBAZGVzY3JpcHRpb25cbiAqIEEgZmx1ZW50LCBzdGF0aWMgYm9vdHN0cmFwIGNsYXNzIGZvciBpbml0aWFsaXppbmcgYW5kIGNvbmZpZ3VyaW5nIGEgTmVzdEpTIGFwcGxpY2F0aW9uLlxuICpcbiAqIEBzdW1tYXJ5XG4gKiBUaGUgYE5lc3RCb290c3RyYXBlcmAgY2xhc3MgcHJvdmlkZXMgYSBjaGFpbmFibGUgQVBJIGZvciBjb25maWd1cmluZ1xuICogYSBOZXN0SlMgYXBwbGljYXRpb24gaW5zdGFuY2UuIEl0IGluY2x1ZGVzIGJ1aWx0LWluIG1ldGhvZHMgZm9yIGVuYWJsaW5nXG4gKiBDT1JTLCBIZWxtZXQgc2VjdXJpdHksIFN3YWdnZXIgZG9jdW1lbnRhdGlvbiwgZ2xvYmFsIHBpcGVzLCBmaWx0ZXJzLFxuICogaW50ZXJjZXB0b3JzLCBhbmQgc3RhcnRpbmcgdGhlIHNlcnZlci5cbiAqXG4gKiBUaGlzIGNsYXNzIHByb21vdGVzIGNvbnNpc3RlbmN5IGFuZCByZWR1Y2VzIHJlcGV0aXRpdmUgc2V0dXAgY29kZVxuICogYWNyb3NzIG11bHRpcGxlIE5lc3RKUyBwcm9qZWN0cy5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHNcbiAqIGltcG9ydCB7IE5lc3RGYWN0b3J5IH0gZnJvbSBcIkBuZXN0anMvY29yZVwiO1xuICogaW1wb3J0IHsgQXBwTW9kdWxlIH0gZnJvbSBcIi4vYXBwLm1vZHVsZVwiO1xuICogaW1wb3J0IHsgTXlMb2dnZXIgfSBmcm9tIFwiLi9NeUxvZ2dlclwiO1xuICogaW1wb3J0IHsgTmVzdEJvb3RzdHJhcGVyIH0gZnJvbSBcIkBkZWNhZi10cy9mb3ItbmVzdFwiO1xuICpcbiAqIGFzeW5jIGZ1bmN0aW9uIGJvb3RzdHJhcCgpIHtcbiAqICAgY29uc3QgYXBwID0gYXdhaXQgTmVzdEZhY3RvcnkuY3JlYXRlKEFwcE1vZHVsZSk7XG4gKlxuICogICBhd2FpdCBOZXN0Qm9vdHN0cmFwZXJcbiAqICAgICAuaW5pdGlhbGl6ZShhcHApXG4gKiAgICAgLmVuYWJsZUxvZ2dlcihuZXcgTXlMb2dnZXIoKSlcbiAqICAgICAuZW5hYmxlQ29ycyhbXCJodHRwOi8vbG9jYWxob3N0OjQyMDBcIl0pXG4gKiAgICAgLnVzZUhlbG1ldCgpXG4gKiAgICAgLnNldHVwU3dhZ2dlcih7XG4gKiAgICAgICB0aXRsZTogXCJPcGVuQVBJIGJ5IFRyYWRlTWFya+KEolwiLFxuICogICAgICAgZGVzY3JpcHRpb246IFwiVHJhZGVNYXJr4oSiIEFQSSBkb2N1bWVudGF0aW9uXCIsXG4gKiAgICAgICB2ZXJzaW9uOiBcIjEuMC4wXCIsXG4gKiAgICAgICBwYXRoOiBcImFwaVwiLFxuICogICAgICAgcGVyc2lzdEF1dGhvcml6YXRpb246IHRydWUsXG4gKiAgICAgICB0b3BiYXJCZ0NvbG9yOiBcIiMyQzNFNTBcIixcbiAqICAgICAgIHRvcGJhckljb25QYXRoOiBcIi9hc3NldHMvbG9nby5zdmdcIixcbiAqICAgICAgIGZhdmljb25QYXRoOiBcIi9hc3NldHMvZmF2aWNvbi5pY29cIlxuICogICAgIH0pXG4gKiAgICAgLnVzZUdsb2JhbEZpbHRlcnMoKVxuICogICAgIC51c2VHbG9iYWxQaXBlcyguLi4pXG4gKiAgICAgLnVzZUdsb2JhbEludGVyY2VwdG9ycyguLi4pXG4gKiAgICAgLnN0YXJ0KDMwMDApO1xuICogfVxuICpcbiAqIGJvb3RzdHJhcCgpO1xuICogYGBgXG4gKiBAY2xhc3NcbiAqL1xuZXhwb3J0IGNsYXNzIE5lc3RCb290c3RyYXBlciB7XG4gIHByaXZhdGUgc3RhdGljIGFwcDogSU5lc3RBcHBsaWNhdGlvbjtcbiAgcHJpdmF0ZSBzdGF0aWMgX2xvZ2dlcjogTG9nZ2VyO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb25cbiAgICogUmV0dXJucyB0aGUgY3VycmVudCBsb2dnZXIgaW5zdGFuY2UsIGNyZWF0aW5nIGEgZGVmYXVsdCBvbmUgaWYgbm90IHNldC5cbiAgICpcbiAgICogQHN1bW1hcnlcbiAgICogRW5zdXJlcyB0aGF0IGEgdmFsaWQgYExvZ2dlcmAgaW5zdGFuY2UgaXMgYWx3YXlzIGF2YWlsYWJsZVxuICAgKiBmb3IgbG9nZ2luZyBib290c3RyYXAtcmVsYXRlZCBtZXNzYWdlcy5cbiAgICpcbiAgICogQHJldHVybiB7TG9nZ2VyfSBUaGUgYWN0aXZlIGxvZ2dlciBpbnN0YW5jZS5cbiAgICovXG4gIHByaXZhdGUgc3RhdGljIGdldCBsb2dnZXIoKTogTG9nZ2VyIHtcbiAgICBpZiAoIXRoaXMuX2xvZ2dlcikge1xuICAgICAgLy8gZmFsbGJhY2tcbiAgICAgIHRoaXMuX2xvZ2dlciA9IG5ldyBMb2dnZXIoXCJOZXN0Qm9vdHN0cmFwXCIpO1xuICAgIH1cbiAgICByZXR1cm4gdGhpcy5fbG9nZ2VyO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvblxuICAgKiBJbml0aWFsaXplcyB0aGUgYm9vdHN0cmFwcGVyIHdpdGggYSBnaXZlbiBOZXN0SlMgYXBwbGljYXRpb24uXG4gICAqXG4gICAqIEBzdW1tYXJ5XG4gICAqIEJpbmRzIHRoZSBwcm92aWRlZCBOZXN0SlMgYXBwIGluc3RhbmNlIHRvIHRoZSBib290c3RyYXBwZXIsIGVuYWJsaW5nXG4gICAqIGNoYWluZWQgY29uZmlndXJhdGlvbiBtZXRob2RzLlxuICAgKlxuICAgKiBAcGFyYW0ge0lOZXN0QXBwbGljYXRpb259IGFwcCAtIFRoZSBOZXN0SlMgYXBwbGljYXRpb24gaW5zdGFuY2UgdG8gaW5pdGlhbGl6ZS5cbiAgICogQHJldHVybiB7TmVzdEJvb3RzdHJhcGVyfSBSZXR1cm5zIHRoZSBjbGFzcyBmb3IgY2hhaW5pbmcgY29uZmlndXJhdGlvbiBtZXRob2RzLlxuICAgKi9cbiAgc3RhdGljIGluaXRpYWxpemUoYXBwOiBJTmVzdEFwcGxpY2F0aW9uKSB7XG4gICAgdGhpcy5hcHAgPSBhcHA7XG4gICAgcmV0dXJuIHRoaXM7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uXG4gICAqIEVuYWJsZXMgb3IgcmVwbGFjZXMgdGhlIGdsb2JhbCBsb2dnZXIgZm9yIHRoZSBOZXN0SlMgYXBwbGljYXRpb24uXG4gICAqXG4gICAqIEBzdW1tYXJ5XG4gICAqIElmIGEgY3VzdG9tIGxvZ2dlciBpcyBwcm92aWRlZCwgaXQgcmVwbGFjZXMgdGhlIGRlZmF1bHQgbG9nZ2VyLiBPdGhlcndpc2UsXG4gICAqIGEgbmV3IGxvZ2dlciBuYW1lZCBgXCJOZXN0Qm9vdHN0cmFwXCJgIGlzIHVzZWQuIFRoaXMgbG9nZ2VyIGlzIGFsc28gcmVnaXN0ZXJlZFxuICAgKiB3aXRoIHRoZSBOZXN0SlMgYXBwbGljYXRpb24uXG4gICAqXG4gICAqIEBwYXJhbSB7TG9nZ2VyfSBbY3VzdG9tTG9nZ2VyXSAtIE9wdGlvbmFsIGN1c3RvbSBsb2dnZXIgaW5zdGFuY2UuXG4gICAqIEByZXR1cm4ge05lc3RCb290c3RyYXBlcn0gUmV0dXJucyB0aGUgY2xhc3MgZm9yIGNoYWluaW5nLlxuICAgKi9cbiAgc3RhdGljIGVuYWJsZUxvZ2dlcihjdXN0b21Mb2dnZXI/OiBMb2dnZXIpIHtcbiAgICB0aGlzLl9sb2dnZXIgPSBjdXN0b21Mb2dnZXIgfHwgbmV3IExvZ2dlcihcIk5lc3RCb290c3RyYXBcIik7XG4gICAgdGhpcy5hcHAudXNlTG9nZ2VyKHRoaXMuX2xvZ2dlcik7XG4gICAgcmV0dXJuIHRoaXM7XG4gIH1cblxuICAvKipcbiAgICogQGRlc2NyaXB0aW9uXG4gICAqIEVuYWJsZXMgQ3Jvc3MtT3JpZ2luIFJlc291cmNlIFNoYXJpbmcgKENPUlMpIGZvciB0aGUgYXBwbGljYXRpb24uXG4gICAqXG4gICAqIEBzdW1tYXJ5XG4gICAqIEFsbG93cyBkZWZpbmluZyBlaXRoZXIgYSB3aWxkY2FyZCBvcmlnaW4gKGBcIipcImApIG9yIGEgbGlzdCBvZiBhbGxvd2VkIG9yaWdpbnMuXG4gICAqIEF1dG9tYXRpY2FsbHkgYWNjZXB0cyBsb2NhbCBkZXZlbG9wbWVudCByZXF1ZXN0cyBhbmQgdGhvc2Ugd2l0aG91dCBvcmlnaW4gaGVhZGVycy5cbiAgICogVGhyb3dzIGEgYENvcnNFcnJvcmAgZm9yIHVuYXV0aG9yaXplZCBvcmlnaW5zLlxuICAgKlxuICAgKiBAcGFyYW0geycqJyB8IHN0cmluZ1tdfSBbb3JpZ2lucz1bXV0gLSBMaXN0IG9mIGFsbG93ZWQgb3JpZ2lucyBvciBgXCIqXCJgIHRvIGFsbG93IGFsbC5cbiAgICogQHBhcmFtIHtzdHJpbmdbXX0gW2FsbG93TWV0aG9kcz1bJ0dFVCcsICdQT1NUJywgJ1BVVCcsICdERUxFVEUnXV0gLSBBbGxvd2VkIEhUVFAgbWV0aG9kcy5cbiAgICogQHJldHVybiB7TmVzdEJvb3RzdHJhcGVyfSBSZXR1cm5zIHRoZSBjbGFzcyBmb3IgY2hhaW5pbmcgY29uZmlndXJhdGlvbi5cbiAgICpcbiAgICovXG4gIHN0YXRpYyBlbmFibGVDb3JzKFxuICAgIG9yaWdpbnM6IFwiKlwiIHwgc3RyaW5nW10gPSBbXSxcbiAgICBhbGxvd01ldGhvZHM6IHN0cmluZ1tdID0gW1wiR0VUXCIsIFwiUE9TVFwiLCBcIlBVVFwiLCBcIkRFTEVURVwiXVxuICApIHtcbiAgICBjb25zdCBhbGxvd2VkT3JpZ2lucyA9XG4gICAgICBvcmlnaW5zID09PSBcIipcIiA/IFwiKlwiIDogb3JpZ2lucy5tYXAoKG8pID0+IG8udHJpbSgpLnRvTG93ZXJDYXNlKCkpO1xuXG4gICAgY29uc3QgY29yc09wdGlvbnM6IENvcnNPcHRpb25zID0ge1xuICAgICAgb3JpZ2luOiAob3JpZ2luLCBjYWxsYmFjaykgPT4ge1xuICAgICAgICAvLyBBbGxvdyByZXF1ZXN0IHdpdGhvdXQgb3JpZ2luLi4uXG4gICAgICAgIGlmICghb3JpZ2luKSByZXR1cm4gY2FsbGJhY2sobnVsbCwgdHJ1ZSk7XG5cbiAgICAgICAgaWYgKFxuICAgICAgICAgIGFsbG93ZWRPcmlnaW5zID09PSBcIipcIiB8fFxuICAgICAgICAgIChBcnJheS5pc0FycmF5KGFsbG93ZWRPcmlnaW5zKSAmJlxuICAgICAgICAgICAgYWxsb3dlZE9yaWdpbnMuaW5jbHVkZXMob3JpZ2luLnRvTG93ZXJDYXNlKCkpKVxuICAgICAgICApIHtcbiAgICAgICAgICByZXR1cm4gY2FsbGJhY2sobnVsbCwgdHJ1ZSk7XG4gICAgICAgIH1cblxuICAgICAgICBjYWxsYmFjayhuZXcgQ29yc0Vycm9yKGBPcmlnaW4gJHtvcmlnaW59IG5vdCBhbGxvd2VkYCkpO1xuICAgICAgfSxcbiAgICAgIGNyZWRlbnRpYWxzOiB0cnVlLFxuICAgICAgbWV0aG9kczogYWxsb3dNZXRob2RzLmpvaW4oXCIsXCIpLFxuICAgIH07XG5cbiAgICB0aGlzLmFwcC5lbmFibGVDb3JzKGNvcnNPcHRpb25zKTtcbiAgICByZXR1cm4gdGhpcztcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb25cbiAgICogQXBwbGllcyB0aGUgSGVsbWV0IG1pZGRsZXdhcmUgZm9yIGVuaGFuY2VkIHNlY3VyaXR5LlxuICAgKlxuICAgKiBAc3VtbWFyeVxuICAgKiBEeW5hbWljYWxseSBsb2FkcyB0aGUgYGhlbG1ldGAgcGFja2FnZSBpZiBhdmFpbGFibGUgYW5kIHJlZ2lzdGVycyBpdFxuICAgKiBhcyBtaWRkbGV3YXJlIHRvIGltcHJvdmUgSFRUUCBoZWFkZXIgc2VjdXJpdHkuIElmIG5vdCBpbnN0YWxsZWQsIGxvZ3MgYSB3YXJuaW5nXG4gICAqIGFuZCBjb250aW51ZXMgZXhlY3V0aW9uIHdpdGhvdXQgdGhyb3dpbmcgZXJyb3JzLlxuICAgKlxuICAgKiBAcGFyYW0ge1JlY29yZDxzdHJpbmcsIGFueT59IFtvcHRpb25zXSAtIE9wdGlvbmFsIGNvbmZpZ3VyYXRpb24gcGFzc2VkIHRvIEhlbG1ldC5cbiAgICogQHJldHVybiB7TmVzdEJvb3RzdHJhcGVyfSBSZXR1cm5zIHRoZSBjbGFzcyBmb3IgY2hhaW5pbmcgY29uZmlndXJhdGlvbi5cbiAgICovXG4gIHN0YXRpYyB1c2VIZWxtZXQob3B0aW9ucz86IFJlY29yZDxzdHJpbmcsIGFueT4pIHtcbiAgICB0cnkge1xuICAgICAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEB0eXBlc2NyaXB0LWVzbGludC9uby1yZXF1aXJlLWltcG9ydHNcbiAgICAgIGNvbnN0IGhlbG1ldCA9IHJlcXVpcmUoXCJoZWxtZXRcIik7IC8vIER5bmFtaWMgaW1wb3J0IHRvIGF2b2lkIGhhcmQgZGVwZW5kZW5jeVxuICAgICAgdGhpcy5hcHAudXNlKGhlbG1ldChvcHRpb25zKSk7XG4gICAgICB0aGlzLmxvZ2dlci5sb2coXCJIZWxtZXQgbWlkZGxld2FyZSBlbmFibGVkIHN1Y2Nlc3NmdWxseS5cIik7XG4gICAgICAvLyBlc2xpbnQtZGlzYWJsZS1uZXh0LWxpbmUgQHR5cGVzY3JpcHQtZXNsaW50L25vLXVudXNlZC12YXJzXG4gICAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgICB0aGlzLmxvZ2dlci53YXJuKFwiSGVsbWV0IG5vdCBpbnN0YWxsZWQuIFNraXBwaW5nIG1pZGRsZXdhcmUuXCIpO1xuICAgIH1cblxuICAgIHJldHVybiB0aGlzO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvblxuICAgKiBDb25maWd1cmVzIGFuZCBpbml0aWFsaXplcyBTd2FnZ2VyIFVJIGZvciBBUEkgZG9jdW1lbnRhdGlvbi5cbiAgICpcbiAgICogQHN1bW1hcnlcbiAgICogVXNlcyB0aGUgYFN3YWdnZXJCdWlsZGVyYCB1dGlsaXR5IHRvIGNvbmZpZ3VyZSBBUEkgZG9jdW1lbnRhdGlvblxuICAgKiB3aXRoIGRldGFpbGVkIGN1c3RvbWl6YXRpb24gZm9yIHRpdGxlLCB2ZXJzaW9uLCBwYXRocywgYW5kIGNvbG9ycy5cbiAgICogU3dhZ2dlciBpcyBhdXRvbWF0aWNhbGx5IGV4cG9zZWQgYXQgdGhlIGNvbmZpZ3VyZWQgcGF0aC5cbiAgICpcbiAgICogQHBhcmFtIHtTd2FnZ2VyU2V0dXBPcHRpb25zfSBvcHRpb25zIC0gU3dhZ2dlciBjb25maWd1cmF0aW9uIG9wdGlvbnMuXG4gICAqIEByZXR1cm4ge05lc3RCb290c3RyYXBlcn0gUmV0dXJucyB0aGUgY2xhc3MgZm9yIGNoYWluaW5nIGNvbmZpZ3VyYXRpb24uXG4gICAqL1xuICBzdGF0aWMgc2V0dXBTd2FnZ2VyKG9wdGlvbnM6IFN3YWdnZXJTZXR1cE9wdGlvbnMpIHtcbiAgICBjb25zdCBzd2FnZ2VyID0gbmV3IFN3YWdnZXJCdWlsZGVyKHRoaXMuYXBwLCB7XG4gICAgICB0aXRsZTogb3B0aW9ucy50aXRsZSxcbiAgICAgIGRlc2NyaXB0aW9uOiBvcHRpb25zLmRlc2NyaXB0aW9uLFxuICAgICAgdmVyc2lvbjogb3B0aW9ucy52ZXJzaW9uLFxuICAgICAgcGF0aDogb3B0aW9ucy5wYXRoIHx8IFwiYXBpXCIsXG4gICAgICBwZXJzaXN0QXV0aG9yaXphdGlvbjogb3B0aW9ucy5wZXJzaXN0QXV0aG9yaXphdGlvbiA/PyB0cnVlLFxuICAgICAgYXNzZXRzUGF0aDogb3B0aW9ucy5hc3NldHNQYXRoLFxuICAgICAgZmF2aWNvbkZpbGVQYXRoOiBvcHRpb25zLmZhdmljb25QYXRoLFxuICAgICAgdG9wYmFySWNvbkZpbGVQYXRoOiBvcHRpb25zLnRvcGJhckljb25QYXRoLFxuICAgICAgdG9wYmFyQmdDb2xvcjogb3B0aW9ucy50b3BiYXJCZ0NvbG9yLFxuICAgICAgb3BlbkFwaUpzb25QYXRoOiBvcHRpb25zLm9wZW5BcGlKc29uUGF0aCxcbiAgICAgIG9wZW5BcGlZYW1sUGF0aDogb3B0aW9ucy5vcGVuQXBpWWFtbFBhdGgsXG4gICAgfSk7XG4gICAgc3dhZ2dlci5zZXR1cFN3YWdnZXIoKTtcbiAgICByZXR1cm4gdGhpcztcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb25cbiAgICogUmVnaXN0ZXJzIG9uZSBvciBtb3JlIGdsb2JhbCB2YWxpZGF0aW9uIHBpcGVzLlxuICAgKlxuICAgKiBAc3VtbWFyeVxuICAgKiBFbmFibGVzIHJlcXVlc3QgcGF5bG9hZCB2YWxpZGF0aW9uIGFuZCB0cmFuc2Zvcm1hdGlvbiBnbG9iYWxseSBhY3Jvc3NcbiAgICogdGhlIGVudGlyZSBOZXN0SlMgYXBwbGljYXRpb24uIE11bHRpcGxlIHBpcGVzIGNhbiBiZSBjaGFpbmVkIHRvZ2V0aGVyXG4gICAqIGZvciBtb2R1bGFyIGlucHV0IHZhbGlkYXRpb24uXG4gICAqXG4gICAqIEBwYXJhbSB7Li4uUGlwZVRyYW5zZm9ybVtdfSBwaXBlcyAtIFBpcGUgaW5zdGFuY2VzIHRvIHJlZ2lzdGVyIGdsb2JhbGx5LlxuICAgKiBAcmV0dXJuIHtOZXN0Qm9vdHN0cmFwZXJ9IFJldHVybnMgdGhlIGNsYXNzIGZvciBjaGFpbmluZy5cbiAgICovXG4gIHN0YXRpYyB1c2VHbG9iYWxQaXBlcyguLi5waXBlczogUGlwZVRyYW5zZm9ybVtdKSB7XG4gICAgaWYgKHBpcGVzLmxlbmd0aCA+IDApIHRoaXMuYXBwLnVzZUdsb2JhbFBpcGVzKC4uLnBpcGVzKTtcbiAgICByZXR1cm4gdGhpcztcbiAgfVxuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb25cbiAgICogUmVnaXN0ZXJzIG9uZSBvciBtb3JlIGdsb2JhbCBleGNlcHRpb24gZmlsdGVycy5cbiAgICpcbiAgICogQHN1bW1hcnlcbiAgICogSWYgbm8gZmlsdGVycyBhcmUgcHJvdmlkZWQsIGl0IGF1dG9tYXRpY2FsbHkgcmVnaXN0ZXJzIGEgZGVmYXVsdFxuICAgKiBzZXQgb2Ygc3RhbmRhcmQgZXhjZXB0aW9uIGZpbHRlcnMgZm9yIGNvbW1vbiBlcnJvciB0eXBlcyBsaWtlXG4gICAqIGBIdHRwRXhjZXB0aW9uYCwgYFZhbGlkYXRpb25FeGNlcHRpb25gLCBgQ29uZmxpY3RFeGNlcHRpb25gLCBhbmQgb3RoZXJzLlxuICAgKlxuICAgKiBAcGFyYW0gey4uLkV4Y2VwdGlvbkZpbHRlcltdfSBmaWx0ZXJzIC0gT3B0aW9uYWwgZmlsdGVycyB0byBhcHBseSBnbG9iYWxseS5cbiAgICovXG4gIHN0YXRpYyB1c2VHbG9iYWxGaWx0ZXJzKC4uLmZpbHRlcnM6IGFueVtdKSB7XG4gICAgdGhpcy5hcHAudXNlR2xvYmFsRmlsdGVycyhcbiAgICAgIC4uLihmaWx0ZXJzLmxlbmd0aCA+IDAgPyBmaWx0ZXJzIDogW25ldyBEZWNhZkV4Y2VwdGlvbkZpbHRlcigpXSlcbiAgICApO1xuICAgIHJldHVybiB0aGlzO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvblxuICAgKiBSZWdpc3RlcnMgZ2xvYmFsIGludGVyY2VwdG9ycyBmb3IgcmVxdWVzdCBhbmQgcmVzcG9uc2UgdHJhbnNmb3JtYXRpb24uXG4gICAqXG4gICAqIEBzdW1tYXJ5XG4gICAqIEludGVyY2VwdG9ycyBhbGxvdyBhZHZhbmNlZCByZXF1ZXN0L3Jlc3BvbnNlIG1hbmlwdWxhdGlvbiBzdWNoIGFzXG4gICAqIHNlcmlhbGl6YXRpb24sIGxvZ2dpbmcsIG9yIHRyYW5zZm9ybWF0aW9uLiBNdWx0aXBsZSBpbnRlcmNlcHRvcnNcbiAgICogY2FuIGJlIGFkZGVkIGZvciBtb2R1bGFyIGNvbmZpZ3VyYXRpb24uXG4gICAqXG4gICAqIEBwYXJhbSB7Li4uTmVzdEludGVyY2VwdG9yW119IGludGVyY2VwdG9ycyAtIEludGVyY2VwdG9yIGluc3RhbmNlcyB0byByZWdpc3Rlci5cbiAgICogQHJldHVybiB7TmVzdEJvb3RzdHJhcGVyfSBSZXR1cm5zIHRoZSBjbGFzcyBmb3IgY2hhaW5pbmcgY29uZmlndXJhdGlvbi5cbiAgICovXG4gIHN0YXRpYyB1c2VHbG9iYWxJbnRlcmNlcHRvcnMoLi4uaW50ZXJjZXB0b3JzOiBOZXN0SW50ZXJjZXB0b3JbXSkge1xuICAgIGlmIChpbnRlcmNlcHRvcnMubGVuZ3RoID4gMClcbiAgICAgIHRoaXMuYXBwLnVzZUdsb2JhbEludGVyY2VwdG9ycyguLi5pbnRlcmNlcHRvcnMpO1xuICAgIHJldHVybiB0aGlzO1xuICB9XG5cbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvblxuICAgKiBTdGFydHMgdGhlIE5lc3RKUyBhcHBsaWNhdGlvbiBhbmQgYmluZHMgaXQgdG8gdGhlIGdpdmVuIHBvcnQgYW5kIGhvc3QuXG4gICAqXG4gICAqIEBzdW1tYXJ5XG4gICAqIExpc3RlbnMgb24gdGhlIHNwZWNpZmllZCBwb3J0IGFuZCBvcHRpb25hbGx5IGEgaG9zdC4gT25jZSBzdGFydGVkLFxuICAgKiBsb2dzIHRoZSBhcHBsaWNhdGlvbiBVUkwgZm9yIGVhc3kgYWNjZXNzLiBUaGUgc3RhcnR1cCBwcm9jZXNzIHJlc29sdmVzXG4gICAqIG9uY2UgdGhlIGFwcGxpY2F0aW9uIGlzIHN1Y2Nlc3NmdWxseSBydW5uaW5nLlxuICAgKlxuICAgKiBAcGFyYW0ge251bWJlcn0gW3BvcnQ9MzAwMF0gLSBQb3J0IG51bWJlciB0byBsaXN0ZW4gb24uXG4gICAqIEBwYXJhbSB7c3RyaW5nfSBbaG9zdF0gLSBPcHRpb25hbCBob3N0IG9yIElQIGFkZHJlc3MgdG8gYmluZCB0by5cbiAgICogQHBhcmFtIHtib29sZWFufSBbbG9nPXRydWVdIC0gV2hldGhlciB0byBsb2cgdGhlIGFwcGxpY2F0aW9uIFVSTCB1cG9uIHN0YXJ0dXAuXG4gICAqIEByZXR1cm4ge1Byb21pc2U8dm9pZD59IFJlc29sdmVzIG9uY2UgdGhlIGFwcGxpY2F0aW9uIHN0YXJ0cyBzdWNjZXNzZnVsbHkuXG4gICAqL1xuICBzdGF0aWMgYXN5bmMgc3RhcnQoXG4gICAgcG9ydDogbnVtYmVyID0gTnVtYmVyKHByb2Nlc3MuZW52LlBPUlQpIHx8IDMwMDAsXG4gICAgaG9zdDogc3RyaW5nIHwgdW5kZWZpbmVkID0gdW5kZWZpbmVkLFxuICAgIGxvZzogYm9vbGVhbiA9IHRydWVcbiAgKSB7XG4gICAgdGhpcy5hcHAubGlzdGVuKHBvcnQsIGhvc3QgYXMgYW55KS50aGVuKGFzeW5jICgpID0+IHtcbiAgICAgIGlmIChsb2cpIHtcbiAgICAgICAgY29uc3QgdXJsID0gYXdhaXQgdGhpcy5hcHAuZ2V0VXJsKCk7XG4gICAgICAgIHRoaXMubG9nZ2VyLmxvZyhg8J+agCBBcHBsaWNhdGlvbiBpcyBydW5uaW5nIGF0OiAke3VybH1gKTtcbiAgICAgIH1cbiAgICB9KTtcbiAgfVxufVxuIiwiaW1wb3J0IHsgSW5qZWN0IH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBDb25zdHJ1Y3RvciB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdGlvblwiO1xuaW1wb3J0IHtcbiAgaW5qZWN0YWJsZVNlcnZpY2VLZXksXG4gIE1vZGVsU2VydmljZSxcbiAgUmVwb3NpdG9yeSBhcyBDb3JlUmVwb3NpdG9yeSxcbiAgU2VydmljZSBhcyBDb3JlU2VydmljZSxcbn0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBNb2RlbCB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdG9yLXZhbGlkYXRpb25cIjtcblxuZXhwb3J0IHR5cGUgRGVjYWZGYWN0b3J5UHJvdmlkZXIgPSB7XG4gIHByb3ZpZGU6IHN0cmluZztcbiAgdXNlRmFjdG9yeTogKCkgPT4gdW5rbm93bjtcbn07XG5cbmNvbnN0IHBlbmRpbmdQcm92aWRlcnMgPSBuZXcgTWFwPHN0cmluZywgRGVjYWZGYWN0b3J5UHJvdmlkZXI+KCk7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIERyYWlucyB0aGUgcHJvdmlkZXJzIHJlZ2lzdGVyZWQgYnkge0BsaW5rIFNlcnZpY2V9L3tAbGluayBSZXBvc2l0b3J5fSB1c2FnZXMuXG4gKiBAc3VtbWFyeSBDb25zdW1lZCBieSB7QGxpbmsgRGVjYWZDb3JlTW9kdWxlLmZvclJvb3R9IHRvIHdpcmUgdGhlIHRva2VucyB0aGF0IHRob3NlIHBhcmFtZXRlclxuICogZGVjb3JhdG9ycyByZWZlcmVuY2UgYmFjayBpbnRvIE5lc3QncyBESSBjb250YWluZXIsIHNpbmNlIGEgcGFyYW1ldGVyIGRlY29yYXRvciBhbG9uZSBjYW5ub3RcbiAqIHN1cHBseSBhIHByb3ZpZGVyLlxuICovXG5leHBvcnQgZnVuY3Rpb24gZ2V0UmVnaXN0ZXJlZERlY2FmUHJvdmlkZXJzKCk6IERlY2FmRmFjdG9yeVByb3ZpZGVyW10ge1xuICByZXR1cm4gQXJyYXkuZnJvbShwZW5kaW5nUHJvdmlkZXJzLnZhbHVlcygpKTtcbn1cblxuZnVuY3Rpb24gcmVnaXN0ZXJQcm92aWRlcih0b2tlbjogc3RyaW5nLCB1c2VGYWN0b3J5OiAoKSA9PiB1bmtub3duKTogdm9pZCB7XG4gIGlmIChwZW5kaW5nUHJvdmlkZXJzLmhhcyh0b2tlbikpIHJldHVybjtcbiAgcGVuZGluZ1Byb3ZpZGVycy5zZXQodG9rZW4sIHsgcHJvdmlkZTogdG9rZW4sIHVzZUZhY3RvcnkgfSk7XG59XG5cbmZ1bmN0aW9uIGlzTW9kZWxDb25zdHJ1Y3Rvcih2YWx1ZTogdW5rbm93bik6IHZhbHVlIGlzIENvbnN0cnVjdG9yPGFueT4ge1xuICByZXR1cm4gdHlwZW9mIHZhbHVlID09PSBcImZ1bmN0aW9uXCIgJiYgdmFsdWUucHJvdG90eXBlIGluc3RhbmNlb2YgTW9kZWw7XG59XG5cbmZ1bmN0aW9uIG1vZGVsU2VydmljZVRva2VuKG1vZGVsOiBDb25zdHJ1Y3Rvcjxhbnk+KTogc3RyaW5nIHtcbiAgcmV0dXJuIGAke21vZGVsLm5hbWV9U2VydmljZWA7XG59XG5cbmZ1bmN0aW9uIG1vZGVsUmVwb3NpdG9yeVRva2VuKFxuICBtb2RlbDogQ29uc3RydWN0b3I8YW55PixcbiAgZmxhdm91cj86IHN0cmluZ1xuKTogc3RyaW5nIHtcbiAgcmV0dXJuIGZsYXZvdXIgPyBgJHttb2RlbC5uYW1lfVJlcG9zaXRvcnlAJHtmbGF2b3VyfWAgOiBgJHttb2RlbC5uYW1lfVJlcG9zaXRvcnlgO1xufVxuXG4vKipcbiAqIEBkZXNjcmlwdGlvbiBDb25zdHJ1Y3RvciBwYXJhbWV0ZXIgZGVjb3JhdG9yIHRoYXQgaW5qZWN0cyBhIGRlY2FmIHNlcnZpY2UuXG4gKiBAc3VtbWFyeSBUcmFuc2xhdGVzIGRlY2FmJ3Mgc2VydmljZSByZXNvbHV0aW9uIEFQSXMgaW50byBhIE5lc3QgYEBJbmplY3QoKWAgY2FsbDpcbiAqIC0gYEBTZXJ2aWNlKFNvbWVNb2RlbClgIGluamVjdHMgdGhlIGBNb2RlbFNlcnZpY2U8U29tZU1vZGVsPmAgc2luZ2xldG9uIGZvciB0aGF0IG1vZGVsXG4gKiAgICh2aWEge0BsaW5rIE1vZGVsU2VydmljZS5mb3JNb2RlbH0pLlxuICogLSBgQFNlcnZpY2UoU29tZVNlcnZpY2VDbGFzcylgIG9yIGBAU2VydmljZShcImFsaWFzXCIpYCBpbmplY3RzIHRoZSBtYXRjaGluZyBgQHNlcnZpY2UoKWAtZGVjb3JhdGVkXG4gKiAgIGRlY2FmIHtAbGluayBDb3JlU2VydmljZX0gKHZpYSB7QGxpbmsgQ29yZVNlcnZpY2UuZ2V0fSkuXG4gKiAtIGBAU2VydmljZSgpYCBpbmZlcnMgdGhlIHBhcmFtZXRlcidzIHR5cGUgZnJvbSBlbWl0dGVkIGNvbnN0cnVjdG9yIG1ldGFkYXRhIGFuZCByZXNvbHZlcyBpdCB0aGVcbiAqICAgc2FtZSB3YXksIHByb3ZpZGVkIHRoZSBwYXJhbWV0ZXIgaXMgdHlwZWQgYXMgYSBjb25jcmV0ZSBjbGFzcyAoZ2VuZXJpY3MgbGlrZSBgTW9kZWxTZXJ2aWNlPFg+YFxuICogICBhcmUgZXJhc2VkIGF0IHJ1bnRpbWUsIHNvIHRoZSBtb2RlbC1zZXJ2aWNlIGZvcm0gYWx3YXlzIHJlcXVpcmVzIGFuIGV4cGxpY2l0IGFyZ3VtZW50KS5cbiAqIEBwYXJhbSBrZXkgdGhlIG1vZGVsIGNsYXNzLCBzZXJ2aWNlIGNsYXNzLCBvciBhbGlhcyB0byByZXNvbHZlLiBPbWl0IHRvIGluZmVyIGZyb20gdGhlIHBhcmFtZXRlciB0eXBlLlxuICovXG5leHBvcnQgZnVuY3Rpb24gU2VydmljZShrZXk/OiBzdHJpbmcgfCBDb25zdHJ1Y3Rvcjxhbnk+KTogUGFyYW1ldGVyRGVjb3JhdG9yIHtcbiAgcmV0dXJuIChcbiAgICB0YXJnZXQ6IG9iamVjdCxcbiAgICBwcm9wZXJ0eUtleTogc3RyaW5nIHwgc3ltYm9sIHwgdW5kZWZpbmVkLFxuICAgIHBhcmFtZXRlckluZGV4OiBudW1iZXJcbiAgKTogdm9pZCA9PiB7XG4gICAgbGV0IHJlc29sdmVkOiBzdHJpbmcgfCBDb25zdHJ1Y3Rvcjxhbnk+IHwgdW5kZWZpbmVkID0ga2V5O1xuICAgIGlmICh0eXBlb2YgcmVzb2x2ZWQgPT09IFwidW5kZWZpbmVkXCIpIHtcbiAgICAgIGNvbnN0IHBhcmFtVHlwZXM6IGFueVtdID1cbiAgICAgICAgUmVmbGVjdC5nZXRNZXRhZGF0YShcImRlc2lnbjpwYXJhbXR5cGVzXCIsIHRhcmdldCkgfHwgW107XG4gICAgICByZXNvbHZlZCA9IHBhcmFtVHlwZXNbcGFyYW1ldGVySW5kZXhdO1xuICAgIH1cbiAgICBpZiAodHlwZW9mIHJlc29sdmVkID09PSBcInVuZGVmaW5lZFwiKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgIGBAU2VydmljZSgpIGNvdWxkIG5vdCBkZXRlcm1pbmUgYW4gaW5qZWN0aW9uIHR5cGUgZm9yIHBhcmFtZXRlciAke3BhcmFtZXRlckluZGV4fSBvZiAkeyh0YXJnZXQgYXMgQ29uc3RydWN0b3I8YW55PikubmFtZX0uIFByb3ZpZGUgYW4gZXhwbGljaXQgYXJndW1lbnQsIGUuZy4gQFNlcnZpY2UoU29tZVNlcnZpY2UpLmBcbiAgICAgICk7XG4gICAgfVxuXG4gICAgY29uc3QgYXNNb2RlbCA9IHR5cGVvZiByZXNvbHZlZCAhPT0gXCJzdHJpbmdcIiAmJiBpc01vZGVsQ29uc3RydWN0b3IocmVzb2x2ZWQpO1xuICAgIGNvbnN0IHRva2VuID0gYXNNb2RlbFxuICAgICAgPyBtb2RlbFNlcnZpY2VUb2tlbihyZXNvbHZlZCBhcyBDb25zdHJ1Y3Rvcjxhbnk+KVxuICAgICAgOiBpbmplY3RhYmxlU2VydmljZUtleShyZXNvbHZlZCBhcyBzdHJpbmcgfCBDb25zdHJ1Y3Rvcjxhbnk+KTtcblxuICAgIHJlZ2lzdGVyUHJvdmlkZXIodG9rZW4sICgpID0+IHtcbiAgICAgIGlmIChhc01vZGVsKSByZXR1cm4gTW9kZWxTZXJ2aWNlLmZvck1vZGVsKHJlc29sdmVkIGFzIENvbnN0cnVjdG9yPGFueT4pO1xuICAgICAgcmV0dXJuIHR5cGVvZiByZXNvbHZlZCA9PT0gXCJzdHJpbmdcIlxuICAgICAgICA/IENvcmVTZXJ2aWNlLmdldChyZXNvbHZlZClcbiAgICAgICAgOiBDb3JlU2VydmljZS5nZXQocmVzb2x2ZWQgYXMgQ29uc3RydWN0b3I8YW55Pik7XG4gICAgfSk7XG5cbiAgICByZXR1cm4gSW5qZWN0KHRva2VuKSh0YXJnZXQsIHByb3BlcnR5S2V5LCBwYXJhbWV0ZXJJbmRleCk7XG4gIH07XG59XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIENvbnN0cnVjdG9yIHBhcmFtZXRlciBkZWNvcmF0b3IgdGhhdCBpbmplY3RzIGEgZGVjYWYge0BsaW5rIENvcmVSZXBvc2l0b3J5fSBmb3IgYG1vZGVsYC5cbiAqIEBzdW1tYXJ5IFVubGlrZSB7QGxpbmsgU2VydmljZX0sIGBtb2RlbGAgaXMgYWx3YXlzIHRyZWF0ZWQgYXMgYSBNb2RlbCBjbGFzczogdGhlIHNhbWUgY2xhc3MgcGFzc2VkXG4gKiB0byBgQFNlcnZpY2UobW9kZWwpYCBhbmQgYEBSZXBvc2l0b3J5KG1vZGVsKWAgcmVzb2x2ZXMgdG8gdHdvIGRpZmZlcmVudCBpbmplY3RlZCBvYmplY3RzIChhXG4gKiBgTW9kZWxTZXJ2aWNlPE0+YCB2cyBhIGBSZXBvc2l0b3J5PE0+YCByZXNwZWN0aXZlbHkpLCB2aWEge0BsaW5rIENvcmVSZXBvc2l0b3J5LmZvck1vZGVsfS5cbiAqIEBwYXJhbSBtb2RlbCB0aGUgbW9kZWwgY2xhc3MgdG8gcmVzb2x2ZSBhIHJlcG9zaXRvcnkgZm9yLlxuICogQHBhcmFtIGZsYXZvdXIgb3B0aW9uYWwgYWRhcHRlciBmbGF2b3VyL2FsaWFzLCBmb3J3YXJkZWQgdG8ge0BsaW5rIENvcmVSZXBvc2l0b3J5LmZvck1vZGVsfS5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIFJlcG9zaXRvcnkoXG4gIG1vZGVsOiBDb25zdHJ1Y3Rvcjxhbnk+LFxuICBmbGF2b3VyPzogc3RyaW5nXG4pOiBQYXJhbWV0ZXJEZWNvcmF0b3Ige1xuICByZXR1cm4gKFxuICAgIHRhcmdldDogb2JqZWN0LFxuICAgIHByb3BlcnR5S2V5OiBzdHJpbmcgfCBzeW1ib2wgfCB1bmRlZmluZWQsXG4gICAgcGFyYW1ldGVySW5kZXg6IG51bWJlclxuICApOiB2b2lkID0+IHtcbiAgICBjb25zdCB0b2tlbiA9IG1vZGVsUmVwb3NpdG9yeVRva2VuKG1vZGVsLCBmbGF2b3VyKTtcbiAgICByZWdpc3RlclByb3ZpZGVyKHRva2VuLCAoKSA9PiBDb3JlUmVwb3NpdG9yeS5mb3JNb2RlbChtb2RlbCwgZmxhdm91cikpO1xuICAgIHJldHVybiBJbmplY3QodG9rZW4pKHRhcmdldCwgcHJvcGVydHlLZXksIHBhcmFtZXRlckluZGV4KTtcbiAgfTtcbn1cbiIsImltcG9ydCB7XG4gIER5bmFtaWNNb2R1bGUsXG4gIEdsb2JhbCxcbiAgSW5qZWN0LFxuICBNb2R1bGUsXG4gIE9uQXBwbGljYXRpb25TaHV0ZG93bixcbn0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBBUFBfSU5URVJDRVBUT1IsIE1vZHVsZVJlZiB9IGZyb20gXCJAbmVzdGpzL2NvcmVcIjtcbmltcG9ydCB0eXBlIHsgRGVjYWZNb2R1bGVPcHRpb25zIH0gZnJvbSBcIi4vdHlwZXNcIjtcbmltcG9ydCB7XG4gIERFQ0FGX0FEQVBURVJfSUQsXG4gIERFQ0FGX0hBTkRMRVJTLFxuICBERUNBRl9NT0RVTEVfT1BUSU9OUyxcbn0gZnJvbSBcIi4vY29uc3RhbnRzXCI7XG5pbXBvcnQge1xuICBEZWNhZlJlcXVlc3RIYW5kbGVySW50ZXJjZXB0b3IsXG59IGZyb20gXCIuL2ludGVyY2VwdG9yc1wiO1xuaW1wb3J0IHsgRGVjYWZIYW5kbGVyRXhlY3V0b3IsIERlY2FmUmVxdWVzdENvbnRleHQgfSBmcm9tIFwiLi9yZXF1ZXN0XCI7XG5pbXBvcnQgeyBDb25zdHJ1Y3RvciB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdGlvblwiO1xuaW1wb3J0IHsgQWRhcHRlciwgUGVyc2lzdGVuY2VTZXJ2aWNlLCBTZXJ2aWNlIH0gZnJvbSBcIkBkZWNhZi10cy9jb3JlXCI7XG5pbXBvcnQgeyBnZXRSZWdpc3RlcmVkRGVjYWZQcm92aWRlcnMgfSBmcm9tIFwiLi9kZWNvcmF0b3JzXCI7XG5pbXBvcnQgeyBJbnRlcm5hbEVycm9yIH0gZnJvbSBcIkBkZWNhZi10cy9kYi1kZWNvcmF0b3JzXCI7XG5pbXBvcnQgeyBMb2dnZXIsIExvZ2dpbmcgfSBmcm9tIFwiQGRlY2FmLXRzL2xvZ2dpbmdcIjtcbmltcG9ydCB7XG4gIFJlcXVlc3RUb0NvbnRleHRUcmFuc2Zvcm1lcixcbiAgcmVxdWVzdFRvQ29udGV4dFRyYW5zZm9ybWVyLFxufSBmcm9tIFwiQGRlY2FmLXRzL2Zvci1odHRwL3NlcnZlclwiO1xuXG5AR2xvYmFsKClcbkBNb2R1bGUoe30pXG5leHBvcnQgY2xhc3MgRGVjYWZDb3JlTW9kdWxlPENPTkYsIEFEQVBURVIgZXh0ZW5kcyBBZGFwdGVyPENPTkYsIGFueSwgYW55LCBhbnk+PlxuICBpbXBsZW1lbnRzIE9uQXBwbGljYXRpb25TaHV0ZG93blxue1xuICBwcml2YXRlIHN0YXRpYyBfbG9nZ2VyOiBMb2dnZXI7XG5cbiAgcHJpdmF0ZSBzdGF0aWMgX3BlcnNpc3RlbmNlPzogUGVyc2lzdGVuY2VTZXJ2aWNlPEFkYXB0ZXI8YW55LCBhbnksIGFueSwgYW55Pj47XG5cbiAgcHJvdGVjdGVkIHN0YXRpYyBnZXQgcGVyc2lzdGVuY2UoKTogUGVyc2lzdGVuY2VTZXJ2aWNlPEFkYXB0ZXI8YW55LCBhbnksIGFueSwgYW55Pj4ge1xuICAgIGlmICghdGhpcy5fcGVyc2lzdGVuY2UpXG4gICAgICB0aHJvdyBuZXcgSW50ZXJuYWxFcnJvcihcIlBlcnNpc3RlbmNlIHNlcnZpY2Ugbm90IGluaXRpYWxpemVkXCIpO1xuICAgIHJldHVybiB0aGlzLl9wZXJzaXN0ZW5jZTtcbiAgfVxuXG4gIHByb3RlY3RlZCBzdGF0aWMgZ2V0IGxvZygpOiBMb2dnZXIge1xuICAgIGlmICghdGhpcy5fbG9nZ2VyKSB0aGlzLl9sb2dnZXIgPSBMb2dnaW5nLmZvcihEZWNhZkNvcmVNb2R1bGUpO1xuICAgIHJldHVybiB0aGlzLl9sb2dnZXI7XG4gIH1cblxuICBjb25zdHJ1Y3RvcihcbiAgICBASW5qZWN0KERFQ0FGX01PRFVMRV9PUFRJT05TKVxuICAgIHByaXZhdGUgcmVhZG9ubHkgb3B0aW9uczogRGVjYWZNb2R1bGVPcHRpb25zPENPTkYsIEFEQVBURVI+LFxuICAgIHByaXZhdGUgcmVhZG9ubHkgbW9kdWxlUmVmOiBNb2R1bGVSZWZcbiAgKSB7fVxuXG4gIHN0YXRpYyBmb3JSb290KG9wdGlvbnM6IERlY2FmTW9kdWxlT3B0aW9ucyk6IER5bmFtaWNNb2R1bGUge1xuICAgIGNvbnN0IGxvZyA9IHRoaXMubG9nLmZvcih0aGlzLmZvclJvb3QpO1xuICAgIGNvbnN0IGRlY2FmUHJvdmlkZXJzID0gZ2V0UmVnaXN0ZXJlZERlY2FmUHJvdmlkZXJzKCk7XG4gICAgcmV0dXJuIHtcbiAgICAgIG1vZHVsZTogRGVjYWZDb3JlTW9kdWxlLFxuICAgICAgcHJvdmlkZXJzOiBbXG4gICAgICAgIHsgcHJvdmlkZTogREVDQUZfTU9EVUxFX09QVElPTlMsIHVzZVZhbHVlOiBvcHRpb25zIH0sXG4gICAgICAgIHsgcHJvdmlkZTogREVDQUZfQURBUFRFUl9JRCwgdXNlVmFsdWU6IHRoaXMucGVyc2lzdGVuY2U/LmNsaWVudCB9LFxuICAgICAgICB7XG4gICAgICAgICAgcHJvdmlkZTogREVDQUZfSEFORExFUlMsXG4gICAgICAgICAgdXNlRmFjdG9yeTogKCkgPT5cbiAgICAgICAgICAgIG9wdGlvbnMuaGFuZGxlcnM/Lm1hcCgoSCkgPT4ge1xuICAgICAgICAgICAgICBsb2cuaW5mbyhgUmVnaXN0ZXJlZCByZXF1ZXN0IGhhbmRsZXI6ICR7SC5uYW1lfWApO1xuICAgICAgICAgICAgICByZXR1cm4gbmV3IEgoKTtcbiAgICAgICAgICAgIH0pID8/IFtdLFxuICAgICAgICB9LFxuICAgICAgICBEZWNhZlJlcXVlc3RDb250ZXh0LFxuICAgICAgICBEZWNhZkhhbmRsZXJFeGVjdXRvcixcbiAgICAgICAge1xuICAgICAgICAgIHByb3ZpZGU6IEFQUF9JTlRFUkNFUFRPUixcbiAgICAgICAgICB1c2VDbGFzczogRGVjYWZSZXF1ZXN0SGFuZGxlckludGVyY2VwdG9yLFxuICAgICAgICB9LFxuICAgICAgICAuLi5kZWNhZlByb3ZpZGVycyxcbiAgICAgIF0sXG4gICAgICBleHBvcnRzOiBbXG4gICAgICAgIERFQ0FGX01PRFVMRV9PUFRJT05TLFxuICAgICAgICBERUNBRl9BREFQVEVSX0lELFxuICAgICAgICBERUNBRl9IQU5ETEVSUyxcbiAgICAgICAgRGVjYWZSZXF1ZXN0Q29udGV4dCxcbiAgICAgICAgRGVjYWZIYW5kbGVyRXhlY3V0b3IsXG4gICAgICAgIC4uLmRlY2FmUHJvdmlkZXJzLm1hcCgocHJvdmlkZXIpID0+IHByb3ZpZGVyLnByb3ZpZGUpLFxuICAgICAgXSxcbiAgICB9O1xuICB9XG5cbiAgc3RhdGljIGFzeW5jIGJvb3RQZXJzaXN0ZW5jZShcbiAgICBvcHRpb25zOiBEZWNhZk1vZHVsZU9wdGlvbnNcbiAgKTogUHJvbWlzZTxBZGFwdGVyPGFueSwgYW55LCBhbnksIGFueT5bXT4ge1xuICAgIGNvbnN0IGxvZyA9IHRoaXMubG9nLmZvcih0aGlzLmJvb3RQZXJzaXN0ZW5jZSk7XG5cbiAgICBpZiAoIXRoaXMuX3BlcnNpc3RlbmNlKSB7XG4gICAgICBjb25zdCB0cmltbWVkID0gb3B0aW9ucy5jb25mLm1hcCgoW2NvbnRyLCBjZmcsIC4uLmFyZ3NdKSA9PiB7XG4gICAgICAgIGNvbnN0IHBvc3NpYmxlID0gYXJncy5wb3AoKTtcbiAgICAgICAgaWYgKCFwb3NzaWJsZSkgcmV0dXJuIFtjb250ciwgY2ZnXTtcbiAgICAgICAgcmV0dXJuIFtjb250ciwgY2ZnLCAuLi5hcmdzXTtcbiAgICAgIH0pO1xuICAgICAgdGhpcy5fcGVyc2lzdGVuY2UgPSBuZXcgUGVyc2lzdGVuY2VTZXJ2aWNlKCk7XG4gICAgICBhd2FpdCB0aGlzLl9wZXJzaXN0ZW5jZS5ib290KHRyaW1tZWQpO1xuICAgICAgY29uc3QgY2xpZW50cyA9IHRoaXMuX3BlcnNpc3RlbmNlLmNsaWVudDtcbiAgICAgIGZvciAobGV0IGkgPSAwOyBpIDwgY2xpZW50cy5sZW5ndGg7IGkrKykge1xuICAgICAgICBjb25zdCBjID0gb3B0aW9ucy5jb25mW2ldO1xuICAgICAgICBjb25zdCBwb3NzaWJsZVRyYW5zZiA9IGMuc2xpY2UoMiwgYy5sZW5ndGgpO1xuICAgICAgICBsZXQgdHJhbnNmb3JtZXIgPSBwb3NzaWJsZVRyYW5zZi5wb3AoKTtcbiAgICAgICAgaWYgKFxuICAgICAgICAgICF0cmFuc2Zvcm1lciB8fFxuICAgICAgICAgICEodHJhbnNmb3JtZXIgYXMgUmVxdWVzdFRvQ29udGV4dFRyYW5zZm9ybWVyPGFueT4pLmZyb21cbiAgICAgICAgKSB7XG4gICAgICAgICAgY29uc3QgY29udHIgPSBBZGFwdGVyLnRyYW5zZm9ybWVyRm9yKGNsaWVudHNbaV0uZmxhdm91cik7XG4gICAgICAgICAgaWYgKCFjb250cilcbiAgICAgICAgICAgIHRocm93IG5ldyBJbnRlcm5hbEVycm9yKFxuICAgICAgICAgICAgICBgTm8gdHJhbnNmb3JtZXIgZm91bmQgZm9yIGZsYXZvdXIgJHtjbGllbnRzW2ldLmZsYXZvdXJ9LiB5b3Ugc2hvdWxkIGVpdGhlciBAcmVxdWVzdFRvQ29udGV4dFRyYW5zZm9ybWVyIG9yIHByb3ZpZGUgYSB0cmFuc2Zvcm1lciBpbiB0aGUgY29uZmlnYFxuICAgICAgICAgICAgKTtcbiAgICAgICAgICB0cnkge1xuICAgICAgICAgICAgdHJhbnNmb3JtZXIgPSAoY29udHIgYXMgUmVxdWVzdFRvQ29udGV4dFRyYW5zZm9ybWVyPGFueT4pLmZyb21cbiAgICAgICAgICAgICAgPyBjb250clxuICAgICAgICAgICAgICA6IG5ldyAoY29udHIgYXMgQ29uc3RydWN0b3I8UmVxdWVzdFRvQ29udGV4dFRyYW5zZm9ybWVyPGFueT4+KSgpO1xuICAgICAgICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBJbnRlcm5hbEVycm9yKFxuICAgICAgICAgICAgICBgRmFpbGVkIHRvIGJvb3QgdHJhbnNmb3JtZXIgZm9yICR7Y2xpZW50c1tpXS5mbGF2b3VyfTogJHtlfWBcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHJlcXVlc3RUb0NvbnRleHRUcmFuc2Zvcm1lcihjbGllbnRzW2ldLmZsYXZvdXIpKHRyYW5zZm9ybWVyKTtcbiAgICAgIH1cbiAgICAgIGxvZy5pbmZvKFwicGVyc2lzdGVuY2UgbGF5ZXIgY3JlYXRlZCBzdWNjZXNzZnVsbHkhXCIpO1xuXG4gICAgICBpZiAob3B0aW9ucy5pbml0aWFsaXphdGlvbikge1xuICAgICAgICB0cnkge1xuICAgICAgICAgIGF3YWl0IG9wdGlvbnMuaW5pdGlhbGl6YXRpb24oKTtcbiAgICAgICAgfSBjYXRjaCAoZTogdW5rbm93bikge1xuICAgICAgICAgIHRocm93IG5ldyBJbnRlcm5hbEVycm9yKGBGYWlsZWQgdG8gaW5pdGlhbGl6ZSBhcHBsaWNhdGlvbjogJHtlfWApO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuXG4gICAgcmV0dXJuIHRoaXMucGVyc2lzdGVuY2UuY2xpZW50O1xuICB9XG5cbiAgYXN5bmMgb25BcHBsaWNhdGlvblNodXRkb3duKCk6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IGxvZyA9IERlY2FmQ29yZU1vZHVsZS5sb2cuZm9yKHRoaXMub25BcHBsaWNhdGlvblNodXRkb3duKTtcbiAgICBjb25zdCBhZGFwdGVyczogQWRhcHRlcjxhbnksIGFueSwgYW55LCBhbnk+W10gPVxuICAgICAgdGhpcy5tb2R1bGVSZWYuZ2V0PGFueT4oREVDQUZfQURBUFRFUl9JRCk7XG4gICAgZm9yIChjb25zdCBhZGFwdGVyIG9mIGFkYXB0ZXJzKVxuICAgICAgdHJ5IHtcbiAgICAgICAgaWYgKGFkYXB0ZXIpIHtcbiAgICAgICAgICBsb2cuaW5mbyhgU2h1dHRpbmcgZG93biAke2FkYXB0ZXIudG9TdHJpbmcoKX1gKTtcbiAgICAgICAgICBhd2FpdCBhZGFwdGVyLnNodXRkb3duKCk7XG4gICAgICAgIH1cbiAgICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgICAgbG9nLmVycm9yKGBGYWlsZWQgdG8gc2h1dGRvd24gYXBwbGljYXRpb25gLCBlIGFzIEVycm9yKTtcbiAgICAgIH1cbiAgICB0cnkge1xuICAgICAgYXdhaXQgU2VydmljZS5zaHV0ZG93bigpO1xuICAgIH0gY2F0Y2ggKGU6IHVua25vd24pIHtcbiAgICAgIGxvZy5lcnJvcihgRmFpbGVkIHRvIHNodXRkb3duIHNlcnZpY2VzYCwgZSBhcyBFcnJvcik7XG4gICAgfVxuICB9XG59XG4iLCJleHBvcnQgY29uc3QgTElTVEVOSU5HX0FEQVBURVJTX0ZMQVZPVVJTID0gU3ltYm9sKFxuICBcIkxJU1RFTklOR19BREFQVEVSU19GTEFWT1VSU1wiXG4pO1xuIiwiaW1wb3J0IHsgTW9kZWwgfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5cbmV4cG9ydCBmdW5jdGlvbiBub3JtYWxpemVFdmVudFJlc3BvbnNlKGFyZ3M6IGFueVtdKTogdW5rbm93bltdIHtcbiAgY29uc3QgW21vZGVsQ29uc3RyLCBvcGVyYXRpb24sIGlkLCBwYXlsb2FkXSA9IGFyZ3M7XG5cbiAgY29uc3QgbW9kZWxOYW1lID0gbW9kZWxDb25zdHI/Lm5hbWUgPz8gbW9kZWxDb25zdHI7XG5cbiAgY29uc3Qgc2VyaWFsaXplZFBheWxvYWQgPSBBcnJheS5pc0FycmF5KHBheWxvYWQpXG4gICAgPyBwYXlsb2FkLm1hcCgoZSkgPT4ge1xuICAgICAgICB0cnkge1xuICAgICAgICAgIGlmICh0eXBlb2YgZS5zZXJpYWxpemUgPT09IFwiZnVuY3Rpb25cIikgcmV0dXJuIGUuc2VyaWFsaXplKCk7XG5cbiAgICAgICAgICBjb25zb2xlLndhcm4oXG4gICAgICAgICAgICBgUGF5bG9hZCBpdGVtIGZvciAke21vZGVsTmFtZX0gZG9lcyBub3QgaGF2ZSBzZXJpYWxpemUgbWV0aG9kIGFuZCBpcyBhbiAke3R5cGVvZiBlfSwgYXR0ZW1wdGluZyB0byBzdHJpbmdpZnkgZGlyZWN0bHkuIEl0ZW06ICR7ZX1gXG4gICAgICAgICAgKTtcbiAgICAgICAgICByZXR1cm4gdHlwZW9mIGUgPT09IFwic3RyaW5nXCIgPyBlIDogSlNPTi5zdHJpbmdpZnkoZSk7XG4gICAgICAgIH0gY2F0Y2ggKGVycjogdW5rbm93bikge1xuICAgICAgICAgIGNvbnNvbGUud2FybihgRmFpbGVkIHRvIHNlcmlhbGl6ZSBwYXlsb2FkIGZvciAke21vZGVsTmFtZX06ICR7ZXJyfWApO1xuICAgICAgICAgIHJldHVybiB1bmRlZmluZWQ7XG4gICAgICAgIH1cbiAgICAgIH0pXG4gICAgOiBwYXlsb2FkICYmIHR5cGVvZiBwYXlsb2FkLnNlcmlhbGl6ZSA9PT0gXCJmdW5jdGlvblwiXG4gICAgICA/IHBheWxvYWQuc2VyaWFsaXplKClcbiAgICAgIDogdHlwZW9mIHBheWxvYWQgPT09IFwic3RyaW5nXCJcbiAgICAgICAgPyBwYXlsb2FkXG4gICAgICAgIDogSlNPTi5zdHJpbmdpZnkocGF5bG9hZCk7XG5cbiAgY29uc29sZS5kZWJ1ZyhcbiAgICBgTm9ybWFsaXplZCBldmVudCByZXNwb25zZSBmb3IgbW9kZWwgJHttb2RlbE5hbWV9LCBvcGVyYXRpb24gJHtvcGVyYXRpb259LCBpZCAke2lkfTpgLFxuICAgIHNlcmlhbGl6ZWRQYXlsb2FkXG4gICk7XG5cbiAgcmV0dXJuIFttb2RlbE5hbWUsIG9wZXJhdGlvbiwgaWQsIHNlcmlhbGl6ZWRQYXlsb2FkXTtcbn1cbiIsImltcG9ydCB7IERlY2FmQ29udHJvbGxlciB9IGZyb20gXCIuLi9jb250cm9sbGVyc1wiO1xuaW1wb3J0IHsgRGVjYWZSZXF1ZXN0Q29udGV4dCB9IGZyb20gXCIuLi9yZXF1ZXN0XCI7XG5pbXBvcnQgeyBBZGFwdGVyLCBPYnNlcnZlciB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgQ29udHJvbGxlciwgSW5qZWN0LCBNZXNzYWdlRXZlbnQsIFF1ZXJ5LCBTc2UgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcbmltcG9ydCB7IGludGVydmFsLCBtZXJnZSwgT2JzZXJ2YWJsZSB9IGZyb20gXCJyeGpzXCI7XG5pbXBvcnQgeyBMb2dnaW5nIH0gZnJvbSBcIkBkZWNhZi10cy9sb2dnaW5nXCI7XG5pbXBvcnQgeyBMSVNURU5JTkdfQURBUFRFUlNfRkxBVk9VUlMgfSBmcm9tIFwiLi9jb25zdGFudFwiO1xuaW1wb3J0IHsgRGVjYWZTZXJ2ZXJDdHggfSBmcm9tIFwiLi4vY29uc3RhbnRzXCI7XG5pbXBvcnQgeyBub3JtYWxpemVFdmVudFJlc3BvbnNlIH0gZnJvbSBcIi4vdXRpbHNcIjtcbmltcG9ydCB7IG1hcCwgdGFwIH0gZnJvbSBcInJ4anMvb3BlcmF0b3JzXCI7XG5cbkBDb250cm9sbGVyKClcbmV4cG9ydCBjbGFzcyBFdmVudHNDb250cm9sbGVyIGV4dGVuZHMgRGVjYWZDb250cm9sbGVyPERlY2FmU2VydmVyQ3R4PiB7XG4gIHByaXZhdGUgcmVhZG9ubHkgYWRhcHRlcnM6IEFkYXB0ZXI8YW55LCBhbnksIGFueSwgYW55PltdO1xuXG4gIGNvbnN0cnVjdG9yKFxuICAgIGNsaWVudENvbnRleHQ6IERlY2FmUmVxdWVzdENvbnRleHQsXG4gICAgQEluamVjdChMSVNURU5JTkdfQURBUFRFUlNfRkxBVk9VUlMpIGZsYXZvdXJzOiBzdHJpbmdbXVxuICApIHtcbiAgICBzdXBlcihjbGllbnRDb250ZXh0LCBFdmVudHNDb250cm9sbGVyLm5hbWUpO1xuICAgIHRoaXMuYWRhcHRlcnMgPSBmbGF2b3Vycy5tYXAoKGZsYXZvdXIpID0+IChBZGFwdGVyIGFzIGFueSkuZ2V0KGZsYXZvdXIpKTsgLy8gY2hhbmdlIHRvIEFkYXB0ZXIuY2FjaGUoXCJcIilcbiAgfVxuXG4gIEBTc2UoKVxuICBsaXN0ZW4oKTogT2JzZXJ2YWJsZTxNZXNzYWdlRXZlbnQ+IHtcbiAgICBjb25zdCBsb2dnZXIgPSBMb2dnaW5nLmZvcihFdmVudHNDb250cm9sbGVyLm5hbWUpO1xuXG4gICAgY29uc3QgZXZlbnRzJCA9IG5ldyBPYnNlcnZhYmxlPE1lc3NhZ2VFdmVudD4oKG9ic2VydmVyKSA9PiB7XG4gICAgICBjb25zdCBvYnNlcnZlcklkID1cbiAgICAgICAgYEItJHtNYXRoLnJhbmRvbSgpLnRvU3RyaW5nKDM2KS5zbGljZSgyLCA4KX1gLnRvVXBwZXJDYXNlKCk7XG5cbiAgICAgIGxvZ2dlci5pbmZvKFxuICAgICAgICBgQ3JlYXRpbmcgU1NFIG9ic2VydmVyOiAke29ic2VydmVySWR9IGZvciBjbGllbnQgJHt0aGlzLmNsaWVudENvbnRleHQudXVpZH1gXG4gICAgICApO1xuICAgICAgY29uc3QgY2IgPSBuZXcgKGNsYXNzIGltcGxlbWVudHMgT2JzZXJ2ZXIge1xuICAgICAgICBvYnNlcnZlcklkID0gb2JzZXJ2ZXJJZDtcbiAgICAgICAgcmVmcmVzaCguLi5hcmdzOiBhbnlbXSk6IFByb21pc2U8dm9pZD4ge1xuICAgICAgICAgIGxvZ2dlci5kZWJ1ZyhcbiAgICAgICAgICAgIGBTU0Ugb2JzZXJ2ZXIgJHt0aGlzLm9ic2VydmVySWR9IHJlY2VpdmVkIHJlZnJlc2ggZXZlbnRgXG4gICAgICAgICAgKTtcbiAgICAgICAgICByZXR1cm4gUHJvbWlzZS5yZXNvbHZlKCkudGhlbigoKSA9PiB7XG4gICAgICAgICAgICBjb25zdCBkYXRhID0gbm9ybWFsaXplRXZlbnRSZXNwb25zZShhcmdzKTtcbiAgICAgICAgICAgIG9ic2VydmVyLm5leHQoeyB0eXBlOiBcIm1lc3NhZ2VcIiwgZGF0YSB9KTtcbiAgICAgICAgICAgIGxvZ2dlci5kZWJ1ZyhcbiAgICAgICAgICAgICAgYFNTRSBvYnNlcnZlciAke3RoaXMub2JzZXJ2ZXJJZH0gZXZlbnQgcHVzaGVkIHRvIGNsaWVudGBcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgfSk7XG4gICAgICAgIH1cbiAgICAgIH0pKCk7XG5cbiAgICAgIGxvZ2dlci52ZXJib3NlKFxuICAgICAgICBgUmVnaXN0ZXJpbmcgb2JzZXJ2ZXIgJHtvYnNlcnZlcklkfSBhY3Jvc3MgJHt0aGlzLmFkYXB0ZXJzLmxlbmd0aH0gYWRhcHRlcihzKWBcbiAgICAgICk7XG4gICAgICBmb3IgKGNvbnN0IGFkYXB0ZXIgb2YgdGhpcy5hZGFwdGVycykge1xuICAgICAgICBjb25zdCBhZGFwdGVyTmFtZSA9IGFkYXB0ZXI/LmNvbnN0cnVjdG9yPy5uYW1lID8/IFwiVW5rbm93bkFkYXB0ZXJcIjtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICBsb2dnZXIuZGVidWcoXG4gICAgICAgICAgICBgUmVnaXN0ZXJpbmcgb2JzZXJ2ZXIgJHtvYnNlcnZlcklkfSBpbiBhZGFwdGVyICR7YWRhcHRlck5hbWV9YFxuICAgICAgICAgICk7XG4gICAgICAgICAgYWRhcHRlci5vYnNlcnZlKGNiKTtcbiAgICAgICAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgICAgICAgbG9nZ2VyLmRlYnVnKFxuICAgICAgICAgICAgYEZhaWxlZCB0byByZWdpc3RlciBvYnNlcnZlciAke29ic2VydmVySWR9IGluIGFkYXB0ZXIgJHthZGFwdGVyTmFtZX06ICR7ZT8ubWVzc2FnZSB8fCBlfWBcbiAgICAgICAgICApO1xuICAgICAgICAgIGxvZ2dlci5lcnJvcihlKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICByZXR1cm4gKCkgPT4ge1xuICAgICAgICBsb2dnZXIuZGVidWcoYENsZWFuaW5nIHVwIFNTRSBvYnNlcnZlciAke29ic2VydmVySWR9YCk7XG5cbiAgICAgICAgZm9yIChjb25zdCBhZGFwdGVyIG9mIHRoaXMuYWRhcHRlcnMpIHtcbiAgICAgICAgICBjb25zdCBhZGFwdGVyTmFtZSA9IGFkYXB0ZXI/LmNvbnN0cnVjdG9yPy5uYW1lID8/IFwiVW5rbm93bkFkYXB0ZXJcIjtcbiAgICAgICAgICB0cnkge1xuICAgICAgICAgICAgbG9nZ2VyLmRlYnVnKFxuICAgICAgICAgICAgICBgVW5yZWdpc3RlcmluZyBvYnNlcnZlciAke29ic2VydmVySWR9IGZyb20gYWRhcHRlciAke2FkYXB0ZXJOYW1lfWBcbiAgICAgICAgICAgICk7XG4gICAgICAgICAgICBhZGFwdGVyLnVuT2JzZXJ2ZShjYik7XG4gICAgICAgICAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgICAgICAgICBsb2dnZXIuZGVidWcoXG4gICAgICAgICAgICAgIGBGYWlsZWQgZHVyaW5nIGNsZWFudXAgb2Ygb2JzZXJ2ZXIgJHtvYnNlcnZlcklkfSBpbiBhZGFwdGVyICR7YWRhcHRlck5hbWV9OiAke2U/Lm1lc3NhZ2UgfHwgZX1gXG4gICAgICAgICAgICApO1xuICAgICAgICAgICAgbG9nZ2VyLmVycm9yKGUpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfTtcbiAgICB9KTtcblxuICAgIGNvbnN0IEhFQVJUQkVBVF9JTlRFUlZBTF9NUyA9IDE1MDAwO1xuICAgIGNvbnN0IGhlYXJ0YmVhdCQgPSBpbnRlcnZhbChIRUFSVEJFQVRfSU5URVJWQUxfTVMpLnBpcGUoXG4gICAgICB0YXAoKCkgPT4ge1xuICAgICAgICBsb2dnZXIuZGVidWcoXCJTZW5kaW5nIGhlYXJ0YmVhdFwiKTtcbiAgICAgIH0pLFxuICAgICAgbWFwKFxuICAgICAgICAoKTogTWVzc2FnZUV2ZW50ID0+ICh7XG4gICAgICAgICAgdHlwZTogXCJoZWFydGJlYXRcIixcbiAgICAgICAgICBkYXRhOiB7XG4gICAgICAgICAgICB0czogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgICAgICAgIH0sXG4gICAgICAgIH0pXG4gICAgICApXG4gICAgKTtcblxuICAgIHJldHVybiBtZXJnZShldmVudHMkLCBoZWFydGJlYXQkKTtcbiAgfVxuXG4gIEBTc2UoXCIvOm1vZGVsXCIpXG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAdHlwZXNjcmlwdC1lc2xpbnQvbm8tdW51c2VkLXZhcnNcbiAgbGlzdGVuRm9yTW9kZWwoQFF1ZXJ5KFwibW9kZWxcIikgbW9kZWw6IHN0cmluZyk6IE9ic2VydmFibGU8TWVzc2FnZUV2ZW50PiB7XG4gICAgY29uc3QgbG9nZ2VyID0gTG9nZ2luZy5mb3IoRXZlbnRzQ29udHJvbGxlci5uYW1lKTtcblxuICAgIHJldHVybiBuZXcgT2JzZXJ2YWJsZTxNZXNzYWdlRXZlbnQ+KChvYnNlcnZlcikgPT4ge1xuICAgICAgY29uc3QgY2IgPSBuZXcgKGNsYXNzIGltcGxlbWVudHMgT2JzZXJ2ZXIge1xuICAgICAgICByZWZyZXNoKC4uLmFyZ3M6IGFueVtdKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgICAgICAgcmV0dXJuIFByb21pc2UucmVzb2x2ZSgpLnRoZW4oKCkgPT4ge1xuICAgICAgICAgICAgb2JzZXJ2ZXIubmV4dCh7IGRhdGE6IGFyZ3MgfSBhcyBhbnkpO1xuICAgICAgICAgIH0pO1xuICAgICAgICB9XG4gICAgICB9KSgpO1xuXG4gICAgICB0cnkge1xuICAgICAgICBmb3IgKGNvbnN0IGFkYXB0ZXIgb2YgdGhpcy5hZGFwdGVycykge1xuICAgICAgICAgIGFkYXB0ZXIub2JzZXJ2ZShjYik7XG4gICAgICAgIH1cbiAgICAgIH0gY2F0Y2ggKGU6IGFueSkge1xuICAgICAgICBvYnNlcnZlci5lcnJvcihgRmFpbGVkIHRvIG9ic2VydmUgZXZlbnQ6ICR7ZS5tZXNzYWdlIHx8IGV9YCk7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiAoKSA9PiB7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgZm9yIChjb25zdCBhZGFwdGVyIG9mIHRoaXMuYWRhcHRlcnMpIHtcbiAgICAgICAgICAgIGFkYXB0ZXIudW5PYnNlcnZlKGNiKTtcbiAgICAgICAgICB9XG4gICAgICAgIH0gY2F0Y2ggKGU6IGFueSkge1xuICAgICAgICAgIGxvZ2dlci5lcnJvcihlKTtcbiAgICAgICAgfVxuICAgICAgfTtcbiAgICB9KTtcbiAgfVxufVxuIiwiaW1wb3J0IHsgRHluYW1pY01vZHVsZSwgTW9kdWxlIH0gZnJvbSBcIkBuZXN0anMvY29tbW9uXCI7XG5pbXBvcnQgeyBFdmVudHNDb250cm9sbGVyIH0gZnJvbSBcIi4vRXZlbnRzQ29udHJvbGxlclwiO1xuaW1wb3J0IHsgUm91dGVyTW9kdWxlIH0gZnJvbSBcIkBuZXN0anMvY29yZVwiO1xuaW1wb3J0IHsgTElTVEVOSU5HX0FEQVBURVJTX0ZMQVZPVVJTIH0gZnJvbSBcIi4vY29uc3RhbnRcIjtcbmltcG9ydCB7IERlY2FmUmVxdWVzdENvbnRleHQgfSBmcm9tIFwiLi4vcmVxdWVzdFwiO1xuXG5ATW9kdWxlKHt9KVxuZXhwb3J0IGNsYXNzIERlY2FmU3RyZWFtTW9kdWxlIHtcbiAgc3RhdGljIGZvckZsYXZvdXJzKFxuICAgIGZsYXZvdXJzOiBzdHJpbmdbXSxcbiAgICBwYXRoOiBzdHJpbmcgPSBcImV2ZW50c1wiXG4gICk6IER5bmFtaWNNb2R1bGUge1xuICAgIHJldHVybiB7XG4gICAgICBtb2R1bGU6IERlY2FmU3RyZWFtTW9kdWxlLFxuICAgICAgY29udHJvbGxlcnM6IFtFdmVudHNDb250cm9sbGVyXSxcbiAgICAgIGltcG9ydHM6IFtcbiAgICAgICAgUm91dGVyTW9kdWxlLnJlZ2lzdGVyKFtcbiAgICAgICAgICB7XG4gICAgICAgICAgICBwYXRoOiBwYXRoLnJlcGxhY2UoL15cXC8vLCBcIlwiKSxcbiAgICAgICAgICAgIG1vZHVsZTogRGVjYWZTdHJlYW1Nb2R1bGUsXG4gICAgICAgICAgfSxcbiAgICAgICAgXSksXG4gICAgICBdLFxuICAgICAgcHJvdmlkZXJzOiBbXG4gICAgICAgIERlY2FmUmVxdWVzdENvbnRleHQsXG4gICAgICAgIHtcbiAgICAgICAgICBwcm92aWRlOiBMSVNURU5JTkdfQURBUFRFUlNfRkxBVk9VUlMsXG4gICAgICAgICAgdXNlVmFsdWU6IGZsYXZvdXJzID8/IFtdLFxuICAgICAgICB9LFxuICAgICAgXSxcbiAgICB9O1xuICB9XG59XG4iLCJpbXBvcnQgeyBEeW5hbWljTW9kdWxlLCBGb3J3YXJkUmVmZXJlbmNlLCBNb2R1bGUsIFR5cGUgfSBmcm9tIFwiQG5lc3Rqcy9jb21tb25cIjtcbmltcG9ydCB7IERlY2FmTW9kdWxlT3B0aW9ucyB9IGZyb20gXCIuL3R5cGVzXCI7XG5pbXBvcnQgeyBEZWNhZkNvcmVNb2R1bGUgfSBmcm9tIFwiLi9jb3JlLW1vZHVsZVwiO1xuaW1wb3J0IHsgQWRhcHRlciB9IGZyb20gXCJAZGVjYWYtdHMvY29yZVwiO1xuaW1wb3J0IHsgZ2V0TW9kdWxlRm9yIH0gZnJvbSBcIi4vZGVjYWYtbW9kZWwvaW5kZXhcIjtcbmltcG9ydCB7IERlY2FmU3RyZWFtTW9kdWxlIH0gZnJvbSBcIi4vZXZlbnRzLW1vZHVsZVwiO1xuXG4vKipcbiAqIEBwdWJsaWNBcGlcbiAqL1xuQE1vZHVsZSh7fSlcbmV4cG9ydCBjbGFzcyBEZWNhZk1vZHVsZSB7XG4gIHN0YXRpYyBhc3luYyBmb3JSb290QXN5bmMoXG4gICAgb3B0aW9uczogRGVjYWZNb2R1bGVPcHRpb25zXG4gICk6IFByb21pc2U8RHluYW1pY01vZHVsZT4ge1xuICAgIGNvbnN0IHsgYXV0b0NvbnRyb2xsZXJzLCBhdXRvU2VydmljZXMgfSA9IG9wdGlvbnM7XG5cbiAgICBjb25zdCBhZGFwdGVyczogQWRhcHRlcjxhbnksIGFueSwgYW55LCBhbnk+W10gPVxuICAgICAgYXdhaXQgRGVjYWZDb3JlTW9kdWxlLmJvb3RQZXJzaXN0ZW5jZShvcHRpb25zKTtcbiAgICBjb25zdCBmbGF2b3VycyA9IGFkYXB0ZXJzLm1hcCgoYWRhcHRlcikgPT4gYWRhcHRlci5mbGF2b3VyKTtcblxuICAgIGNvbnN0IGltcG9ydHM6XG4gICAgICB8IChcbiAgICAgICAgICB8IER5bmFtaWNNb2R1bGVcbiAgICAgICAgICB8IFR5cGU8YW55PlxuICAgICAgICAgIHwgUHJvbWlzZTxEeW5hbWljTW9kdWxlPlxuICAgICAgICAgIHwgRm9yd2FyZFJlZmVyZW5jZTxhbnk+XG4gICAgICAgIClbXVxuICAgICAgfCB1bmRlZmluZWQgPSBbRGVjYWZDb3JlTW9kdWxlLmZvclJvb3Qob3B0aW9ucyldO1xuXG4gICAgaWYgKGF1dG9Db250cm9sbGVycykge1xuICAgICAgICBmbGF2b3Vycy5mb3JFYWNoKChmbGF2b3VyKSA9PiB7XG4gICAgICAgICAgaW1wb3J0cy5wdXNoKFxuICAgICAgICAgICAgZ2V0TW9kdWxlRm9yKGZsYXZvdXIpLmZvclJvb3QoZmxhdm91ciwge1xuICAgICAgICAgICAgICBhdXRvU2VydmljZXMsXG4gICAgICAgICAgICAgIGNvbnRyb2xsZXJFeHBvc3VyZTogb3B0aW9ucy5jb250cm9sbGVyRXhwb3N1cmUsXG4gICAgICAgICAgICAgIGNvbnRyb2xsZXJDb25maWc6IG9wdGlvbnMuY29udHJvbGxlckNvbmZpZyxcbiAgICAgICAgICAgIH0pXG4gICAgICAgICAgKTtcbiAgICAgICAgfSk7XG4gICAgICB9XG5cbiAgICBpZiAob3B0aW9ucy5vYnNlcnZlck9wdGlvbnM/LmVuYWJsZU9ic2VydmVyRXZlbnRzKSB7XG4gICAgICBpbXBvcnRzLnB1c2goXG4gICAgICAgIERlY2FmU3RyZWFtTW9kdWxlLmZvckZsYXZvdXJzKFxuICAgICAgICAgIG9wdGlvbnMub2JzZXJ2ZXJPcHRpb25zLm9ic2VydmVyRmxhdm91cnMgfHwgZmxhdm91cnNcbiAgICAgICAgKVxuICAgICAgKTtcbiAgICB9XG5cbiAgICByZXR1cm4ge1xuICAgICAgbW9kdWxlOiBEZWNhZk1vZHVsZSxcbiAgICAgIGltcG9ydHM6IGltcG9ydHMsXG4gICAgfTtcbiAgfVxufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gcnVuTWlncmF0aW9ucygpOiBQcm9taXNlPHZvaWQ+IHt9XG4iLCJpbXBvcnQge01vZGVsfSBmcm9tIFwiQGRlY2FmLXRzL2RlY29yYXRvci12YWxpZGF0aW9uXCI7XG5pbXBvcnQge0ludGVybmFsRXJyb3J9IGZyb20gXCJAZGVjYWYtdHMvZGItZGVjb3JhdG9yc1wiO1xuaW1wb3J0IHtSZXBvc2l0b3J5fSBmcm9tIFwiQGRlY2FmLXRzL2NvcmVcIjtcblxuZXhwb3J0IGZ1bmN0aW9uIHJlcG9Gb3JNb2RlbChtb2RlbDogc3RyaW5nKSB7XG5cdGNvbnN0IG0gPSBNb2RlbC5nZXQobW9kZWwpO1xuXHRpZiAoIW0pXG5cdFx0dGhyb3cgbmV3IEludGVybmFsRXJyb3IoYEZhaWxlZCB0byBmaW5kIHJlcG9zaXRvcnkgZm9yICR7bW9kZWx9YClcblx0Y29uc3QgcmVwbyA9IFJlcG9zaXRvcnkuZm9yTW9kZWwobSk7XG5cdHJldHVybiByZXBvO1xufSIsIi8qKlxuICogQG1vZHVsZSBmb3ItbmVzdFxuICogQGRlc2NyaXB0aW9uIFRoaXMgbW9kdWxlIHNlcnZlcyBhcyB0aGUgbWFpbiBlbnRyeSBwb2ludCBmb3IgdGhlIHRzLXdvcmtzcGFjZSBsaWJyYXJ5LiBJdCBhZ2dyZWdhdGVzIGFuZCBleHBvcnRzXG4gKiBmdW5jdGlvbmFsaXR5IGZyb20gdmFyaW91cyBzdWJtb2R1bGVzIGFuZCB1dGlsaXRpZXMgd2l0aGluIHRoZSBwcm9qZWN0LlxuICpcbiAqIFRoZSBtb2R1bGUgaW5jbHVkZXM6XG4gKiAxLiBVdGlsaXR5IGZ1bmN0aW9ucyBhbmQgdHlwZXMgZnJvbSB0aGUgXCIuL3V0aWxzXCIgZGlyZWN0b3J5OlxuICogICAgLSBUaGVzZSBsaWtlbHkgY29udGFpbiBoZWxwZXIgZnVuY3Rpb25zLCBjb21tb24gdHlwZXMsIGFuZCBzaGFyZWQgZnVuY3Rpb25hbGl0eSB1c2VkIHRocm91Z2hvdXQgdGhlIHByb2plY3QuXG4gKiAgICAtIE1heSBpbmNsdWRlIG9wZXJhdGlvbnMgZm9yIGRhdGEgbWFuaXB1bGF0aW9uLCB0eXBlIGNoZWNraW5nLCBvciBvdGhlciBnZW5lcmFsLXB1cnBvc2UgdXRpbGl0aWVzLlxuICpcbiAqIDIuIEEgbmFtZXNwYWNlIGFuZCByZWxhdGVkIHR5cGVzIGZyb20gdGhlIFwiLi9uYW1lc3BhY2VcIiBkaXJlY3Rvcnk6XG4gKiAgICAtIFRoaXMgY291bGQgY29udGFpbiBkb21haW4tc3BlY2lmaWMgY29kZSBvciBhIGNvbGxlY3Rpb24gb2YgcmVsYXRlZCBmdW5jdGlvbmFsaXR5LlxuICogICAgLSBNaWdodCBpbmNsdWRlIGludGVyZmFjZXMsIHR5cGVzLCBvciBjbGFzc2VzIHRoYXQgcmVwcmVzZW50IGNvcmUgY29uY2VwdHMgaW4gdGhlIGxpYnJhcnkuXG4gKlxuICogMy4gQSBWRVJTSU9OIGNvbnN0YW50OlxuICogICAgLSBSZXByZXNlbnRzIHRoZSBjdXJyZW50IHZlcnNpb24gb2YgdGhlIG1vZHVsZS5cbiAqICAgIC0gVXNlZnVsIGZvciB2ZXJzaW9uIGNoZWNraW5nIGFuZCBjb21wYXRpYmlsaXR5IHB1cnBvc2VzLlxuICpcbiAqIFRoaXMgc3RydWN0dXJlIHByb3ZpZGVzIGEgY2xlYW4gYW5kIG9yZ2FuaXplZCBleHBvcnQgb2YgdGhlIG1vZHVsZSdzIGZ1bmN0aW9uYWxpdHksIGFsbG93aW5nIGNvbnN1bWVyc1xuICogdG8gZWFzaWx5IGltcG9ydCBhbmQgdXNlIHNwZWNpZmljIHBhcnRzIG9mIHRoZSBsaWJyYXJ5IGFzIG5lZWRlZC5cbiAqL1xuXG5pbXBvcnQgeyBNZXRhZGF0YSB9IGZyb20gXCJAZGVjYWYtdHMvZGVjb3JhdGlvblwiO1xuaW1wb3J0IFwiLi9kZWNvcmF0aW9uXCI7XG5cbmV4cG9ydCAqIGZyb20gXCIuL2RlY29yYXRpb25cIjsgLy8gb24gdG9wIG9uIHB1cnBvc2VcbmV4cG9ydCAqIGZyb20gXCIuL2RlY2FmLW1vZGVsXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9hdXRoXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9mYWN0b3J5XCI7XG5leHBvcnQgKiBmcm9tIFwiLi9pbnRlcmNlcHRvcnNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL292ZXJyaWRlc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vcmVxdWVzdFwiO1xuZXhwb3J0ICogZnJvbSBcIi4vY29uc3RhbnRzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9jb250cm9sbGVyc1wiO1xuZXhwb3J0ICogZnJvbSBcIi4vbW9kdWxlXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9jb3JlLW1vZHVsZVwiO1xuZXhwb3J0ICogZnJvbSBcIi4vdHlwZXNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL2RlY29yYXRvcnNcIjtcbmV4cG9ydCAqIGZyb20gXCIuL3V0aWxzXCI7XG5leHBvcnQgKiBmcm9tIFwiLi9ldmVudHMtbW9kdWxlXCI7XG4vLyBleHBvcnQgKiBmcm9tIFwiLi93ZWJob29rc1wiO1xuXG4vKipcbiAqIFJlcHJlc2VudHMgdGhlIGN1cnJlbnQgdmVyc2lvbiBvZiB0aGUgdHMtd29ya3NwYWNlIG1vZHVsZS5cbiAqIFRoZSBhY3R1YWwgdmVyc2lvbiBudW1iZXIgaXMgcmVwbGFjZWQgZHVyaW5nIHRoZSBidWlsZCBwcm9jZXNzLlxuICogQGNvbnN0YW50XG4gKiBAdHlwZSB7c3RyaW5nfVxuICovXG5leHBvcnQgY29uc3QgVkVSU0lPTiA9IFwiIyNWRVJTSU9OIyNcIjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gUmVwcmVzZW50cyB0aGUgY3VycmVudCBjb21taXQgaGFzaCBvZiB0aGUgbW9kdWxlIGJ1aWxkLlxuICogQHN1bW1hcnkgU3RvcmVzIHRoZSBjdXJyZW50IGdpdCBjb21taXQgaGFzaCBmb3IgdGhlIHBhY2thZ2UuIFRoZSBidWlsZCByZXBsYWNlc1xuICogdGhlIHBsYWNlaG9sZGVyIHdpdGggdGhlIGFjdHVhbCBjb21taXQgaGFzaCBhdCBwdWJsaXNoIHRpbWUuXG4gKiBAY29uc3QgQ09NTUlUXG4gKi9cbmV4cG9ydCBjb25zdCBDT01NSVQgPSBcIiMjQ09NTUlUIyNcIjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gUmVwcmVzZW50cyB0aGUgZnVsbCB2ZXJzaW9uIHN0cmluZyBvZiB0aGUgbW9kdWxlLlxuICogQHN1bW1hcnkgU3RvcmVzIHRoZSBzZW12ZXIgdmVyc2lvbiBhbmQgY29tbWl0IGhhc2ggZm9yIHRoZSBwYWNrYWdlLlxuICogVGhlIGJ1aWxkIHJlcGxhY2VzIHRoZSBwbGFjZWhvbGRlciB3aXRoIHRoZSBhY3R1YWwgYDx2ZXJzaW9uPi08Y29tbWl0PmAgdmFsdWUgYXQgcHVibGlzaCB0aW1lLlxuICogQGNvbnN0IEZVTExfVkVSU0lPTlxuICovXG5leHBvcnQgY29uc3QgRlVMTF9WRVJTSU9OID0gXCIjI0ZVTExfVkVSU0lPTiMjXCI7XG5cbmV4cG9ydCBjb25zdCBQQUNLQUdFX05BTUUgPSBcIiMjUEFDS0FHRSMjXCI7XG5cbk1ldGFkYXRhLmFsbG93UmVyZWdpc3RyYXRpb24odHJ1ZSk7XG5NZXRhZGF0YS5yZWdpc3RlckxpYnJhcnkoUEFDS0FHRV9OQU1FLCBWRVJTSU9OKTtcbk1ldGFkYXRhLmFsbG93UmVyZWdpc3RyYXRpb24oZmFsc2UpO1xuIl0sIm5hbWVzIjpbIkRFQ09SQVRPUlNfUFJFRklYIiwiREVDT1JBVE9SUyIsIkFQSV9PUEVSQVRJT04iLCJBUElfUkVTUE9OU0UiLCJBUElfUFJPRFVDRVMiLCJBUElfQ09OU1VNRVMiLCJBUElfVEFHUyIsIkFQSV9DQUxMQkFDS1MiLCJBUElfUEFSQU1FVEVSUyIsIkFQSV9IRUFERVJTIiwiQVBJX01PREVMX1BST1BFUlRJRVMiLCJBUElfTU9ERUxfUFJPUEVSVElFU19BUlJBWSIsIkFQSV9TRUNVUklUWSIsIkFQSV9FWENMVURFX0VORFBPSU5UIiwiQVBJX0VYQ0xVREVfQ09OVFJPTExFUiIsIkFQSV9FWFRSQV9NT0RFTFMiLCJBUElfRVhURU5TSU9OIiwiQVBJX1NDSEVNQSIsIkFQSV9ERUZBVUxUX0dFVFRFUiIsIkFQSV9MSU5LIiwiaXNDb25zdHJ1Y3RvciIsInZhbCIsIk1FVEFEQVRBX0ZBQ1RPUllfTkFNRSIsIk1FVEhPRF9NRVRBREFUQSIsImNyZWF0ZU1ldGhvZERlY29yYXRvciIsIm1ldGFrZXkiLCJtZXRhZGF0YSIsIm92ZXJyaWRlRXhpc3RpbmciLCJ0YXJnZXQiLCJrZXkiLCJkZXNjcmlwdG9yIiwicHJldlZhbHVlIiwiUmVmbGVjdCIsImdldE1ldGFkYXRhIiwidmFsdWUiLCJkZWZpbmVNZXRhZGF0YSIsImNyZWF0ZUNsYXNzRGVjb3JhdG9yIiwiY3JlYXRlUHJvcGVydHlEZWNvcmF0b3IiLCJwcm9wZXJ0eUtleSIsInByb3BlcnRpZXMiLCJpbmNsdWRlcyIsImV4aXN0aW5nTWV0YWRhdGEiLCJuZXdNZXRhZGF0YSIsInBpY2tCeSIsIm5lZ2F0ZSIsImlzVW5kZWZpbmVkIiwibWV0YWRhdGFUb1NhdmUiLCJ0eXBlIiwiY29uc3RydWN0b3IiLCJjcmVhdGVNaXhlZERlY29yYXRvciIsIm1ldGFkYXRhcyIsIkFycmF5IiwiaXNBcnJheSIsInByZXZpb3VzTWV0YWRhdGEiLCJPYmplY3QiLCJhc3NpZ24iLCJjcmVhdGVQYXJhbURlY29yYXRvciIsImluaXRpYWwiLCJwYXJhbU9wdGlvbnMiLCJwYXJhbWV0ZXJzIiwicHJvcGVydHlLZXlzIiwiZ2V0T3duUHJvcGVydHlOYW1lcyIsInByb3RvdHlwZSIsIm1ldGhvZERlc2NyaXB0b3IiLCJnZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3IiLCJpc0FwaU1ldGhvZCIsImhhc01ldGFkYXRhIiwiZ2V0VHlwZUlzQXJyYXlUdXBsZSIsImlucHV0IiwiaXNBcnJheUZsYWciLCJpc0lucHV0QXJyYXkiLCJvYmoiLCJnZXRFbnVtVmFsdWVzIiwiZW51bVR5cGUiLCJudW1lcmljVmFsdWVzIiwidmFsdWVzIiwiZmlsdGVyIiwibWFwIiwidG9TdHJpbmciLCJrZXlzIiwiZ2V0RW51bVR5cGUiLCJoYXNTdHJpbmciLCJpc1N0cmluZyIsImxlbmd0aCIsInJlcXVpcmVkIiwiaXNFbnVtQXJyYXkiLCJvcHRzIiwiZW51bSIsInVuZGVmaW5lZCIsIkFwaVByb3BlcnR5Iiwib3B0aW9ucyIsImNyZWF0ZUFwaVByb3BlcnR5RGVjb3JhdG9yIiwiZW51bVZhbHVlcyIsIml0ZW1zIiwiQXBpT3BlcmF0aW9uRnJvbU1vZGVsIiwiTW9kZWxDb25zdHJ1Y3RvciIsInZlcmIiLCJwYXRoIiwiaHR0cFRvQ3J1ZCIsIkdFVCIsIk9wZXJhdGlvbktleXMiLCJSRUFEIiwiR2V0IiwiUE9TVCIsIkNSRUFURSIsIlBvc3QiLCJQVVQiLCJVUERBVEUiLCJQdXQiLCJQQVRDSCIsIlBhdGNoIiwiREVMRVRFIiwiRGVsZXRlIiwiY3J1ZE9wIiwiSHR0cE1ldGhvZERlY29yYXRvciIsInJlc29sdmVCbG9ja1RhcmdldCIsImNvcmVJc09wZXJhdGlvbkJsb2NrZWQiLCJraW5kIiwiYXBwbHkiLCJBcGlFeGNsdWRlRW5kcG9pbnQiLCJCdWxrQXBpT3BlcmF0aW9uRnJvbU1vZGVsIiwiQnVsa0NydWRPcGVyYXRpb25LZXlzIiwiUkVBRF9BTEwiLCJDUkVBVEVfQUxMIiwiVVBEQVRFX0FMTCIsIkRFTEVURV9BTEwiLCJub3JtYWxpemVkIiwicmVwbGFjZSIsInN0YXRlbWVudFRhcmdldHMiLCJQcmVwYXJlZFN0YXRlbWVudEtleXMiLCJMSVNUX0JZIiwiUEFHRV9CWSIsIkZJTkQiLCJQQUdFIiwiRklORF9PTkVfQlkiLCJGSU5EX0JZIiwiQ09VTlRfT0YiLCJNQVhfT0YiLCJNSU5fT0YiLCJBVkdfT0YiLCJTVU1fT0YiLCJESVNUSU5DVF9PRiIsIkdST1VQX09GIiwic3RhcnRzV2l0aCIsInN0YXRlbWVudFZhbHVlIiwiQXBpUGFyYW1zRnJvbU1vZGVsIiwicHJvcHMiLCJkZWNvcmF0b3JzIiwicCIsIkFwaVBhcmFtRGVjIiwibmFtZSIsImRlc2NyaXB0aW9uIiwiU3RyaW5nIiwiYXBwbHlEZWNvcmF0b3JzIiwiRGVjYWZCb2R5IiwiZGF0YSIsImN0eCIsInJlcXVlc3QiLCJzd2l0Y2hUb0h0dHAiLCJnZXRSZXF1ZXN0IiwiYm9keSIsImNvbnRyb2xsZXIiLCJnZXRDbGFzcyIsIk1vZGVsQ29uc3RyIiwiY2xhc3MiLCJJbnRlcm5hbFNlcnZlckVycm9yRXhjZXB0aW9uIiwiaXRlbSIsIk9yZGVyZWRQYXJhbXMiLCJvcmRlciIsInJlcSIsIm9yaWdpbmFsIiwicGFyYW1zIiwib3JkZXJMaXN0Iiwib3JkZXJlZCIsImsiLCJyYXciLCJ2YWx1ZXNJbk9yZGVyIiwia2V5c0luT3JkZXIiLCJEZWNhZlBhcmFtcyIsIkRlY2FmUXVlcnkiLCJfIiwicXVlcnkiLCJwYXJzZWQiLCJsaW1pdCIsIm4iLCJOdW1iZXIiLCJpc05hTiIsIm9mZnNldCIsImJvb2ttYXJrIiwiREVDQUZfTU9EVUxFX09QVElPTlMiLCJERUNBRl9BREFQVEVSX0lEIiwiREVDQUZfVEFTS19TRVJWSUNFX0lEIiwiREVDQUZfUk9VVEUiLCJERUNBRl9IQU5ETEVSUyIsIlN5bWJvbCIsIkRFQ0FGX0FEQVBURVJfT1BUSU9OUyIsIkRFQ0FGX0VYUE9TRSIsIkRFQ0FGX0NPTlRST0xMRVJfQ09ORklHIiwiREVDQUZfQ09OVEVYVF9LRVkiLCJleHBvc2UiLCJmbGF2b3VycyIsIk1ldGFkYXRhIiwic2V0IiwiY29udHJvbGxlckNvbmZpZyIsImNvbmZpZyIsImNvbnRyb2xsZXJDb25maWdEZWNvcmF0b3IiLCJBVVRIX0hBTkRMRVIiLCJBVVRIX01FVEFfS0VZIiwiSVNfUFVCTElDX0tFWSIsIlJFUVVJUkVEX1JPTEVTX0tFWSIsIkRlY2FmUmVxdWVzdENvbnRleHQiLCJSZXF1ZXN0Q29udGV4dCIsInN1cGVyIiwiaGVhZGVyc09mIiwiaGVhZGVycyIsInRoaXMiLCJ1dWlkIiwiVVVJRCIsImluc3RhbmNlIiwiZ2VuZXJhdGUiLCJwdXQiLCJyZWNvcmQiLCJvdmVycmlkZXMiLCJnZXQiLCJlIiwiYWNjdW11bGF0ZSIsIl9fZGVjb3JhdGUiLCJJbmplY3RhYmxlIiwic2NvcGUiLCJTY29wZSIsIlJFUVVFU1QiLCJfX3BhcmFtIiwiSW5qZWN0IiwiQXV0aEludGVyY2VwdG9yIiwicmVmbGVjdG9yIiwicmVxdWVzdENvbnRleHQiLCJhdXRoSGFuZGxlciIsImludGVyY2VwdCIsIm5leHQiLCJsb2ciLCJMb2dnaW5nIiwiZm9yIiwiaXNQdWJsaWMiLCJnZXRBbGxBbmRPdmVycmlkZSIsImdldEhhbmRsZXIiLCJtb2RlbE5hbWUiLCJyZXF1aXJlZFJvbGVzIiwidmVyYm9zZSIsImRlYnVnIiwiYXV0aG9yaXplIiwiYXBwbHlUcmFuc2Zvcm1lcnMiLCJ1c2VyIiwiZ2V0T3JVbmRlZmluZWQiLCJvcmdhbml6YXRpb24iLCJjdXJyZW50TG9nIiwiY2hpbGRMb2ciLCJsb2dnZXIiLCJoYW5kbGUiLCJBZGFwdGVyIiwiZmxhdm91cnNUb1RyYW5zZm9ybSIsImZsYXZvdXIiLCJ0cmFuc2Zvcm1lciIsInRyYW5zZm9ybWVyRm9yIiwiZnJvbSIsIk9wdGlvbmFsIiwiUmVmbGVjdG9yIiwiRGVjYWZBdXRoSGFuZGxlciIsIkF1dGhIYW5kbGVyIiwicGFyc2VSZXF1ZXN0IiwidXNlclJvbGUiLCJhdXRob3JpemF0aW9uIiwic3BsaXQiLCJleHRyYWN0RnJvbUF1dGgiLCJBdXRob3JpemF0aW9uRXJyb3IiLCJyb2xlcyIsIkRlY2FmUm9sZUF1dGhIYW5kbGVyIiwiQXV0aCIsIm1vZGVsIiwicmVzb3VyY2UiLCJkZWNzIiwiQXBpQmVhcmVyQXV0aCIsIlVzZUludGVyY2VwdG9ycyIsInB1c2giLCJTZXRNZXRhZGF0YSIsIlB1YmxpYyIsIlJlcXVpcmVSb2xlcyIsIkFwaVNlY3VyaXR5IiwiRGVjYWZIYW5kbGVyRXhlY3V0b3IiLCJEZWNhZkhhbmRsZXJFeGVjdXRvcl8xIiwiaGFuZGxlcnMiLCJjb250ZXh0IiwiZXhlYyIsInJlcyIsIm1ldGhvZCIsInVybCIsImhhbmRsZXIiLCJEZWNhZlJlc3BvbnNlSW50ZXJjZXB0b3IiLCJyZXNwb25zZSIsImdldFJlc3BvbnNlIiwicGlwZSIsInRhcCIsInRvUmVzcG9uc2UiLCJ0b0NvbnRleHRGbGFncyIsImFkYXB0ZXIiLCJhbGlhcyIsImJpbmQiLCJyZXF1ZXN0VHJhbnNmb3JtZXJzIiwibWV0YSIsIkNvbnRleHQiLCJwZW5kaW5nIiwiaGVhZGVyIiwiSlNPTiIsInN0cmluZ2lmeSIsIk1vZGVsQnVpbGRlciIsImRlY29yYXRlQ2xhc3MiLCJkZWNvcmF0b3IiLCJfY2xhc3NEZWNvcmF0b3JzIiwiX19oYXNDbGFzc0RlY29yYXRvclN1cHBvcnQiLCJvcmlnaW5hbEJ1aWxkIiwiYnVpbGQiLCJyZXN1bHQiLCJjYWxsIiwiZGVjb3JhdGVkIiwiRGVjYWZSZXF1ZXN0SGFuZGxlckludGVyY2VwdG9yIiwiRGVjYWZSZXF1ZXN0SGFuZGxlckludGVyY2VwdG9yXzEiLCJleGVjdXRvciIsImNvbnRleHR1YWxpemUiLCJmbGFncyIsImlwIiwiZXh0cmFjdElwIiwiRGVmYXVsdEFkYXB0ZXJGbGFncyIsInRpbWVzdGFtcCIsIkRhdGUiLCJvcGVyYXRpb24iLCJwYXJzZUlwSGVhZGVyIiwiY2FuZGlkYXRlIiwic2VnbWVudCIsInRyaW0iLCJCb29sZWFuIiwiRGVjYWZBdXRoTW9kdWxlIiwiRGVjYWZBdXRoTW9kdWxlXzEiLCJmb3JSb290IiwicHJvdmlkZXJzIiwicHJvdmlkZSIsIkFQUF9JTlRFUkNFUFRPUiIsInVzZUNsYXNzIiwiZ2xvYmFsIiwidXNlRXhpc3RpbmciLCJtb2R1bGUiLCJleHBvcnRzIiwiTW9kdWxlIiwiSHR0cFZlcmJUb0RlY29yYXRvciIsIkVycm9yIiwiZXh0cmFjdFBhdGhQYXJhbXMiLCJyb3V0ZVBhdGgiLCJzbGljZSIsImFwaVBhcmFtU3BlYyIsImdldEFwaURlY29yYXRvcnMiLCJtZXRob2ROYW1lIiwiaHR0cFZlcmIiLCJpbmNsdWRlUXVlcnlQYXJhbXMiLCJOZXN0SHR0cFJvdXRlRGVjIiwiYXBpUGF0aFBhcmFtcyIsInN3YWdnZXJRdWVyeVBhcmFtcyIsIkFwaVF1ZXJ5IiwiT3JkZXJEaXJlY3Rpb24iLCJBcGlQYXJhbSIsIkFwaU9wZXJhdGlvbiIsInN1bW1hcnkiLCJBcGlPa1Jlc3BvbnNlIiwiQXBpTm9Db250ZW50UmVzcG9uc2UiLCJRdWVyeSIsImFwcGx5QXBpRGVjb3JhdG9ycyIsInByb3RvIiwiZm9yRWFjaCIsImQiLCJpbmRleCIsInJlc29sdmVQZXJzaXN0ZW5jZU1ldGhvZCIsInBlcnNpc3RlbmNlIiwiYXJncyIsIk1vZGVsU2VydmljZSIsInN0YXRlbWVudCIsImNyZWF0ZVJvdXRlSGFuZGxlciIsImFzeW5jIiwicGF0aFBhcmFtcyIsInF1ZXJ5UGFyYW1zIiwiZGlyZWN0aW9uIiwiZXJyb3IiLCJkZWZpbmVSb3V0ZU1ldGhvZCIsIkNvbnRyb2xsZXJDbGFzcyIsImRlZmluZVByb3BlcnR5Iiwid3JpdGFibGUiLCJEZWNhZkNvbnRyb2xsZXIiLCJIdHRwRGVjYWZDb250cm9sbGVyIiwiY2xpZW50Q29udGV4dCIsIl9uYW1lIiwiRGVjYWZNb2RlbENvbnRyb2xsZXIiLCJfcGVyc2lzdGVuY2UiLCJTZXJ2aWNlIiwiZ2V0U2VydmljZSIsIlJlcG9zaXRvcnkiLCJmb3JNb2RlbCIsImNlcnRzIiwib3ZlcnJpZGUiLCJkdG9DYWNoZSIsIk1hcCIsIkR0b0ZvciIsIm9wIiwic3RhY2siLCJTZXQiLCJUcmFuc2FjdGlvbk9wZXJhdGlvbktleXMiLCJjYWNoZSIsImdldER0b0NhY2hlIiwiY2FjaGVkIiwiaXNVcGRhdGVPcCIsIkR5bmFtaWNEVE8iLCJ0b1Bhc2NhbENhc2UiLCJwa1Byb3AiLCJNb2RlbCIsInBrIiwicGtQcm9wc01ldGFkYXRhIiwicGtQcm9wcyIsInBrRGVzaWduVHlwZSIsInBrVHlwZUlzTnVtZXJpYyIsIkJpZ0ludCIsInBrSXNHZW5lcmF0ZWQiLCJnZW5lcmF0ZWQiLCJnZW5lcmF0ZWRCeVNlcXVlbmNlIiwiaXNQcm9wZXJ0eUdlbmVyYXRlZEFjcm9zc0luaGVyaXRhbmNlIiwiYWxsUHJvcHMiLCJyZWxhdGlvbnMiLCJzY2FsYXJQcm9wcyIsInByb3AiLCJoYXMiLCJ2YWxpZGF0aW9uIiwiZ2V0VmFsaWRhdGlvbkFjcm9zc0luaGVyaXRhbmNlIiwiaXNSZXF1aXJlZCIsIlZhbGlkYXRpb25LZXlzIiwiUkVRVUlSRUQiLCJ0eXBlSGludCIsImdldFR5cGVBY3Jvc3NJbmhlcml0YW5jZSIsImFwaU9wdGlvbnMiLCJkZXNpZ25UeXBlIiwiZW51bWVyYWJsZSIsImNvbmZpZ3VyYWJsZSIsInJlbGF0aW9uIiwicmVsYXRpb25NZXRhIiwiSW50ZXJuYWxFcnJvciIsInJlbGF0aW9uVHlwZSIsInZhbGlkYXRpb25Gb3IiLCJMSVNUIiwiaXNDaXJjdWxhciIsInBrVHlwZU5hbWUiLCJnZXRQa09wZW5BcGlUeXBlIiwiYWRkUmVsYXRpb25Qa1JlZiIsInJlbGF0aW9uRHRvIiwiYWRkIiwiYWRkUmVsYXRpb25VcGRhdGUiLCJhZGRSZWxhdGlvbiIsIkR0b0NsYXNzIiwiZXh0cmFNb2RlbHMiLCJkdG9SZWYiLCJvbmVPZkl0ZW1zIiwiJHJlZiIsIm9uZU9mIiwicGtUeXBlIiwiY3VycmVudCIsIkZ1bmN0aW9uIiwiZ2V0UHJvdG90eXBlT2YiLCJXZWFrTWFwIiwiRnJvbU1vZGVsQ29udHJvbGxlciIsImdldFBlcnNpc3RlbmNlIiwiTW9kZWxDbGF6eiIsImUyIiwiY3JlYXRlUXVlcnlSb3V0ZXNGcm9tUmVwb3NpdG9yeSIsInByZWZpeCIsIlBlcnNpc3RlbmNlS2V5cyIsIlFVRVJZIiwicmVwbyIsInF1ZXJ5TWV0aG9kcyIsInJvdXRlTWV0aG9kcyIsIlF1ZXJ5Q29udHJvbGxlciIsImVudHJpZXMiLCJqb2luIiwiY3JlYXRlQ29tcGxleFF1ZXJ5SGFuZGxlciIsImRlZmluZU1ldGhvZCIsImh0dHBEZWNvcmF0b3IiLCJodHRwTWV0aG9kIiwiZ2V0UXVlcnlEZWNvcmF0b3JzIiwib2JqVmFsdWVzIiwiZmllbGRzIiwiZiIsImNyZWF0ZSIsIm1vZHVsZUNvbmZpZ092ZXJyaWRlcyIsInRhYmxlTmFtZSIsInRvS2ViYWJDYXNlIiwibW9kZWxDbGF6ek5hbWUiLCJmYWN0b3J5UGVyc2lzdGVuY2UiLCJkZWNvcmF0b3JDb25maWciLCJtb2R1bGVPdmVycmlkZSIsIm1lcmdlZENvbmZpZyIsIkZhY3RvcnlDb250cm9sbGVyIiwiTW9kZWxDb250cm9sbGVyRmFjdG9yeSIsImZhY3RvcnlSb3V0ZXMiLCJfX3JvdXRlc19fIiwiZ2V0UEsiLCJhcGlQcm9wZXJ0aWVzIiwicGtQYXRoIiwiZ2V0Um91dGVQYXJhbWV0ZXJzRnJvbU1vZGVsIiwiRHluYW1pY01vZGVsQ29udHJvbGxlciIsIkR5bmFtaWNNb2RlbENvbnRyb2xsZXJfMSIsImluZm8iLCJDb250cm9sbGVyIiwiQXBpVGFncyIsIkFwaUV4dHJhTW9kZWxzIiwic29ydGVkUm91dGVzIiwic29ydCIsImEiLCJiIiwiYVNlZ21lbnRzIiwiYlNlZ21lbnRzIiwiYVBhcmFtQ291bnQiLCJzIiwiYlBhcmFtQ291bnQiLCJhTGl0ZXJhbENvdW50IiwiYkxpdGVyYWxDb3VudCIsInJvdXRlIiwicmVnaXN0cmF0aW9uIiwibWF0Y2hSb3V0ZSIsInBhcmFtRGVjb3JhdG9ycyIsImNvbXBvc2VkIiwiREJLZXlzIiwiQ09NUE9TRUQiLCJjb21wb3NlZEtleXMiLCJ1bmlxdWVLZXlzIiwic2VwYXJhdG9yIiwibWV0aG9kRGVjb3JhdG9ycyIsIm5vcm1hbGl6ZWRQYXRoIiwiY3JlYXRlUmVnaXN0cmF0aW9uIiwiY3JlYXRlQ3JlYXRlSGFuZGxlciIsImNyZWF0ZUNyZWF0ZURlY29yYXRvcnMiLCJSZXNwb25zZSIsInBhc3N0aHJvdWdoIiwiY3JlYXRlQnVsa0NyZWF0ZUhhbmRsZXIiLCJidWxrQ3JlYXRlRGVjb3JhdG9ycyIsImNyZWF0ZUJ1bGtSZWFkSGFuZGxlciIsImJ1bGtSZWFkRGVjb3JhdG9ycyIsImNyZWF0ZUJ1bGtVcGRhdGVIYW5kbGVyIiwiYnVsa1VwZGF0ZURlY29yYXRvcnMiLCJjcmVhdGVCdWxrRGVsZXRlSGFuZGxlciIsImJ1bGtEZWxldGVEZWNvcmF0b3JzIiwiY3JlYXRlUmVhZEhhbmRsZXIiLCJyZWFkRGVjb3JhdG9ycyIsImNyZWF0ZVVwZGF0ZUhhbmRsZXIiLCJ1cGRhdGVEZWNvcmF0b3JzIiwiY3JlYXRlRGVsZXRlSGFuZGxlciIsImRlbGV0ZURlY29yYXRvcnMiLCJmYWxsYmFja1NlZ21lbnRzIiwiaXNBbGxQYXJhbXMiLCJldmVyeSIsImZhbGxiYWNrQXBpUHJvcHMiLCJzdWZmaXgiLCJjcmVhdGVTdGF0ZW1lbnRIYW5kbGVyIiwic3RhdGVtZW50RGVjb3JhdG9ycyIsIlBhcmFtIiwic3RhdGVtZW50Um91dGVzIiwic3RhdGVtZW50S2V5Iiwic3RhdGVtZW50TWV0aG9kTmFtZSIsImNyZWF0ZVN0YXRlbWVudFNob3J0Y3V0SGFuZGxlciIsInN0YXRlbWVudFNob3J0Y3V0RGVjb3JhdG9ycyIsInN0YXRlbWVudFNob3J0Y3V0UGFyYW1zIiwicXVlcnlNZXRob2QiLCJjb21wbGV4UXVlcnlQYXJhbXMiLCJwYXRoU2VnbWVudHMiLCJrbm93blByZWZpeGVzIiwicm91dGVNZXRhZGF0YSIsIm1hdGNoZWRFbnRyeSIsImZpbmQiLCJhY3R1YWxNZXRob2ROYW1lIiwicGFyYW1TZWdtZW50cyIsImNyZWF0ZUN1c3RvbVJvdXRlSGFuZGxlciIsImZpcnN0U2VnbWVudCIsInNlZ21lbnRzIiwic2VnIiwiaSIsInJlc3AiLCJsb2dDdHgiLCJjcmVhdGVkIiwiY3JlYXRlQWxsIiwicmVhZEFsbCIsImlkcyIsIm5vcm1hbGl6ZWRJZHMiLCJyZWFkIiwidXBkYXRlQWxsIiwidXBkYXRlZCIsImRlbGV0ZUFsbCIsInJvdXRlUGFyYW1zIiwiaWQiLCJWYWxpZGF0aW9uRXJyb3IiLCJyZWFkUmVzdWx0IiwidXBkYXRlIiwicGF5bG9hZCIsInBhcnNlIiwicmVtb3ZlIiwiZGVsIiwiZGVsZXRlIiwiZGV0YWlscyIsIlNUQVRFTUVOVCIsInBhcnNlSW50IiwicGF0aERpcmVjdGlvbiIsInJlc29sdmVkRGlyZWN0aW9uIiwic3RhdGVtZW50U2hvcnRjdXQiLCJsaXN0QnkiLCJwYWdlIiwicGFnaW5hdGVCeSIsIkFTQyIsInJlZiIsImZpbmRPbmVCeSIsInRvT3ZlcnJpZGVzIiwiZmluZEJ5IiwiZmllbGQiLCJjb21wbGV4UXVlcnkiLCJzcHJlYWRBcmdzIiwiZXh0cmFjdFF1ZXJ5QXJncyIsImN1c3RvbVJvdXRlIiwibGFzdCIsInF1ZXJ5T2JqIiwicG9wIiwiaGFzRGlyZWN0aW9uIiwiaGFzTGltaXQiLCJoYXNPZmZzZXQiLCJleHRyYXMiLCJBcGlCb2R5IiwiQXBpQ3JlYXRlZFJlc3BvbnNlIiwic2NoZW1hIiwiZ2V0U2NoZW1hUGF0aCIsIkFwaUJhZFJlcXVlc3RSZXNwb25zZSIsIkFwaVVucHJvY2Vzc2FibGVFbnRpdHlSZXNwb25zZSIsIkFwaU5vdEZvdW5kUmVzcG9uc2UiLCJiYXNlIiwiZ2V0TW9kdWxlRm9yIiwiRGVjYWZNb2RlbE1vZHVsZSIsIkRlY2FmTW9kZWxNb2R1bGVfMSIsImNyZWF0ZU1vZGVsU2VydmljZXMiLCJtb2RlbHMiLCJ1c2VGYWN0b3J5IiwiaXNFeHBvc2VkIiwiZXhwb3N1cmUiLCJ0cmFja2VkTW9kZWxzIiwiY29udHJvbGxlckV4cG9zdXJlIiwibW9kZWxTZXJ2aWNlcyIsImF1dG9TZXJ2aWNlcyIsImNvbnRyb2xsZXJzIiwiRGVjb3JhdGlvbiIsIkluamVjdGFibGVzS2V5cyIsIklOSkVDVEFCTEUiLCJleHRlbmQiLCJuZXN0SW5qZWN0YWJsZSIsImNhdGVnb3J5IiwiY2ZnIiwiRGVmYXVsdEluamVjdGFibGVzQ29uZmlnIiwic2luZ2xldG9uIiwiREVGQVVMVCIsImR1cmFibGUiLCJJTkpFQ1QiLCJuZXN0SW5qZWN0IiwiaW5uZXJOZXN0SW5qZWN0IiwiTUFYIiwibWF4RGVjIiwibWF4IiwibWF4aW11bSIsIk1JTiIsIm1pbkRlYyIsIm1pbiIsIm1pbmltdW0iLCJNQVhfTEVOR1RIIiwibWF4TGVuZ3RoRGVjIiwibWF4TGVuZ3RoIiwiTUlOX0xFTkdUSCIsIm1pbkxlbmd0aERlYyIsIm1pbkxlbmd0aCIsIlRZUEUiLCJ0eXBlRGVjIiwiREFURSIsImRhdGVEZWMiLCJmb3JtYXQiLCJFTlVNIiwib3B0aW9uRGVjIiwiUEFUVEVSTiIsInBhdHRlcm5EZWMiLCJwYXQiLCJwYXR0ZXJuIiwic291cmNlIiwiQ09MVU1OIiwiY29sdW1uRGVjIiwiRGVjb3JhdGlvbktleXMiLCJERVNDUklQVElPTiIsImRlc2NyaXB0aW9uRGVjIiwiQVVUSCIsIkNvcnNFcnJvciIsIkZvcmJpZGRlbkVycm9yIiwibXNnIiwiVG9NYW55UmVxdWVzdHNFcnJvciIsIkJhc2VFcnJvciIsIkRlY2FmRXhjZXB0aW9uRmlsdGVyIiwiZXhjZXB0aW9uIiwiaG9zdCIsImlzUHJvZHVjdGlvbiIsIkxvZ2dlZEVudmlyb25tZW50IiwiZW52Iiwic3RhdHVzQ29kZSIsIk5vdEZvdW5kRXhjZXB0aW9uIiwiVW5zdXBwb3J0ZWRFcnJvciIsIk5vdEFjY2VwdGFibGVFeGNlcHRpb24iLCJtZXNzYWdlIiwiZ2V0U3RhdHVzIiwic3RhdHVzIiwiY29kZSIsImpzb24iLCJ0b0lTT1N0cmluZyIsIkNhdGNoIiwiVXNlRGVjYWZGaWx0ZXIiLCJVc2VGaWx0ZXJzIiwiVXNlRGVjYWZIZWFkZXJzIiwiU1dBR0dFUl9VSV9DT05TVEFOVFMiLCJ0aXRsZSIsInZlcnNpb24iLCJmYXZpY29uRmlsZVBhdGgiLCJ0b3BiYXJJY29uRmlsZVBhdGgiLCJhdXRoIiwic2NoZW1lIiwiYmVhcmVyRm9ybWF0IiwiaW4iLCJwZXJzaXN0QXV0aG9yaXphdGlvbiIsInRvcGJhckJnQ29sb3IiLCJTd2FnZ2VyQ3VzdG9tVUkiLCJjdXN0b21DU1MiLCJjc3MiLCJ0b3BiYXJJY29uUGF0aCIsImltZyIsImI2NCIsImdldEN1c3RvbU9wdGlvbnMiLCJmYXZpY29uIiwiZmF2aWNvblBhdGgiLCJjdXN0b21TaXRlVGl0bGUiLCJjdXN0b21Dc3MiLCJzd2FnZ2VyT3B0aW9ucyIsImZpbGUiLCJmaWxlUGF0aCIsImFzc2V0c1BhdGgiLCJyZWFkRmlsZVN5bmMiLCJlbmNvZGluZyIsIlN3YWdnZXJCdWlsZGVyIiwiYXBwIiwiY3JlYXRlRG9jdW1lbnQiLCJvcGVuQXBpSnNvblBhdGgiLCJvcGVuQXBpWWFtbFBhdGgiLCJEb2N1bWVudEJ1aWxkZXIiLCJzZXRUaXRsZSIsInNldERlc2NyaXB0aW9uIiwic2V0VmVyc2lvbiIsImFkZEJlYXJlckF1dGgiLCJTd2FnZ2VyTW9kdWxlIiwicmVnaXN0ZXJPcGVuQXBpUm91dGUiLCJjb250ZW50VHlwZSIsImJvZHlGYWN0b3J5IiwiaHR0cEFkYXB0ZXIiLCJnZXRIdHRwQWRhcHRlciIsIl9yZXEiLCJyZXBseSIsInNldHVwU3dhZ2dlciIsImRvY3VtZW50Iiwic3dhZ2dlclVJIiwic2V0dXAiLCJqc29uRG9jdW1lbnRVcmwiLCJ5YW1sRG9jdW1lbnRVcmwiLCJZQU1MIiwiTmVzdEJvb3RzdHJhcGVyIiwiX2xvZ2dlciIsIkxvZ2dlciIsImluaXRpYWxpemUiLCJlbmFibGVMb2dnZXIiLCJjdXN0b21Mb2dnZXIiLCJ1c2VMb2dnZXIiLCJlbmFibGVDb3JzIiwib3JpZ2lucyIsImFsbG93TWV0aG9kcyIsImFsbG93ZWRPcmlnaW5zIiwibyIsInRvTG93ZXJDYXNlIiwiY29yc09wdGlvbnMiLCJvcmlnaW4iLCJjYWxsYmFjayIsImNyZWRlbnRpYWxzIiwibWV0aG9kcyIsInVzZUhlbG1ldCIsImhlbG1ldCIsInJlcXVpcmUiLCJ1c2UiLCJ3YXJuIiwic3dhZ2dlciIsInVzZUdsb2JhbFBpcGVzIiwicGlwZXMiLCJ1c2VHbG9iYWxGaWx0ZXJzIiwiZmlsdGVycyIsInVzZUdsb2JhbEludGVyY2VwdG9ycyIsImludGVyY2VwdG9ycyIsInN0YXJ0IiwicG9ydCIsInByb2Nlc3MiLCJQT1JUIiwibGlzdGVuIiwidGhlbiIsImdldFVybCIsInBlbmRpbmdQcm92aWRlcnMiLCJnZXRSZWdpc3RlcmVkRGVjYWZQcm92aWRlcnMiLCJyZWdpc3RlclByb3ZpZGVyIiwidG9rZW4iLCJpc01vZGVsQ29uc3RydWN0b3IiLCJtb2RlbFNlcnZpY2VUb2tlbiIsIm1vZGVsUmVwb3NpdG9yeVRva2VuIiwicGFyYW1ldGVySW5kZXgiLCJyZXNvbHZlZCIsInBhcmFtVHlwZXMiLCJhc01vZGVsIiwiaW5qZWN0YWJsZVNlcnZpY2VLZXkiLCJDb3JlU2VydmljZSIsIkNvcmVSZXBvc2l0b3J5IiwiRGVjYWZDb3JlTW9kdWxlIiwiRGVjYWZDb3JlTW9kdWxlXzEiLCJtb2R1bGVSZWYiLCJkZWNhZlByb3ZpZGVycyIsInVzZVZhbHVlIiwiY2xpZW50IiwiSCIsInByb3ZpZGVyIiwiYm9vdFBlcnNpc3RlbmNlIiwidHJpbW1lZCIsImNvbmYiLCJjb250ciIsInBvc3NpYmxlIiwiUGVyc2lzdGVuY2VTZXJ2aWNlIiwiYm9vdCIsImNsaWVudHMiLCJjIiwicG9zc2libGVUcmFuc2YiLCJyZXF1ZXN0VG9Db250ZXh0VHJhbnNmb3JtZXIiLCJpbml0aWFsaXphdGlvbiIsIm9uQXBwbGljYXRpb25TaHV0ZG93biIsImFkYXB0ZXJzIiwic2h1dGRvd24iLCJHbG9iYWwiLCJNb2R1bGVSZWYiLCJMSVNURU5JTkdfQURBUFRFUlNfRkxBVk9VUlMiLCJub3JtYWxpemVFdmVudFJlc3BvbnNlIiwibW9kZWxDb25zdHIiLCJzZXJpYWxpemVkUGF5bG9hZCIsInNlcmlhbGl6ZSIsImNvbnNvbGUiLCJlcnIiLCJFdmVudHNDb250cm9sbGVyIiwiRXZlbnRzQ29udHJvbGxlcl8xIiwiZXZlbnRzJCIsIk9ic2VydmFibGUiLCJvYnNlcnZlciIsIm9ic2VydmVySWQiLCJNYXRoIiwicmFuZG9tIiwidG9VcHBlckNhc2UiLCJjYiIsInJlZnJlc2giLCJQcm9taXNlIiwicmVzb2x2ZSIsImFkYXB0ZXJOYW1lIiwib2JzZXJ2ZSIsInVuT2JzZXJ2ZSIsIkhFQVJUQkVBVF9JTlRFUlZBTF9NUyIsImhlYXJ0YmVhdCQiLCJpbnRlcnZhbCIsInRzIiwibWVyZ2UiLCJsaXN0ZW5Gb3JNb2RlbCIsIlNzZSIsIkRlY2FmU3RyZWFtTW9kdWxlIiwiRGVjYWZTdHJlYW1Nb2R1bGVfMSIsImZvckZsYXZvdXJzIiwiaW1wb3J0cyIsIlJvdXRlck1vZHVsZSIsInJlZ2lzdGVyIiwiRGVjYWZNb2R1bGUiLCJEZWNhZk1vZHVsZV8xIiwiZm9yUm9vdEFzeW5jIiwiYXV0b0NvbnRyb2xsZXJzIiwib2JzZXJ2ZXJPcHRpb25zIiwiZW5hYmxlT2JzZXJ2ZXJFdmVudHMiLCJvYnNlcnZlckZsYXZvdXJzIiwicnVuTWlncmF0aW9ucyIsInJlcG9Gb3JNb2RlbCIsIm0iLCJWRVJTSU9OIiwiQ09NTUlUIiwiRlVMTF9WRVJTSU9OIiwiUEFDS0FHRV9OQU1FIiwiYWxsb3dSZXJlZ2lzdHJhdGlvbiIsInJlZ2lzdGVyTGlicmFyeSJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUFPLE1BQU1BLG9CQUFvQjs7QUFDMUIsTUFBTUMsYUFBYTtJQUN4QkMsZUFBZSxHQUFHRjtJQUNsQkcsY0FBYyxHQUFHSDtJQUNqQkksY0FBYyxHQUFHSjtJQUNqQkssY0FBYyxHQUFHTDtJQUNqQk0sVUFBVSxHQUFHTjtJQUNiTyxlQUFlLEdBQUdQO0lBQ2xCUSxnQkFBZ0IsR0FBR1I7SUFDbkJTLGFBQWEsR0FBR1Q7SUFDaEJVLHNCQUFzQixHQUFHVjtJQUN6QlcsNEJBQTRCLEdBQUdYO0lBQy9CWSxjQUFjLEdBQUdaO0lBQ2pCYSxzQkFBc0IsR0FBR2I7SUFDekJjLHdCQUF3QixHQUFHZDtJQUMzQmUsa0JBQWtCLEdBQUdmO0lBQ3JCZ0IsZUFBZSxHQUFHaEI7SUFDbEJpQixZ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
|