aws-cdk 0.0.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/CONTRIBUTING.md +276 -0
- package/LICENSE +202 -0
- package/NOTICE +16 -0
- package/README.md +1274 -0
- package/THIRD_PARTY_LICENSES +26821 -0
- package/bin/cdk +6 -0
- package/build-info.json +4 -0
- package/db.json.gz +0 -0
- package/generate.sh +25 -0
- package/images/garbage-collection.png +0 -0
- package/lib/api/aws-auth/account-cache.d.ts +39 -0
- package/lib/api/aws-auth/account-cache.js +102 -0
- package/lib/api/aws-auth/awscli-compatible.d.ts +42 -0
- package/lib/api/aws-auth/awscli-compatible.js +264 -0
- package/lib/api/aws-auth/cached.d.ts +11 -0
- package/lib/api/aws-auth/cached.js +26 -0
- package/lib/api/aws-auth/credential-plugins.d.ts +36 -0
- package/lib/api/aws-auth/credential-plugins.js +153 -0
- package/lib/api/aws-auth/index.d.ts +3 -0
- package/lib/api/aws-auth/index.js +20 -0
- package/lib/api/aws-auth/provider-caching.d.ts +13 -0
- package/lib/api/aws-auth/provider-caching.js +24 -0
- package/lib/api/aws-auth/sdk-logger.d.ts +69 -0
- package/lib/api/aws-auth/sdk-logger.js +130 -0
- package/lib/api/aws-auth/sdk-provider.d.ts +207 -0
- package/lib/api/aws-auth/sdk-provider.js +359 -0
- package/lib/api/aws-auth/sdk.d.ts +223 -0
- package/lib/api/aws-auth/sdk.js +366 -0
- package/lib/api/aws-auth/tracing.d.ts +11 -0
- package/lib/api/aws-auth/tracing.js +61 -0
- package/lib/api/aws-auth/user-agent.d.ts +7 -0
- package/lib/api/aws-auth/user-agent.js +21 -0
- package/lib/api/aws-auth/util.d.ts +6 -0
- package/lib/api/aws-auth/util.js +23 -0
- package/lib/api/bootstrap/bootstrap-environment.d.ts +33 -0
- package/lib/api/bootstrap/bootstrap-environment.js +322 -0
- package/lib/api/bootstrap/bootstrap-props.d.ts +130 -0
- package/lib/api/bootstrap/bootstrap-props.js +14 -0
- package/lib/api/bootstrap/bootstrap-template.yaml +692 -0
- package/lib/api/bootstrap/deploy-bootstrap.d.ts +37 -0
- package/lib/api/bootstrap/deploy-bootstrap.js +143 -0
- package/lib/api/bootstrap/index.d.ts +2 -0
- package/lib/api/bootstrap/index.js +19 -0
- package/lib/api/bootstrap/legacy-template.d.ts +2 -0
- package/lib/api/bootstrap/legacy-template.js +82 -0
- package/lib/api/context.d.ts +40 -0
- package/lib/api/context.js +82 -0
- package/lib/api/cxapp/cloud-assembly.d.ts +150 -0
- package/lib/api/cxapp/cloud-assembly.js +305 -0
- package/lib/api/cxapp/cloud-executable.d.ts +44 -0
- package/lib/api/cxapp/cloud-executable.js +90 -0
- package/lib/api/cxapp/environments.d.ts +9 -0
- package/lib/api/cxapp/environments.js +66 -0
- package/lib/api/cxapp/exec.d.ts +56 -0
- package/lib/api/cxapp/exec.js +275 -0
- package/lib/api/deployments/asset-manifest-builder.d.ts +8 -0
- package/lib/api/deployments/asset-manifest-builder.js +35 -0
- package/lib/api/deployments/asset-publishing.d.ts +77 -0
- package/lib/api/deployments/asset-publishing.js +163 -0
- package/lib/api/deployments/assets.d.ts +10 -0
- package/lib/api/deployments/assets.js +111 -0
- package/lib/api/deployments/checks.d.ts +8 -0
- package/lib/api/deployments/checks.js +73 -0
- package/lib/api/deployments/cloudformation.d.ts +235 -0
- package/lib/api/deployments/cloudformation.js +598 -0
- package/lib/api/deployments/deploy-stack.d.ts +177 -0
- package/lib/api/deployments/deploy-stack.js +484 -0
- package/lib/api/deployments/deployment-method.d.ts +24 -0
- package/lib/api/deployments/deployment-method.js +3 -0
- package/lib/api/deployments/deployment-result.d.ts +21 -0
- package/lib/api/deployments/deployment-result.js +10 -0
- package/lib/api/deployments/deployments.d.ts +340 -0
- package/lib/api/deployments/deployments.js +369 -0
- package/lib/api/deployments/hotswap-deployments.d.ts +14 -0
- package/lib/api/deployments/hotswap-deployments.js +357 -0
- package/lib/api/deployments/index.d.ts +6 -0
- package/lib/api/deployments/index.js +23 -0
- package/lib/api/deployments/nested-stack-helpers.d.ts +25 -0
- package/lib/api/deployments/nested-stack-helpers.js +88 -0
- package/lib/api/environment-access.d.ts +138 -0
- package/lib/api/environment-access.js +203 -0
- package/lib/api/environment-resources.d.ts +73 -0
- package/lib/api/environment-resources.js +208 -0
- package/lib/api/evaluate-cloudformation-template.d.ts +84 -0
- package/lib/api/evaluate-cloudformation-template.js +443 -0
- package/lib/api/garbage-collection/garbage-collector.d.ts +152 -0
- package/lib/api/garbage-collection/garbage-collector.js +607 -0
- package/lib/api/garbage-collection/progress-printer.d.ts +21 -0
- package/lib/api/garbage-collection/progress-printer.js +69 -0
- package/lib/api/garbage-collection/stack-refresh.d.ts +44 -0
- package/lib/api/garbage-collection/stack-refresh.js +154 -0
- package/lib/api/hotswap/appsync-mapping-templates.d.ts +3 -0
- package/lib/api/hotswap/appsync-mapping-templates.js +157 -0
- package/lib/api/hotswap/code-build-projects.d.ts +3 -0
- package/lib/api/hotswap/code-build-projects.js +55 -0
- package/lib/api/hotswap/common.d.ts +126 -0
- package/lib/api/hotswap/common.js +170 -0
- package/lib/api/hotswap/ecs-services.d.ts +3 -0
- package/lib/api/hotswap/ecs-services.js +140 -0
- package/lib/api/hotswap/lambda-functions.d.ts +3 -0
- package/lib/api/hotswap/lambda-functions.js +309 -0
- package/lib/api/hotswap/s3-bucket-deployments.d.ts +9 -0
- package/lib/api/hotswap/s3-bucket-deployments.js +112 -0
- package/lib/api/hotswap/stepfunctions-state-machines.d.ts +3 -0
- package/lib/api/hotswap/stepfunctions-state-machines.js +42 -0
- package/lib/api/index.d.ts +5 -0
- package/lib/api/index.js +22 -0
- package/lib/api/logs/find-cloudwatch-logs.d.ts +24 -0
- package/lib/api/logs/find-cloudwatch-logs.js +97 -0
- package/lib/api/logs/logs-monitor.d.ts +53 -0
- package/lib/api/logs/logs-monitor.js +169 -0
- package/lib/api/plugin/context-provider-plugin.d.ts +6 -0
- package/lib/api/plugin/context-provider-plugin.js +7 -0
- package/lib/api/plugin/index.d.ts +3 -0
- package/lib/api/plugin/index.js +20 -0
- package/lib/api/plugin/mode.d.ts +4 -0
- package/lib/api/plugin/mode.js +9 -0
- package/lib/api/plugin/plugin.d.ts +63 -0
- package/lib/api/plugin/plugin.js +106 -0
- package/lib/api/settings.d.ts +29 -0
- package/lib/api/settings.js +141 -0
- package/lib/api/tags.d.ts +9 -0
- package/lib/api/tags.js +10 -0
- package/lib/api/toolkit-info.d.ts +51 -0
- package/lib/api/toolkit-info.js +156 -0
- package/lib/api/util/cloudformation/stack-activity-monitor.d.ts +237 -0
- package/lib/api/util/cloudformation/stack-activity-monitor.js +550 -0
- package/lib/api/util/cloudformation/stack-event-poller.d.ts +63 -0
- package/lib/api/util/cloudformation/stack-event-poller.js +129 -0
- package/lib/api/util/cloudformation/stack-status.d.ts +42 -0
- package/lib/api/util/cloudformation/stack-status.js +88 -0
- package/lib/api/util/display.d.ts +13 -0
- package/lib/api/util/display.js +80 -0
- package/lib/api/util/placeholders.d.ts +10 -0
- package/lib/api/util/placeholders.js +24 -0
- package/lib/api/util/rwlock.d.ts +65 -0
- package/lib/api/util/rwlock.js +179 -0
- package/lib/api/util/string-manipulation.d.ts +10 -0
- package/lib/api/util/string-manipulation.js +33 -0
- package/lib/api/util/template-body-parameter.d.ts +21 -0
- package/lib/api/util/template-body-parameter.js +104 -0
- package/lib/cli/cdk-toolkit.d.ts +594 -0
- package/lib/cli/cdk-toolkit.js +1019 -0
- package/lib/cli/cli-config.d.ts +10 -0
- package/lib/cli/cli-config.js +406 -0
- package/lib/cli/cli.d.ts +4 -0
- package/lib/cli/cli.js +538 -0
- package/lib/cli/convert-to-user-input.d.ts +3 -0
- package/lib/cli/convert-to-user-input.js +434 -0
- package/lib/cli/parse-command-line-arguments.d.ts +1 -0
- package/lib/cli/parse-command-line-arguments.js +806 -0
- package/lib/cli/platform-warnings.d.ts +2 -0
- package/lib/cli/platform-warnings.js +45 -0
- package/lib/cli/user-configuration.d.ts +90 -0
- package/lib/cli/user-configuration.js +272 -0
- package/lib/cli/user-input.d.ts +1163 -0
- package/lib/cli/user-input.js +3 -0
- package/lib/cli/util/console-formatters.d.ts +18 -0
- package/lib/cli/util/console-formatters.js +42 -0
- package/lib/cli/util/npm.d.ts +1 -0
- package/lib/cli/util/npm.js +22 -0
- package/lib/cli/util/yargs-helpers.d.ts +22 -0
- package/lib/cli/util/yargs-helpers.js +49 -0
- package/lib/cli/version.d.ts +13 -0
- package/lib/cli/version.js +120 -0
- package/lib/commands/context.d.ts +35 -0
- package/lib/commands/context.js +156 -0
- package/lib/commands/docs.d.ts +13 -0
- package/lib/commands/docs.js +32 -0
- package/lib/commands/doctor.d.ts +1 -0
- package/lib/commands/doctor.js +69 -0
- package/lib/commands/migrate.d.ts +327 -0
- package/lib/commands/migrate.js +804 -0
- package/lib/context-providers/ami.d.ts +11 -0
- package/lib/context-providers/ami.js +50 -0
- package/lib/context-providers/availability-zones.d.ts +11 -0
- package/lib/context-providers/availability-zones.js +27 -0
- package/lib/context-providers/endpoint-service-availability-zones.d.ts +11 -0
- package/lib/context-providers/endpoint-service-availability-zones.js +33 -0
- package/lib/context-providers/hosted-zones.d.ts +10 -0
- package/lib/context-providers/hosted-zones.js +67 -0
- package/lib/context-providers/index.d.ts +30 -0
- package/lib/context-providers/index.js +109 -0
- package/lib/context-providers/keys.d.ts +11 -0
- package/lib/context-providers/keys.js +52 -0
- package/lib/context-providers/load-balancers.d.ts +20 -0
- package/lib/context-providers/load-balancers.js +159 -0
- package/lib/context-providers/security-groups.d.ts +9 -0
- package/lib/context-providers/security-groups.js +70 -0
- package/lib/context-providers/ssm-parameters.d.ts +23 -0
- package/lib/context-providers/ssm-parameters.js +59 -0
- package/lib/context-providers/vpcs.d.ts +11 -0
- package/lib/context-providers/vpcs.js +288 -0
- package/lib/diff.d.ts +28 -0
- package/lib/diff.js +165 -0
- package/lib/import.d.ts +182 -0
- package/lib/import.js +335 -0
- package/lib/index.d.ts +3 -0
- package/lib/index.js +348304 -0
- package/lib/index_bg.wasm +0 -0
- package/lib/init-hooks.d.ts +40 -0
- package/lib/init-hooks.js +64 -0
- package/lib/init-templates/.init-version.json +1 -0
- package/lib/init-templates/.no-packagejson-validator +0 -0
- package/lib/init-templates/.recommended-feature-flags.json +68 -0
- package/lib/init-templates/LICENSE +16 -0
- package/lib/init-templates/app/csharp/.template.gitignore +342 -0
- package/lib/init-templates/app/csharp/README.md +14 -0
- package/lib/init-templates/app/csharp/cdk.template.json +15 -0
- package/lib/init-templates/app/csharp/src/%name.PascalCased%/%name.PascalCased%.template.csproj +20 -0
- package/lib/init-templates/app/csharp/src/%name.PascalCased%/%name.PascalCased%Stack.template.cs +13 -0
- package/lib/init-templates/app/csharp/src/%name.PascalCased%/GlobalSuppressions.cs +1 -0
- package/lib/init-templates/app/csharp/src/%name.PascalCased%/Program.template.cs +44 -0
- package/lib/init-templates/app/csharp/src/%name.PascalCased%.template.sln +18 -0
- package/lib/init-templates/app/fsharp/.template.gitignore +342 -0
- package/lib/init-templates/app/fsharp/README.md +18 -0
- package/lib/init-templates/app/fsharp/cdk.template.json +14 -0
- package/lib/init-templates/app/fsharp/src/%name.PascalCased%/%name.PascalCased%.template.fsproj +25 -0
- package/lib/init-templates/app/fsharp/src/%name.PascalCased%/%name.PascalCased%Stack.template.fs +8 -0
- package/lib/init-templates/app/fsharp/src/%name.PascalCased%/Program.template.fs +11 -0
- package/lib/init-templates/app/fsharp/src/%name.PascalCased%.template.sln +18 -0
- package/lib/init-templates/app/go/%name%.template.go +70 -0
- package/lib/init-templates/app/go/%name%_test.template.go +26 -0
- package/lib/init-templates/app/go/.template.gitignore +19 -0
- package/lib/init-templates/app/go/README.md +12 -0
- package/lib/init-templates/app/go/cdk.template.json +13 -0
- package/lib/init-templates/app/go/go.template.mod +9 -0
- package/lib/init-templates/app/info.json +4 -0
- package/lib/init-templates/app/java/.template.gitignore +13 -0
- package/lib/init-templates/app/java/README.md +18 -0
- package/lib/init-templates/app/java/cdk.json +13 -0
- package/lib/init-templates/app/java/pom.xml +60 -0
- package/lib/init-templates/app/java/src/main/java/com/myorg/%name.PascalCased%App.template.java +42 -0
- package/lib/init-templates/app/java/src/main/java/com/myorg/%name.PascalCased%Stack.template.java +24 -0
- package/lib/init-templates/app/java/src/test/java/com/myorg/%name.PascalCased%Test.template.java +26 -0
- package/lib/init-templates/app/javascript/.template.gitignore +5 -0
- package/lib/init-templates/app/javascript/.template.npmignore +3 -0
- package/lib/init-templates/app/javascript/README.md +12 -0
- package/lib/init-templates/app/javascript/bin/%name%.template.js +21 -0
- package/lib/init-templates/app/javascript/cdk.template.json +15 -0
- package/lib/init-templates/app/javascript/jest.config.js +3 -0
- package/lib/init-templates/app/javascript/lib/%name%-stack.template.js +23 -0
- package/lib/init-templates/app/javascript/package.json +20 -0
- package/lib/init-templates/app/javascript/test/%name%.test.template.js +17 -0
- package/lib/init-templates/app/python/%name.PythonModule%/%name.PythonModule%_stack.template.py +19 -0
- package/lib/init-templates/app/python/%name.PythonModule%/__init__.py +0 -0
- package/lib/init-templates/app/python/.template.gitignore +10 -0
- package/lib/init-templates/app/python/README.template.md +58 -0
- package/lib/init-templates/app/python/app.template.py +28 -0
- package/lib/init-templates/app/python/cdk.template.json +15 -0
- package/lib/init-templates/app/python/requirements-dev.txt +1 -0
- package/lib/init-templates/app/python/requirements.txt +2 -0
- package/lib/init-templates/app/python/source.bat +13 -0
- package/lib/init-templates/app/python/tests/__init__.py +0 -0
- package/lib/init-templates/app/python/tests/unit/__init__.py +0 -0
- package/lib/init-templates/app/python/tests/unit/test_%name.PythonModule%_stack.template.py +15 -0
- package/lib/init-templates/app/typescript/.template.gitignore +8 -0
- package/lib/init-templates/app/typescript/.template.npmignore +6 -0
- package/lib/init-templates/app/typescript/README.md +14 -0
- package/lib/init-templates/app/typescript/bin/%name%.template.ts +20 -0
- package/lib/init-templates/app/typescript/cdk.template.json +17 -0
- package/lib/init-templates/app/typescript/jest.config.js +8 -0
- package/lib/init-templates/app/typescript/lib/%name%-stack.template.ts +16 -0
- package/lib/init-templates/app/typescript/package.json +26 -0
- package/lib/init-templates/app/typescript/test/%name%.test.template.ts +17 -0
- package/lib/init-templates/app/typescript/tsconfig.json +31 -0
- package/lib/init-templates/lib/info.json +4 -0
- package/lib/init-templates/lib/typescript/.template.gitignore +8 -0
- package/lib/init-templates/lib/typescript/.template.npmignore +6 -0
- package/lib/init-templates/lib/typescript/README.template.md +12 -0
- package/lib/init-templates/lib/typescript/jest.config.js +8 -0
- package/lib/init-templates/lib/typescript/lib/index.template.ts +21 -0
- package/lib/init-templates/lib/typescript/package.json +24 -0
- package/lib/init-templates/lib/typescript/test/%name%.test.template.ts +18 -0
- package/lib/init-templates/lib/typescript/tsconfig.json +31 -0
- package/lib/init-templates/sample-app/csharp/.template.gitignore +342 -0
- package/lib/init-templates/sample-app/csharp/README.template.md +19 -0
- package/lib/init-templates/sample-app/csharp/cdk.template.json +15 -0
- package/lib/init-templates/sample-app/csharp/src/%name.PascalCased%/%name.PascalCased%.template.csproj +20 -0
- package/lib/init-templates/sample-app/csharp/src/%name.PascalCased%/%name.PascalCased%Stack.template.cs +24 -0
- package/lib/init-templates/sample-app/csharp/src/%name.PascalCased%/GlobalSuppressions.cs +1 -0
- package/lib/init-templates/sample-app/csharp/src/%name.PascalCased%/Program.template.cs +15 -0
- package/lib/init-templates/sample-app/csharp/src/%name.PascalCased%.template.sln +18 -0
- package/lib/init-templates/sample-app/fsharp/.template.gitignore +342 -0
- package/lib/init-templates/sample-app/fsharp/README.template.md +20 -0
- package/lib/init-templates/sample-app/fsharp/cdk.template.json +14 -0
- package/lib/init-templates/sample-app/fsharp/src/%name.PascalCased%/%name.PascalCased%.template.fsproj +25 -0
- package/lib/init-templates/sample-app/fsharp/src/%name.PascalCased%/%name.PascalCased%Stack.template.fs +14 -0
- package/lib/init-templates/sample-app/fsharp/src/%name.PascalCased%/Program.template.fs +11 -0
- package/lib/init-templates/sample-app/fsharp/src/%name.PascalCased%.template.sln +18 -0
- package/lib/init-templates/sample-app/go/%name%.template.go +73 -0
- package/lib/init-templates/sample-app/go/%name%_test.template.go +25 -0
- package/lib/init-templates/sample-app/go/.template.gitignore +19 -0
- package/lib/init-templates/sample-app/go/README.md +12 -0
- package/lib/init-templates/sample-app/go/cdk.template.json +13 -0
- package/lib/init-templates/sample-app/go/go.template.mod +9 -0
- package/lib/init-templates/sample-app/info.json +4 -0
- package/lib/init-templates/sample-app/java/.template.gitignore +13 -0
- package/lib/init-templates/sample-app/java/README.template.md +19 -0
- package/lib/init-templates/sample-app/java/cdk.json +13 -0
- package/lib/init-templates/sample-app/java/pom.xml +55 -0
- package/lib/init-templates/sample-app/java/src/main/java/com/myorg/%name.PascalCased%App.template.java +13 -0
- package/lib/init-templates/sample-app/java/src/main/java/com/myorg/%name.PascalCased%Stack.template.java +29 -0
- package/lib/init-templates/sample-app/java/src/test/java/com/myorg/%name.PascalCased%StackTest.template.java +27 -0
- package/lib/init-templates/sample-app/javascript/.template.gitignore +5 -0
- package/lib/init-templates/sample-app/javascript/.template.npmignore +3 -0
- package/lib/init-templates/sample-app/javascript/README.template.md +13 -0
- package/lib/init-templates/sample-app/javascript/bin/%name%.template.js +6 -0
- package/lib/init-templates/sample-app/javascript/cdk.template.json +15 -0
- package/lib/init-templates/sample-app/javascript/jest.config.js +3 -0
- package/lib/init-templates/sample-app/javascript/lib/%name%-stack.template.js +25 -0
- package/lib/init-templates/sample-app/javascript/package.json +20 -0
- package/lib/init-templates/sample-app/javascript/test/%name%.test.template.js +16 -0
- package/lib/init-templates/sample-app/javascript/tsconfig.json +34 -0
- package/lib/init-templates/sample-app/python/%name.PythonModule%/%name.PythonModule%_stack.template.py +26 -0
- package/lib/init-templates/sample-app/python/%name.PythonModule%/__init__.py +0 -0
- package/lib/init-templates/sample-app/python/.template.gitignore +22 -0
- package/lib/init-templates/sample-app/python/README.template.md +65 -0
- package/lib/init-templates/sample-app/python/app.template.py +11 -0
- package/lib/init-templates/sample-app/python/cdk.template.json +15 -0
- package/lib/init-templates/sample-app/python/requirements-dev.txt +1 -0
- package/lib/init-templates/sample-app/python/requirements.txt +2 -0
- package/lib/init-templates/sample-app/python/source.bat +13 -0
- package/lib/init-templates/sample-app/python/tests/__init__.py +0 -0
- package/lib/init-templates/sample-app/python/tests/unit/__init__.py +0 -0
- package/lib/init-templates/sample-app/python/tests/unit/test_%name.PythonModule%_stack.template.py +21 -0
- package/lib/init-templates/sample-app/typescript/.template.gitignore +8 -0
- package/lib/init-templates/sample-app/typescript/.template.npmignore +6 -0
- package/lib/init-templates/sample-app/typescript/README.template.md +15 -0
- package/lib/init-templates/sample-app/typescript/bin/%name%.template.ts +6 -0
- package/lib/init-templates/sample-app/typescript/cdk.template.json +17 -0
- package/lib/init-templates/sample-app/typescript/jest.config.js +8 -0
- package/lib/init-templates/sample-app/typescript/lib/%name%-stack.template.ts +19 -0
- package/lib/init-templates/sample-app/typescript/package.json +26 -0
- package/lib/init-templates/sample-app/typescript/test/%name%.test.template.ts +17 -0
- package/lib/init-templates/sample-app/typescript/tsconfig.json +31 -0
- package/lib/init.d.ts +52 -0
- package/lib/init.js +430 -0
- package/lib/legacy-exports-source.d.ts +27 -0
- package/lib/legacy-exports-source.js +88 -0
- package/lib/legacy-exports.d.ts +10 -0
- package/lib/legacy-exports.js +28 -0
- package/lib/legacy-logging-source.d.ts +32 -0
- package/lib/legacy-logging-source.js +107 -0
- package/lib/list-stacks.d.ts +22 -0
- package/lib/list-stacks.js +23 -0
- package/lib/logging.d.ts +109 -0
- package/lib/logging.js +159 -0
- package/lib/migrator.d.ts +25 -0
- package/lib/migrator.js +67 -0
- package/lib/notices.d.ts +156 -0
- package/lib/notices.js +373 -0
- package/lib/os.d.ts +7 -0
- package/lib/os.js +92 -0
- package/lib/serialize.d.ts +27 -0
- package/lib/serialize.js +86 -0
- package/lib/toolkit/cli-io-host.d.ts +208 -0
- package/lib/toolkit/cli-io-host.js +282 -0
- package/lib/toolkit/error.d.ts +44 -0
- package/lib/toolkit/error.js +78 -0
- package/lib/tree.d.ts +31 -0
- package/lib/tree.js +40 -0
- package/lib/util/archive.d.ts +1 -0
- package/lib/util/archive.js +86 -0
- package/lib/util/arrays.d.ts +14 -0
- package/lib/util/arrays.js +36 -0
- package/lib/util/bool.d.ts +7 -0
- package/lib/util/bool.js +13 -0
- package/lib/util/bytes.d.ts +8 -0
- package/lib/util/bytes.js +21 -0
- package/lib/util/content-hash.d.ts +5 -0
- package/lib/util/content-hash.js +43 -0
- package/lib/util/directories.d.ts +23 -0
- package/lib/util/directories.js +57 -0
- package/lib/util/error.d.ts +9 -0
- package/lib/util/error.js +22 -0
- package/lib/util/index.d.ts +5 -0
- package/lib/util/index.js +22 -0
- package/lib/util/objects.d.ts +52 -0
- package/lib/util/objects.js +183 -0
- package/lib/util/parallel.d.ts +6 -0
- package/lib/util/parallel.js +44 -0
- package/lib/util/tables.d.ts +1 -0
- package/lib/util/tables.js +10 -0
- package/lib/util/type-brands.d.ts +39 -0
- package/lib/util/type-brands.js +38 -0
- package/lib/util/types.d.ts +27 -0
- package/lib/util/types.js +25 -0
- package/lib/util/validate-notification-arn.d.ts +4 -0
- package/lib/util/validate-notification-arn.js +10 -0
- package/lib/util/version-range.d.ts +2 -0
- package/lib/util/version-range.js +36 -0
- package/lib/util/work-graph-builder.d.ts +32 -0
- package/lib/util/work-graph-builder.js +167 -0
- package/lib/util/work-graph-types.d.ts +50 -0
- package/lib/util/work-graph-types.js +14 -0
- package/lib/util/work-graph.d.ts +70 -0
- package/lib/util/work-graph.js +344 -0
- package/lib/util/yaml-cfn.d.ts +15 -0
- package/lib/util/yaml-cfn.js +56 -0
- package/package.json +197 -0
- package/scripts/user-input-gen +2 -0
|
@@ -0,0 +1,357 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.tryHotswapDeployment = tryHotswapDeployment;
|
|
4
|
+
const cfn_diff = require("@aws-cdk/cloudformation-diff");
|
|
5
|
+
const chalk = require("chalk");
|
|
6
|
+
const nested_stack_helpers_1 = require("./nested-stack-helpers");
|
|
7
|
+
const logging_1 = require("../../logging");
|
|
8
|
+
const error_1 = require("../../toolkit/error");
|
|
9
|
+
const error_2 = require("../../util/error");
|
|
10
|
+
const evaluate_cloudformation_template_1 = require("../evaluate-cloudformation-template");
|
|
11
|
+
const appsync_mapping_templates_1 = require("../hotswap/appsync-mapping-templates");
|
|
12
|
+
const code_build_projects_1 = require("../hotswap/code-build-projects");
|
|
13
|
+
const common_1 = require("../hotswap/common");
|
|
14
|
+
const ecs_services_1 = require("../hotswap/ecs-services");
|
|
15
|
+
const lambda_functions_1 = require("../hotswap/lambda-functions");
|
|
16
|
+
const s3_bucket_deployments_1 = require("../hotswap/s3-bucket-deployments");
|
|
17
|
+
const stepfunctions_state_machines_1 = require("../hotswap/stepfunctions-state-machines");
|
|
18
|
+
const plugin_1 = require("../plugin");
|
|
19
|
+
// Must use a require() otherwise esbuild complains about calling a namespace
|
|
20
|
+
// eslint-disable-next-line @typescript-eslint/no-require-imports
|
|
21
|
+
const pLimit = require('p-limit');
|
|
22
|
+
const RESOURCE_DETECTORS = {
|
|
23
|
+
// Lambda
|
|
24
|
+
'AWS::Lambda::Function': lambda_functions_1.isHotswappableLambdaFunctionChange,
|
|
25
|
+
'AWS::Lambda::Version': lambda_functions_1.isHotswappableLambdaFunctionChange,
|
|
26
|
+
'AWS::Lambda::Alias': lambda_functions_1.isHotswappableLambdaFunctionChange,
|
|
27
|
+
// AppSync
|
|
28
|
+
'AWS::AppSync::Resolver': appsync_mapping_templates_1.isHotswappableAppSyncChange,
|
|
29
|
+
'AWS::AppSync::FunctionConfiguration': appsync_mapping_templates_1.isHotswappableAppSyncChange,
|
|
30
|
+
'AWS::AppSync::GraphQLSchema': appsync_mapping_templates_1.isHotswappableAppSyncChange,
|
|
31
|
+
'AWS::AppSync::ApiKey': appsync_mapping_templates_1.isHotswappableAppSyncChange,
|
|
32
|
+
'AWS::ECS::TaskDefinition': ecs_services_1.isHotswappableEcsServiceChange,
|
|
33
|
+
'AWS::CodeBuild::Project': code_build_projects_1.isHotswappableCodeBuildProjectChange,
|
|
34
|
+
'AWS::StepFunctions::StateMachine': stepfunctions_state_machines_1.isHotswappableStateMachineChange,
|
|
35
|
+
'Custom::CDKBucketDeployment': s3_bucket_deployments_1.isHotswappableS3BucketDeploymentChange,
|
|
36
|
+
'AWS::IAM::Policy': async (logicalId, change, evaluateCfnTemplate) => {
|
|
37
|
+
// If the policy is for a S3BucketDeploymentChange, we can ignore the change
|
|
38
|
+
if (await (0, s3_bucket_deployments_1.skipChangeForS3DeployCustomResourcePolicy)(logicalId, change, evaluateCfnTemplate)) {
|
|
39
|
+
return [];
|
|
40
|
+
}
|
|
41
|
+
return (0, common_1.reportNonHotswappableResource)(change, 'This resource type is not supported for hotswap deployments');
|
|
42
|
+
},
|
|
43
|
+
'AWS::CDK::Metadata': async () => [],
|
|
44
|
+
};
|
|
45
|
+
/**
|
|
46
|
+
* Perform a hotswap deployment, short-circuiting CloudFormation if possible.
|
|
47
|
+
* If it's not possible to short-circuit the deployment
|
|
48
|
+
* (because the CDK Stack contains changes that cannot be deployed without CloudFormation),
|
|
49
|
+
* returns `undefined`.
|
|
50
|
+
*/
|
|
51
|
+
async function tryHotswapDeployment(sdkProvider, assetParams, cloudFormationStack, stackArtifact, hotswapMode, hotswapPropertyOverrides) {
|
|
52
|
+
// resolve the environment, so we can substitute things like AWS::Region in CFN expressions
|
|
53
|
+
const resolvedEnv = await sdkProvider.resolveEnvironment(stackArtifact.environment);
|
|
54
|
+
// create a new SDK using the CLI credentials, because the default one will not work for new-style synthesis -
|
|
55
|
+
// it assumes the bootstrap deploy Role, which doesn't have permissions to update Lambda functions
|
|
56
|
+
const sdk = (await sdkProvider.forEnvironment(resolvedEnv, plugin_1.Mode.ForWriting)).sdk;
|
|
57
|
+
const currentTemplate = await (0, nested_stack_helpers_1.loadCurrentTemplateWithNestedStacks)(stackArtifact, sdk);
|
|
58
|
+
const evaluateCfnTemplate = new evaluate_cloudformation_template_1.EvaluateCloudFormationTemplate({
|
|
59
|
+
stackName: stackArtifact.stackName,
|
|
60
|
+
template: stackArtifact.template,
|
|
61
|
+
parameters: assetParams,
|
|
62
|
+
account: resolvedEnv.account,
|
|
63
|
+
region: resolvedEnv.region,
|
|
64
|
+
partition: (await sdk.currentAccount()).partition,
|
|
65
|
+
sdk,
|
|
66
|
+
nestedStacks: currentTemplate.nestedStacks,
|
|
67
|
+
});
|
|
68
|
+
const stackChanges = cfn_diff.fullDiff(currentTemplate.deployedRootTemplate, stackArtifact.template);
|
|
69
|
+
const { hotswappableChanges, nonHotswappableChanges } = await classifyResourceChanges(stackChanges, evaluateCfnTemplate, sdk, currentTemplate.nestedStacks, hotswapPropertyOverrides);
|
|
70
|
+
logNonHotswappableChanges(nonHotswappableChanges, hotswapMode);
|
|
71
|
+
// preserve classic hotswap behavior
|
|
72
|
+
if (hotswapMode === common_1.HotswapMode.FALL_BACK) {
|
|
73
|
+
if (nonHotswappableChanges.length > 0) {
|
|
74
|
+
return undefined;
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
// apply the short-circuitable changes
|
|
78
|
+
await applyAllHotswappableChanges(sdk, hotswappableChanges);
|
|
79
|
+
return {
|
|
80
|
+
type: 'did-deploy-stack',
|
|
81
|
+
noOp: hotswappableChanges.length === 0,
|
|
82
|
+
stackArn: cloudFormationStack.stackId,
|
|
83
|
+
outputs: cloudFormationStack.outputs,
|
|
84
|
+
};
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* Classifies all changes to all resources as either hotswappable or not.
|
|
88
|
+
* Metadata changes are excluded from the list of (non)hotswappable resources.
|
|
89
|
+
*/
|
|
90
|
+
async function classifyResourceChanges(stackChanges, evaluateCfnTemplate, sdk, nestedStackNames, hotswapPropertyOverrides) {
|
|
91
|
+
var _a, _b;
|
|
92
|
+
const resourceDifferences = getStackResourceDifferences(stackChanges);
|
|
93
|
+
const promises = [];
|
|
94
|
+
const hotswappableResources = new Array();
|
|
95
|
+
const nonHotswappableResources = new Array();
|
|
96
|
+
for (const logicalId of Object.keys(stackChanges.outputs.changes)) {
|
|
97
|
+
nonHotswappableResources.push({
|
|
98
|
+
hotswappable: false,
|
|
99
|
+
reason: 'output was changed',
|
|
100
|
+
logicalId,
|
|
101
|
+
rejectedChanges: [],
|
|
102
|
+
resourceType: 'Stack Output',
|
|
103
|
+
});
|
|
104
|
+
}
|
|
105
|
+
// gather the results of the detector functions
|
|
106
|
+
for (const [logicalId, change] of Object.entries(resourceDifferences)) {
|
|
107
|
+
if (((_a = change.newValue) === null || _a === void 0 ? void 0 : _a.Type) === 'AWS::CloudFormation::Stack' && ((_b = change.oldValue) === null || _b === void 0 ? void 0 : _b.Type) === 'AWS::CloudFormation::Stack') {
|
|
108
|
+
const nestedHotswappableResources = await findNestedHotswappableChanges(logicalId, change, nestedStackNames, evaluateCfnTemplate, sdk, hotswapPropertyOverrides);
|
|
109
|
+
hotswappableResources.push(...nestedHotswappableResources.hotswappableChanges);
|
|
110
|
+
nonHotswappableResources.push(...nestedHotswappableResources.nonHotswappableChanges);
|
|
111
|
+
continue;
|
|
112
|
+
}
|
|
113
|
+
const hotswappableChangeCandidate = isCandidateForHotswapping(change, logicalId);
|
|
114
|
+
// we don't need to run this through the detector functions, we can already judge this
|
|
115
|
+
if ('hotswappable' in hotswappableChangeCandidate) {
|
|
116
|
+
if (!hotswappableChangeCandidate.hotswappable) {
|
|
117
|
+
nonHotswappableResources.push(hotswappableChangeCandidate);
|
|
118
|
+
}
|
|
119
|
+
continue;
|
|
120
|
+
}
|
|
121
|
+
const resourceType = hotswappableChangeCandidate.newValue.Type;
|
|
122
|
+
if (resourceType in RESOURCE_DETECTORS) {
|
|
123
|
+
// run detector functions lazily to prevent unhandled promise rejections
|
|
124
|
+
promises.push(() => RESOURCE_DETECTORS[resourceType](logicalId, hotswappableChangeCandidate, evaluateCfnTemplate, hotswapPropertyOverrides));
|
|
125
|
+
}
|
|
126
|
+
else {
|
|
127
|
+
(0, common_1.reportNonHotswappableChange)(nonHotswappableResources, hotswappableChangeCandidate, undefined, 'This resource type is not supported for hotswap deployments');
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
// resolve all detector results
|
|
131
|
+
const changesDetectionResults = [];
|
|
132
|
+
for (const detectorResultPromises of promises) {
|
|
133
|
+
// Constant set of promises per resource
|
|
134
|
+
// eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism
|
|
135
|
+
const hotswapDetectionResults = await Promise.all(await detectorResultPromises());
|
|
136
|
+
changesDetectionResults.push(hotswapDetectionResults);
|
|
137
|
+
}
|
|
138
|
+
for (const resourceDetectionResults of changesDetectionResults) {
|
|
139
|
+
for (const propertyResult of resourceDetectionResults) {
|
|
140
|
+
propertyResult.hotswappable
|
|
141
|
+
? hotswappableResources.push(propertyResult)
|
|
142
|
+
: nonHotswappableResources.push(propertyResult);
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
return {
|
|
146
|
+
hotswappableChanges: hotswappableResources,
|
|
147
|
+
nonHotswappableChanges: nonHotswappableResources,
|
|
148
|
+
};
|
|
149
|
+
}
|
|
150
|
+
/**
|
|
151
|
+
* Returns all changes to resources in the given Stack.
|
|
152
|
+
*
|
|
153
|
+
* @param stackChanges the collection of all changes to a given Stack
|
|
154
|
+
*/
|
|
155
|
+
function getStackResourceDifferences(stackChanges) {
|
|
156
|
+
// we need to collapse logical ID rename changes into one change,
|
|
157
|
+
// as they are represented in stackChanges as a pair of two changes: one addition and one removal
|
|
158
|
+
const allResourceChanges = stackChanges.resources.changes;
|
|
159
|
+
const allRemovalChanges = filterDict(allResourceChanges, (resChange) => resChange.isRemoval);
|
|
160
|
+
const allNonRemovalChanges = filterDict(allResourceChanges, (resChange) => !resChange.isRemoval);
|
|
161
|
+
for (const [logId, nonRemovalChange] of Object.entries(allNonRemovalChanges)) {
|
|
162
|
+
if (nonRemovalChange.isAddition) {
|
|
163
|
+
const addChange = nonRemovalChange;
|
|
164
|
+
// search for an identical removal change
|
|
165
|
+
const identicalRemovalChange = Object.entries(allRemovalChanges).find(([_, remChange]) => {
|
|
166
|
+
return changesAreForSameResource(remChange, addChange);
|
|
167
|
+
});
|
|
168
|
+
// if we found one, then this means this is a rename change
|
|
169
|
+
if (identicalRemovalChange) {
|
|
170
|
+
const [removedLogId, removedResourceChange] = identicalRemovalChange;
|
|
171
|
+
allNonRemovalChanges[logId] = makeRenameDifference(removedResourceChange, addChange);
|
|
172
|
+
// delete the removal change that forms the rename pair
|
|
173
|
+
delete allRemovalChanges[removedLogId];
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
// the final result are all of the remaining removal changes,
|
|
178
|
+
// plus all of the non-removal changes
|
|
179
|
+
// (we saved the rename changes in that object already)
|
|
180
|
+
return {
|
|
181
|
+
...allRemovalChanges,
|
|
182
|
+
...allNonRemovalChanges,
|
|
183
|
+
};
|
|
184
|
+
}
|
|
185
|
+
/** Filters an object with string keys based on whether the callback returns 'true' for the given value in the object. */
|
|
186
|
+
function filterDict(dict, func) {
|
|
187
|
+
return Object.entries(dict).reduce((acc, [key, t]) => {
|
|
188
|
+
if (func(t)) {
|
|
189
|
+
acc[key] = t;
|
|
190
|
+
}
|
|
191
|
+
return acc;
|
|
192
|
+
}, {});
|
|
193
|
+
}
|
|
194
|
+
/** Finds any hotswappable changes in all nested stacks. */
|
|
195
|
+
async function findNestedHotswappableChanges(logicalId, change, nestedStackTemplates, evaluateCfnTemplate, sdk, hotswapPropertyOverrides) {
|
|
196
|
+
var _a, _b;
|
|
197
|
+
const nestedStack = nestedStackTemplates[logicalId];
|
|
198
|
+
if (!nestedStack.physicalName) {
|
|
199
|
+
return {
|
|
200
|
+
hotswappableChanges: [],
|
|
201
|
+
nonHotswappableChanges: [
|
|
202
|
+
{
|
|
203
|
+
hotswappable: false,
|
|
204
|
+
logicalId,
|
|
205
|
+
reason: `physical name for AWS::CloudFormation::Stack '${logicalId}' could not be found in CloudFormation, so this is a newly created nested stack and cannot be hotswapped`,
|
|
206
|
+
rejectedChanges: [],
|
|
207
|
+
resourceType: 'AWS::CloudFormation::Stack',
|
|
208
|
+
},
|
|
209
|
+
],
|
|
210
|
+
};
|
|
211
|
+
}
|
|
212
|
+
const evaluateNestedCfnTemplate = await evaluateCfnTemplate.createNestedEvaluateCloudFormationTemplate(nestedStack.physicalName, nestedStack.generatedTemplate, (_b = (_a = change.newValue) === null || _a === void 0 ? void 0 : _a.Properties) === null || _b === void 0 ? void 0 : _b.Parameters);
|
|
213
|
+
const nestedDiff = cfn_diff.fullDiff(nestedStackTemplates[logicalId].deployedTemplate, nestedStackTemplates[logicalId].generatedTemplate);
|
|
214
|
+
return classifyResourceChanges(nestedDiff, evaluateNestedCfnTemplate, sdk, nestedStackTemplates[logicalId].nestedStackTemplates, hotswapPropertyOverrides);
|
|
215
|
+
}
|
|
216
|
+
/** Returns 'true' if a pair of changes is for the same resource. */
|
|
217
|
+
function changesAreForSameResource(oldChange, newChange) {
|
|
218
|
+
return (oldChange.oldResourceType === newChange.newResourceType &&
|
|
219
|
+
// this isn't great, but I don't want to bring in something like underscore just for this comparison
|
|
220
|
+
JSON.stringify(oldChange.oldProperties) === JSON.stringify(newChange.newProperties));
|
|
221
|
+
}
|
|
222
|
+
function makeRenameDifference(remChange, addChange) {
|
|
223
|
+
return new cfn_diff.ResourceDifference(
|
|
224
|
+
// we have to fill in the old value, because otherwise this will be classified as a non-hotswappable change
|
|
225
|
+
remChange.oldValue, addChange.newValue, {
|
|
226
|
+
resourceType: {
|
|
227
|
+
oldType: remChange.oldResourceType,
|
|
228
|
+
newType: addChange.newResourceType,
|
|
229
|
+
},
|
|
230
|
+
propertyDiffs: addChange.propertyDiffs,
|
|
231
|
+
otherDiffs: addChange.otherDiffs,
|
|
232
|
+
});
|
|
233
|
+
}
|
|
234
|
+
/**
|
|
235
|
+
* Returns a `HotswappableChangeCandidate` if the change is hotswappable
|
|
236
|
+
* Returns an empty `HotswappableChange` if the change is to CDK::Metadata
|
|
237
|
+
* Returns a `NonHotswappableChange` if the change is not hotswappable
|
|
238
|
+
*/
|
|
239
|
+
function isCandidateForHotswapping(change, logicalId) {
|
|
240
|
+
var _a, _b, _c, _d, _e;
|
|
241
|
+
// a resource has been removed OR a resource has been added; we can't short-circuit that change
|
|
242
|
+
if (!change.oldValue) {
|
|
243
|
+
return {
|
|
244
|
+
hotswappable: false,
|
|
245
|
+
resourceType: change.newValue.Type,
|
|
246
|
+
logicalId,
|
|
247
|
+
rejectedChanges: [],
|
|
248
|
+
reason: `resource '${logicalId}' was created by this deployment`,
|
|
249
|
+
};
|
|
250
|
+
}
|
|
251
|
+
else if (!change.newValue) {
|
|
252
|
+
return {
|
|
253
|
+
hotswappable: false,
|
|
254
|
+
resourceType: change.oldValue.Type,
|
|
255
|
+
logicalId,
|
|
256
|
+
rejectedChanges: [],
|
|
257
|
+
reason: `resource '${logicalId}' was destroyed by this deployment`,
|
|
258
|
+
};
|
|
259
|
+
}
|
|
260
|
+
// a resource has had its type changed
|
|
261
|
+
if (((_a = change.newValue) === null || _a === void 0 ? void 0 : _a.Type) !== ((_b = change.oldValue) === null || _b === void 0 ? void 0 : _b.Type)) {
|
|
262
|
+
return {
|
|
263
|
+
hotswappable: false,
|
|
264
|
+
resourceType: (_c = change.newValue) === null || _c === void 0 ? void 0 : _c.Type,
|
|
265
|
+
logicalId,
|
|
266
|
+
rejectedChanges: [],
|
|
267
|
+
reason: `resource '${logicalId}' had its type changed from '${(_d = change.oldValue) === null || _d === void 0 ? void 0 : _d.Type}' to '${(_e = change.newValue) === null || _e === void 0 ? void 0 : _e.Type}'`,
|
|
268
|
+
};
|
|
269
|
+
}
|
|
270
|
+
return {
|
|
271
|
+
logicalId,
|
|
272
|
+
oldValue: change.oldValue,
|
|
273
|
+
newValue: change.newValue,
|
|
274
|
+
propertyUpdates: change.propertyUpdates,
|
|
275
|
+
};
|
|
276
|
+
}
|
|
277
|
+
async function applyAllHotswappableChanges(sdk, hotswappableChanges) {
|
|
278
|
+
if (hotswappableChanges.length > 0) {
|
|
279
|
+
(0, logging_1.info)(`\n${common_1.ICON} hotswapping resources:`);
|
|
280
|
+
}
|
|
281
|
+
const limit = pLimit(10);
|
|
282
|
+
// eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism
|
|
283
|
+
return Promise.all(hotswappableChanges.map(hotswapOperation => limit(() => {
|
|
284
|
+
return applyHotswappableChange(sdk, hotswapOperation);
|
|
285
|
+
})));
|
|
286
|
+
}
|
|
287
|
+
async function applyHotswappableChange(sdk, hotswapOperation) {
|
|
288
|
+
// note the type of service that was successfully hotswapped in the User-Agent
|
|
289
|
+
const customUserAgent = `cdk-hotswap/success-${hotswapOperation.service}`;
|
|
290
|
+
sdk.appendCustomUserAgent(customUserAgent);
|
|
291
|
+
for (const name of hotswapOperation.resourceNames) {
|
|
292
|
+
(0, logging_1.info)(` ${common_1.ICON} %s`, chalk.bold(name));
|
|
293
|
+
}
|
|
294
|
+
// if the SDK call fails, an error will be thrown by the SDK
|
|
295
|
+
// and will prevent the green 'hotswapped!' text from being displayed
|
|
296
|
+
try {
|
|
297
|
+
await hotswapOperation.apply(sdk);
|
|
298
|
+
}
|
|
299
|
+
catch (e) {
|
|
300
|
+
if (e.name === 'TimeoutError' || e.name === 'AbortError') {
|
|
301
|
+
const result = JSON.parse((0, error_2.formatErrorMessage)(e));
|
|
302
|
+
const error = new error_1.ToolkitError(formatWaiterErrorResult(result));
|
|
303
|
+
error.name = e.name;
|
|
304
|
+
throw error;
|
|
305
|
+
}
|
|
306
|
+
throw e;
|
|
307
|
+
}
|
|
308
|
+
for (const name of hotswapOperation.resourceNames) {
|
|
309
|
+
(0, logging_1.info)(`${common_1.ICON} %s %s`, chalk.bold(name), chalk.green('hotswapped!'));
|
|
310
|
+
}
|
|
311
|
+
sdk.removeCustomUserAgent(customUserAgent);
|
|
312
|
+
}
|
|
313
|
+
function formatWaiterErrorResult(result) {
|
|
314
|
+
const main = [
|
|
315
|
+
`Resource is not in the expected state due to waiter status: ${result.state}`,
|
|
316
|
+
result.reason ? `${result.reason}.` : '',
|
|
317
|
+
].join('. ');
|
|
318
|
+
if (result.observedResponses != null) {
|
|
319
|
+
const observedResponses = Object
|
|
320
|
+
.entries(result.observedResponses)
|
|
321
|
+
.map(([msg, count]) => ` - ${msg} (${count})`)
|
|
322
|
+
.join('\n');
|
|
323
|
+
return `${main} Observed responses:\n${observedResponses}`;
|
|
324
|
+
}
|
|
325
|
+
return main;
|
|
326
|
+
}
|
|
327
|
+
function logNonHotswappableChanges(nonHotswappableChanges, hotswapMode) {
|
|
328
|
+
if (nonHotswappableChanges.length === 0) {
|
|
329
|
+
return;
|
|
330
|
+
}
|
|
331
|
+
/**
|
|
332
|
+
* EKS Services can have a task definition that doesn't refer to the task definition being updated.
|
|
333
|
+
* We have to log this as a non-hotswappable change to the task definition, but when we do,
|
|
334
|
+
* we wind up hotswapping the task definition and logging it as a non-hotswappable change.
|
|
335
|
+
*
|
|
336
|
+
* This logic prevents us from logging that change as non-hotswappable when we hotswap it.
|
|
337
|
+
*/
|
|
338
|
+
if (hotswapMode === common_1.HotswapMode.HOTSWAP_ONLY) {
|
|
339
|
+
nonHotswappableChanges = nonHotswappableChanges.filter((change) => change.hotswapOnlyVisible === true);
|
|
340
|
+
if (nonHotswappableChanges.length === 0) {
|
|
341
|
+
return;
|
|
342
|
+
}
|
|
343
|
+
}
|
|
344
|
+
if (hotswapMode === common_1.HotswapMode.HOTSWAP_ONLY) {
|
|
345
|
+
(0, logging_1.info)('\n%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found. To reconcile these using CloudFormation, specify --hotswap-fallback'));
|
|
346
|
+
}
|
|
347
|
+
else {
|
|
348
|
+
(0, logging_1.info)('\n%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found:'));
|
|
349
|
+
}
|
|
350
|
+
for (const change of nonHotswappableChanges) {
|
|
351
|
+
change.rejectedChanges.length > 0
|
|
352
|
+
? (0, logging_1.info)(' logicalID: %s, type: %s, rejected changes: %s, reason: %s', chalk.bold(change.logicalId), chalk.bold(change.resourceType), chalk.bold(change.rejectedChanges), chalk.red(change.reason))
|
|
353
|
+
: (0, logging_1.info)(' logicalID: %s, type: %s, reason: %s', chalk.bold(change.logicalId), chalk.bold(change.resourceType), chalk.red(change.reason));
|
|
354
|
+
}
|
|
355
|
+
(0, logging_1.info)(''); // newline
|
|
356
|
+
}
|
|
357
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hotswap-deployments.js","sourceRoot":"","sources":["hotswap-deployments.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cfn_diff from '@aws-cdk/cloudformation-diff';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport { WaiterResult } from '@smithy/util-waiter';\nimport * as chalk from 'chalk';\nimport type { SDK, SdkProvider } from '../aws-auth';\nimport type { CloudFormationStack } from './cloudformation';\nimport { NestedStackTemplates, loadCurrentTemplateWithNestedStacks } from './nested-stack-helpers';\nimport { info } from '../../logging';\nimport { ToolkitError } from '../../toolkit/error';\nimport { formatErrorMessage } from '../../util/error';\nimport { EvaluateCloudFormationTemplate } from '../evaluate-cloudformation-template';\nimport { isHotswappableAppSyncChange } from '../hotswap/appsync-mapping-templates';\nimport { isHotswappableCodeBuildProjectChange } from '../hotswap/code-build-projects';\nimport {\n  ICON,\n  ChangeHotswapResult,\n  HotswapMode,\n  HotswappableChange,\n  NonHotswappableChange,\n  HotswappableChangeCandidate,\n  HotswapPropertyOverrides, ClassifiedResourceChanges,\n  reportNonHotswappableChange,\n  reportNonHotswappableResource,\n} from '../hotswap/common';\nimport { isHotswappableEcsServiceChange } from '../hotswap/ecs-services';\nimport { isHotswappableLambdaFunctionChange } from '../hotswap/lambda-functions';\nimport {\n  skipChangeForS3DeployCustomResourcePolicy,\n  isHotswappableS3BucketDeploymentChange,\n} from '../hotswap/s3-bucket-deployments';\nimport { isHotswappableStateMachineChange } from '../hotswap/stepfunctions-state-machines';\nimport { Mode } from '../plugin';\nimport { SuccessfulDeployStackResult } from './deployment-result';\n\n// Must use a require() otherwise esbuild complains about calling a namespace\n// eslint-disable-next-line @typescript-eslint/no-require-imports\nconst pLimit: typeof import('p-limit') = require('p-limit');\n\ntype HotswapDetector = (\n  logicalId: string,\n  change: HotswappableChangeCandidate,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n) => Promise<ChangeHotswapResult>;\n\nconst RESOURCE_DETECTORS: { [key: string]: HotswapDetector } = {\n  // Lambda\n  'AWS::Lambda::Function': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Version': isHotswappableLambdaFunctionChange,\n  'AWS::Lambda::Alias': isHotswappableLambdaFunctionChange,\n\n  // AppSync\n  'AWS::AppSync::Resolver': isHotswappableAppSyncChange,\n  'AWS::AppSync::FunctionConfiguration': isHotswappableAppSyncChange,\n  'AWS::AppSync::GraphQLSchema': isHotswappableAppSyncChange,\n  'AWS::AppSync::ApiKey': isHotswappableAppSyncChange,\n\n  'AWS::ECS::TaskDefinition': isHotswappableEcsServiceChange,\n  'AWS::CodeBuild::Project': isHotswappableCodeBuildProjectChange,\n  'AWS::StepFunctions::StateMachine': isHotswappableStateMachineChange,\n  'Custom::CDKBucketDeployment': isHotswappableS3BucketDeploymentChange,\n  'AWS::IAM::Policy': async (\n    logicalId: string,\n    change: HotswappableChangeCandidate,\n    evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  ): Promise<ChangeHotswapResult> => {\n    // If the policy is for a S3BucketDeploymentChange, we can ignore the change\n    if (await skipChangeForS3DeployCustomResourcePolicy(logicalId, change, evaluateCfnTemplate)) {\n      return [];\n    }\n\n    return reportNonHotswappableResource(change, 'This resource type is not supported for hotswap deployments');\n  },\n\n  'AWS::CDK::Metadata': async () => [],\n};\n\n/**\n * Perform a hotswap deployment, short-circuiting CloudFormation if possible.\n * If it's not possible to short-circuit the deployment\n * (because the CDK Stack contains changes that cannot be deployed without CloudFormation),\n * returns `undefined`.\n */\nexport async function tryHotswapDeployment(\n  sdkProvider: SdkProvider,\n  assetParams: { [key: string]: string },\n  cloudFormationStack: CloudFormationStack,\n  stackArtifact: cxapi.CloudFormationStackArtifact,\n  hotswapMode: HotswapMode, hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<SuccessfulDeployStackResult | undefined> {\n  // resolve the environment, so we can substitute things like AWS::Region in CFN expressions\n  const resolvedEnv = await sdkProvider.resolveEnvironment(stackArtifact.environment);\n  // create a new SDK using the CLI credentials, because the default one will not work for new-style synthesis -\n  // it assumes the bootstrap deploy Role, which doesn't have permissions to update Lambda functions\n  const sdk = (await sdkProvider.forEnvironment(resolvedEnv, Mode.ForWriting)).sdk;\n\n  const currentTemplate = await loadCurrentTemplateWithNestedStacks(stackArtifact, sdk);\n\n  const evaluateCfnTemplate = new EvaluateCloudFormationTemplate({\n    stackName: stackArtifact.stackName,\n    template: stackArtifact.template,\n    parameters: assetParams,\n    account: resolvedEnv.account,\n    region: resolvedEnv.region,\n    partition: (await sdk.currentAccount()).partition,\n    sdk,\n    nestedStacks: currentTemplate.nestedStacks,\n  });\n\n  const stackChanges = cfn_diff.fullDiff(currentTemplate.deployedRootTemplate, stackArtifact.template);\n  const { hotswappableChanges, nonHotswappableChanges } = await classifyResourceChanges(\n    stackChanges,\n    evaluateCfnTemplate,\n    sdk,\n    currentTemplate.nestedStacks, hotswapPropertyOverrides,\n  );\n\n  logNonHotswappableChanges(nonHotswappableChanges, hotswapMode);\n\n  // preserve classic hotswap behavior\n  if (hotswapMode === HotswapMode.FALL_BACK) {\n    if (nonHotswappableChanges.length > 0) {\n      return undefined;\n    }\n  }\n\n  // apply the short-circuitable changes\n  await applyAllHotswappableChanges(sdk, hotswappableChanges);\n\n  return {\n    type: 'did-deploy-stack',\n    noOp: hotswappableChanges.length === 0,\n    stackArn: cloudFormationStack.stackId,\n    outputs: cloudFormationStack.outputs,\n  };\n}\n\n/**\n * Classifies all changes to all resources as either hotswappable or not.\n * Metadata changes are excluded from the list of (non)hotswappable resources.\n */\nasync function classifyResourceChanges(\n  stackChanges: cfn_diff.TemplateDiff,\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: SDK,\n  nestedStackNames: { [nestedStackName: string]: NestedStackTemplates },\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<ClassifiedResourceChanges> {\n  const resourceDifferences = getStackResourceDifferences(stackChanges);\n\n  const promises: Array<() => Promise<ChangeHotswapResult>> = [];\n  const hotswappableResources = new Array<HotswappableChange>();\n  const nonHotswappableResources = new Array<NonHotswappableChange>();\n  for (const logicalId of Object.keys(stackChanges.outputs.changes)) {\n    nonHotswappableResources.push({\n      hotswappable: false,\n      reason: 'output was changed',\n      logicalId,\n      rejectedChanges: [],\n      resourceType: 'Stack Output',\n    });\n  }\n  // gather the results of the detector functions\n  for (const [logicalId, change] of Object.entries(resourceDifferences)) {\n    if (change.newValue?.Type === 'AWS::CloudFormation::Stack' && change.oldValue?.Type === 'AWS::CloudFormation::Stack') {\n      const nestedHotswappableResources = await findNestedHotswappableChanges(\n        logicalId,\n        change,\n        nestedStackNames,\n        evaluateCfnTemplate,\n        sdk,\n        hotswapPropertyOverrides,\n      );\n      hotswappableResources.push(...nestedHotswappableResources.hotswappableChanges);\n      nonHotswappableResources.push(...nestedHotswappableResources.nonHotswappableChanges);\n\n      continue;\n    }\n\n    const hotswappableChangeCandidate = isCandidateForHotswapping(change, logicalId);\n    // we don't need to run this through the detector functions, we can already judge this\n    if ('hotswappable' in hotswappableChangeCandidate) {\n      if (!hotswappableChangeCandidate.hotswappable) {\n        nonHotswappableResources.push(hotswappableChangeCandidate);\n      }\n\n      continue;\n    }\n\n    const resourceType: string = hotswappableChangeCandidate.newValue.Type;\n    if (resourceType in RESOURCE_DETECTORS) {\n      // run detector functions lazily to prevent unhandled promise rejections\n      promises.push(() =>\n        RESOURCE_DETECTORS[resourceType](logicalId, hotswappableChangeCandidate, evaluateCfnTemplate, hotswapPropertyOverrides),\n      );\n    } else {\n      reportNonHotswappableChange(\n        nonHotswappableResources,\n        hotswappableChangeCandidate,\n        undefined,\n        'This resource type is not supported for hotswap deployments',\n      );\n    }\n  }\n\n  // resolve all detector results\n  const changesDetectionResults: Array<ChangeHotswapResult> = [];\n  for (const detectorResultPromises of promises) {\n    // Constant set of promises per resource\n    // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n    const hotswapDetectionResults = await Promise.all(await detectorResultPromises());\n    changesDetectionResults.push(hotswapDetectionResults);\n  }\n\n  for (const resourceDetectionResults of changesDetectionResults) {\n    for (const propertyResult of resourceDetectionResults) {\n      propertyResult.hotswappable\n        ? hotswappableResources.push(propertyResult)\n        : nonHotswappableResources.push(propertyResult);\n    }\n  }\n\n  return {\n    hotswappableChanges: hotswappableResources,\n    nonHotswappableChanges: nonHotswappableResources,\n  };\n}\n\n/**\n * Returns all changes to resources in the given Stack.\n *\n * @param stackChanges the collection of all changes to a given Stack\n */\nfunction getStackResourceDifferences(stackChanges: cfn_diff.TemplateDiff): {\n  [logicalId: string]: cfn_diff.ResourceDifference;\n} {\n  // we need to collapse logical ID rename changes into one change,\n  // as they are represented in stackChanges as a pair of two changes: one addition and one removal\n  const allResourceChanges: { [logId: string]: cfn_diff.ResourceDifference } = stackChanges.resources.changes;\n  const allRemovalChanges = filterDict(allResourceChanges, (resChange) => resChange.isRemoval);\n  const allNonRemovalChanges = filterDict(allResourceChanges, (resChange) => !resChange.isRemoval);\n  for (const [logId, nonRemovalChange] of Object.entries(allNonRemovalChanges)) {\n    if (nonRemovalChange.isAddition) {\n      const addChange = nonRemovalChange;\n      // search for an identical removal change\n      const identicalRemovalChange = Object.entries(allRemovalChanges).find(([_, remChange]) => {\n        return changesAreForSameResource(remChange, addChange);\n      });\n      // if we found one, then this means this is a rename change\n      if (identicalRemovalChange) {\n        const [removedLogId, removedResourceChange] = identicalRemovalChange;\n        allNonRemovalChanges[logId] = makeRenameDifference(removedResourceChange, addChange);\n        // delete the removal change that forms the rename pair\n        delete allRemovalChanges[removedLogId];\n      }\n    }\n  }\n  // the final result are all of the remaining removal changes,\n  // plus all of the non-removal changes\n  // (we saved the rename changes in that object already)\n  return {\n    ...allRemovalChanges,\n    ...allNonRemovalChanges,\n  };\n}\n\n/** Filters an object with string keys based on whether the callback returns 'true' for the given value in the object. */\nfunction filterDict<T>(dict: { [key: string]: T }, func: (t: T) => boolean): { [key: string]: T } {\n  return Object.entries(dict).reduce(\n    (acc, [key, t]) => {\n      if (func(t)) {\n        acc[key] = t;\n      }\n      return acc;\n    },\n    {} as { [key: string]: T },\n  );\n}\n\n/** Finds any hotswappable changes in all nested stacks. */\nasync function findNestedHotswappableChanges(\n  logicalId: string,\n  change: cfn_diff.ResourceDifference,\n  nestedStackTemplates: { [nestedStackName: string]: NestedStackTemplates },\n  evaluateCfnTemplate: EvaluateCloudFormationTemplate,\n  sdk: SDK,\n  hotswapPropertyOverrides: HotswapPropertyOverrides,\n): Promise<ClassifiedResourceChanges> {\n  const nestedStack = nestedStackTemplates[logicalId];\n  if (!nestedStack.physicalName) {\n    return {\n      hotswappableChanges: [],\n      nonHotswappableChanges: [\n        {\n          hotswappable: false,\n          logicalId,\n          reason: `physical name for AWS::CloudFormation::Stack '${logicalId}' could not be found in CloudFormation, so this is a newly created nested stack and cannot be hotswapped`,\n          rejectedChanges: [],\n          resourceType: 'AWS::CloudFormation::Stack',\n        },\n      ],\n    };\n  }\n\n  const evaluateNestedCfnTemplate = await evaluateCfnTemplate.createNestedEvaluateCloudFormationTemplate(\n    nestedStack.physicalName,\n    nestedStack.generatedTemplate,\n    change.newValue?.Properties?.Parameters,\n  );\n\n  const nestedDiff = cfn_diff.fullDiff(\n    nestedStackTemplates[logicalId].deployedTemplate,\n    nestedStackTemplates[logicalId].generatedTemplate,\n  );\n\n  return classifyResourceChanges(\n    nestedDiff,\n    evaluateNestedCfnTemplate,\n    sdk,\n    nestedStackTemplates[logicalId].nestedStackTemplates,\n    hotswapPropertyOverrides);\n}\n\n/** Returns 'true' if a pair of changes is for the same resource. */\nfunction changesAreForSameResource(\n  oldChange: cfn_diff.ResourceDifference,\n  newChange: cfn_diff.ResourceDifference,\n): boolean {\n  return (\n    oldChange.oldResourceType === newChange.newResourceType &&\n    // this isn't great, but I don't want to bring in something like underscore just for this comparison\n    JSON.stringify(oldChange.oldProperties) === JSON.stringify(newChange.newProperties)\n  );\n}\n\nfunction makeRenameDifference(\n  remChange: cfn_diff.ResourceDifference,\n  addChange: cfn_diff.ResourceDifference,\n): cfn_diff.ResourceDifference {\n  return new cfn_diff.ResourceDifference(\n    // we have to fill in the old value, because otherwise this will be classified as a non-hotswappable change\n    remChange.oldValue,\n    addChange.newValue,\n    {\n      resourceType: {\n        oldType: remChange.oldResourceType,\n        newType: addChange.newResourceType,\n      },\n      propertyDiffs: (addChange as any).propertyDiffs,\n      otherDiffs: (addChange as any).otherDiffs,\n    },\n  );\n}\n\n/**\n * Returns a `HotswappableChangeCandidate` if the change is hotswappable\n * Returns an empty `HotswappableChange` if the change is to CDK::Metadata\n * Returns a `NonHotswappableChange` if the change is not hotswappable\n */\nfunction isCandidateForHotswapping(\n  change: cfn_diff.ResourceDifference,\n  logicalId: string,\n): HotswappableChange | NonHotswappableChange | HotswappableChangeCandidate {\n  // a resource has been removed OR a resource has been added; we can't short-circuit that change\n  if (!change.oldValue) {\n    return {\n      hotswappable: false,\n      resourceType: change.newValue!.Type,\n      logicalId,\n      rejectedChanges: [],\n      reason: `resource '${logicalId}' was created by this deployment`,\n    };\n  } else if (!change.newValue) {\n    return {\n      hotswappable: false,\n      resourceType: change.oldValue!.Type,\n      logicalId,\n      rejectedChanges: [],\n      reason: `resource '${logicalId}' was destroyed by this deployment`,\n    };\n  }\n\n  // a resource has had its type changed\n  if (change.newValue?.Type !== change.oldValue?.Type) {\n    return {\n      hotswappable: false,\n      resourceType: change.newValue?.Type,\n      logicalId,\n      rejectedChanges: [],\n      reason: `resource '${logicalId}' had its type changed from '${change.oldValue?.Type}' to '${change.newValue?.Type}'`,\n    };\n  }\n\n  return {\n    logicalId,\n    oldValue: change.oldValue,\n    newValue: change.newValue,\n    propertyUpdates: change.propertyUpdates,\n  };\n}\n\nasync function applyAllHotswappableChanges(sdk: SDK, hotswappableChanges: HotswappableChange[]): Promise<void[]> {\n  if (hotswappableChanges.length > 0) {\n    info(`\\n${ICON} hotswapping resources:`);\n  }\n  const limit = pLimit(10);\n  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  return Promise.all(hotswappableChanges.map(hotswapOperation => limit(() => {\n    return applyHotswappableChange(sdk, hotswapOperation);\n  })));\n}\n\nasync function applyHotswappableChange(sdk: SDK, hotswapOperation: HotswappableChange): Promise<void> {\n  // note the type of service that was successfully hotswapped in the User-Agent\n  const customUserAgent = `cdk-hotswap/success-${hotswapOperation.service}`;\n  sdk.appendCustomUserAgent(customUserAgent);\n\n  for (const name of hotswapOperation.resourceNames) {\n    info(`   ${ICON} %s`, chalk.bold(name));\n  }\n\n  // if the SDK call fails, an error will be thrown by the SDK\n  // and will prevent the green 'hotswapped!' text from being displayed\n  try {\n    await hotswapOperation.apply(sdk);\n  } catch (e: any) {\n    if (e.name === 'TimeoutError' || e.name === 'AbortError') {\n      const result: WaiterResult = JSON.parse(formatErrorMessage(e));\n      const error = new ToolkitError(formatWaiterErrorResult(result));\n      error.name = e.name;\n      throw error;\n    }\n    throw e;\n  }\n\n  for (const name of hotswapOperation.resourceNames) {\n    info(`${ICON} %s %s`, chalk.bold(name), chalk.green('hotswapped!'));\n  }\n\n  sdk.removeCustomUserAgent(customUserAgent);\n}\n\nfunction formatWaiterErrorResult(result: WaiterResult) {\n  const main = [\n    `Resource is not in the expected state due to waiter status: ${result.state}`,\n    result.reason ? `${result.reason}.` : '',\n  ].join('. ');\n\n  if (result.observedResponses != null) {\n    const observedResponses = Object\n      .entries(result.observedResponses)\n      .map(([msg, count]) => `  - ${msg} (${count})`)\n      .join('\\n');\n\n    return `${main} Observed responses:\\n${observedResponses}`;\n  }\n\n  return main;\n}\n\nfunction logNonHotswappableChanges(nonHotswappableChanges: NonHotswappableChange[], hotswapMode: HotswapMode): void {\n  if (nonHotswappableChanges.length === 0) {\n    return;\n  }\n  /**\n   * EKS Services can have a task definition that doesn't refer to the task definition being updated.\n   * We have to log this as a non-hotswappable change to the task definition, but when we do,\n   * we wind up hotswapping the task definition and logging it as a non-hotswappable change.\n   *\n   * This logic prevents us from logging that change as non-hotswappable when we hotswap it.\n   */\n  if (hotswapMode === HotswapMode.HOTSWAP_ONLY) {\n    nonHotswappableChanges = nonHotswappableChanges.filter((change) => change.hotswapOnlyVisible === true);\n\n    if (nonHotswappableChanges.length === 0) {\n      return;\n    }\n  }\n  if (hotswapMode === HotswapMode.HOTSWAP_ONLY) {\n    info(\n      '\\n%s %s',\n      chalk.red('⚠️'),\n      chalk.red(\n        'The following non-hotswappable changes were found. To reconcile these using CloudFormation, specify --hotswap-fallback',\n      ),\n    );\n  } else {\n    info('\\n%s %s', chalk.red('⚠️'), chalk.red('The following non-hotswappable changes were found:'));\n  }\n\n  for (const change of nonHotswappableChanges) {\n    change.rejectedChanges.length > 0\n      ? info(\n        '    logicalID: %s, type: %s, rejected changes: %s, reason: %s',\n        chalk.bold(change.logicalId),\n        chalk.bold(change.resourceType),\n        chalk.bold(change.rejectedChanges),\n        chalk.red(change.reason),\n      )\n      : info(\n        '    logicalID: %s, type: %s, reason: %s',\n        chalk.bold(change.logicalId),\n        chalk.bold(change.resourceType),\n        chalk.red(change.reason),\n      );\n  }\n\n  info(''); // newline\n}\n"]}
|
|
@@ -0,0 +1,23 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
+
};
|
|
16
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
+
__exportStar(require("./cloudformation"), exports);
|
|
18
|
+
__exportStar(require("./deployments"), exports);
|
|
19
|
+
__exportStar(require("./deployment-result"), exports);
|
|
20
|
+
__exportStar(require("./deployment-method"), exports);
|
|
21
|
+
__exportStar(require("./nested-stack-helpers"), exports);
|
|
22
|
+
__exportStar(require("./asset-manifest-builder"), exports);
|
|
23
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJpbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7O0FBQUEsbURBQWlDO0FBQ2pDLGdEQUE4QjtBQUM5QixzREFBb0M7QUFDcEMsc0RBQW9DO0FBQ3BDLHlEQUF1QztBQUN2QywyREFBeUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2Nsb3VkZm9ybWF0aW9uJztcbmV4cG9ydCAqIGZyb20gJy4vZGVwbG95bWVudHMnO1xuZXhwb3J0ICogZnJvbSAnLi9kZXBsb3ltZW50LXJlc3VsdCc7XG5leHBvcnQgKiBmcm9tICcuL2RlcGxveW1lbnQtbWV0aG9kJztcbmV4cG9ydCAqIGZyb20gJy4vbmVzdGVkLXN0YWNrLWhlbHBlcnMnO1xuZXhwb3J0ICogZnJvbSAnLi9hc3NldC1tYW5pZmVzdC1idWlsZGVyJztcbiJdfQ==
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import type { CloudFormationStackArtifact } from '@aws-cdk/cx-api';
|
|
2
|
+
import type { SDK } from '../aws-auth';
|
|
3
|
+
import { type Template } from './cloudformation';
|
|
4
|
+
export interface NestedStackTemplates {
|
|
5
|
+
readonly physicalName: string | undefined;
|
|
6
|
+
readonly deployedTemplate: Template;
|
|
7
|
+
readonly generatedTemplate: Template;
|
|
8
|
+
readonly nestedStackTemplates: {
|
|
9
|
+
[nestedStackLogicalId: string]: NestedStackTemplates;
|
|
10
|
+
};
|
|
11
|
+
}
|
|
12
|
+
export interface RootTemplateWithNestedStacks {
|
|
13
|
+
readonly deployedRootTemplate: Template;
|
|
14
|
+
readonly nestedStacks: {
|
|
15
|
+
[nestedStackLogicalId: string]: NestedStackTemplates;
|
|
16
|
+
};
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Reads the currently deployed template and all of its nested stack templates from CloudFormation.
|
|
20
|
+
*/
|
|
21
|
+
export declare function loadCurrentTemplateWithNestedStacks(rootStackArtifact: CloudFormationStackArtifact, sdk: SDK, retrieveProcessedTemplate?: boolean): Promise<RootTemplateWithNestedStacks>;
|
|
22
|
+
/**
|
|
23
|
+
* Returns the currently deployed template from CloudFormation that corresponds to `stackArtifact`.
|
|
24
|
+
*/
|
|
25
|
+
export declare function loadCurrentTemplate(stackArtifact: CloudFormationStackArtifact, sdk: SDK, retrieveProcessedTemplate?: boolean): Promise<Template>;
|
|
@@ -0,0 +1,88 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.loadCurrentTemplateWithNestedStacks = loadCurrentTemplateWithNestedStacks;
|
|
4
|
+
exports.loadCurrentTemplate = loadCurrentTemplate;
|
|
5
|
+
const path = require("path");
|
|
6
|
+
const fs = require("fs-extra");
|
|
7
|
+
const cloudformation_1 = require("./cloudformation");
|
|
8
|
+
const error_1 = require("../../util/error");
|
|
9
|
+
const evaluate_cloudformation_template_1 = require("../evaluate-cloudformation-template");
|
|
10
|
+
/**
|
|
11
|
+
* Reads the currently deployed template and all of its nested stack templates from CloudFormation.
|
|
12
|
+
*/
|
|
13
|
+
async function loadCurrentTemplateWithNestedStacks(rootStackArtifact, sdk, retrieveProcessedTemplate = false) {
|
|
14
|
+
const deployedRootTemplate = await loadCurrentTemplate(rootStackArtifact, sdk, retrieveProcessedTemplate);
|
|
15
|
+
const nestedStacks = await loadNestedStacks(rootStackArtifact, sdk, {
|
|
16
|
+
generatedTemplate: rootStackArtifact.template,
|
|
17
|
+
deployedTemplate: deployedRootTemplate,
|
|
18
|
+
deployedStackName: rootStackArtifact.stackName,
|
|
19
|
+
});
|
|
20
|
+
return {
|
|
21
|
+
deployedRootTemplate,
|
|
22
|
+
nestedStacks,
|
|
23
|
+
};
|
|
24
|
+
}
|
|
25
|
+
/**
|
|
26
|
+
* Returns the currently deployed template from CloudFormation that corresponds to `stackArtifact`.
|
|
27
|
+
*/
|
|
28
|
+
async function loadCurrentTemplate(stackArtifact, sdk, retrieveProcessedTemplate = false) {
|
|
29
|
+
return loadCurrentStackTemplate(stackArtifact.stackName, sdk, retrieveProcessedTemplate);
|
|
30
|
+
}
|
|
31
|
+
async function loadCurrentStackTemplate(stackName, sdk, retrieveProcessedTemplate = false) {
|
|
32
|
+
const cfn = sdk.cloudFormation();
|
|
33
|
+
const stack = await cloudformation_1.CloudFormationStack.lookup(cfn, stackName, retrieveProcessedTemplate);
|
|
34
|
+
return stack.template();
|
|
35
|
+
}
|
|
36
|
+
async function loadNestedStacks(rootStackArtifact, sdk, parentTemplates) {
|
|
37
|
+
var _a;
|
|
38
|
+
const listStackResources = parentTemplates.deployedStackName
|
|
39
|
+
? new evaluate_cloudformation_template_1.LazyListStackResources(sdk, parentTemplates.deployedStackName)
|
|
40
|
+
: undefined;
|
|
41
|
+
const nestedStacks = {};
|
|
42
|
+
for (const [nestedStackLogicalId, generatedNestedStackResource] of Object.entries((_a = parentTemplates.generatedTemplate.Resources) !== null && _a !== void 0 ? _a : {})) {
|
|
43
|
+
if (!isCdkManagedNestedStack(generatedNestedStackResource)) {
|
|
44
|
+
continue;
|
|
45
|
+
}
|
|
46
|
+
const assetPath = generatedNestedStackResource.Metadata['aws:asset:path'];
|
|
47
|
+
const nestedStackTemplates = await getNestedStackTemplates(rootStackArtifact, assetPath, nestedStackLogicalId, listStackResources, sdk);
|
|
48
|
+
nestedStacks[nestedStackLogicalId] = {
|
|
49
|
+
deployedTemplate: nestedStackTemplates.deployedTemplate,
|
|
50
|
+
generatedTemplate: nestedStackTemplates.generatedTemplate,
|
|
51
|
+
physicalName: nestedStackTemplates.deployedStackName,
|
|
52
|
+
nestedStackTemplates: await loadNestedStacks(rootStackArtifact, sdk, nestedStackTemplates),
|
|
53
|
+
};
|
|
54
|
+
}
|
|
55
|
+
return nestedStacks;
|
|
56
|
+
}
|
|
57
|
+
async function getNestedStackTemplates(rootStackArtifact, nestedTemplateAssetPath, nestedStackLogicalId, listStackResources, sdk) {
|
|
58
|
+
const nestedTemplatePath = path.join(rootStackArtifact.assembly.directory, nestedTemplateAssetPath);
|
|
59
|
+
// CFN generates the nested stack name in the form `ParentStackName-NestedStackLogicalID-SomeHashWeCan'tCompute,
|
|
60
|
+
// the arn is of the form: arn:aws:cloudformation:region:123456789012:stack/NestedStackName/AnotherHashWeDon'tNeed
|
|
61
|
+
// so we get the ARN and manually extract the name.
|
|
62
|
+
const nestedStackArn = await getNestedStackArn(nestedStackLogicalId, listStackResources);
|
|
63
|
+
const deployedStackName = nestedStackArn === null || nestedStackArn === void 0 ? void 0 : nestedStackArn.slice(nestedStackArn.indexOf('/') + 1, nestedStackArn.lastIndexOf('/'));
|
|
64
|
+
return {
|
|
65
|
+
generatedTemplate: JSON.parse(fs.readFileSync(nestedTemplatePath, 'utf-8')),
|
|
66
|
+
deployedTemplate: deployedStackName ? await loadCurrentStackTemplate(deployedStackName, sdk) : {},
|
|
67
|
+
deployedStackName,
|
|
68
|
+
};
|
|
69
|
+
}
|
|
70
|
+
async function getNestedStackArn(nestedStackLogicalId, listStackResources) {
|
|
71
|
+
var _a;
|
|
72
|
+
try {
|
|
73
|
+
const stackResources = await (listStackResources === null || listStackResources === void 0 ? void 0 : listStackResources.listStackResources());
|
|
74
|
+
return (_a = stackResources === null || stackResources === void 0 ? void 0 : stackResources.find((sr) => sr.LogicalResourceId === nestedStackLogicalId)) === null || _a === void 0 ? void 0 : _a.PhysicalResourceId;
|
|
75
|
+
}
|
|
76
|
+
catch (e) {
|
|
77
|
+
if ((0, error_1.formatErrorMessage)(e).startsWith('Stack with id ') && (0, error_1.formatErrorMessage)(e).endsWith(' does not exist')) {
|
|
78
|
+
return;
|
|
79
|
+
}
|
|
80
|
+
throw e;
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
function isCdkManagedNestedStack(stackResource) {
|
|
84
|
+
return (stackResource.Type === 'AWS::CloudFormation::Stack' &&
|
|
85
|
+
stackResource.Metadata &&
|
|
86
|
+
stackResource.Metadata['aws:asset:path']);
|
|
87
|
+
}
|
|
88
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"nested-stack-helpers.js","sourceRoot":"","sources":["nested-stack-helpers.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport type { CloudFormationStackArtifact } from '@aws-cdk/cx-api';\nimport * as fs from 'fs-extra';\nimport type { SDK } from '../aws-auth';\nimport { CloudFormationStack, type Template } from './cloudformation';\nimport { formatErrorMessage } from '../../util/error';\nimport { LazyListStackResources, type ListStackResources } from '../evaluate-cloudformation-template';\n\nexport interface NestedStackTemplates {\n  readonly physicalName: string | undefined;\n  readonly deployedTemplate: Template;\n  readonly generatedTemplate: Template;\n  readonly nestedStackTemplates: {\n    [nestedStackLogicalId: string]: NestedStackTemplates;\n  };\n}\n\nexport interface RootTemplateWithNestedStacks {\n  readonly deployedRootTemplate: Template;\n  readonly nestedStacks: {\n    [nestedStackLogicalId: string]: NestedStackTemplates;\n  };\n}\n\n/**\n * Reads the currently deployed template and all of its nested stack templates from CloudFormation.\n */\nexport async function loadCurrentTemplateWithNestedStacks(\n  rootStackArtifact: CloudFormationStackArtifact,\n  sdk: SDK,\n  retrieveProcessedTemplate: boolean = false,\n): Promise<RootTemplateWithNestedStacks> {\n  const deployedRootTemplate = await loadCurrentTemplate(rootStackArtifact, sdk, retrieveProcessedTemplate);\n  const nestedStacks = await loadNestedStacks(rootStackArtifact, sdk, {\n    generatedTemplate: rootStackArtifact.template,\n    deployedTemplate: deployedRootTemplate,\n    deployedStackName: rootStackArtifact.stackName,\n  });\n\n  return {\n    deployedRootTemplate,\n    nestedStacks,\n  };\n}\n\n/**\n * Returns the currently deployed template from CloudFormation that corresponds to `stackArtifact`.\n */\nexport async function loadCurrentTemplate(\n  stackArtifact: CloudFormationStackArtifact,\n  sdk: SDK,\n  retrieveProcessedTemplate: boolean = false,\n): Promise<Template> {\n  return loadCurrentStackTemplate(stackArtifact.stackName, sdk, retrieveProcessedTemplate);\n}\n\nasync function loadCurrentStackTemplate(\n  stackName: string,\n  sdk: SDK,\n  retrieveProcessedTemplate: boolean = false,\n): Promise<Template> {\n  const cfn = sdk.cloudFormation();\n  const stack = await CloudFormationStack.lookup(cfn, stackName, retrieveProcessedTemplate);\n  return stack.template();\n}\n\nasync function loadNestedStacks(\n  rootStackArtifact: CloudFormationStackArtifact,\n  sdk: SDK,\n  parentTemplates: StackTemplates,\n): Promise<{ [nestedStackLogicalId: string]: NestedStackTemplates }> {\n  const listStackResources = parentTemplates.deployedStackName\n    ? new LazyListStackResources(sdk, parentTemplates.deployedStackName)\n    : undefined;\n  const nestedStacks: { [nestedStackLogicalId: string]: NestedStackTemplates } = {};\n  for (const [nestedStackLogicalId, generatedNestedStackResource] of Object.entries(\n    parentTemplates.generatedTemplate.Resources ?? {},\n  )) {\n    if (!isCdkManagedNestedStack(generatedNestedStackResource)) {\n      continue;\n    }\n\n    const assetPath = generatedNestedStackResource.Metadata['aws:asset:path'];\n    const nestedStackTemplates = await getNestedStackTemplates(\n      rootStackArtifact,\n      assetPath,\n      nestedStackLogicalId,\n      listStackResources,\n      sdk,\n    );\n\n    nestedStacks[nestedStackLogicalId] = {\n      deployedTemplate: nestedStackTemplates.deployedTemplate,\n      generatedTemplate: nestedStackTemplates.generatedTemplate,\n      physicalName: nestedStackTemplates.deployedStackName,\n      nestedStackTemplates: await loadNestedStacks(rootStackArtifact, sdk, nestedStackTemplates),\n    };\n  }\n\n  return nestedStacks;\n}\n\nasync function getNestedStackTemplates(\n  rootStackArtifact: CloudFormationStackArtifact,\n  nestedTemplateAssetPath: string,\n  nestedStackLogicalId: string,\n  listStackResources: ListStackResources | undefined,\n  sdk: SDK,\n): Promise<StackTemplates> {\n  const nestedTemplatePath = path.join(rootStackArtifact.assembly.directory, nestedTemplateAssetPath);\n\n  // CFN generates the nested stack name in the form `ParentStackName-NestedStackLogicalID-SomeHashWeCan'tCompute,\n  // the arn is of the form: arn:aws:cloudformation:region:123456789012:stack/NestedStackName/AnotherHashWeDon'tNeed\n  // so we get the ARN and manually extract the name.\n  const nestedStackArn = await getNestedStackArn(nestedStackLogicalId, listStackResources);\n  const deployedStackName = nestedStackArn?.slice(nestedStackArn.indexOf('/') + 1, nestedStackArn.lastIndexOf('/'));\n\n  return {\n    generatedTemplate: JSON.parse(fs.readFileSync(nestedTemplatePath, 'utf-8')),\n    deployedTemplate: deployedStackName ? await loadCurrentStackTemplate(deployedStackName, sdk) : {},\n    deployedStackName,\n  };\n}\n\nasync function getNestedStackArn(\n  nestedStackLogicalId: string,\n  listStackResources?: ListStackResources,\n): Promise<string | undefined> {\n  try {\n    const stackResources = await listStackResources?.listStackResources();\n    return stackResources?.find((sr) => sr.LogicalResourceId === nestedStackLogicalId)?.PhysicalResourceId;\n  } catch (e: any) {\n    if (formatErrorMessage(e).startsWith('Stack with id ') && formatErrorMessage(e).endsWith(' does not exist')) {\n      return;\n    }\n    throw e;\n  }\n}\n\nfunction isCdkManagedNestedStack(stackResource: any): stackResource is NestedStackResource {\n  return (\n    stackResource.Type === 'AWS::CloudFormation::Stack' &&\n    stackResource.Metadata &&\n    stackResource.Metadata['aws:asset:path']\n  );\n}\n\ninterface StackTemplates {\n  readonly generatedTemplate: any;\n  readonly deployedTemplate: any;\n  readonly deployedStackName: string | undefined;\n}\n\ninterface NestedStackResource {\n  readonly Metadata: { 'aws:asset:path': string };\n  readonly Properties: any;\n}\n"]}
|