@dyanet/nestjs-config-aws 1.0.1 → 1.2.0
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 +213 -1013
- package/dist/cjs/config-aws/src/config-manager.js +366 -0
- package/dist/cjs/config-aws/src/errors/index.js +77 -0
- package/dist/cjs/config-aws/src/index.js +37 -0
- package/dist/cjs/config-aws/src/interfaces/config-loader.interface.js +3 -0
- package/dist/cjs/config-aws/src/interfaces/config-manager.interface.js +3 -0
- package/dist/cjs/config-aws/src/interfaces/env-file-loader.interface.js +3 -0
- package/dist/cjs/config-aws/src/interfaces/environment-loader.interface.js +3 -0
- package/dist/cjs/config-aws/src/interfaces/s3-loader.interface.js +3 -0
- package/dist/cjs/config-aws/src/interfaces/secrets-manager-loader.interface.js +3 -0
- package/dist/cjs/config-aws/src/interfaces/ssm-parameter-store-loader.interface.js +3 -0
- package/dist/cjs/config-aws/src/loaders/env-file.loader.js +169 -0
- package/dist/cjs/config-aws/src/loaders/environment.loader.js +85 -0
- package/dist/cjs/config-aws/src/loaders/s3.loader.js +145 -0
- package/dist/cjs/config-aws/src/loaders/secrets-manager.loader.js +169 -0
- package/dist/cjs/config-aws/src/loaders/ssm-parameter-store.loader.js +199 -0
- package/dist/cjs/config-aws/src/utils/env-file-parser.util.js +98 -0
- package/dist/cjs/config-aws/src/utils/validation.util.js +116 -0
- package/dist/cjs/nestjs-config-aws/src/config.module.js +175 -0
- package/dist/cjs/nestjs-config-aws/src/index.js +61 -0
- package/dist/cjs/{integration → nestjs-config-aws/src/integration}/index.js +1 -1
- package/dist/cjs/nestjs-config-aws/src/integration/interfaces/configuration-factory.interface.js +3 -0
- package/dist/cjs/nestjs-config-aws/src/integration/interfaces/configuration-source.interface.js +3 -0
- package/dist/cjs/{integration → nestjs-config-aws/src/integration}/interfaces/index.js +1 -1
- package/dist/cjs/nestjs-config-aws/src/integration/interfaces/integration-options.interface.js +3 -0
- package/dist/cjs/nestjs-config-aws/src/integration/interfaces/integration-state.interface.js +3 -0
- package/dist/cjs/nestjs-config-aws/src/integration/interfaces/nestjs-config-compatibility.interface.js +73 -0
- package/dist/cjs/nestjs-config-aws/src/integration/interfaces/nestjs-config-integration.interface.js +3 -0
- package/dist/cjs/nestjs-config-aws/src/integration/interfaces/typed-configuration.interface.js +4 -0
- package/dist/cjs/nestjs-config-aws/src/integration/interfaces/utility-types.interface.js +52 -0
- package/dist/cjs/nestjs-config-aws/src/integration/nestjs-config-integration.module.js +124 -0
- package/dist/cjs/nestjs-config-aws/src/integration/providers/aws-configuration-loader.service.js +592 -0
- package/dist/cjs/nestjs-config-aws/src/integration/providers/configuration-factory.provider.js +385 -0
- package/dist/cjs/{integration → nestjs-config-aws/src/integration}/providers/index.js +1 -1
- package/dist/cjs/nestjs-config-aws/src/integration/services/async-config-helper.service.js +366 -0
- package/dist/cjs/nestjs-config-aws/src/integration/services/error-handler.service.js +267 -0
- package/dist/cjs/nestjs-config-aws/src/integration/services/factory-registration.service.js +517 -0
- package/dist/cjs/{integration → nestjs-config-aws/src/integration}/services/index.js +1 -1
- package/dist/cjs/nestjs-config-aws/src/integration/services/integration-state.service.js +81 -0
- package/dist/cjs/nestjs-config-aws/src/integration/services/namespace-handler.service.js +465 -0
- package/dist/cjs/nestjs-config-aws/src/integration/services/nestjs-config-integration.service.js +318 -0
- package/dist/cjs/nestjs-config-aws/src/integration/services/precedence-handler.service.js +292 -0
- package/dist/cjs/nestjs-config-aws/src/integration/services/validation-integration.service.js +595 -0
- package/dist/cjs/nestjs-config-aws/src/integration/utils/config-integration.util.js +283 -0
- package/dist/cjs/{integration → nestjs-config-aws/src/integration}/utils/index.js +1 -1
- package/dist/cjs/nestjs-config-aws/src/interfaces/config-service.interface.js +11 -0
- package/dist/cjs/nestjs-config-aws/src/interfaces/default-schema.interface.js +63 -0
- package/dist/cjs/nestjs-config-aws/src/interfaces/index.js +30 -0
- package/dist/cjs/nestjs-config-aws/src/interfaces/module-options.interface.js +3 -0
- package/dist/cjs/nestjs-config-aws/src/services/config.service.js +142 -0
- package/dist/esm/config-aws/src/config-manager.js +362 -0
- package/dist/esm/config-aws/src/errors/index.js +69 -0
- package/dist/esm/config-aws/src/index.js +21 -0
- package/dist/esm/config-aws/src/interfaces/config-loader.interface.js +2 -0
- package/dist/esm/config-aws/src/interfaces/config-manager.interface.js +2 -0
- package/dist/esm/config-aws/src/interfaces/env-file-loader.interface.js +2 -0
- package/dist/esm/config-aws/src/interfaces/environment-loader.interface.js +2 -0
- package/dist/esm/config-aws/src/interfaces/s3-loader.interface.js +2 -0
- package/dist/esm/config-aws/src/interfaces/secrets-manager-loader.interface.js +2 -0
- package/dist/esm/config-aws/src/interfaces/ssm-parameter-store-loader.interface.js +2 -0
- package/dist/esm/config-aws/src/loaders/env-file.loader.js +132 -0
- package/dist/esm/config-aws/src/loaders/environment.loader.js +81 -0
- package/dist/esm/config-aws/src/loaders/s3.loader.js +141 -0
- package/dist/esm/config-aws/src/loaders/secrets-manager.loader.js +165 -0
- package/dist/esm/config-aws/src/loaders/ssm-parameter-store.loader.js +195 -0
- package/dist/esm/config-aws/src/utils/env-file-parser.util.js +94 -0
- package/dist/esm/config-aws/src/utils/validation.util.js +112 -0
- package/dist/esm/nestjs-config-aws/src/config.module.js +172 -0
- package/dist/esm/nestjs-config-aws/src/index.js +23 -0
- package/dist/esm/nestjs-config-aws/src/integration/index.js +7 -0
- package/dist/esm/nestjs-config-aws/src/integration/interfaces/configuration-factory.interface.js +2 -0
- package/dist/esm/nestjs-config-aws/src/integration/interfaces/configuration-source.interface.js +2 -0
- package/dist/esm/nestjs-config-aws/src/integration/interfaces/index.js +10 -0
- package/dist/esm/nestjs-config-aws/src/integration/interfaces/integration-options.interface.js +2 -0
- package/dist/esm/nestjs-config-aws/src/integration/interfaces/integration-state.interface.js +2 -0
- package/dist/esm/nestjs-config-aws/src/integration/interfaces/nestjs-config-compatibility.interface.js +64 -0
- package/dist/esm/nestjs-config-aws/src/integration/interfaces/nestjs-config-integration.interface.js +2 -0
- package/dist/esm/nestjs-config-aws/src/integration/interfaces/typed-configuration.interface.js +3 -0
- package/dist/esm/nestjs-config-aws/src/integration/interfaces/utility-types.interface.js +44 -0
- package/dist/esm/nestjs-config-aws/src/integration/nestjs-config-integration.module.js +121 -0
- package/dist/esm/nestjs-config-aws/src/integration/providers/aws-configuration-loader.service.js +589 -0
- package/dist/esm/nestjs-config-aws/src/integration/providers/configuration-factory.provider.js +382 -0
- package/dist/esm/nestjs-config-aws/src/integration/providers/index.js +4 -0
- package/dist/esm/nestjs-config-aws/src/integration/services/async-config-helper.service.js +363 -0
- package/dist/esm/nestjs-config-aws/src/integration/services/error-handler.service.js +264 -0
- package/dist/esm/nestjs-config-aws/src/integration/services/factory-registration.service.js +514 -0
- package/dist/esm/nestjs-config-aws/src/integration/services/index.js +10 -0
- package/dist/esm/nestjs-config-aws/src/integration/services/integration-state.service.js +78 -0
- package/dist/esm/nestjs-config-aws/src/integration/services/namespace-handler.service.js +462 -0
- package/dist/esm/nestjs-config-aws/src/integration/services/nestjs-config-integration.service.js +315 -0
- package/dist/esm/nestjs-config-aws/src/integration/services/precedence-handler.service.js +289 -0
- package/dist/esm/nestjs-config-aws/src/integration/services/validation-integration.service.js +589 -0
- package/dist/esm/nestjs-config-aws/src/integration/utils/config-integration.util.js +240 -0
- package/dist/esm/nestjs-config-aws/src/integration/utils/index.js +3 -0
- package/dist/esm/nestjs-config-aws/src/interfaces/config-service.interface.js +7 -0
- package/dist/esm/nestjs-config-aws/src/interfaces/default-schema.interface.js +59 -0
- package/dist/esm/nestjs-config-aws/src/interfaces/index.js +8 -0
- package/dist/esm/nestjs-config-aws/src/interfaces/module-options.interface.js +2 -0
- package/dist/esm/nestjs-config-aws/src/services/config.service.js +139 -0
- package/dist/types/config-aws/src/config-manager.d.ts +119 -0
- package/dist/types/config-aws/src/config-manager.d.ts.map +1 -0
- package/dist/types/config-aws/src/errors/index.d.ts +43 -0
- package/dist/types/config-aws/src/errors/index.d.ts.map +1 -0
- package/dist/types/config-aws/src/index.d.ts +24 -0
- package/dist/types/config-aws/src/index.d.ts.map +1 -0
- package/dist/types/config-aws/src/interfaces/config-loader.interface.d.ts +33 -0
- package/dist/types/config-aws/src/interfaces/config-loader.interface.d.ts.map +1 -0
- package/dist/types/config-aws/src/interfaces/config-manager.interface.d.ts +86 -0
- package/dist/types/config-aws/src/interfaces/config-manager.interface.d.ts.map +1 -0
- package/dist/types/config-aws/src/interfaces/env-file-loader.interface.d.ts +12 -0
- package/dist/types/config-aws/src/interfaces/env-file-loader.interface.d.ts.map +1 -0
- package/dist/types/config-aws/src/interfaces/environment-loader.interface.d.ts +10 -0
- package/dist/types/config-aws/src/interfaces/environment-loader.interface.d.ts.map +1 -0
- package/dist/types/config-aws/src/interfaces/s3-loader.interface.d.ts +14 -0
- package/dist/types/config-aws/src/interfaces/s3-loader.interface.d.ts.map +1 -0
- package/dist/types/config-aws/src/interfaces/secrets-manager-loader.interface.d.ts +12 -0
- package/dist/types/config-aws/src/interfaces/secrets-manager-loader.interface.d.ts.map +1 -0
- package/dist/types/config-aws/src/interfaces/ssm-parameter-store-loader.interface.d.ts +14 -0
- package/dist/types/config-aws/src/interfaces/ssm-parameter-store-loader.interface.d.ts.map +1 -0
- package/dist/types/config-aws/src/loaders/env-file.loader.d.ts +69 -0
- package/dist/types/config-aws/src/loaders/env-file.loader.d.ts.map +1 -0
- package/dist/types/config-aws/src/loaders/environment.loader.d.ts +46 -0
- package/dist/types/config-aws/src/loaders/environment.loader.d.ts.map +1 -0
- package/dist/types/config-aws/src/loaders/s3.loader.d.ts +62 -0
- package/dist/types/config-aws/src/loaders/s3.loader.d.ts.map +1 -0
- package/dist/types/config-aws/src/loaders/secrets-manager.loader.d.ts +68 -0
- package/dist/types/config-aws/src/loaders/secrets-manager.loader.d.ts.map +1 -0
- package/dist/types/config-aws/src/loaders/ssm-parameter-store.loader.d.ts +78 -0
- package/dist/types/config-aws/src/loaders/ssm-parameter-store.loader.d.ts.map +1 -0
- package/dist/types/config-aws/src/utils/env-file-parser.util.d.ts +45 -0
- package/dist/types/config-aws/src/utils/env-file-parser.util.d.ts.map +1 -0
- package/dist/types/{utils → config-aws/src/utils}/validation.util.d.ts +10 -10
- package/dist/types/config-aws/src/utils/validation.util.d.ts.map +1 -0
- package/dist/types/{config.module.d.ts → nestjs-config-aws/src/config.module.d.ts} +1 -0
- package/dist/types/nestjs-config-aws/src/config.module.d.ts.map +1 -0
- package/dist/types/{index.d.ts → nestjs-config-aws/src/index.d.ts} +2 -4
- package/dist/types/nestjs-config-aws/src/index.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/index.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/interfaces/configuration-factory.interface.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/interfaces/configuration-source.interface.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/interfaces/index.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/interfaces/integration-options.interface.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/interfaces/integration-state.interface.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/interfaces/nestjs-config-compatibility.interface.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/interfaces/nestjs-config-integration.interface.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/interfaces/typed-configuration.interface.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/interfaces/utility-types.interface.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/nestjs-config-integration.module.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/providers/aws-configuration-loader.service.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/providers/configuration-factory.provider.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/providers/index.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/services/async-config-helper.service.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/services/error-handler.service.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/services/factory-registration.service.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/services/index.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/services/integration-state.service.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/services/namespace-handler.service.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/services/nestjs-config-integration.service.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/services/precedence-handler.service.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/services/validation-integration.service.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/utils/config-integration.util.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/integration/utils/index.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/interfaces/config-service.interface.d.ts.map +1 -0
- package/dist/types/{interfaces → nestjs-config-aws/src/interfaces}/default-schema.interface.d.ts +28 -28
- package/dist/types/nestjs-config-aws/src/interfaces/default-schema.interface.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/interfaces/index.d.ts +5 -0
- package/dist/types/nestjs-config-aws/src/interfaces/index.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/interfaces/module-options.interface.d.ts.map +1 -0
- package/dist/types/nestjs-config-aws/src/services/config.service.d.ts +88 -0
- package/dist/types/nestjs-config-aws/src/services/config.service.d.ts.map +1 -0
- package/package.json +20 -27
- package/LICENSE +0 -21
- package/dist/cjs/config.module.js +0 -178
- package/dist/cjs/config.module.js.map +0 -1
- package/dist/cjs/index.js +0 -47
- package/dist/cjs/index.js.map +0 -1
- package/dist/cjs/integration/index.js.map +0 -1
- package/dist/cjs/integration/interfaces/configuration-factory.interface.js +0 -3
- package/dist/cjs/integration/interfaces/configuration-factory.interface.js.map +0 -1
- package/dist/cjs/integration/interfaces/configuration-source.interface.js +0 -3
- package/dist/cjs/integration/interfaces/configuration-source.interface.js.map +0 -1
- package/dist/cjs/integration/interfaces/index.js.map +0 -1
- package/dist/cjs/integration/interfaces/integration-options.interface.js +0 -3
- package/dist/cjs/integration/interfaces/integration-options.interface.js.map +0 -1
- package/dist/cjs/integration/interfaces/integration-state.interface.js +0 -3
- package/dist/cjs/integration/interfaces/integration-state.interface.js.map +0 -1
- package/dist/cjs/integration/interfaces/nestjs-config-compatibility.interface.js +0 -73
- package/dist/cjs/integration/interfaces/nestjs-config-compatibility.interface.js.map +0 -1
- package/dist/cjs/integration/interfaces/nestjs-config-integration.interface.js +0 -3
- package/dist/cjs/integration/interfaces/nestjs-config-integration.interface.js.map +0 -1
- package/dist/cjs/integration/interfaces/typed-configuration.interface.js +0 -4
- package/dist/cjs/integration/interfaces/typed-configuration.interface.js.map +0 -1
- package/dist/cjs/integration/interfaces/utility-types.interface.js +0 -52
- package/dist/cjs/integration/interfaces/utility-types.interface.js.map +0 -1
- package/dist/cjs/integration/nestjs-config-integration.module.js +0 -124
- package/dist/cjs/integration/nestjs-config-integration.module.js.map +0 -1
- package/dist/cjs/integration/providers/aws-configuration-loader.service.js +0 -591
- package/dist/cjs/integration/providers/aws-configuration-loader.service.js.map +0 -1
- package/dist/cjs/integration/providers/configuration-factory.provider.js +0 -383
- package/dist/cjs/integration/providers/configuration-factory.provider.js.map +0 -1
- package/dist/cjs/integration/providers/index.js.map +0 -1
- package/dist/cjs/integration/services/async-config-helper.service.js +0 -356
- package/dist/cjs/integration/services/async-config-helper.service.js.map +0 -1
- package/dist/cjs/integration/services/error-handler.service.js +0 -265
- package/dist/cjs/integration/services/error-handler.service.js.map +0 -1
- package/dist/cjs/integration/services/factory-registration.service.js +0 -512
- package/dist/cjs/integration/services/factory-registration.service.js.map +0 -1
- package/dist/cjs/integration/services/index.js.map +0 -1
- package/dist/cjs/integration/services/integration-state.service.js +0 -83
- package/dist/cjs/integration/services/integration-state.service.js.map +0 -1
- package/dist/cjs/integration/services/namespace-handler.service.js +0 -467
- package/dist/cjs/integration/services/namespace-handler.service.js.map +0 -1
- package/dist/cjs/integration/services/nestjs-config-integration.service.js +0 -316
- package/dist/cjs/integration/services/nestjs-config-integration.service.js.map +0 -1
- package/dist/cjs/integration/services/precedence-handler.service.js +0 -294
- package/dist/cjs/integration/services/precedence-handler.service.js.map +0 -1
- package/dist/cjs/integration/services/validation-integration.service.js +0 -591
- package/dist/cjs/integration/services/validation-integration.service.js.map +0 -1
- package/dist/cjs/integration/utils/config-integration.util.js +0 -283
- package/dist/cjs/integration/utils/config-integration.util.js.map +0 -1
- package/dist/cjs/integration/utils/index.js.map +0 -1
- package/dist/cjs/interfaces/config-loader.interface.js +0 -3
- package/dist/cjs/interfaces/config-loader.interface.js.map +0 -1
- package/dist/cjs/interfaces/config-service.interface.js +0 -11
- package/dist/cjs/interfaces/config-service.interface.js.map +0 -1
- package/dist/cjs/interfaces/default-schema.interface.js +0 -63
- package/dist/cjs/interfaces/default-schema.interface.js.map +0 -1
- package/dist/cjs/interfaces/errors.interface.js +0 -77
- package/dist/cjs/interfaces/errors.interface.js.map +0 -1
- package/dist/cjs/interfaces/index.js +0 -25
- package/dist/cjs/interfaces/index.js.map +0 -1
- package/dist/cjs/interfaces/module-options.interface.js +0 -3
- package/dist/cjs/interfaces/module-options.interface.js.map +0 -1
- package/dist/cjs/loaders/environment.loader.js +0 -59
- package/dist/cjs/loaders/environment.loader.js.map +0 -1
- package/dist/cjs/loaders/secrets-manager.loader.js +0 -122
- package/dist/cjs/loaders/secrets-manager.loader.js.map +0 -1
- package/dist/cjs/loaders/ssm-parameter-store.loader.js +0 -146
- package/dist/cjs/loaders/ssm-parameter-store.loader.js.map +0 -1
- package/dist/cjs/services/config.service.js +0 -297
- package/dist/cjs/services/config.service.js.map +0 -1
- package/dist/cjs/utils/validation.util.js +0 -114
- package/dist/cjs/utils/validation.util.js.map +0 -1
- package/dist/esm/config.module.js +0 -175
- package/dist/esm/config.module.js.map +0 -1
- package/dist/esm/index.js +0 -18
- package/dist/esm/index.js.map +0 -1
- package/dist/esm/integration/index.js +0 -7
- package/dist/esm/integration/index.js.map +0 -1
- package/dist/esm/integration/interfaces/configuration-factory.interface.js +0 -2
- package/dist/esm/integration/interfaces/configuration-factory.interface.js.map +0 -1
- package/dist/esm/integration/interfaces/configuration-source.interface.js +0 -2
- package/dist/esm/integration/interfaces/configuration-source.interface.js.map +0 -1
- package/dist/esm/integration/interfaces/index.js +0 -10
- package/dist/esm/integration/interfaces/index.js.map +0 -1
- package/dist/esm/integration/interfaces/integration-options.interface.js +0 -2
- package/dist/esm/integration/interfaces/integration-options.interface.js.map +0 -1
- package/dist/esm/integration/interfaces/integration-state.interface.js +0 -2
- package/dist/esm/integration/interfaces/integration-state.interface.js.map +0 -1
- package/dist/esm/integration/interfaces/nestjs-config-compatibility.interface.js +0 -64
- package/dist/esm/integration/interfaces/nestjs-config-compatibility.interface.js.map +0 -1
- package/dist/esm/integration/interfaces/nestjs-config-integration.interface.js +0 -2
- package/dist/esm/integration/interfaces/nestjs-config-integration.interface.js.map +0 -1
- package/dist/esm/integration/interfaces/typed-configuration.interface.js +0 -3
- package/dist/esm/integration/interfaces/typed-configuration.interface.js.map +0 -1
- package/dist/esm/integration/interfaces/utility-types.interface.js +0 -44
- package/dist/esm/integration/interfaces/utility-types.interface.js.map +0 -1
- package/dist/esm/integration/nestjs-config-integration.module.js +0 -121
- package/dist/esm/integration/nestjs-config-integration.module.js.map +0 -1
- package/dist/esm/integration/providers/aws-configuration-loader.service.js +0 -588
- package/dist/esm/integration/providers/aws-configuration-loader.service.js.map +0 -1
- package/dist/esm/integration/providers/configuration-factory.provider.js +0 -380
- package/dist/esm/integration/providers/configuration-factory.provider.js.map +0 -1
- package/dist/esm/integration/providers/index.js +0 -4
- package/dist/esm/integration/providers/index.js.map +0 -1
- package/dist/esm/integration/services/async-config-helper.service.js +0 -353
- package/dist/esm/integration/services/async-config-helper.service.js.map +0 -1
- package/dist/esm/integration/services/error-handler.service.js +0 -262
- package/dist/esm/integration/services/error-handler.service.js.map +0 -1
- package/dist/esm/integration/services/factory-registration.service.js +0 -509
- package/dist/esm/integration/services/factory-registration.service.js.map +0 -1
- package/dist/esm/integration/services/index.js +0 -10
- package/dist/esm/integration/services/index.js.map +0 -1
- package/dist/esm/integration/services/integration-state.service.js +0 -80
- package/dist/esm/integration/services/integration-state.service.js.map +0 -1
- package/dist/esm/integration/services/namespace-handler.service.js +0 -464
- package/dist/esm/integration/services/namespace-handler.service.js.map +0 -1
- package/dist/esm/integration/services/nestjs-config-integration.service.js +0 -313
- package/dist/esm/integration/services/nestjs-config-integration.service.js.map +0 -1
- package/dist/esm/integration/services/precedence-handler.service.js +0 -291
- package/dist/esm/integration/services/precedence-handler.service.js.map +0 -1
- package/dist/esm/integration/services/validation-integration.service.js +0 -585
- package/dist/esm/integration/services/validation-integration.service.js.map +0 -1
- package/dist/esm/integration/utils/config-integration.util.js +0 -240
- package/dist/esm/integration/utils/config-integration.util.js.map +0 -1
- package/dist/esm/integration/utils/index.js +0 -3
- package/dist/esm/integration/utils/index.js.map +0 -1
- package/dist/esm/interfaces/config-loader.interface.js +0 -2
- package/dist/esm/interfaces/config-loader.interface.js.map +0 -1
- package/dist/esm/interfaces/config-service.interface.js +0 -7
- package/dist/esm/interfaces/config-service.interface.js.map +0 -1
- package/dist/esm/interfaces/default-schema.interface.js +0 -59
- package/dist/esm/interfaces/default-schema.interface.js.map +0 -1
- package/dist/esm/interfaces/errors.interface.js +0 -69
- package/dist/esm/interfaces/errors.interface.js.map +0 -1
- package/dist/esm/interfaces/index.js +0 -9
- package/dist/esm/interfaces/index.js.map +0 -1
- package/dist/esm/interfaces/module-options.interface.js +0 -2
- package/dist/esm/interfaces/module-options.interface.js.map +0 -1
- package/dist/esm/loaders/environment.loader.js +0 -55
- package/dist/esm/loaders/environment.loader.js.map +0 -1
- package/dist/esm/loaders/secrets-manager.loader.js +0 -118
- package/dist/esm/loaders/secrets-manager.loader.js.map +0 -1
- package/dist/esm/loaders/ssm-parameter-store.loader.js +0 -142
- package/dist/esm/loaders/ssm-parameter-store.loader.js.map +0 -1
- package/dist/esm/services/config.service.js +0 -261
- package/dist/esm/services/config.service.js.map +0 -1
- package/dist/esm/utils/validation.util.js +0 -110
- package/dist/esm/utils/validation.util.js.map +0 -1
- package/dist/types/config.module.d.ts.map +0 -1
- package/dist/types/index.d.ts.map +0 -1
- package/dist/types/integration/index.d.ts.map +0 -1
- package/dist/types/integration/interfaces/configuration-factory.interface.d.ts.map +0 -1
- package/dist/types/integration/interfaces/configuration-source.interface.d.ts.map +0 -1
- package/dist/types/integration/interfaces/index.d.ts.map +0 -1
- package/dist/types/integration/interfaces/integration-options.interface.d.ts.map +0 -1
- package/dist/types/integration/interfaces/integration-state.interface.d.ts.map +0 -1
- package/dist/types/integration/interfaces/nestjs-config-compatibility.interface.d.ts.map +0 -1
- package/dist/types/integration/interfaces/nestjs-config-integration.interface.d.ts.map +0 -1
- package/dist/types/integration/interfaces/typed-configuration.interface.d.ts.map +0 -1
- package/dist/types/integration/interfaces/utility-types.interface.d.ts.map +0 -1
- package/dist/types/integration/nestjs-config-integration.module.d.ts.map +0 -1
- package/dist/types/integration/providers/aws-configuration-loader.service.d.ts.map +0 -1
- package/dist/types/integration/providers/configuration-factory.provider.d.ts.map +0 -1
- package/dist/types/integration/providers/index.d.ts.map +0 -1
- package/dist/types/integration/services/async-config-helper.service.d.ts.map +0 -1
- package/dist/types/integration/services/error-handler.service.d.ts.map +0 -1
- package/dist/types/integration/services/factory-registration.service.d.ts.map +0 -1
- package/dist/types/integration/services/index.d.ts.map +0 -1
- package/dist/types/integration/services/integration-state.service.d.ts.map +0 -1
- package/dist/types/integration/services/namespace-handler.service.d.ts.map +0 -1
- package/dist/types/integration/services/nestjs-config-integration.service.d.ts.map +0 -1
- package/dist/types/integration/services/precedence-handler.service.d.ts.map +0 -1
- package/dist/types/integration/services/validation-integration.service.d.ts.map +0 -1
- package/dist/types/integration/utils/config-integration.util.d.ts.map +0 -1
- package/dist/types/integration/utils/index.d.ts.map +0 -1
- package/dist/types/interfaces/config-loader.interface.d.ts +0 -22
- package/dist/types/interfaces/config-loader.interface.d.ts.map +0 -1
- package/dist/types/interfaces/config-service.interface.d.ts.map +0 -1
- package/dist/types/interfaces/default-schema.interface.d.ts.map +0 -1
- package/dist/types/interfaces/errors.interface.d.ts +0 -38
- package/dist/types/interfaces/errors.interface.d.ts.map +0 -1
- package/dist/types/interfaces/index.d.ts +0 -6
- package/dist/types/interfaces/index.d.ts.map +0 -1
- package/dist/types/interfaces/module-options.interface.d.ts.map +0 -1
- package/dist/types/loaders/environment.loader.d.ts +0 -26
- package/dist/types/loaders/environment.loader.d.ts.map +0 -1
- package/dist/types/loaders/secrets-manager.loader.d.ts +0 -52
- package/dist/types/loaders/secrets-manager.loader.d.ts.map +0 -1
- package/dist/types/loaders/ssm-parameter-store.loader.d.ts +0 -68
- package/dist/types/loaders/ssm-parameter-store.loader.d.ts.map +0 -1
- package/dist/types/services/config.service.d.ts +0 -94
- package/dist/types/services/config.service.d.ts.map +0 -1
- package/dist/types/utils/validation.util.d.ts.map +0 -1
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/index.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/interfaces/configuration-factory.interface.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/interfaces/configuration-source.interface.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/interfaces/index.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/interfaces/integration-options.interface.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/interfaces/integration-state.interface.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/interfaces/nestjs-config-compatibility.interface.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/interfaces/nestjs-config-integration.interface.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/interfaces/typed-configuration.interface.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/interfaces/utility-types.interface.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/nestjs-config-integration.module.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/providers/aws-configuration-loader.service.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/providers/configuration-factory.provider.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/providers/index.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/services/async-config-helper.service.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/services/error-handler.service.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/services/factory-registration.service.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/services/index.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/services/integration-state.service.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/services/namespace-handler.service.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/services/nestjs-config-integration.service.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/services/precedence-handler.service.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/services/validation-integration.service.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/utils/config-integration.util.d.ts +0 -0
- /package/dist/types/{integration → nestjs-config-aws/src/integration}/utils/index.d.ts +0 -0
- /package/dist/types/{interfaces → nestjs-config-aws/src/interfaces}/config-service.interface.d.ts +0 -0
- /package/dist/types/{interfaces → nestjs-config-aws/src/interfaces}/module-options.interface.d.ts +0 -0
|
@@ -0,0 +1,165 @@
|
|
|
1
|
+
import { GetSecretValueCommand, SecretsManagerClient } from '@aws-sdk/client-secrets-manager';
|
|
2
|
+
import { fromNodeProviderChain } from '@aws-sdk/credential-providers';
|
|
3
|
+
import { AWSServiceError, ConfigurationLoadError } from '../errors';
|
|
4
|
+
/**
|
|
5
|
+
* Loader that reads configuration from AWS Secrets Manager.
|
|
6
|
+
* Supports environment-aware path construction.
|
|
7
|
+
*
|
|
8
|
+
* @example
|
|
9
|
+
* ```typescript
|
|
10
|
+
* // Basic usage
|
|
11
|
+
* const loader = new SecretsManagerLoader({
|
|
12
|
+
* secretName: '/my-app/config',
|
|
13
|
+
* region: 'us-east-1'
|
|
14
|
+
* });
|
|
15
|
+
*
|
|
16
|
+
* // With environment mapping
|
|
17
|
+
* const loader = new SecretsManagerLoader({
|
|
18
|
+
* secretName: '/my-app/config',
|
|
19
|
+
* environmentMapping: {
|
|
20
|
+
* development: 'dev',
|
|
21
|
+
* staging: 'stg',
|
|
22
|
+
* production: 'prod'
|
|
23
|
+
* }
|
|
24
|
+
* });
|
|
25
|
+
* ```
|
|
26
|
+
*/
|
|
27
|
+
export class SecretsManagerLoader {
|
|
28
|
+
/** @internal */
|
|
29
|
+
_client;
|
|
30
|
+
/** @internal */
|
|
31
|
+
_config;
|
|
32
|
+
/** @internal */
|
|
33
|
+
_appEnv;
|
|
34
|
+
constructor(config = {}) {
|
|
35
|
+
this._appEnv = process.env['APP_ENV'] || process.env['NODE_ENV'] || 'local';
|
|
36
|
+
// Set default configuration
|
|
37
|
+
this._config = {
|
|
38
|
+
secretName: config.secretName || '/nestjs-config-aws',
|
|
39
|
+
region: config.region || process.env['AWS_REGION'] || 'us-east-1',
|
|
40
|
+
environmentMapping: config.environmentMapping || {
|
|
41
|
+
development: 'dev',
|
|
42
|
+
test: 'test',
|
|
43
|
+
production: 'production',
|
|
44
|
+
},
|
|
45
|
+
};
|
|
46
|
+
// Initialize AWS Secrets Manager client
|
|
47
|
+
this._client = new SecretsManagerClient({
|
|
48
|
+
credentials: fromNodeProviderChain(),
|
|
49
|
+
region: this._config.region,
|
|
50
|
+
});
|
|
51
|
+
}
|
|
52
|
+
/**
|
|
53
|
+
* Get the name of this loader for logging and debugging.
|
|
54
|
+
* @returns The loader name with secret path or base secret name if path cannot be built
|
|
55
|
+
*/
|
|
56
|
+
getName() {
|
|
57
|
+
// Avoid calling buildSecretName() to prevent stack overflow when
|
|
58
|
+
// environment mapping is missing - buildSecretName() throws an error
|
|
59
|
+
// that includes getName() in the message, causing infinite recursion.
|
|
60
|
+
const envPrefix = this._config.environmentMapping[this._appEnv];
|
|
61
|
+
if (envPrefix) {
|
|
62
|
+
return `SecretsManagerLoader(/${envPrefix}${this._config.secretName})`;
|
|
63
|
+
}
|
|
64
|
+
// Fallback to base secret name when environment mapping is unavailable
|
|
65
|
+
return `SecretsManagerLoader(${this._config.secretName})`;
|
|
66
|
+
}
|
|
67
|
+
/**
|
|
68
|
+
* Check if this loader is available in the current environment.
|
|
69
|
+
* @returns Promise resolving to true if not in local environment and AWS credentials are available
|
|
70
|
+
*/
|
|
71
|
+
async isAvailable() {
|
|
72
|
+
// Skip in local environment
|
|
73
|
+
if (this._appEnv === 'local') {
|
|
74
|
+
return false;
|
|
75
|
+
}
|
|
76
|
+
try {
|
|
77
|
+
// Test AWS credentials by attempting to get caller identity
|
|
78
|
+
await this._client.config.credentials();
|
|
79
|
+
return true;
|
|
80
|
+
}
|
|
81
|
+
catch {
|
|
82
|
+
return false;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Load configuration from AWS Secrets Manager.
|
|
87
|
+
* @returns Promise resolving to configuration key-value pairs from the secret
|
|
88
|
+
* @throws AWSServiceError if AWS operation fails
|
|
89
|
+
* @throws ConfigurationLoadError if secret cannot be parsed
|
|
90
|
+
*/
|
|
91
|
+
async load() {
|
|
92
|
+
// Skip loading in local environment
|
|
93
|
+
if (this._appEnv === 'local') {
|
|
94
|
+
return {};
|
|
95
|
+
}
|
|
96
|
+
const secretName = this.buildSecretName();
|
|
97
|
+
try {
|
|
98
|
+
const command = new GetSecretValueCommand({ SecretId: secretName });
|
|
99
|
+
const response = await this._client.send(command);
|
|
100
|
+
if (!response.SecretString) {
|
|
101
|
+
return {};
|
|
102
|
+
}
|
|
103
|
+
// Try to parse as JSON, fallback to string value
|
|
104
|
+
try {
|
|
105
|
+
const parsed = JSON.parse(response.SecretString);
|
|
106
|
+
// Ensure we return an object for configuration merging
|
|
107
|
+
if (typeof parsed === 'object' && parsed !== null && !Array.isArray(parsed)) {
|
|
108
|
+
return parsed;
|
|
109
|
+
}
|
|
110
|
+
else {
|
|
111
|
+
// If it's not an object, wrap it in a configuration object
|
|
112
|
+
return { SECRET_VALUE: parsed };
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
catch {
|
|
116
|
+
// If JSON parsing fails, treat as a single string value
|
|
117
|
+
return { SECRET_VALUE: response.SecretString };
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
catch (error) {
|
|
121
|
+
// Handle specific AWS errors
|
|
122
|
+
if (error instanceof Error) {
|
|
123
|
+
if (error.name === 'ResourceNotFoundException') {
|
|
124
|
+
// Secret doesn't exist - this is not necessarily an error in all environments
|
|
125
|
+
return {};
|
|
126
|
+
}
|
|
127
|
+
if (error.name === 'AccessDeniedException') {
|
|
128
|
+
throw new AWSServiceError(`Access denied when retrieving secret '${secretName}'. Check AWS credentials and permissions.`, 'SecretsManager', 'GetSecretValue', error);
|
|
129
|
+
}
|
|
130
|
+
if (error.name === 'InvalidRequestException') {
|
|
131
|
+
throw new AWSServiceError(`Invalid request when retrieving secret '${secretName}'. Check secret name format.`, 'SecretsManager', 'GetSecretValue', error);
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
// For other errors, wrap in AWSServiceError
|
|
135
|
+
throw new AWSServiceError(`Failed to retrieve secret '${secretName}' from AWS Secrets Manager: ${error instanceof Error ? error.message : String(error)}`, 'SecretsManager', 'GetSecretValue', error instanceof Error ? error : undefined);
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
/**
|
|
139
|
+
* Build the environment-aware secret name/path.
|
|
140
|
+
* @returns The full secret name with environment prefix
|
|
141
|
+
*/
|
|
142
|
+
buildSecretName() {
|
|
143
|
+
const envPrefix = this._config.environmentMapping[this._appEnv];
|
|
144
|
+
if (!envPrefix) {
|
|
145
|
+
throw new ConfigurationLoadError(`No environment mapping found for APP_ENV '${this._appEnv}'. ` +
|
|
146
|
+
`Available environments: ${Object.keys(this._config.environmentMapping).join(', ')}`, this.getName());
|
|
147
|
+
}
|
|
148
|
+
return `/${envPrefix}${this._config.secretName}`;
|
|
149
|
+
}
|
|
150
|
+
/**
|
|
151
|
+
* Get the current app environment.
|
|
152
|
+
* @returns The current APP_ENV or NODE_ENV value
|
|
153
|
+
*/
|
|
154
|
+
getAppEnv() {
|
|
155
|
+
return this._appEnv;
|
|
156
|
+
}
|
|
157
|
+
/**
|
|
158
|
+
* Get the environment mapping configuration.
|
|
159
|
+
* @returns The environment mapping record
|
|
160
|
+
*/
|
|
161
|
+
getEnvironmentMapping() {
|
|
162
|
+
return { ...this._config.environmentMapping };
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VjcmV0cy1tYW5hZ2VyLmxvYWRlci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL2NvbmZpZy1hd3Mvc3JjL2xvYWRlcnMvc2VjcmV0cy1tYW5hZ2VyLmxvYWRlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUscUJBQXFCLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUM5RixPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQztBQUl0RSxPQUFPLEVBQUUsZUFBZSxFQUFFLHNCQUFzQixFQUFFLE1BQU0sV0FBVyxDQUFDO0FBRXBFOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBc0JHO0FBQ0gsTUFBTSxPQUFPLG9CQUFvQjtJQUMvQixnQkFBZ0I7SUFDRyxPQUFPLENBQXVCO0lBQ2pELGdCQUFnQjtJQUNHLE9BQU8sQ0FBdUM7SUFDakUsZ0JBQWdCO0lBQ0csT0FBTyxDQUFTO0lBRW5DLFlBQVksU0FBcUMsRUFBRTtRQUNqRCxJQUFJLENBQUMsT0FBTyxHQUFHLE9BQU8sQ0FBQyxHQUFHLENBQUMsU0FBUyxDQUFDLElBQUksT0FBTyxDQUFDLEdBQUcsQ0FBQyxVQUFVLENBQUMsSUFBSSxPQUFPLENBQUM7UUFFNUUsNEJBQTRCO1FBQzVCLElBQUksQ0FBQyxPQUFPLEdBQUc7WUFDYixVQUFVLEVBQUUsTUFBTSxDQUFDLFVBQVUsSUFBSSxvQkFBb0I7WUFDckQsTUFBTSxFQUFFLE1BQU0sQ0FBQyxNQUFNLElBQUksT0FBTyxDQUFDLEdBQUcsQ0FBQyxZQUFZLENBQUMsSUFBSSxXQUFXO1lBQ2pFLGtCQUFrQixFQUFFLE1BQU0sQ0FBQyxrQkFBa0IsSUFBSTtnQkFDL0MsV0FBVyxFQUFFLEtBQUs7Z0JBQ2xCLElBQUksRUFBRSxNQUFNO2dCQUNaLFVBQVUsRUFBRSxZQUFZO2FBQ3pCO1NBQ0YsQ0FBQztRQUVGLHdDQUF3QztRQUN4QyxJQUFJLENBQUMsT0FB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QUVEOzs7T0FHRztJQUNILFNBQVM7UUFDUCxPQUFPLElBQUksQ0FBQyxPQUFPLENBQUM7SUFDdEIsQ0FBQztJQUVEOzs7T0FHRztJQUNILHFCQUFxQjtRQUNuQixPQUFPLEVBQUUsR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLGtCQUFrQixFQUFFLENBQUM7SUFDaEQsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgR2V0U2VjcmV0VmFsdWVDb21tYW5kLCBTZWNyZXRzTWFuYWdlckNsaWVudCB9IGZyb20gJ0Bhd3Mtc2RrL2NsaWVudC1zZWNyZXRzLW1hbmFnZXInO1xyXG5pbXBvcnQgeyBmcm9tTm9kZVByb3ZpZGVyQ2hhaW4gfSBmcm9tICdAYXdzLXNkay9jcmVkZW50aWFsLXByb3ZpZGVycyc7XHJcblxyXG5pbXBvcnQgdHlwZSB7IENvbmZpZ0xvYWRlciB9IGZyb20gJy4uL2ludGVyZmFjZXMvY29uZmlnLWxvYWRlci5pbnRlcmZhY2UnO1xyXG5pbXBvcnQgdHlwZSB7IFNlY3JldHNNYW5hZ2VyTG9hZGVyQ29uZmlnIH0gZnJvbSAnLi4vaW50ZXJmYWNlcy9zZWNyZXRzLW1hbmFnZXItbG9hZGVyLmludGVyZmFjZSc7XHJcbmltcG9ydCB7IEFXU1NlcnZpY2VFcnJvciwgQ29uZmlndXJhdGlvbkxvYWRFcnJvciB9IGZyb20gJy4uL2Vycm9ycyc7XHJcblxyXG4vKipcclxuICogTG9hZGVyIHRoYXQgcmVhZHMgY29uZmlndXJhdGlvbiBmcm9tIEFXUyBTZWNyZXRzIE1hbmFnZXIuXHJcbiAqIFN1cHBvcnRzIGVudmlyb25tZW50LWF3YXJlIHBhdGggY29uc3RydWN0aW9uLlxyXG4gKlxyXG4gKiBAZXhhbXBsZVxyXG4gKiBgYGB0eXBlc2NyaXB0XHJcbiAqIC8vIEJhc2ljIHVzYWdlXHJcbiAqIGNvbnN0IGxvYWRlciA9IG5ldyBTZWNyZXRzTWFuYWdlckxvYWRlcih7XHJcbiAqICAgc2VjcmV0TmFtZTogJy9teS1hcHAvY29uZmlnJyxcclxuICogICByZWdpb246ICd1cy1lYXN0LTEnXHJcbiAqIH0pO1xyXG4gKlxyXG4gKiAvLyBXaXRoIGVudmlyb25tZW50IG1hcHBpbmdcclxuICogY29uc3QgbG9hZGVyID0gbmV3IFNlY3JldHNNYW5hZ2VyTG9hZGVyKHtcclxuICogICBzZWNyZXROYW1lOiAnL215LWFwcC9jb25maWcnLFxyXG4gKiAgIGVudmlyb25tZW50TWFwcGluZzoge1xyXG4gKiAgICAgZGV2ZWxvcG1lbnQ6ICdkZXYnLFxyXG4gKiAgICAgc3RhZ2luZzogJ3N0ZycsXHJcbiAqICAgICBwcm9kdWN0aW9uOiAncHJvZCdcclxuICogICB9XHJcbiAqIH0pO1xyXG4gKiBgYGBcclxuICovXHJcbmV4cG9ydCBjbGFzcyBTZWNyZXRzTWFuYWdlckxvYWRlciBpbXBsZW1lbnRzIENvbmZpZ0xvYWRlciB7XHJcbiAgLyoqIEBpbnRlcm5hbCAqL1xyXG4gIHByb3RlY3RlZCByZWFkb25seSBfY2xpZW50OiBTZWNyZXRzTWFuYWdlckNsaWVudDtcclxuICAvKiogQGludGVybmFsICovXHJcbiAgcHJvdGVjdGVkIHJlYWRvbmx5IF9jb25maWc6IFJlcXVpcmVkPFNlY3JldHNNYW5hZ2VyTG9hZGVyQ29uZmlnPjtcclxuICAvKiogQGludGVybmFsICovXHJcbiAgcHJvdGVjdGVkIHJlYWRvbmx5IF9hcHBFbnY6IHN0cmluZztcclxuXHJcbiAgY29uc3RydWN0b3IoY29uZmlnOiBTZWNyZXRzTWFuYWdlckxvYWRlckNvbmZpZyA9IHt9KSB7XHJcbiAgICB0aGlzLl9hcHBFbnYgPSBwcm9jZXNzLmVudlsnQVBQX0VOViddIHx8IHByb2Nlc3MuZW52WydOT0RFX0VOViddIHx8ICdsb2NhbCc7XHJcblxyXG4gICAgLy8gU2V0IGRlZmF1bHQgY29uZmlndXJhdGlvblxyXG4gICAgdGhpcy5fY29uZmlnID0ge1xyXG4gICAgICBzZWNyZXROYW1lOiBjb25maWcuc2VjcmV0TmFtZSB8fCAnL25lc3Rqcy1jb25maWctYXdzJyxcclxuICAgICAgcmVnaW9uOiBjb25maWcucmVnaW9uIHx8IHByb2Nlc3MuZW52WydBV1NfUkVHSU9OJ10gfHwgJ3VzLWVhc3QtMScsXHJcbiAgICAgIGVudmlyb25tZW50TWFwcGluZzogY29uZmlnLmVudmlyb25tZW50TWFwcGluZyB8fCB7XHJcbiAgICAgICAgZGV2ZWxvcG1lbnQ6ICdkZXYnLFxyXG4gICAgICAgIHRlc3Q6ICd0ZXN0JyxcclxuICAgICAgICBwcm9kdWN0aW9uOiAncHJvZHVjdGlvbicsXHJcbiAgICAgIH0sXHJcbiAgICB9O1xyXG5cclxuICAgIC8vIEluaXRpYWxpemUgQVdTIFNlY3JldHMgTWFuYWdlciBjbGllbnRcclxuICAgIHRoaXMuX2NsaWVudCA9IG5ldyBTZWNyZXRzTWFuYWdlckNsaWVudCh7XHJcbiAgICAgIGNyZWRlbnRpYWxzOiBmcm9tTm9kZVByb3ZpZGVyQ2hhaW4oKSxcclxuICAgICAgcmVnaW9uOiB0aGlzLl9jb25maWcucmVnaW9uLFxyXG4gICAgfSk7XHJcbiAgfVxyXG5cclxuXHJcbiAgLyoqXHJcbiAgICogR2V0IHRoZSBuYW1lIG9mIHRoaXMgbG9hZGVyIGZvciBsb2dnaW5nIGFuZCBkZWJ1Z2dpbmcuXHJcbiAgICogQHJldHVybnMgVGhlIGxvYWRlciBuYW1lIHdpdGggc2VjcmV0IHBhdGggb3IgYmFzZSBzZWNyZXQgbmFtZSBpZiBwYXRoIGNhbm5vdCBiZSBidWlsdFxyXG4gICAqL1xyXG4gIGdldE5hbWUoKTogc3RyaW5nIHtcclxuICAgIC8vIEF2b2lkIGNhbGxpbmcgYnVpbGRTZWNyZXROYW1lKCkgdG8gcHJldmVudCBzdGFjayBvdmVyZmxvdyB3aGVuXHJcbiAgICAvLyBlbnZpcm9ubWVudCBtYXBwaW5nIGlzIG1pc3NpbmcgLSBidWlsZFNlY3JldE5hbWUoKSB0aHJvd3MgYW4gZXJyb3JcclxuICAgIC8vIHRoYXQgaW5jbHVkZXMgZ2V0TmFtZSgpIGluIHRoZSBtZXNzYWdlLCBjYXVzaW5nIGluZmluaXRlIHJlY3Vyc2lvbi5cclxuICAgIGNvbnN0IGVudlByZWZpeCA9IHRoaXMuX2NvbmZpZy5lbnZpcm9ubWVudE1hcHBpbmdbdGhpcy5fYXBwRW52XTtcclxuICAgIGlmIChlbnZQcmVmaXgpIHtcclxuICAgICAgcmV0dXJuIGBTZWNyZXRzTWFuYWdlckxvYWRlcigvJHtlbnZQcmVmaXh9JHt0aGlzLl9jb25maWcuc2VjcmV0TmFtZX0pYDtcclxuICAgIH1cclxuICAgIC8vIEZhbGxiYWNrIHRvIGJhc2Ugc2VjcmV0IG5hbWUgd2hlbiBlbnZpcm9ubWVudCBtYXBwaW5nIGlzIHVuYXZhaWxhYmxlXHJcbiAgICByZXR1cm4gYFNlY3JldHNNYW5hZ2VyTG9hZGVyKCR7dGhpcy5fY29uZmlnLnNlY3JldE5hbWV9KWA7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBDaGVjayBpZiB0aGlzIGxvYWRlciBpcyBhdmFpbGFibGUgaW4gdGhlIGN1cnJlbnQgZW52aXJvbm1lbnQuXHJcbiAgICogQHJldHVybnMgUHJvbWlzZSByZXNvbHZpbmcgdG8gdHJ1ZSBpZiBub3QgaW4gbG9jYWwgZW52aXJvbm1lbnQgYW5kIEFXUyBjcmVkZW50aWFscyBhcmUgYXZhaWxhYmxlXHJcbiAgICovXHJcbiAgYXN5bmMgaXNBdmFpbGFibGUoKTogUHJvbWlzZTxib29sZWFuPiB7XHJcbiAgICAvLyBTa2lwIGluIGxvY2FsIGVudmlyb25tZW50XHJcbiAgICBpZiAodGhpcy5fYXBwRW52ID09PSAnbG9jYWwnKSB7XHJcbiAgICAgIHJldHVybiBmYWxzZTtcclxuICAgIH1cclxuXHJcbiAgICB0cnkge1xyXG4gICAgICAvLyBUZXN0IEFXUyBjcmVkZW50aWFscyBieSBhdHRlbXB0aW5nIHRvIGdldCBjYWxsZXIgaWRlbnRpdHlcclxuICAgICAgYXdhaXQgdGhpcy5fY2xpZW50LmNvbmZpZy5jcmVkZW50aWFscygpO1xyXG4gICAgICByZXR1cm4gdHJ1ZTtcclxuICAgIH0gY2F0Y2gge1xyXG4gICAgICByZXR1cm4gZmFsc2U7XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBMb2FkIGNvbmZpZ3VyYXRpb24gZnJvbSBBV1MgU2VjcmV0cyBNYW5hZ2VyLlxyXG4gICAqIEByZXR1cm5zIFByb21pc2UgcmVzb2x2aW5nIHRvIGNvbmZpZ3VyYXRpb24ga2V5LXZhbHVlIHBhaXJzIGZyb20gdGhlIHNlY3JldFxyXG4gICAqIEB0aHJvd3MgQVdTU2VydmljZUVycm9yIGlmIEFXUyBvcGVyYXRpb24gZmFpbHNcclxuICAgKiBAdGhyb3dzIENvbmZpZ3VyYXRpb25Mb2FkRXJyb3IgaWYgc2VjcmV0IGNhbm5vdCBiZSBwYXJzZWRcclxuICAgKi9cclxuICBhc3luYyBsb2FkKCk6IFByb21pc2U8UmVjb3JkPHN0cmluZywgdW5rbm93bj4+IHtcclxuICAgIC8vIFNraXAgbG9hZGluZyBpbiBsb2NhbCBlbnZpcm9ubWVudFxyXG4gICAgaWYgKHRoaXMuX2FwcEVudiA9PT0gJ2xvY2FsJykge1xyXG4gICAgICByZXR1cm4ge307XHJcbiAgICB9XHJcblxyXG4gICAgY29uc3Qgc2VjcmV0TmFtZSA9IHRoaXMuYnVpbGRTZWNyZXROYW1lKCk7XHJcblxyXG4gICAgdHJ5IHtcclxuICAgICAgY29uc3QgY29tbWFuZCA9IG5ldyBHZXRTZWNyZXRWYWx1ZUNvbW1hbmQoeyBTZWNyZXRJZDogc2VjcmV0TmFtZSB9KTtcclxuICAgICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCB0aGlzLl9jbGllbnQuc2VuZChjb21tYW5kKTtcclxuXHJcbiAgICAgIGlmICghcmVzcG9uc2UuU2VjcmV0U3RyaW5nKSB7XHJcbiAgICAgICAgcmV0dXJuIHt9O1xyXG4gICAgICB9XHJcblxyXG4gICAgICAvLyBUcnkgdG8gcGFyc2UgYXMgSlNPTiwgZmFsbGJhY2sgdG8gc3RyaW5nIHZhbHVlXHJcbiAgICAgIHRyeSB7XHJcbiAgICAgICAgY29uc3QgcGFyc2VkID0gSlNPTi5wYXJzZShyZXNwb25zZS5TZWNyZXRTdHJpbmcpO1xyXG5cclxuICAgICAgICAvLyBFbnN1cmUgd2UgcmV0dXJuIGFuIG9iamVjdCBmb3IgY29uZmlndXJhdGlvbiBtZXJnaW5nXHJcbiAgICAgICAgaWYgKHR5cGVvZiBwYXJzZWQgPT09ICdvYmplY3QnICYmIHBhcnNlZCAhPT0gbnVsbCAmJiAhQXJyYXkuaXNBcnJheShwYXJzZWQpKSB7XHJcbiAgICAgICAgICByZXR1cm4gcGFyc2VkO1xyXG4gICAgICAgIH0gZWxzZSB7XHJcbiAgICAgICAgICAvLyBJZiBpdCdzIG5vdCBhbiBvYmplY3QsIHdyYXAgaXQgaW4gYSBjb25maWd1cmF0aW9uIG9iamVjdFxyXG4gICAgICAgICAgcmV0dXJuIHsgU0VDUkVUX1ZBTFVFOiBwYXJzZWQgfTtcclxuICAgICAgICB9XHJcbiAgICAgIH0gY2F0Y2gge1xyXG4gICAgICAgIC8vIElmIEpTT04gcGFyc2luZyBmYWlscywgdHJlYXQgYXMgYSBzaW5nbGUgc3RyaW5nIHZhbHVlXHJcbiAgICAgICAgcmV0dXJuIHsgU0VDUkVUX1ZBTFVFOiByZXNwb25zZS5TZWNyZXRTdHJpbmcgfTtcclxuICAgICAgfVxyXG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcclxuICAgICAgLy8gSGFuZGxlIHNwZWNpZmljIEFXUyBlcnJvcnNcclxuICAgICAgaWYgKGVycm9yIGluc3RhbmNlb2YgRXJyb3IpIHtcclxuICAgICAgICBpZiAoZXJyb3IubmFtZSA9PT0gJ1Jlc291cmNlTm90Rm91bmRFeGNlcHRpb24nKSB7XHJcbiAgICAgICAgICAvLyBTZWNyZXQgZG9lc24ndCBleGlzdCAtIHRoaXMgaXMgbm90IG5lY2Vzc2FyaWx5IGFuIGVycm9yIGluIGFsbCBlbnZpcm9ubWVudHNcclxuICAgICAgICAgIHJldHVybiB7fTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGlmIChlcnJvci5uYW1lID09PSAnQWNjZXNzRGVuaWVkRXhjZXB0aW9uJykge1xyXG4gICAgICAgICAgdGhyb3cgbmV3IEFXU1NlcnZpY2VFcnJvcihcclxuICAgICAgICAgICAgYEFjY2VzcyBkZW5pZWQgd2hlbiByZXRyaWV2aW5nIHNlY3JldCAnJHtzZWNyZXROYW1lfScuIENoZWNrIEFXUyBjcmVkZW50aWFscyBhbmQgcGVybWlzc2lvbnMuYCxcclxuICAgICAgICAgICAgJ1NlY3JldHNNYW5hZ2VyJyxcclxuICAgICAgICAgICAgJ0dldFNlY3JldFZhbHVlJyxcclxuICAgICAgICAgICAgZXJyb3IsXHJcbiAgICAgICAgICApO1xyXG4gICAgICAgIH1cclxuXHJcbiAgICAgICAgaWYgKGVycm9yLm5hbWUgPT09ICdJbnZhbGlkUmVxdWVzdEV4Y2VwdGlvbicpIHtcclxuICAgICAgICAgIHRocm93IG5ldyBBV1NTZXJ2aWNlRXJyb3IoXHJcbiAgICAgICAgICAgIGBJbnZhbGlkIHJlcXVlc3Qgd2hlbiByZXRyaWV2aW5nIHNlY3JldCAnJHtzZWNyZXROYW1lfScuIENoZWNrIHNlY3JldCBuYW1lIGZvcm1hdC5gLFxyXG4gICAgICAgICAgICAnU2VjcmV0c01hbmFnZXInLFxyXG4gICAgICAgICAgICAnR2V0U2VjcmV0VmFsdWUnLFxyXG4gICAgICAgICAgICBlcnJvcixcclxuICAgICAgICAgICk7XHJcbiAgICAgICAgfVxyXG4gICAgICB9XHJcblxyXG4gICAgICAvLyBGb3Igb3RoZXIgZXJyb3JzLCB3cmFwIGluIEFXU1NlcnZpY2VFcnJvclxyXG4gICAgICB0aHJvdyBuZXcgQVdTU2VydmljZUVycm9yKFxyXG4gICAgICAgIGBGYWlsZWQgdG8gcmV0cmlldmUgc2VjcmV0ICcke3NlY3JldE5hbWV9JyBmcm9tIEFXUyBTZWNyZXRzIE1hbmFnZXI6ICR7ZXJyb3IgaW5zdGFuY2VvZiBFcnJvciA/IGVycm9yLm1lc3NhZ2UgOiBTdHJpbmcoZXJyb3IpfWAsXHJcbiAgICAgICAgJ1NlY3JldHNNYW5hZ2VyJyxcclxuICAgICAgICAnR2V0U2VjcmV0VmFsdWUnLFxyXG4gICAgICAgIGVycm9yIGluc3RhbmNlb2YgRXJyb3IgPyBlcnJvciA6IHVuZGVmaW5lZCxcclxuICAgICAgKTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIEJ1aWxkIHRoZSBlbnZpcm9ubWVudC1hd2FyZSBzZWNyZXQgbmFtZS9wYXRoLlxyXG4gICAqIEByZXR1cm5zIFRoZSBmdWxsIHNlY3JldCBuYW1lIHdpdGggZW52aXJvbm1lbnQgcHJlZml4XHJcbiAgICovXHJcbiAgYnVpbGRTZWNyZXROYW1lKCk6IHN0cmluZyB7XHJcbiAgICBjb25zdCBlbnZQcmVmaXggPSB0aGlzLl9jb25maWcuZW52aXJvbm1lbnRNYXBwaW5nW3RoaXMuX2FwcEVudl07XHJcblxyXG4gICAgaWYgKCFlbnZQcmVmaXgpIHtcclxuICAgICAgdGhyb3cgbmV3IENvbmZpZ3VyYXRpb25Mb2FkRXJyb3IoXHJcbiAgICAgICAgYE5vIGVudmlyb25tZW50IG1hcHBpbmcgZm91bmQgZm9yIEFQUF9FTlYgJyR7dGhpcy5fYXBwRW52fScuIGAgK1xyXG4gICAgICAgICAgYEF2YWlsYWJsZSBlbnZpcm9ubWVudHM6ICR7T2JqZWN0LmtleXModGhpcy5fY29uZmlnLmVudmlyb25tZW50TWFwcGluZykuam9pbignLCAnKX1gLFxyXG4gICAgICAgIHRoaXMuZ2V0TmFtZSgpLFxyXG4gICAgICApO1xyXG4gICAgfVxyXG5cclxuICAgIHJldHVybiBgLyR7ZW52UHJlZml4fSR7dGhpcy5fY29uZmlnLnNlY3JldE5hbWV9YDtcclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIEdldCB0aGUgY3VycmVudCBhcHAgZW52aXJvbm1lbnQuXHJcbiAgICogQHJldHVybnMgVGhlIGN1cnJlbnQgQVBQX0VOViBvciBOT0RFX0VOViB2YWx1ZVxyXG4gICAqL1xyXG4gIGdldEFwcEVudigpOiBzdHJpbmcge1xyXG4gICAgcmV0dXJuIHRoaXMuX2FwcEVudjtcclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIEdldCB0aGUgZW52aXJvbm1lbnQgbWFwcGluZyBjb25maWd1cmF0aW9uLlxyXG4gICAqIEByZXR1cm5zIFRoZSBlbnZpcm9ubWVudCBtYXBwaW5nIHJlY29yZFxyXG4gICAqL1xyXG4gIGdldEVudmlyb25tZW50TWFwcGluZygpOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+IHtcclxuICAgIHJldHVybiB7IC4uLnRoaXMuX2NvbmZpZy5lbnZpcm9ubWVudE1hcHBpbmcgfTtcclxuICB9XHJcbn1cclxuIl19
|
|
@@ -0,0 +1,195 @@
|
|
|
1
|
+
import { GetParametersByPathCommand, SSMClient } from '@aws-sdk/client-ssm';
|
|
2
|
+
import { fromNodeProviderChain } from '@aws-sdk/credential-providers';
|
|
3
|
+
import { AWSServiceError, ConfigurationLoadError } from '../errors';
|
|
4
|
+
/**
|
|
5
|
+
* Loader that reads configuration from AWS SSM Parameter Store.
|
|
6
|
+
* Supports environment-aware path construction, pagination, and decryption options.
|
|
7
|
+
*
|
|
8
|
+
* @example
|
|
9
|
+
* ```typescript
|
|
10
|
+
* // Basic usage
|
|
11
|
+
* const loader = new SSMParameterStoreLoader({
|
|
12
|
+
* parameterPath: '/my-app/config',
|
|
13
|
+
* region: 'us-east-1'
|
|
14
|
+
* });
|
|
15
|
+
*
|
|
16
|
+
* // With environment mapping
|
|
17
|
+
* const loader = new SSMParameterStoreLoader({
|
|
18
|
+
* parameterPath: '/my-app/config',
|
|
19
|
+
* environmentMapping: {
|
|
20
|
+
* development: 'dev',
|
|
21
|
+
* staging: 'stg',
|
|
22
|
+
* production: 'prod'
|
|
23
|
+
* }
|
|
24
|
+
* });
|
|
25
|
+
* ```
|
|
26
|
+
*/
|
|
27
|
+
export class SSMParameterStoreLoader {
|
|
28
|
+
/** @internal */
|
|
29
|
+
_client;
|
|
30
|
+
/** @internal */
|
|
31
|
+
_config;
|
|
32
|
+
/** @internal */
|
|
33
|
+
_appEnv;
|
|
34
|
+
constructor(config = {}) {
|
|
35
|
+
this._appEnv = process.env['APP_ENV'] || process.env['NODE_ENV'] || 'local';
|
|
36
|
+
// Set default configuration
|
|
37
|
+
this._config = {
|
|
38
|
+
parameterPath: config.parameterPath || '/config-aws',
|
|
39
|
+
region: config.region || process.env['AWS_REGION'] || 'us-east-1',
|
|
40
|
+
environmentMapping: config.environmentMapping || {
|
|
41
|
+
development: 'dev',
|
|
42
|
+
test: 'test',
|
|
43
|
+
production: 'production',
|
|
44
|
+
},
|
|
45
|
+
withDecryption: config.withDecryption ?? true,
|
|
46
|
+
};
|
|
47
|
+
// Initialize AWS SSM client
|
|
48
|
+
this._client = new SSMClient({
|
|
49
|
+
credentials: fromNodeProviderChain(),
|
|
50
|
+
region: this._config.region,
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
/**
|
|
54
|
+
* Get the name of this loader for logging and debugging.
|
|
55
|
+
* @returns The loader name with parameter path
|
|
56
|
+
*/
|
|
57
|
+
getName() {
|
|
58
|
+
try {
|
|
59
|
+
const parameterPath = this.buildParameterPath();
|
|
60
|
+
return `SSMParameterStoreLoader(${parameterPath})`;
|
|
61
|
+
}
|
|
62
|
+
catch {
|
|
63
|
+
// Fallback if path construction fails (e.g., missing env mapping)
|
|
64
|
+
return `SSMParameterStoreLoader(${this._config.parameterPath})`;
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
/**
|
|
68
|
+
* Check if this loader is available in the current environment.
|
|
69
|
+
* @returns Promise resolving to true if not in local environment and AWS credentials are available
|
|
70
|
+
*/
|
|
71
|
+
async isAvailable() {
|
|
72
|
+
// Skip in local environment
|
|
73
|
+
if (this._appEnv === 'local') {
|
|
74
|
+
return false;
|
|
75
|
+
}
|
|
76
|
+
try {
|
|
77
|
+
// Test AWS credentials by attempting to get caller identity
|
|
78
|
+
await this._client.config.credentials();
|
|
79
|
+
return true;
|
|
80
|
+
}
|
|
81
|
+
catch {
|
|
82
|
+
return false;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Load configuration from AWS SSM Parameter Store.
|
|
87
|
+
* Implements recursive parameter fetching with NextToken handling for pagination.
|
|
88
|
+
* @returns Promise resolving to configuration key-value pairs from parameters
|
|
89
|
+
* @throws AWSServiceError if AWS operation fails
|
|
90
|
+
* @throws ConfigurationLoadError if parameter path cannot be constructed
|
|
91
|
+
*/
|
|
92
|
+
async load() {
|
|
93
|
+
// Skip loading in local environment
|
|
94
|
+
if (this._appEnv === 'local') {
|
|
95
|
+
return {};
|
|
96
|
+
}
|
|
97
|
+
const parameterPath = this.buildParameterPath();
|
|
98
|
+
const result = {};
|
|
99
|
+
let nextToken;
|
|
100
|
+
try {
|
|
101
|
+
do {
|
|
102
|
+
const command = new GetParametersByPathCommand({
|
|
103
|
+
Path: parameterPath,
|
|
104
|
+
Recursive: true,
|
|
105
|
+
WithDecryption: this._config.withDecryption,
|
|
106
|
+
NextToken: nextToken,
|
|
107
|
+
});
|
|
108
|
+
const response = await this._client.send(command);
|
|
109
|
+
if (!response.Parameters) {
|
|
110
|
+
// No parameters found - this is not necessarily an error
|
|
111
|
+
break;
|
|
112
|
+
}
|
|
113
|
+
// Process each parameter
|
|
114
|
+
for (const param of response.Parameters) {
|
|
115
|
+
const key = this.transformParameterName(param.Name, parameterPath);
|
|
116
|
+
if (key && param.Value !== undefined) {
|
|
117
|
+
result[key] = param.Value;
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
nextToken = response.NextToken;
|
|
121
|
+
} while (nextToken);
|
|
122
|
+
return result;
|
|
123
|
+
}
|
|
124
|
+
catch (error) {
|
|
125
|
+
// Handle specific AWS errors
|
|
126
|
+
if (error instanceof Error) {
|
|
127
|
+
if (error.name === 'ResourceNotFoundException' || error.name === 'ParameterNotFound') {
|
|
128
|
+
// No parameters found at path - this is not necessarily an error
|
|
129
|
+
return {};
|
|
130
|
+
}
|
|
131
|
+
if (error.name === 'AccessDeniedException') {
|
|
132
|
+
throw new AWSServiceError(`Access denied when retrieving parameters from path '${parameterPath}'. Check AWS credentials and permissions.`, 'SSM', 'GetParametersByPath', error);
|
|
133
|
+
}
|
|
134
|
+
if (error.name === 'InvalidFilterKey' || error.name === 'InvalidFilterValue') {
|
|
135
|
+
throw new AWSServiceError(`Invalid parameter path '${parameterPath}'. Check path format.`, 'SSM', 'GetParametersByPath', error);
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
// For other errors, wrap in AWSServiceError
|
|
139
|
+
throw new AWSServiceError(`Failed to retrieve parameters from path '${parameterPath}' in AWS SSM Parameter Store: ${error instanceof Error ? error.message : String(error)}`, 'SSM', 'GetParametersByPath', error instanceof Error ? error : undefined);
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
/**
|
|
143
|
+
* Build the environment-aware parameter path.
|
|
144
|
+
* @returns The full parameter path with environment prefix
|
|
145
|
+
* @throws ConfigurationLoadError if no environment mapping found
|
|
146
|
+
*/
|
|
147
|
+
buildParameterPath() {
|
|
148
|
+
const envPrefix = this._config.environmentMapping[this._appEnv];
|
|
149
|
+
if (!envPrefix) {
|
|
150
|
+
throw new ConfigurationLoadError(`No environment mapping found for APP_ENV '${this._appEnv}'. ` +
|
|
151
|
+
`Available environments: ${Object.keys(this._config.environmentMapping).join(', ')}`, this.getName());
|
|
152
|
+
}
|
|
153
|
+
return `/${envPrefix}${this._config.parameterPath}`;
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* Transform parameter name by removing the prefix and converting to uppercase.
|
|
157
|
+
* Example: '/dev/config-aws/database/host' -> 'DATABASE_HOST'
|
|
158
|
+
* @param parameterName The full parameter name from AWS
|
|
159
|
+
* @param pathPrefix The path prefix to remove
|
|
160
|
+
* @returns The transformed parameter name or null if invalid
|
|
161
|
+
*/
|
|
162
|
+
transformParameterName(parameterName, pathPrefix) {
|
|
163
|
+
if (!parameterName) {
|
|
164
|
+
return null;
|
|
165
|
+
}
|
|
166
|
+
// Remove the path prefix
|
|
167
|
+
let key = parameterName;
|
|
168
|
+
if (key.startsWith(pathPrefix)) {
|
|
169
|
+
key = key.substring(pathPrefix.length);
|
|
170
|
+
}
|
|
171
|
+
// Remove leading slash if present
|
|
172
|
+
if (key.startsWith('/')) {
|
|
173
|
+
key = key.substring(1);
|
|
174
|
+
}
|
|
175
|
+
// Convert slashes to underscores and uppercase
|
|
176
|
+
key = key.replace(/\//g, '_').toUpperCase();
|
|
177
|
+
// Return null for empty keys
|
|
178
|
+
return key || null;
|
|
179
|
+
}
|
|
180
|
+
/**
|
|
181
|
+
* Get the current app environment.
|
|
182
|
+
* @returns The current APP_ENV or NODE_ENV value
|
|
183
|
+
*/
|
|
184
|
+
getAppEnv() {
|
|
185
|
+
return this._appEnv;
|
|
186
|
+
}
|
|
187
|
+
/**
|
|
188
|
+
* Get the environment mapping configuration.
|
|
189
|
+
* @returns The environment mapping record
|
|
190
|
+
*/
|
|
191
|
+
getEnvironmentMapping() {
|
|
192
|
+
return { ...this._config.environmentMapping };
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3NtLXBhcmFtZXRlci1zdG9yZS5sb2FkZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9jb25maWctYXdzL3NyYy9sb2FkZXJzL3NzbS1wYXJhbWV0ZXItc3RvcmUubG9hZGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSwwQkFBMEIsRUFBRSxTQUFTLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQztBQUM1RSxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSwrQkFBK0IsQ0FBQztBQUl0RSxPQUFPLEVBQUUsZUFBZSxFQUFFLHNCQUFzQixFQUFFLE1BQU0sV0FBVyxDQUFDO0FBRXBFOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBc0JHO0FBQ0gsTUFBTSxPQUFPLHVCQUF1QjtJQUNsQyxnQkFBZ0I7SUFDRyxPQUFPLENBQVk7SUFDdEMsZ0JBQWdCO0lBQ0csT0FBTyxDQUEwQztJQUNwRSxnQkFBZ0I7SUFDRyxPQUFPLENBQVM7SUFFbkMsWUFBWSxTQUF3QyxFQUFFO1FBQ3BELElBQUksQ0FBQyxPQUFPLEdBQUcsT0FBTyxDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUMsSUFBSSxPQUFPLENBQUMsR0FBRyxDQUFDLFVBQVUsQ0FBQyxJQUFJLE9BQU8sQ0FBQztRQUU1RSw0QkFBNEI7UUFDNUIsSUFBSSxDQUFDLE9BQU8sR0FBRztZQUNiLGFBQWEsRUFBRSxNQUFNLENBQUMsYUFBYSxJQUFJLGFBQWE7WUFDcEQsTUFBTSxFQUFFLE1BQU0sQ0FBQyxNQUFNLElBQUksT0FBTyxDQUFDLEdBQUcsQ0FBQyxZQUFZLENBQUMsSUFBSSxXQUFXO1lBQ2pFLGtCQUFrQixFQUFFLE1BQU0sQ0FBQyxrQkFBa0IsSUFBSTtnQkFDL0MsV0FBVyxFQUFFLEtBQUs7Z0JBQ2xCLElBQUksRUFBRSxNQUFNO2dCQUNaLFVBQVUsRUFBRSxZQUFZO2FBQ3pCO1lBQ0QsY0FBYyxFQUFFLE1BQU0sQ0FBQyxjQUFjLElBQUksSUFB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QUFFLEVBQ3RGLElBQUksQ0FBQyxPQUFPLEVBQUUsQ0FDZixDQUFDO1FBQ0osQ0FBQztRQUVELE9BQU8sSUFBSSxTQUFTLEdBQUcsSUFBSSxDQUFDLE9BQU8sQ0FBQyxhQUFhLEVBQUUsQ0FBQztJQUN0RCxDQUFDO0lBRUQ7Ozs7OztPQU1HO0lBQ0ssc0JBQXNCLENBQUMsYUFBaUMsRUFBRSxVQUFrQjtRQUNsRixJQUFJLENBQUMsYUFBYSxFQUFFLENBQUM7WUFDbkIsT0FBTyxJQUFJLENBQUM7UUFDZCxDQUFDO1FBRUQseUJBQXlCO1FBQ3pCLElBQUksR0FBRyxHQUFHLGFBQWEsQ0FBQztRQUN4QixJQUFJLEdBQUcsQ0FBQyxVQUFVLENBQUMsVUFBVSxDQUFDLEVBQUUsQ0FBQztZQUMvQixHQUFHLEdBQUcsR0FBRyxDQUFDLFNBQVMsQ0FBQyxVQUFVLENBQUMsTUFBTSxDQUFDLENBQUM7UUFDekMsQ0FBQztRQUVELGtDQUFrQztRQUNsQyxJQUFJLEdBQUcsQ0FBQyxVQUFVLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQztZQUN4QixHQUFHLEdBQUcsR0FBRyxDQUFDLFNBQVMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUN6QixDQUFDO1FBRUQsK0NBQStDO1FBQy9DLEdBQUcsR0FBRyxHQUFHLENBQUMsT0FBTyxDQUFDLEtBQUssRUFBRSxHQUFHLENBQUMsQ0FBQyxXQUFXLEVBQUUsQ0FBQztRQUU1Qyw2QkFBNkI7UUFDN0IsT0FBTyxHQUFHLElBQUksSUFBSSxDQUFDO0lBQ3JCLENBQUM7SUFFRDs7O09BR0c7SUFDSCxTQUFTO1FBQ1AsT0FBTyxJQUFJLENBQUMsT0FBTyxDQUFDO0lBQ3RCLENBQUM7SUFFRDs7O09BR0c7SUFDSCxxQkFBcUI7UUFDbkIsT0FBTyxFQUFFLEdBQUcsSUFBSSxDQUFDLE9BQU8sQ0FBQyxrQkFBa0IsRUFBRSxDQUFDO0lBQ2hELENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEdldFBhcmFtZXRlcnNCeVBhdGhDb21tYW5kLCBTU01DbGllbnQgfSBmcm9tICdAYXdzLXNkay9jbGllbnQtc3NtJztcclxuaW1wb3J0IHsgZnJvbU5vZGVQcm92aWRlckNoYWluIH0gZnJvbSAnQGF3cy1zZGsvY3JlZGVudGlhbC1wcm92aWRlcnMnO1xyXG5cclxuaW1wb3J0IHR5cGUgeyBDb25maWdMb2FkZXIgfSBmcm9tICcuLi9pbnRlcmZhY2VzL2NvbmZpZy1sb2FkZXIuaW50ZXJmYWNlJztcclxuaW1wb3J0IHR5cGUgeyBTU01QYXJhbWV0ZXJTdG9yZUxvYWRlckNvbmZpZyB9IGZyb20gJy4uL2ludGVyZmFjZXMvc3NtLXBhcmFtZXRlci1zdG9yZS1sb2FkZXIuaW50ZXJmYWNlJztcclxuaW1wb3J0IHsgQVdTU2VydmljZUVycm9yLCBDb25maWd1cmF0aW9uTG9hZEVycm9yIH0gZnJvbSAnLi4vZXJyb3JzJztcclxuXHJcbi8qKlxyXG4gKiBMb2FkZXIgdGhhdCByZWFkcyBjb25maWd1cmF0aW9uIGZyb20gQVdTIFNTTSBQYXJhbWV0ZXIgU3RvcmUuXHJcbiAqIFN1cHBvcnRzIGVudmlyb25tZW50LWF3YXJlIHBhdGggY29uc3RydWN0aW9uLCBwYWdpbmF0aW9uLCBhbmQgZGVjcnlwdGlvbiBvcHRpb25zLlxyXG4gKlxyXG4gKiBAZXhhbXBsZVxyXG4gKiBgYGB0eXBlc2NyaXB0XHJcbiAqIC8vIEJhc2ljIHVzYWdlXHJcbiAqIGNvbnN0IGxvYWRlciA9IG5ldyBTU01QYXJhbWV0ZXJTdG9yZUxvYWRlcih7XHJcbiAqICAgcGFyYW1ldGVyUGF0aDogJy9teS1hcHAvY29uZmlnJyxcclxuICogICByZWdpb246ICd1cy1lYXN0LTEnXHJcbiAqIH0pO1xyXG4gKlxyXG4gKiAvLyBXaXRoIGVudmlyb25tZW50IG1hcHBpbmdcclxuICogY29uc3QgbG9hZGVyID0gbmV3IFNTTVBhcmFtZXRlclN0b3JlTG9hZGVyKHtcclxuICogICBwYXJhbWV0ZXJQYXRoOiAnL215LWFwcC9jb25maWcnLFxyXG4gKiAgIGVudmlyb25tZW50TWFwcGluZzoge1xyXG4gKiAgICAgZGV2ZWxvcG1lbnQ6ICdkZXYnLFxyXG4gKiAgICAgc3RhZ2luZzogJ3N0ZycsXHJcbiAqICAgICBwcm9kdWN0aW9uOiAncHJvZCdcclxuICogICB9XHJcbiAqIH0pO1xyXG4gKiBgYGBcclxuICovXHJcbmV4cG9ydCBjbGFzcyBTU01QYXJhbWV0ZXJTdG9yZUxvYWRlciBpbXBsZW1lbnRzIENvbmZpZ0xvYWRlciB7XHJcbiAgLyoqIEBpbnRlcm5hbCAqL1xyXG4gIHByb3RlY3RlZCByZWFkb25seSBfY2xpZW50OiBTU01DbGllbnQ7XHJcbiAgLyoqIEBpbnRlcm5hbCAqL1xyXG4gIHByb3RlY3RlZCByZWFkb25seSBfY29uZmlnOiBSZXF1aXJlZDxTU01QYXJhbWV0ZXJTdG9yZUxvYWRlckNvbmZpZz47XHJcbiAgLyoqIEBpbnRlcm5hbCAqL1xyXG4gIHByb3RlY3RlZCByZWFkb25seSBfYXBwRW52OiBzdHJpbmc7XHJcblxyXG4gIGNvbnN0cnVjdG9yKGNvbmZpZzogU1NNUGFyYW1ldGVyU3RvcmVMb2FkZXJDb25maWcgPSB7fSkge1xyXG4gICAgdGhpcy5fYXBwRW52ID0gcHJvY2Vzcy5lbnZbJ0FQUF9FTlYnXSB8fCBwcm9jZXNzLmVudlsnTk9ERV9FTlYnXSB8fCAnbG9jYWwnO1xyXG5cclxuICAgIC8vIFNldCBkZWZhdWx0IGNvbmZpZ3VyYXRpb25cclxuICAgIHRoaXMuX2NvbmZpZyA9IHtcclxuICAgICAgcGFyYW1ldGVyUGF0aDogY29uZmlnLnBhcmFtZXRlclBhdGggfHwgJy9jb25maWctYXdzJyxcclxuICAgICAgcmVnaW9uOiBjb25maWcucmVnaW9uIHx8IHByb2Nlc3MuZW52WydBV1NfUkVHSU9OJ10gfHwgJ3VzLWVhc3QtMScsXHJcbiAgICAgIGVudmlyb25tZW50TWFwcGluZzogY29uZmlnLmVudmlyb25tZW50TWFwcGluZyB8fCB7XHJcbiAgICAgICAgZGV2ZWxvcG1lbnQ6ICdkZXYnLFxyXG4gICAgICAgIHRlc3Q6ICd0ZXN0JyxcclxuICAgICAgICBwcm9kdWN0aW9uOiAncHJvZHVjdGlvbicsXHJcbiAgICAgIH0sXHJcbiAgICAgIHdpdGhEZWNyeXB0aW9uOiBjb25maWcud2l0aERlY3J5cHRpb24gPz8gdHJ1ZSxcclxuICAgIH07XHJcblxyXG4gICAgLy8gSW5pdGlhbGl6ZSBBV1MgU1NNIGNsaWVudFxyXG4gICAgdGhpcy5fY2xpZW50ID0gbmV3IFNTTUNsaWVudCh7XHJcbiAgICAgIGNyZWRlbnRpYWxzOiBmcm9tTm9kZVByb3ZpZGVyQ2hhaW4oKSxcclxuICAgICAgcmVnaW9uOiB0aGlzLl9jb25maWcucmVnaW9uLFxyXG4gICAgfSk7XHJcbiAgfVxyXG5cclxuXHJcbiAgLyoqXHJcbiAgICogR2V0IHRoZSBuYW1lIG9mIHRoaXMgbG9hZGVyIGZvciBsb2dnaW5nIGFuZCBkZWJ1Z2dpbmcuXHJcbiAgICogQHJldHVybnMgVGhlIGxvYWRlciBuYW1lIHdpdGggcGFyYW1ldGVyIHBhdGhcclxuICAgKi9cclxuICBnZXROYW1lKCk6IHN0cmluZyB7XHJcbiAgICB0cnkge1xyXG4gICAgICBjb25zdCBwYXJhbWV0ZXJQYXRoID0gdGhpcy5idWlsZFBhcmFtZXRlclBhdGgoKTtcclxuICAgICAgcmV0dXJuIGBTU01QYXJhbWV0ZXJTdG9yZUxvYWRlcigke3BhcmFtZXRlclBhdGh9KWA7XHJcbiAgICB9IGNhdGNoIHtcclxuICAgICAgLy8gRmFsbGJhY2sgaWYgcGF0aCBjb25zdHJ1Y3Rpb24gZmFpbHMgKGUuZy4sIG1pc3NpbmcgZW52IG1hcHBpbmcpXHJcbiAgICAgIHJldHVybiBgU1NNUGFyYW1ldGVyU3RvcmVMb2FkZXIoJHt0aGlzLl9jb25maWcucGFyYW1ldGVyUGF0aH0pYDtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIENoZWNrIGlmIHRoaXMgbG9hZGVyIGlzIGF2YWlsYWJsZSBpbiB0aGUgY3VycmVudCBlbnZpcm9ubWVudC5cclxuICAgKiBAcmV0dXJucyBQcm9taXNlIHJlc29sdmluZyB0byB0cnVlIGlmIG5vdCBpbiBsb2NhbCBlbnZpcm9ubWVudCBhbmQgQVdTIGNyZWRlbnRpYWxzIGFyZSBhdmFpbGFibGVcclxuICAgKi9cclxuICBhc3luYyBpc0F2YWlsYWJsZSgpOiBQcm9taXNlPGJvb2xlYW4+IHtcclxuICAgIC8vIFNraXAgaW4gbG9jYWwgZW52aXJvbm1lbnRcclxuICAgIGlmICh0aGlzLl9hcHBFbnYgPT09ICdsb2NhbCcpIHtcclxuICAgICAgcmV0dXJuIGZhbHNlO1xyXG4gICAgfVxyXG5cclxuICAgIHRyeSB7XHJcbiAgICAgIC8vIFRlc3QgQVdTIGNyZWRlbnRpYWxzIGJ5IGF0dGVtcHRpbmcgdG8gZ2V0IGNhbGxlciBpZGVudGl0eVxyXG4gICAgICBhd2FpdCB0aGlzLl9jbGllbnQuY29uZmlnLmNyZWRlbnRpYWxzKCk7XHJcbiAgICAgIHJldHVybiB0cnVlO1xyXG4gICAgfSBjYXRjaCB7XHJcbiAgICAgIHJldHVybiBmYWxzZTtcclxuICAgIH1cclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIExvYWQgY29uZmlndXJhdGlvbiBmcm9tIEFXUyBTU00gUGFyYW1ldGVyIFN0b3JlLlxyXG4gICAqIEltcGxlbWVudHMgcmVjdXJzaXZlIHBhcmFtZXRlciBmZXRjaGluZyB3aXRoIE5leHRUb2tlbiBoYW5kbGluZyBmb3IgcGFnaW5hdGlvbi5cclxuICAgKiBAcmV0dXJucyBQcm9taXNlIHJlc29sdmluZyB0byBjb25maWd1cmF0aW9uIGtleS12YWx1ZSBwYWlycyBmcm9tIHBhcmFtZXRlcnNcclxuICAgKiBAdGhyb3dzIEFXU1NlcnZpY2VFcnJvciBpZiBBV1Mgb3BlcmF0aW9uIGZhaWxzXHJcbiAgICogQHRocm93cyBDb25maWd1cmF0aW9uTG9hZEVycm9yIGlmIHBhcmFtZXRlciBwYXRoIGNhbm5vdCBiZSBjb25zdHJ1Y3RlZFxyXG4gICAqL1xyXG4gIGFzeW5jIGxvYWQoKTogUHJvbWlzZTxSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4ge1xyXG4gICAgLy8gU2tpcCBsb2FkaW5nIGluIGxvY2FsIGVudmlyb25tZW50XHJcbiAgICBpZiAodGhpcy5fYXBwRW52ID09PSAnbG9jYWwnKSB7XHJcbiAgICAgIHJldHVybiB7fTtcclxuICAgIH1cclxuXHJcbiAgICBjb25zdCBwYXJhbWV0ZXJQYXRoID0gdGhpcy5idWlsZFBhcmFtZXRlclBhdGgoKTtcclxuICAgIGNvbnN0IHJlc3VsdDogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHt9O1xyXG4gICAgbGV0IG5leHRUb2tlbjogc3RyaW5nIHwgdW5kZWZpbmVkO1xyXG5cclxuICAgIHRyeSB7XHJcbiAgICAgIGRvIHtcclxuICAgICAgICBjb25zdCBjb21tYW5kID0gbmV3IEdldFBhcmFtZXRlcnNCeVBhdGhDb21tYW5kKHtcclxuICAgICAgICAgIFBhdGg6IHBhcmFtZXRlclBhdGgsXHJcbiAgICAgICAgICBSZWN1cnNpdmU6IHRydWUsXHJcbiAgICAgICAgICBXaXRoRGVjcnlwdGlvbjogdGhpcy5fY29uZmlnLndpdGhEZWNyeXB0aW9uLFxyXG4gICAgICAgICAgTmV4dFRva2VuOiBuZXh0VG9rZW4sXHJcbiAgICAgICAgfSk7XHJcblxyXG4gICAgICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgdGhpcy5fY2xpZW50LnNlbmQoY29tbWFuZCk7XHJcblxyXG4gICAgICAgIGlmICghcmVzcG9uc2UuUGFyYW1ldGVycykge1xyXG4gICAgICAgICAgLy8gTm8gcGFyYW1ldGVycyBmb3VuZCAtIHRoaXMgaXMgbm90IG5lY2Vzc2FyaWx5IGFuIGVycm9yXHJcbiAgICAgICAgICBicmVhaztcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIC8vIFByb2Nlc3MgZWFjaCBwYXJhbWV0ZXJcclxuICAgICAgICBmb3IgKGNvbnN0IHBhcmFtIG9mIHJlc3BvbnNlLlBhcmFtZXRlcnMpIHtcclxuICAgICAgICAgIGNvbnN0IGtleSA9IHRoaXMudHJhbnNmb3JtUGFyYW1ldGVyTmFtZShwYXJhbS5OYW1lLCBwYXJhbWV0ZXJQYXRoKTtcclxuXHJcbiAgICAgICAgICBpZiAoa2V5ICYmIHBhcmFtLlZhbHVlICE9PSB1bmRlZmluZWQpIHtcclxuICAgICAgICAgICAgcmVzdWx0W2tleV0gPSBwYXJhbS5WYWx1ZTtcclxuICAgICAgICAgIH1cclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIG5leHRUb2tlbiA9IHJlc3BvbnNlLk5leHRUb2tlbjtcclxuICAgICAgfSB3aGlsZSAobmV4dFRva2VuKTtcclxuXHJcbiAgICAgIHJldHVybiByZXN1bHQ7XHJcbiAgICB9IGNhdGNoIChlcnJvcikge1xyXG4gICAgICAvLyBIYW5kbGUgc3BlY2lmaWMgQVdTIGVycm9yc1xyXG4gICAgICBpZiAoZXJyb3IgaW5zdGFuY2VvZiBFcnJvcikge1xyXG4gICAgICAgIGlmIChlcnJvci5uYW1lID09PSAnUmVzb3VyY2VOb3RGb3VuZEV4Y2VwdGlvbicgfHwgZXJyb3IubmFtZSA9PT0gJ1BhcmFtZXRlck5vdEZvdW5kJykge1xyXG4gICAgICAgICAgLy8gTm8gcGFyYW1ldGVycyBmb3VuZCBhdCBwYXRoIC0gdGhpcyBpcyBub3QgbmVjZXNzYXJpbHkgYW4gZXJyb3JcclxuICAgICAgICAgIHJldHVybiB7fTtcclxuICAgICAgICB9XHJcblxyXG4gICAgICAgIGlmIChlcnJvci5uYW1lID09PSAnQWNjZXNzRGVuaWVkRXhjZXB0aW9uJykge1xyXG4gICAgICAgICAgdGhyb3cgbmV3IEFXU1NlcnZpY2VFcnJvcihcclxuICAgICAgICAgICAgYEFjY2VzcyBkZW5pZWQgd2hlbiByZXRyaWV2aW5nIHBhcmFtZXRlcnMgZnJvbSBwYXRoICcke3BhcmFtZXRlclBhdGh9Jy4gQ2hlY2sgQVdTIGNyZWRlbnRpYWxzIGFuZCBwZXJtaXNzaW9ucy5gLFxyXG4gICAgICAgICAgICAnU1NNJyxcclxuICAgICAgICAgICAgJ0dldFBhcmFtZXRlcnNCeVBhdGgnLFxyXG4gICAgICAgICAgICBlcnJvcixcclxuICAgICAgICAgICk7XHJcbiAgICAgICAgfVxyXG5cclxuICAgICAgICBpZiAoZXJyb3IubmFtZSA9PT0gJ0ludmFsaWRGaWx0ZXJLZXknIHx8IGVycm9yLm5hbWUgPT09ICdJbnZhbGlkRmlsdGVyVmFsdWUnKSB7XHJcbiAgICAgICAgICB0aHJvdyBuZXcgQVdTU2VydmljZUVycm9yKFxyXG4gICAgICAgICAgICBgSW52YWxpZCBwYXJhbWV0ZXIgcGF0aCAnJHtwYXJhbWV0ZXJQYXRofScuIENoZWNrIHBhdGggZm9ybWF0LmAsXHJcbiAgICAgICAgICAgICdTU00nLFxyXG4gICAgICAgICAgICAnR2V0UGFyYW1ldGVyc0J5UGF0aCcsXHJcbiAgICAgICAgICAgIGVycm9yLFxyXG4gICAgICAgICAgKTtcclxuICAgICAgICB9XHJcbiAgICAgIH1cclxuXHJcbiAgICAgIC8vIEZvciBvdGhlciBlcnJvcnMsIHdyYXAgaW4gQVdTU2VydmljZUVycm9yXHJcbiAgICAgIHRocm93IG5ldyBBV1NTZXJ2aWNlRXJyb3IoXHJcbiAgICAgICAgYEZhaWxlZCB0byByZXRyaWV2ZSBwYXJhbWV0ZXJzIGZyb20gcGF0aCAnJHtwYXJhbWV0ZXJQYXRofScgaW4gQVdTIFNTTSBQYXJhbWV0ZXIgU3RvcmU6ICR7ZXJyb3IgaW5zdGFuY2VvZiBFcnJvciA/IGVycm9yLm1lc3NhZ2UgOiBTdHJpbmcoZXJyb3IpfWAsXHJcbiAgICAgICAgJ1NTTScsXHJcbiAgICAgICAgJ0dldFBhcmFtZXRlcnNCeVBhdGgnLFxyXG4gICAgICAgIGVycm9yIGluc3RhbmNlb2YgRXJyb3IgPyBlcnJvciA6IHVuZGVmaW5lZCxcclxuICAgICAgKTtcclxuICAgIH1cclxuICB9XHJcblxyXG5cclxuICAvKipcclxuICAgKiBCdWlsZCB0aGUgZW52aXJvbm1lbnQtYXdhcmUgcGFyYW1ldGVyIHBhdGguXHJcbiAgICogQHJldHVybnMgVGhlIGZ1bGwgcGFyYW1ldGVyIHBhdGggd2l0aCBlbnZpcm9ubWVudCBwcmVmaXhcclxuICAgKiBAdGhyb3dzIENvbmZpZ3VyYXRpb25Mb2FkRXJyb3IgaWYgbm8gZW52aXJvbm1lbnQgbWFwcGluZyBmb3VuZFxyXG4gICAqL1xyXG4gIGJ1aWxkUGFyYW1ldGVyUGF0aCgpOiBzdHJpbmcge1xyXG4gICAgY29uc3QgZW52UHJlZml4ID0gdGhpcy5fY29uZmlnLmVudmlyb25tZW50TWFwcGluZ1t0aGlzLl9hcHBFbnZdO1xyXG5cclxuICAgIGlmICghZW52UHJlZml4KSB7XHJcbiAgICAgIHRocm93IG5ldyBDb25maWd1cmF0aW9uTG9hZEVycm9yKFxyXG4gICAgICAgIGBObyBlbnZpcm9ubWVudCBtYXBwaW5nIGZvdW5kIGZvciBBUFBfRU5WICcke3RoaXMuX2FwcEVudn0nLiBgICtcclxuICAgICAgICAgIGBBdmFpbGFibGUgZW52aXJvbm1lbnRzOiAke09iamVjdC5rZXlzKHRoaXMuX2NvbmZpZy5lbnZpcm9ubWVudE1hcHBpbmcpLmpvaW4oJywgJyl9YCxcclxuICAgICAgICB0aGlzLmdldE5hbWUoKSxcclxuICAgICAgKTtcclxuICAgIH1cclxuXHJcbiAgICByZXR1cm4gYC8ke2VudlByZWZpeH0ke3RoaXMuX2NvbmZpZy5wYXJhbWV0ZXJQYXRofWA7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBUcmFuc2Zvcm0gcGFyYW1ldGVyIG5hbWUgYnkgcmVtb3ZpbmcgdGhlIHByZWZpeCBhbmQgY29udmVydGluZyB0byB1cHBlcmNhc2UuXHJcbiAgICogRXhhbXBsZTogJy9kZXYvY29uZmlnLWF3cy9kYXRhYmFzZS9ob3N0JyAtPiAnREFUQUJBU0VfSE9TVCdcclxuICAgKiBAcGFyYW0gcGFyYW1ldGVyTmFtZSBUaGUgZnVsbCBwYXJhbWV0ZXIgbmFtZSBmcm9tIEFXU1xyXG4gICAqIEBwYXJhbSBwYXRoUHJlZml4IFRoZSBwYXRoIHByZWZpeCB0byByZW1vdmVcclxuICAgKiBAcmV0dXJucyBUaGUgdHJhbnNmb3JtZWQgcGFyYW1ldGVyIG5hbWUgb3IgbnVsbCBpZiBpbnZhbGlkXHJcbiAgICovXHJcbiAgcHJpdmF0ZSB0cmFuc2Zvcm1QYXJhbWV0ZXJOYW1lKHBhcmFtZXRlck5hbWU6IHN0cmluZyB8IHVuZGVmaW5lZCwgcGF0aFByZWZpeDogc3RyaW5nKTogc3RyaW5nIHwgbnVsbCB7XHJcbiAgICBpZiAoIXBhcmFtZXRlck5hbWUpIHtcclxuICAgICAgcmV0dXJuIG51bGw7XHJcbiAgICB9XHJcblxyXG4gICAgLy8gUmVtb3ZlIHRoZSBwYXRoIHByZWZpeFxyXG4gICAgbGV0IGtleSA9IHBhcmFtZXRlck5hbWU7XHJcbiAgICBpZiAoa2V5LnN0YXJ0c1dpdGgocGF0aFByZWZpeCkpIHtcclxuICAgICAga2V5ID0ga2V5LnN1YnN0cmluZyhwYXRoUHJlZml4Lmxlbmd0aCk7XHJcbiAgICB9XHJcblxyXG4gICAgLy8gUmVtb3ZlIGxlYWRpbmcgc2xhc2ggaWYgcHJlc2VudFxyXG4gICAgaWYgKGtleS5zdGFydHNXaXRoKCcvJykpIHtcclxuICAgICAga2V5ID0ga2V5LnN1YnN0cmluZygxKTtcclxuICAgIH1cclxuXHJcbiAgICAvLyBDb252ZXJ0IHNsYXNoZXMgdG8gdW5kZXJzY29yZXMgYW5kIHVwcGVyY2FzZVxyXG4gICAga2V5ID0ga2V5LnJlcGxhY2UoL1xcLy9nLCAnXycpLnRvVXBwZXJDYXNlKCk7XHJcblxyXG4gICAgLy8gUmV0dXJuIG51bGwgZm9yIGVtcHR5IGtleXNcclxuICAgIHJldHVybiBrZXkgfHwgbnVsbDtcclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIEdldCB0aGUgY3VycmVudCBhcHAgZW52aXJvbm1lbnQuXHJcbiAgICogQHJldHVybnMgVGhlIGN1cnJlbnQgQVBQX0VOViBvciBOT0RFX0VOViB2YWx1ZVxyXG4gICAqL1xyXG4gIGdldEFwcEVudigpOiBzdHJpbmcge1xyXG4gICAgcmV0dXJuIHRoaXMuX2FwcEVudjtcclxuICB9XHJcblxyXG4gIC8qKlxyXG4gICAqIEdldCB0aGUgZW52aXJvbm1lbnQgbWFwcGluZyBjb25maWd1cmF0aW9uLlxyXG4gICAqIEByZXR1cm5zIFRoZSBlbnZpcm9ubWVudCBtYXBwaW5nIHJlY29yZFxyXG4gICAqL1xyXG4gIGdldEVudmlyb25tZW50TWFwcGluZygpOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+IHtcclxuICAgIHJldHVybiB7IC4uLnRoaXMuX2NvbmZpZy5lbnZpcm9ubWVudE1hcHBpbmcgfTtcclxuICB9XHJcbn1cclxuIl19
|
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Parser for AWS ECS-compatible environment files.
|
|
3
|
+
*
|
|
4
|
+
* Format rules (per AWS ECS documentation):
|
|
5
|
+
* - Lines beginning with # are comments and ignored
|
|
6
|
+
* - Blank lines are ignored
|
|
7
|
+
* - Format: VARIABLE=VALUE (no spaces around =)
|
|
8
|
+
* - Variable names must match /^[a-zA-Z_][a-zA-Z0-9_]*$/
|
|
9
|
+
* - Values are literal (no quote processing, no interpolation)
|
|
10
|
+
* - One variable per line
|
|
11
|
+
* - Lines without = are ignored
|
|
12
|
+
* - Maximum line length: 32KB
|
|
13
|
+
*
|
|
14
|
+
* @see https://docs.aws.amazon.com/AmazonECS/latest/developerguide/use-environment-file.html
|
|
15
|
+
*/
|
|
16
|
+
export class EnvFileParser {
|
|
17
|
+
/** Maximum line length per AWS ECS specification */
|
|
18
|
+
static MAX_LINE_LENGTH = 32 * 1024; // 32KB
|
|
19
|
+
/** Valid variable name pattern: alphanumeric + underscore, cannot start with digit */
|
|
20
|
+
static VARIABLE_NAME_PATTERN = /^[a-zA-Z_][a-zA-Z0-9_]*$/;
|
|
21
|
+
/**
|
|
22
|
+
* Parse environment file content into key-value pairs.
|
|
23
|
+
* @param content The raw content of the environment file
|
|
24
|
+
* @returns Record of environment variable names to values
|
|
25
|
+
*/
|
|
26
|
+
static parse(content) {
|
|
27
|
+
const result = {};
|
|
28
|
+
// Handle empty content
|
|
29
|
+
if (!content) {
|
|
30
|
+
return result;
|
|
31
|
+
}
|
|
32
|
+
const lines = content.split(/\r?\n/);
|
|
33
|
+
for (const line of lines) {
|
|
34
|
+
// Skip lines exceeding max length
|
|
35
|
+
if (line.length > this.MAX_LINE_LENGTH) {
|
|
36
|
+
continue;
|
|
37
|
+
}
|
|
38
|
+
// Skip blank lines
|
|
39
|
+
if (line.trim() === '') {
|
|
40
|
+
continue;
|
|
41
|
+
}
|
|
42
|
+
// Skip comment lines (lines starting with # after optional whitespace)
|
|
43
|
+
if (line.trimStart().startsWith('#')) {
|
|
44
|
+
continue;
|
|
45
|
+
}
|
|
46
|
+
// Find the first = sign
|
|
47
|
+
const equalsIndex = line.indexOf('=');
|
|
48
|
+
if (equalsIndex === -1) {
|
|
49
|
+
// Lines without = are ignored
|
|
50
|
+
continue;
|
|
51
|
+
}
|
|
52
|
+
const key = line.substring(0, equalsIndex);
|
|
53
|
+
const value = line.substring(equalsIndex + 1);
|
|
54
|
+
// Validate variable name (must not be empty and must match pattern)
|
|
55
|
+
if (key.length === 0 || !this.VARIABLE_NAME_PATTERN.test(key)) {
|
|
56
|
+
continue;
|
|
57
|
+
}
|
|
58
|
+
result[key] = value;
|
|
59
|
+
}
|
|
60
|
+
return result;
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Check if a variable name is valid per AWS ECS format.
|
|
64
|
+
* Variable names must:
|
|
65
|
+
* - Start with a letter (a-z, A-Z) or underscore (_)
|
|
66
|
+
* - Contain only alphanumeric characters and underscores
|
|
67
|
+
* - Not be empty
|
|
68
|
+
*
|
|
69
|
+
* @param name The variable name to check
|
|
70
|
+
* @returns true if the name is valid
|
|
71
|
+
*/
|
|
72
|
+
static isValidVariableName(name) {
|
|
73
|
+
if (!name || name.length === 0) {
|
|
74
|
+
return false;
|
|
75
|
+
}
|
|
76
|
+
return this.VARIABLE_NAME_PATTERN.test(name);
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Serialize a configuration object to AWS ECS-compatible env file format.
|
|
80
|
+
* @param config The configuration object to serialize
|
|
81
|
+
* @returns The serialized env file content
|
|
82
|
+
*/
|
|
83
|
+
static serialize(config) {
|
|
84
|
+
const lines = [];
|
|
85
|
+
for (const [key, value] of Object.entries(config)) {
|
|
86
|
+
// Only include valid variable names
|
|
87
|
+
if (this.isValidVariableName(key)) {
|
|
88
|
+
lines.push(`${key}=${value}`);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
return lines.join('\n');
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZW52LWZpbGUtcGFyc2VyLnV0aWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9jb25maWctYXdzL3NyYy91dGlscy9lbnYtZmlsZS1wYXJzZXIudXRpbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7Ozs7Ozs7R0FjRztBQUNILE1BQU0sT0FBTyxhQUFhO0lBQ3hCLG9EQUFvRDtJQUM1QyxNQUFNLENBQVUsZUFBZSxHQUFHLEVBQUUsR0FBRyxJQUFJLENBQUMsQ0FBQyxPQUFPO0lBRTVELHNGQUFzRjtJQUM5RSxNQUFNLENBQVUscUJBQXFCLEdBQUcsMEJBQTBCLENBQUM7SUFFM0U7Ozs7T0FJRztJQUNILE1BQU0sQ0FBQyxLQUFLLENBQUMsT0FBZTtRQUMxQixNQUFNLE1BQU0sR0FBMkIsRUFBRSxDQUFDO1FBRTFDLHVCQUF1QjtRQUN2QixJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7WUFDYixPQUFPLE1BQU0sQ0FBQztRQUNoQixDQUFDO1FBRUQsTUFBTSxLQUFLLEdBQUcsT0FBTyxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUVyQyxLQUFLLE1BQU0sSUFBSSxJQUFJLEtBQUssRUFBRSxDQUFDO1lBQ3pCLGtDQUFrQztZQUNsQyxJQUFJLElBQUksQ0FBQyxNQUFNLEdBQUcsSUFBSSxDQUFDLGVBQWUsRUFBRSxDQUFDO2dCQUN2QyxTQUFTO1lBQ1gsQ0FBQztZQUVELG1CQUFtQjtZQUNuQixJQUFJLElBQUksQ0FBQyxJQUFJLEVBQUUsS0FBSyxFQUFFLEVBQUUsQ0FBQztnQkFDdkIsU0FBUztZQUNYLENBQUM7WUFFRCx1RUFBdUU7WUFDdkUsSUFBSSxJQUFJLENBQUMsU0FBUyxFQUFFLENBQUMsVUFBVSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUM7Z0JBQ3JDLFNBQVM7WUFDWCxDQUFDO1lBRUQsd0JBQXdCO1lBQ3hCLE1BQU0sV0FBVyxHQUFHLElBQUksQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLENBQUM7WUFDdEMsSUFBSSxXQUFXLEtBQUssQ0FBQyxDQUFDLEVBQUUsQ0FBQztnQkFDdkIsOEJBQThCO2dCQUM5QixTQUFTO1lBQ1gsQ0FBQztZQUVELE1BQU0sR0FBRyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsQ0FBQyxFQUFFLFdBQVcsQ0FBQyxDQUFDO1lBQzNDLE1BQU0sS0FBSyxHQUFHLElBQUksQ0FBQyxTQUFTLENBQUMsV0FBVyxHQUFHLENBQUMsQ0FBQyxDQUFDO1lBRTlDLG9FQUFvRTtZQUNwRSxJQUFJLEdBQUcsQ0FBQyxNQUFNLEtBQUssQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLHFCQUFxQixDQUFDLElBQUksQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDO2dCQUM5RCxTQUFTO1lBQ1gsQ0FBQztZQUVELE1BQU0sQ0FBQyxHQUFHLENBQUMsR0FBRyxLQUFLLENBQUM7UUFDdEIsQ0FBQztRQUVELE9BQU8sTUFBTSxDQUFDO0lBQ2hCLENBQUM7SUFFRDs7Ozs7Ozs7O09BU0c7SUFDSCxNQUFNLENBQUMsbUJBQW1CLENBQUMsSUFBWTtRQUNyQyxJQUFJLENBQUMsSUFBSSxJQUFJLElBQUksQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFLENBQUM7WUFDL0IsT0FBTyxLQUFLLENBQUM7UUFDZixDQUFDO1FBQ0QsT0FBTyxJQUFJLENBQUMscUJBQXFCLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQy9DLENBQUM7SUFFRDs7OztPQUlHO0lBQ0gsTUFBTSxDQUFDLFNBQVMsQ0FBQyxNQUE4QjtRQUM3QyxNQUFNLEtBQUssR0FBYSxFQUFFLENBQUM7UUFFM0IsS0FBSyxNQUFNLENBQUMsR0FBRyxFQUFFLEtBQUssQ0FBQyxJQUFJLE1BQU0sQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQztZQUNsRCxvQ0FBb0M7WUFDcEMsSUFBSSxJQUFJLENBQUMsbUJBQW1CLENBQUMsR0FBRyxDQUFDLEVBQUUsQ0FBQztnQkFDbEMsS0FBSyxDQUFDLElBQUksQ0FBQyxHQUFHLEdBQUcsSUFBSSxLQUFLLEVBQUUsQ0FBQyxDQUFDO1lBQ2hDLENBQUM7UUFDSCxDQUFDO1FBRUQsT0FBTyxLQUFLLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQzFCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcclxuICogUGFyc2VyIGZvciBBV1MgRUNTLWNvbXBhdGlibGUgZW52aXJvbm1lbnQgZmlsZXMuXHJcbiAqXHJcbiAqIEZvcm1hdCBydWxlcyAocGVyIEFXUyBFQ1MgZG9jdW1lbnRhdGlvbik6XHJcbiAqIC0gTGluZXMgYmVnaW5uaW5nIHdpdGggIyBhcmUgY29tbWVudHMgYW5kIGlnbm9yZWRcclxuICogLSBCbGFuayBsaW5lcyBhcmUgaWdub3JlZFxyXG4gKiAtIEZvcm1hdDogVkFSSUFCTEU9VkFMVUUgKG5vIHNwYWNlcyBhcm91bmQgPSlcclxuICogLSBWYXJpYWJsZSBuYW1lcyBtdXN0IG1hdGNoIC9eW2EtekEtWl9dW2EtekEtWjAtOV9dKiQvXHJcbiAqIC0gVmFsdWVzIGFyZSBsaXRlcmFsIChubyBxdW90ZSBwcm9jZXNzaW5nLCBubyBpbnRlcnBvbGF0aW9uKVxyXG4gKiAtIE9uZSB2YXJpYWJsZSBwZXIgbGluZVxyXG4gKiAtIExpbmVzIHdpdGhvdXQgPSBhcmUgaWdub3JlZFxyXG4gKiAtIE1heGltdW0gbGluZSBsZW5ndGg6IDMyS0JcclxuICpcclxuICogQHNlZSBodHRwczovL2RvY3MuYXdzLmFtYXpvbi5jb20vQW1hem9uRUNTL2xhdGVzdC9kZXZlbG9wZXJndWlkZS91c2UtZW52aXJvbm1lbnQtZmlsZS5odG1sXHJcbiAqL1xyXG5leHBvcnQgY2xhc3MgRW52RmlsZVBhcnNlciB7XHJcbiAgLyoqIE1heGltdW0gbGluZSBsZW5ndGggcGVyIEFXUyBFQ1Mgc3BlY2lmaWNhdGlvbiAqL1xyXG4gIHByaXZhdGUgc3RhdGljIHJlYWRvbmx5IE1BWF9MSU5FX0xFTkdUSCA9IDMyICogMTAyNDsgLy8gMzJLQlxyXG5cclxuICAvKiogVmFsaWQgdmFyaWFibGUgbmFtZSBwYXR0ZXJuOiBhbHBoYW51bWVyaWMgKyB1bmRlcnNjb3JlLCBjYW5ub3Qgc3RhcnQgd2l0aCBkaWdpdCAqL1xyXG4gIHByaXZhdGUgc3RhdGljIHJlYWRvbmx5IFZBUklBQkxFX05BTUVfUEFUVEVSTiA9IC9eW2EtekEtWl9dW2EtekEtWjAtOV9dKiQvO1xyXG5cclxuICAvKipcclxuICAgKiBQYXJzZSBlbnZpcm9ubWVudCBmaWxlIGNvbnRlbnQgaW50byBrZXktdmFsdWUgcGFpcnMuXHJcbiAgICogQHBhcmFtIGNvbnRlbnQgVGhlIHJhdyBjb250ZW50IG9mIHRoZSBlbnZpcm9ubWVudCBmaWxlXHJcbiAgICogQHJldHVybnMgUmVjb3JkIG9mIGVudmlyb25tZW50IHZhcmlhYmxlIG5hbWVzIHRvIHZhbHVlc1xyXG4gICAqL1xyXG4gIHN0YXRpYyBwYXJzZShjb250ZW50OiBzdHJpbmcpOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+IHtcclxuICAgIGNvbnN0IHJlc3VsdDogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHt9O1xyXG5cclxuICAgIC8vIEhhbmRsZSBlbXB0eSBjb250ZW50XHJcbiAgICBpZiAoIWNvbnRlbnQpIHtcclxuICAgICAgcmV0dXJuIHJlc3VsdDtcclxuICAgIH1cclxuXHJcbiAgICBjb25zdCBsaW5lcyA9IGNvbnRlbnQuc3BsaXQoL1xccj9cXG4vKTtcclxuXHJcbiAgICBmb3IgKGNvbnN0IGxpbmUgb2YgbGluZXMpIHtcclxuICAgICAgLy8gU2tpcCBsaW5lcyBleGNlZWRpbmcgbWF4IGxlbmd0aFxyXG4gICAgICBpZiAobGluZS5sZW5ndGggPiB0aGlzLk1BWF9MSU5FX0xFTkdUSCkge1xyXG4gICAgICAgIGNvbnRpbnVlO1xyXG4gICAgICB9XHJcblxyXG4gICAgICAvLyBTa2lwIGJsYW5rIGxpbmVzXHJcbiAgICAgIGlmIChsaW5lLnRyaW0oKSA9PT0gJycpIHtcclxuICAgICAgICBjb250aW51ZTtcclxuICAgICAgfVxyXG5cclxuICAgICAgLy8gU2tpcCBjb21tZW50IGxpbmVzIChsaW5lcyBzdGFydGluZyB3aXRoICMgYWZ0ZXIgb3B0aW9uYWwgd2hpdGVzcGFjZSlcclxuICAgICAgaWYgKGxpbmUudHJpbVN0YXJ0KCkuc3RhcnRzV2l0aCgnIycpKSB7XHJcbiAgICAgICAgY29udGludWU7XHJcbiAgICAgIH1cclxuXHJcbiAgICAgIC8vIEZpbmQgdGhlIGZpcnN0ID0gc2lnblxyXG4gICAgICBjb25zdCBlcXVhbHNJbmRleCA9IGxpbmUuaW5kZXhPZignPScpO1xyXG4gICAgICBpZiAoZXF1YWxzSW5kZXggPT09IC0xKSB7XHJcbiAgICAgICAgLy8gTGluZXMgd2l0aG91dCA9IGFyZSBpZ25vcmVkXHJcbiAgICAgICAgY29udGludWU7XHJcbiAgICAgIH1cclxuXHJcbiAgICAgIGNvbnN0IGtleSA9IGxpbmUuc3Vic3RyaW5nKDAsIGVxdWFsc0luZGV4KTtcclxuICAgICAgY29uc3QgdmFsdWUgPSBsaW5lLnN1YnN0cmluZyhlcXVhbHNJbmRleCArIDEpO1xyXG5cclxuICAgICAgLy8gVmFsaWRhdGUgdmFyaWFibGUgbmFtZSAobXVzdCBub3QgYmUgZW1wdHkgYW5kIG11c3QgbWF0Y2ggcGF0dGVybilcclxuICAgICAgaWYgKGtleS5sZW5ndGggPT09IDAgfHwgIXRoaXMuVkFSSUFCTEVfTkFNRV9QQVRURVJOLnRlc3Qoa2V5KSkge1xyXG4gICAgICAgIGNvbnRpbnVlO1xyXG4gICAgICB9XHJcblxyXG4gICAgICByZXN1bHRba2V5XSA9IHZhbHVlO1xyXG4gICAgfVxyXG5cclxuICAgIHJldHVybiByZXN1bHQ7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBDaGVjayBpZiBhIHZhcmlhYmxlIG5hbWUgaXMgdmFsaWQgcGVyIEFXUyBFQ1MgZm9ybWF0LlxyXG4gICAqIFZhcmlhYmxlIG5hbWVzIG11c3Q6XHJcbiAgICogLSBTdGFydCB3aXRoIGEgbGV0dGVyIChhLXosIEEtWikgb3IgdW5kZXJzY29yZSAoXylcclxuICAgKiAtIENvbnRhaW4gb25seSBhbHBoYW51bWVyaWMgY2hhcmFjdGVycyBhbmQgdW5kZXJzY29yZXNcclxuICAgKiAtIE5vdCBiZSBlbXB0eVxyXG4gICAqXHJcbiAgICogQHBhcmFtIG5hbWUgVGhlIHZhcmlhYmxlIG5hbWUgdG8gY2hlY2tcclxuICAgKiBAcmV0dXJucyB0cnVlIGlmIHRoZSBuYW1lIGlzIHZhbGlkXHJcbiAgICovXHJcbiAgc3RhdGljIGlzVmFsaWRWYXJpYWJsZU5hbWUobmFtZTogc3RyaW5nKTogYm9vbGVhbiB7XHJcbiAgICBpZiAoIW5hbWUgfHwgbmFtZS5sZW5ndGggPT09IDApIHtcclxuICAgICAgcmV0dXJuIGZhbHNlO1xyXG4gICAgfVxyXG4gICAgcmV0dXJuIHRoaXMuVkFSSUFCTEVfTkFNRV9QQVRURVJOLnRlc3QobmFtZSk7XHJcbiAgfVxyXG5cclxuICAvKipcclxuICAgKiBTZXJpYWxpemUgYSBjb25maWd1cmF0aW9uIG9iamVjdCB0byBBV1MgRUNTLWNvbXBhdGlibGUgZW52IGZpbGUgZm9ybWF0LlxyXG4gICAqIEBwYXJhbSBjb25maWcgVGhlIGNvbmZpZ3VyYXRpb24gb2JqZWN0IHRvIHNlcmlhbGl6ZVxyXG4gICAqIEByZXR1cm5zIFRoZSBzZXJpYWxpemVkIGVudiBmaWxlIGNvbnRlbnRcclxuICAgKi9cclxuICBzdGF0aWMgc2VyaWFsaXplKGNvbmZpZzogUmVjb3JkPHN0cmluZywgc3RyaW5nPik6IHN0cmluZyB7XHJcbiAgICBjb25zdCBsaW5lczogc3RyaW5nW10gPSBbXTtcclxuXHJcbiAgICBmb3IgKGNvbnN0IFtrZXksIHZhbHVlXSBvZiBPYmplY3QuZW50cmllcyhjb25maWcpKSB7XHJcbiAgICAgIC8vIE9ubHkgaW5jbHVkZSB2YWxpZCB2YXJpYWJsZSBuYW1lc1xyXG4gICAgICBpZiAodGhpcy5pc1ZhbGlkVmFyaWFibGVOYW1lKGtleSkpIHtcclxuICAgICAgICBsaW5lcy5wdXNoKGAke2tleX09JHt2YWx1ZX1gKTtcclxuICAgICAgfVxyXG4gICAgfVxyXG5cclxuICAgIHJldHVybiBsaW5lcy5qb2luKCdcXG4nKTtcclxuICB9XHJcbn1cclxuIl19
|