@aws-cdk/toolkit-lib 0.3.6 → 0.4.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 +5 -0
- package/api-extractor.json +1 -1
- package/build-info.json +2 -2
- package/db.json.gz +0 -0
- package/lib/actions/deploy/index.d.ts +106 -34
- package/lib/actions/deploy/index.js +2 -17
- package/lib/actions/deploy/private/deploy-options.d.ts +1 -90
- package/lib/actions/deploy/private/deploy-options.js +1 -1
- package/lib/actions/deploy/private/helpers.d.ts +1 -6
- package/lib/actions/deploy/private/helpers.js +1 -9
- package/lib/actions/destroy/index.d.ts +4 -2
- package/lib/actions/destroy/index.js +1 -1
- package/lib/actions/diff/index.d.ts +4 -10
- package/lib/actions/diff/index.js +1 -1
- package/lib/actions/diff/private/helpers.d.ts +2 -1
- package/lib/actions/diff/private/helpers.js +6 -6
- package/lib/actions/drift/index.d.ts +50 -0
- package/lib/actions/drift/index.js +3 -0
- package/lib/actions/index.d.ts +1 -0
- package/lib/actions/index.js +2 -1
- package/lib/actions/rollback/index.d.ts +3 -1
- package/lib/actions/rollback/index.js +1 -1
- package/lib/actions/watch/index.d.ts +7 -1
- package/lib/actions/watch/index.js +1 -1
- package/lib/api/aws-auth/awscli-compatible.d.ts +5 -5
- package/lib/api/aws-auth/awscli-compatible.js +13 -17
- package/lib/api/aws-auth/base-credentials.d.ts +109 -0
- package/lib/api/aws-auth/base-credentials.js +79 -0
- package/lib/api/aws-auth/credential-plugins.js +3 -3
- package/lib/api/aws-auth/index.d.ts +1 -0
- package/lib/api/aws-auth/index.js +2 -1
- package/lib/api/aws-auth/private/index.d.ts +0 -1
- package/lib/api/aws-auth/private/index.js +1 -2
- package/lib/api/aws-auth/sdk-provider.d.ts +23 -27
- package/lib/api/aws-auth/sdk-provider.js +19 -16
- package/lib/api/aws-auth/sdk.d.ts +9 -5
- package/lib/api/aws-auth/sdk.js +6 -3
- package/lib/api/aws-auth/types.d.ts +7 -87
- package/lib/api/aws-auth/types.js +1 -74
- package/lib/api/bootstrap/bootstrap-environment.d.ts +1 -1
- package/lib/api/bootstrap/bootstrap-environment.js +8 -9
- package/lib/api/bootstrap/deploy-bootstrap.d.ts +1 -1
- package/lib/api/bootstrap/deploy-bootstrap.js +4 -5
- package/lib/api/cloud-assembly/context-store.d.ts +78 -0
- package/lib/api/cloud-assembly/context-store.js +160 -0
- package/lib/api/cloud-assembly/environment.d.ts +19 -16
- package/lib/api/cloud-assembly/environment.js +21 -12
- package/lib/api/cloud-assembly/index.d.ts +1 -0
- package/lib/api/cloud-assembly/index.js +2 -1
- package/lib/api/cloud-assembly/private/context-aware-source.d.ts +3 -10
- package/lib/api/cloud-assembly/private/context-aware-source.js +6 -11
- package/lib/api/cloud-assembly/private/exec.d.ts +1 -1
- package/lib/api/cloud-assembly/private/exec.js +2 -5
- package/lib/api/cloud-assembly/private/helpers.d.ts +9 -0
- package/lib/api/cloud-assembly/private/helpers.js +44 -0
- package/lib/api/cloud-assembly/private/index.d.ts +0 -1
- package/lib/api/cloud-assembly/private/index.js +1 -2
- package/lib/api/cloud-assembly/private/prepare-source.d.ts +27 -16
- package/lib/api/cloud-assembly/private/prepare-source.js +49 -46
- package/lib/api/cloud-assembly/private/stack-assembly.d.ts +1 -1
- package/lib/api/cloud-assembly/private/stack-assembly.js +1 -1
- package/lib/api/cloud-assembly/source-builder.d.ts +142 -14
- package/lib/api/cloud-assembly/source-builder.js +307 -1
- package/lib/api/cloud-assembly/stack-assembly.js +3 -4
- package/lib/api/cloudformation/template-body-parameter.d.ts +1 -1
- package/lib/api/cloudformation/template-body-parameter.js +4 -5
- package/lib/api/context.d.ts +1 -1
- package/lib/api/context.js +1 -1
- package/lib/api/deployments/asset-publishing.js +15 -16
- package/lib/api/deployments/assets.d.ts +1 -1
- package/lib/api/deployments/assets.js +4 -5
- package/lib/api/deployments/cfn-api.d.ts +1 -1
- package/lib/api/deployments/cfn-api.js +14 -15
- package/lib/api/deployments/checks.d.ts +1 -1
- package/lib/api/deployments/checks.js +3 -4
- package/lib/api/deployments/deploy-stack.d.ts +14 -3
- package/lib/api/deployments/deploy-stack.js +73 -49
- package/lib/api/deployments/deployments.d.ts +13 -2
- package/lib/api/deployments/deployments.js +10 -9
- package/lib/api/deployments/index.d.ts +0 -1
- package/lib/api/deployments/index.js +1 -2
- package/lib/api/diff/diff-formatter.d.ts +3 -9
- package/lib/api/diff/diff-formatter.js +7 -14
- package/lib/api/drift/drift-formatter.d.ts +81 -0
- package/lib/api/drift/drift-formatter.js +201 -0
- package/lib/api/drift/drift.d.ts +12 -0
- package/lib/api/drift/drift.js +63 -0
- package/lib/api/drift/index.d.ts +2 -0
- package/lib/api/{io/private/testing → drift}/index.js +3 -3
- package/lib/api/environment/environment-access.d.ts +1 -1
- package/lib/api/environment/environment-access.js +3 -4
- package/lib/api/environment/environment-resources.d.ts +1 -1
- package/lib/api/environment/environment-resources.js +5 -6
- package/lib/api/garbage-collection/garbage-collector.js +44 -35
- package/lib/api/garbage-collection/progress-printer.d.ts +1 -1
- package/lib/api/garbage-collection/progress-printer.js +3 -4
- package/lib/api/garbage-collection/stack-refresh.d.ts +1 -1
- package/lib/api/garbage-collection/stack-refresh.js +3 -4
- package/lib/api/hotswap/common.d.ts +9 -4
- package/lib/api/hotswap/common.js +11 -4
- package/lib/api/hotswap/ecs-services.js +2 -2
- package/lib/api/hotswap/hotswap-deployments.js +3 -3
- package/lib/api/index.d.ts +1 -1
- package/lib/api/index.js +2 -2
- package/lib/api/io/io-host.d.ts +15 -1
- package/lib/api/io/io-host.js +1 -1
- package/lib/api/io/io-message.d.ts +7 -10
- package/lib/api/io/io-message.js +1 -1
- package/lib/api/io/private/index.d.ts +0 -1
- package/lib/api/io/private/index.js +1 -2
- package/lib/api/io/private/io-default-messages.d.ts +14 -10
- package/lib/api/io/private/io-default-messages.js +30 -28
- package/lib/api/io/private/io-helper.d.ts +9 -4
- package/lib/api/io/private/io-helper.js +24 -14
- package/lib/api/io/private/message-maker.d.ts +5 -1
- package/lib/api/io/private/message-maker.js +21 -1
- package/lib/api/io/private/messages.d.ts +20 -15
- package/lib/api/io/private/messages.js +57 -54
- package/lib/api/io/private/span.d.ts +18 -11
- package/lib/api/io/private/span.js +60 -42
- package/lib/api/io/toolkit-action.d.ts +1 -1
- package/lib/api/io/toolkit-action.js +1 -1
- package/lib/api/logs-monitor/find-cloudwatch-logs.js +2 -3
- package/lib/api/notices/cached-data-source.d.ts +3 -3
- package/lib/api/notices/cached-data-source.js +9 -8
- package/lib/api/notices/filter.d.ts +4 -4
- package/lib/api/notices/filter.js +27 -25
- package/lib/api/notices/notices.d.ts +18 -6
- package/lib/api/notices/notices.js +5 -7
- package/lib/api/notices/web-data-source.d.ts +30 -3
- package/lib/api/notices/web-data-source.js +37 -10
- package/lib/api/plugin/plugin.d.ts +1 -1
- package/lib/api/plugin/plugin.js +3 -3
- package/lib/api/refactoring/cloudformation.d.ts +7 -5
- package/lib/api/refactoring/cloudformation.js +1 -1
- package/lib/api/refactoring/digest.d.ts +2 -2
- package/lib/api/refactoring/digest.js +25 -61
- package/lib/api/refactoring/graph.d.ts +15 -0
- package/lib/api/refactoring/graph.js +108 -0
- package/lib/api/refactoring/index.js +12 -6
- package/lib/api/resource-import/importer.d.ts +2 -1
- package/lib/api/resource-import/importer.js +28 -28
- package/lib/api/resource-import/migrator.js +4 -4
- package/lib/api/toolkit-info.d.ts +1 -1
- package/lib/api/toolkit-info.js +3 -4
- package/lib/api/tree.d.ts +1 -1
- package/lib/api/tree.js +3 -3
- package/lib/api/work-graph/work-graph.d.ts +1 -1
- package/lib/api/work-graph/work-graph.js +4 -5
- package/lib/context-providers/index.d.ts +1 -2
- package/lib/context-providers/index.js +6 -4
- package/lib/index_bg.wasm +0 -0
- package/lib/payloads/context.d.ts +0 -1
- package/lib/payloads/context.js +1 -1
- package/lib/payloads/deploy.d.ts +2 -2
- package/lib/payloads/deploy.js +1 -1
- package/lib/payloads/destroy.d.ts +3 -3
- package/lib/payloads/destroy.js +1 -1
- package/lib/payloads/diff.d.ts +46 -5
- package/lib/payloads/diff.js +1 -1
- package/lib/payloads/drift.d.ts +12 -0
- package/lib/payloads/drift.js +3 -0
- package/lib/payloads/gc.d.ts +12 -0
- package/lib/payloads/gc.js +3 -0
- package/lib/payloads/import.d.ts +45 -0
- package/lib/payloads/import.js +3 -0
- package/lib/payloads/index.d.ts +4 -1
- package/lib/payloads/index.js +5 -2
- package/lib/payloads/rollback.d.ts +2 -2
- package/lib/payloads/rollback.js +1 -1
- package/lib/payloads/{sdk-trace.d.ts → sdk.d.ts} +10 -0
- package/lib/payloads/sdk.js +3 -0
- package/lib/payloads/stack-activity.d.ts +2 -2
- package/lib/payloads/stack-activity.js +1 -1
- package/lib/payloads/types.d.ts +21 -0
- package/lib/payloads/types.js +1 -1
- package/lib/toolkit/non-interactive-io-host.js +2 -2
- package/lib/toolkit/private/index.d.ts +2 -1
- package/lib/toolkit/private/index.js +1 -1
- package/lib/toolkit/toolkit.d.ts +14 -7
- package/lib/toolkit/toolkit.js +180 -103
- package/lib/util/objects.d.ts +4 -0
- package/lib/util/objects.js +8 -1
- package/package.json +18 -21
- package/lib/actions/import/index.d.ts +0 -21
- package/lib/actions/import/index.js +0 -3
- package/lib/api/aws-auth/proxy-agent.d.ts +0 -13
- package/lib/api/aws-auth/proxy-agent.js +0 -54
- package/lib/api/cloud-assembly/private/source-builder.d.ts +0 -52
- package/lib/api/cloud-assembly/private/source-builder.js +0 -262
- package/lib/api/deployments/deployment-method.d.ts +0 -24
- package/lib/api/deployments/deployment-method.js +0 -3
- package/lib/api/io/private/testing/fake-io-host.d.ts +0 -28
- package/lib/api/io/private/testing/fake-io-host.js +0 -41
- package/lib/api/io/private/testing/index.d.ts +0 -2
- package/lib/api/io/private/testing/test-io-host.d.ts +0 -27
- package/lib/api/io/private/testing/test-io-host.js +0 -61
- package/lib/api/require-approval.d.ts +0 -17
- package/lib/api/require-approval.js +0 -22
- package/lib/api/shared-private.d.ts +0 -8
- package/lib/api/shared-private.js +0 -32
- package/lib/payloads/sdk-trace.js +0 -3
|
@@ -1,22 +1,49 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.WebsiteNoticeDataSource = void 0;
|
|
3
|
+
exports.WebsiteNoticeDataSource = exports.WebsiteNoticeDataSourceProps = void 0;
|
|
4
4
|
const https = require("node:https");
|
|
5
5
|
const toolkit_error_1 = require("../../toolkit/toolkit-error");
|
|
6
6
|
const util_1 = require("../../util");
|
|
7
|
-
|
|
8
|
-
|
|
7
|
+
/**
|
|
8
|
+
* A data source that fetches notices from the CDK notices data source
|
|
9
|
+
*/
|
|
10
|
+
class WebsiteNoticeDataSourceProps {
|
|
11
|
+
/**
|
|
12
|
+
* The URL to load notices from.
|
|
13
|
+
*
|
|
14
|
+
* Note this must be a valid JSON document in the CDK notices data schema.
|
|
15
|
+
*
|
|
16
|
+
* @see https://github.com/cdklabs/aws-cdk-notices
|
|
17
|
+
*
|
|
18
|
+
* @default - official CDK notices
|
|
19
|
+
*/
|
|
20
|
+
url;
|
|
21
|
+
/**
|
|
22
|
+
* The agent responsible for making the network requests.
|
|
23
|
+
*
|
|
24
|
+
* Use this so set up a proxy connection.
|
|
25
|
+
*
|
|
26
|
+
* @default - uses the shared global node agent
|
|
27
|
+
*/
|
|
28
|
+
agent;
|
|
29
|
+
}
|
|
30
|
+
exports.WebsiteNoticeDataSourceProps = WebsiteNoticeDataSourceProps;
|
|
9
31
|
class WebsiteNoticeDataSource {
|
|
10
32
|
ioHelper;
|
|
11
|
-
|
|
12
|
-
|
|
33
|
+
/**
|
|
34
|
+
* The URL notices are loaded from.
|
|
35
|
+
*/
|
|
36
|
+
url;
|
|
37
|
+
agent;
|
|
38
|
+
constructor(ioHelper, props = {}) {
|
|
13
39
|
this.ioHelper = ioHelper;
|
|
14
|
-
this.
|
|
40
|
+
this.agent = props.agent;
|
|
41
|
+
this.url = props.url ?? 'https://cli.cdk.dev-tools.aws.dev/notices.json';
|
|
15
42
|
}
|
|
16
43
|
async fetch() {
|
|
17
44
|
const timeout = 3000;
|
|
18
45
|
const options = {
|
|
19
|
-
agent:
|
|
46
|
+
agent: this.agent,
|
|
20
47
|
};
|
|
21
48
|
const notices = await new Promise((resolve, reject) => {
|
|
22
49
|
let req;
|
|
@@ -27,7 +54,7 @@ class WebsiteNoticeDataSource {
|
|
|
27
54
|
}, timeout);
|
|
28
55
|
timer.unref();
|
|
29
56
|
try {
|
|
30
|
-
req = https.get(
|
|
57
|
+
req = https.get(this.url, options, res => {
|
|
31
58
|
if (res.statusCode === 200) {
|
|
32
59
|
res.setEncoding('utf8');
|
|
33
60
|
let rawData = '';
|
|
@@ -62,9 +89,9 @@ class WebsiteNoticeDataSource {
|
|
|
62
89
|
reject(toolkit_error_1.ToolkitError.withCause((0, util_1.formatErrorMessage)(e), e));
|
|
63
90
|
}
|
|
64
91
|
});
|
|
65
|
-
await this.ioHelper.
|
|
92
|
+
await this.ioHelper.defaults.debug('Notices refreshed');
|
|
66
93
|
return notices;
|
|
67
94
|
}
|
|
68
95
|
}
|
|
69
96
|
exports.WebsiteNoticeDataSource = WebsiteNoticeDataSource;
|
|
70
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
97
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid2ViLWRhdGEtc291cmNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsid2ViLWRhdGEtc291cmNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7OztBQUVBLG9DQUFvQztBQUVwQywrREFBMkQ7QUFDM0QscUNBQXlGO0FBR3pGOztHQUVHO0FBQ0gsTUFBYSw0QkFBNEI7SUFDdkM7Ozs7Ozs7O09BUUc7SUFDTSxHQUFHLENBQWdCO0lBQzVCOzs7Ozs7T0FNRztJQUNNLEtBQUssQ0FBZTtDQUM5QjtBQW5CRCxvRUFtQkM7QUFFRCxNQUFhLHVCQUF1QjtJQVFMO0lBUDdCOztPQUVHO0lBQ2EsR0FBRyxDQUFNO0lBRVIsS0FBSyxDQUFlO0lBRXJDLFlBQTZCLFFBQWtCLEVBQUUsUUFBc0MsRUFBRTtRQUE1RCxhQUFRLEdBQVIsUUFBUSxDQUFVO1FBQzdDLElBQUksQ0FBQyxLQUFLLEdBQUcsS0FBSyxDQUFDLEtBQUssQ0FBQztRQUN6QixJQUFJLENBQUMsR0FBRyxHQUFHLEtBQUssQ0FBQyxHQUFHLElBQUksZ0RBQWdELENBQUM7SUFDM0UsQ0FBQztJQUVELEtBQUssQ0FBQyxLQUFLO1FBQ1QsTUFBTSxPQUFPLEdBQUcsSUFBSSxDQUFDO1FBRXJCLE1BQU0sT0FBTyxHQUFtQjtZQUM5QixLQUFLLEVBQUUsSUFBSSxDQUFDLEtBQUs7U0FDbEIsQ0FBQztRQUVGLE1BQU0sT0FBTyxHQUFHLE1BQU0sSUFBSSxPQUFPLENBQVcsQ0FBQyxPQUFPLEVBQUUsTUFBTSxFQUFFLEVBQUU7WUFDOUQsSUFBSSxHQUE4QixDQUFDO1lBRW5DLElBQUksS0FBSyxHQUFHLFVBQVUsQ0FBQyxHQUFHLEVBQUU7Z0JBQzFCLElBQUksR0FBRyxFQUFFLENBQUM7b0JBQ1IsR0FBRyxDQUFDLE9BQU8sQ0FBQyxJQUFJLDRCQUFZLENBQUMsbUJBQW1CLENBQUMsQ0FBQyxDQUFDO2dCQUNyRCxDQUFDO1lBQ0gsQ0FBQyxFQUFFLE9BQU8sQ0FBQyxDQUFDO1lBRVosS0FBSyxDQUFDLEtBQUssRUFBRSxDQUFDO1lBRWQsSUFBSSxDQUFDO2dCQUNILEdBQUcsR0FBRyxLQUFLLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxHQUFHLEVBQ3RCLE9BQU8sRUFDUCxHQUFHLENBQUMsRUFBRTtvQkFDSixJQUFJLEdBQUcsQ0FBQyxVQUFVLEtBQUssR0FBRyxFQUFFLENBQUM7d0JBQzNCLEdBQUcsQ0FBQyxXQUFXLENBQUMsTUFBTSxDQUFDLENBQUM7d0JBQ3hCLElBQUksT0FBTyxHQUFHLEVBQUUsQ0FBQzt3QkFDakIsR0FBRyxDQUFDLEVBQUUsQ0FBQyxNQUFNLEVBQUUsQ0FBQyxLQUFLLEVBQUUsRUFBRTs0QkFDdkIsT0FBTyxJQUFJLEtBQUssQ0FBQzt3QkFDbkIsQ0FBQyxDQUFDLENBQUM7d0JBQ0gsR0FBRyxDQUFDLEVBQUUsQ0FBQyxLQUFLLEVBQUUsR0FBRyxFQUFFOzRCQUNqQixJQUFJLENBQUM7Z0NBQ0gsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQyxPQUFtQixDQUFDO2dDQUNyRCxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7b0NBQ1YsTUFBTSxJQUFJLDRCQUFZLENBQUMsNkNBQTZDLENBQUMsQ0FBQztnQ0FDeEUsQ0FBQztnQ0FDRCxPQUFPLENBQUMsSUFBSSxJQUFJLEVBQUUsQ0FBQyxDQUFDOzRCQUN0QixDQUFDOzRCQUFDLE9BQU8sQ0FBTSxFQUFFLENBQUM7Z0NBQ2hCLE1BQU0sQ0FBQyw0QkFBWSxDQUFDLFNBQVMsQ0FBQyxnQkFBZ0IsSUFBQSx5QkFBa0IsRUFBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7NEJBQzdFLENBQUM7d0JBQ0gsQ0FBQyxDQUFDLENBQUM7d0JBQ0gsR0FBRyxDQUFDLEVBQUUsQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQUFDLEVBQUU7NEJBQ2xCLE1BQU0sQ0FBQyw0QkFBWSxDQUFDLFNBQVMsQ0FBQyxJQUFBLHlCQUFrQixFQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7d0JBQzNELENBQUMsQ0FBQyxDQUFDO29CQUNMLENBQUM7eUJBQU0sQ0FBQzt3QkFDTixNQUFNLENBQUMsSUFBSSw0QkFBWSxDQUFDLEdBQUcsSUFBQSwyQkFBb0IsRUFBQyxHQUFHLENBQUMsVUFBVyxDQUFDLGtCQUFrQixHQUFHLENBQUMsVUFBVSxHQUFHLENBQUMsQ0FBQyxDQUFDO29CQUN4RyxDQUFDO2dCQUNILENBQUMsQ0FBQyxDQUFDO2dCQUNMLEdBQUcsQ0FBQyxFQUFFLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQyxFQUFFO29CQUNsQixNQUFNLENBQUMsNEJBQVksQ0FBQyxTQUFTLENBQUMsSUFBQSx3QkFBaUIsRUFBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO2dCQUMxRCxDQUFDLENBQUMsQ0FBQztZQUNMLENBQUM7WUFBQyxPQUFPLENBQU0sRUFBRSxDQUFDO2dCQUNoQixNQUFNLENBQUMsNEJBQVksQ0FBQyxTQUFTLENBQUMsSUFBQSx5QkFBa0IsRUFBQyxDQUFDLENBQUMsRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDO1lBQzNELENBQUM7UUFDSCxDQUFDLENBQUMsQ0FBQztRQUVILE1BQU0sSUFBSSxDQUFDLFFBQVEsQ0FBQyxRQUFRLENBQUMsS0FBSyxDQUFDLG1CQUFtQixDQUFDLENBQUM7UUFDeEQsT0FBTyxPQUFPLENBQUM7SUFDakIsQ0FBQztDQUNGO0FBdEVELDBEQXNFQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgQ2xpZW50UmVxdWVzdCB9IGZyb20gJ25vZGU6aHR0cCc7XG5pbXBvcnQgdHlwZSB7IFJlcXVlc3RPcHRpb25zIH0gZnJvbSAnbm9kZTpodHRwcyc7XG5pbXBvcnQgKiBhcyBodHRwcyBmcm9tICdub2RlOmh0dHBzJztcbmltcG9ydCB0eXBlIHsgTm90aWNlLCBOb3RpY2VEYXRhU291cmNlIH0gZnJvbSAnLi90eXBlcyc7XG5pbXBvcnQgeyBUb29sa2l0RXJyb3IgfSBmcm9tICcuLi8uLi90b29sa2l0L3Rvb2xraXQtZXJyb3InO1xuaW1wb3J0IHsgZm9ybWF0RXJyb3JNZXNzYWdlLCBodW1hbkh0dHBTdGF0dXNFcnJvciwgaHVtYW5OZXR3b3JrRXJyb3IgfSBmcm9tICcuLi8uLi91dGlsJztcbmltcG9ydCB0eXBlIHsgSW9IZWxwZXIgfSBmcm9tICcuLi9pby9wcml2YXRlJztcblxuLyoqXG4gKiBBIGRhdGEgc291cmNlIHRoYXQgZmV0Y2hlcyBub3RpY2VzIGZyb20gdGhlIENESyBub3RpY2VzIGRhdGEgc291cmNlXG4gKi9cbmV4cG9ydCBjbGFzcyBXZWJzaXRlTm90aWNlRGF0YVNvdXJjZVByb3BzIHtcbiAgLyoqXG4gICAqIFRoZSBVUkwgdG8gbG9hZCBub3RpY2VzIGZyb20uXG4gICAqXG4gICAqIE5vdGUgdGhpcyBtdXN0IGJlIGEgdmFsaWQgSlNPTiBkb2N1bWVudCBpbiB0aGUgQ0RLIG5vdGljZXMgZGF0YSBzY2hlbWEuXG4gICAqXG4gICAqIEBzZWUgaHR0cHM6Ly9naXRodWIuY29tL2Nka2xhYnMvYXdzLWNkay1ub3RpY2VzXG4gICAqXG4gICAqIEBkZWZhdWx0IC0gb2ZmaWNpYWwgQ0RLIG5vdGljZXNcbiAgICovXG4gIHJlYWRvbmx5IHVybD86IHN0cmluZyB8IFVSTDtcbiAgLyoqXG4gICAqIFRoZSBhZ2VudCByZXNwb25zaWJsZSBmb3IgbWFraW5nIHRoZSBuZXR3b3JrIHJlcXVlc3RzLlxuICAgKlxuICAgKiBVc2UgdGhpcyBzbyBzZXQgdXAgYSBwcm94eSBjb25uZWN0aW9uLlxuICAgKlxuICAgKiBAZGVmYXVsdCAtIHVzZXMgdGhlIHNoYXJlZCBnbG9iYWwgbm9kZSBhZ2VudFxuICAgKi9cbiAgcmVhZG9ubHkgYWdlbnQ/OiBodHRwcy5BZ2VudDtcbn1cblxuZXhwb3J0IGNsYXNzIFdlYnNpdGVOb3RpY2VEYXRhU291cmNlIGltcGxlbWVudHMgTm90aWNlRGF0YVNvdXJjZSB7XG4gIC8qKlxuICAgKiBUaGUgVVJMIG5vdGljZXMgYXJlIGxvYWRlZCBmcm9tLlxuICAgKi9cbiAgcHVibGljIHJlYWRvbmx5IHVybDogYW55O1xuXG4gIHByaXZhdGUgcmVhZG9ubHkgYWdlbnQ/OiBodHRwcy5BZ2VudDtcblxuICBjb25zdHJ1Y3Rvcihwcml2YXRlIHJlYWRvbmx5IGlvSGVscGVyOiBJb0hlbHBlciwgcHJvcHM6IFdlYnNpdGVOb3RpY2VEYXRhU291cmNlUHJvcHMgPSB7fSkge1xuICAgIHRoaXMuYWdlbnQgPSBwcm9wcy5hZ2VudDtcbiAgICB0aGlzLnVybCA9IHByb3BzLnVybCA/PyAnaHR0cHM6Ly9jbGkuY2RrLmRldi10b29scy5hd3MuZGV2L25vdGljZXMuanNvbic7XG4gIH1cblxuICBhc3luYyBmZXRjaCgpOiBQcm9taXNlPE5vdGljZVtdPiB7XG4gICAgY29uc3QgdGltZW91dCA9IDMwMDA7XG5cbiAgICBjb25zdCBvcHRpb25zOiBSZXF1ZXN0T3B0aW9ucyA9IHtcbiAgICAgIGFnZW50OiB0aGlzLmFnZW50LFxuICAgIH07XG5cbiAgICBjb25zdCBub3RpY2VzID0gYXdhaXQgbmV3IFByb21pc2U8Tm90aWNlW10+KChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICAgIGxldCByZXE6IENsaWVudFJlcXVlc3QgfCB1bmRlZmluZWQ7XG5cbiAgICAgIGxldCB0aW1lciA9IHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgICBpZiAocmVxKSB7XG4gICAgICAgICAgcmVxLmRlc3Ryb3kobmV3IFRvb2xraXRFcnJvcignUmVxdWVzdCB0aW1lZCBvdXQnKSk7XG4gICAgICAgIH1cbiAgICAgIH0sIHRpbWVvdXQpO1xuXG4gICAgICB0aW1lci51bnJlZigpO1xuXG4gICAgICB0cnkge1xuICAgICAgICByZXEgPSBodHRwcy5nZXQodGhpcy51cmwsXG4gICAgICAgICAgb3B0aW9ucyxcbiAgICAgICAgICByZXMgPT4ge1xuICAgICAgICAgICAgaWYgKHJlcy5zdGF0dXNDb2RlID09PSAyMDApIHtcbiAgICAgICAgICAgICAgcmVzLnNldEVuY29kaW5nKCd1dGY4Jyk7XG4gICAgICAgICAgICAgIGxldCByYXdEYXRhID0gJyc7XG4gICAgICAgICAgICAgIHJlcy5vbignZGF0YScsIChjaHVuaykgPT4ge1xuICAgICAgICAgICAgICAgIHJhd0RhdGEgKz0gY2h1bms7XG4gICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgICByZXMub24oJ2VuZCcsICgpID0+IHtcbiAgICAgICAgICAgICAgICB0cnkge1xuICAgICAgICAgICAgICAgICAgY29uc3QgZGF0YSA9IEpTT04ucGFyc2UocmF3RGF0YSkubm90aWNlcyBhcyBOb3RpY2VbXTtcbiAgICAgICAgICAgICAgICAgIGlmICghZGF0YSkge1xuICAgICAgICAgICAgICAgICAgICB0aHJvdyBuZXcgVG9vbGtpdEVycm9yKFwiJ25vdGljZXMnIGtleSBpcyBtaXNzaW5nIGZyb20gcmVjZWl2ZWQgZGF0YVwiKTtcbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICAgIHJlc29sdmUoZGF0YSA/PyBbXSk7XG4gICAgICAgICAgICAgICAgfSBjYXRjaCAoZTogYW55KSB7XG4gICAgICAgICAgICAgICAgICByZWplY3QoVG9vbGtpdEVycm9yLndpdGhDYXVzZShgUGFyc2UgZXJyb3I6ICR7Zm9ybWF0RXJyb3JNZXNzYWdlKGUpfWAsIGUpKTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgICByZXMub24oJ2Vycm9yJywgZSA9PiB7XG4gICAgICAgICAgICAgICAgcmVqZWN0KFRvb2xraXRFcnJvci53aXRoQ2F1c2UoZm9ybWF0RXJyb3JNZXNzYWdlKGUpLCBlKSk7XG4gICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgcmVqZWN0KG5ldyBUb29sa2l0RXJyb3IoYCR7aHVtYW5IdHRwU3RhdHVzRXJyb3IocmVzLnN0YXR1c0NvZGUhKX0gKFN0YXR1cyBjb2RlOiAke3Jlcy5zdGF0dXNDb2RlfSlgKSk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSk7XG4gICAgICAgIHJlcS5vbignZXJyb3InLCBlID0+IHtcbiAgICAgICAgICByZWplY3QoVG9vbGtpdEVycm9yLndpdGhDYXVzZShodW1hbk5ldHdvcmtFcnJvcihlKSwgZSkpO1xuICAgICAgICB9KTtcbiAgICAgIH0gY2F0Y2ggKGU6IGFueSkge1xuICAgICAgICByZWplY3QoVG9vbGtpdEVycm9yLndpdGhDYXVzZShmb3JtYXRFcnJvck1lc3NhZ2UoZSksIGUpKTtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIGF3YWl0IHRoaXMuaW9IZWxwZXIuZGVmYXVsdHMuZGVidWcoJ05vdGljZXMgcmVmcmVzaGVkJyk7XG4gICAgcmV0dXJuIG5vdGljZXM7XG4gIH1cbn1cbiJdfQ==
|
|
@@ -27,7 +27,7 @@ export declare class PluginHost implements IPluginHost {
|
|
|
27
27
|
* @param moduleSpec - the specification (path or name) of the plug-in module to be loaded.
|
|
28
28
|
* @param ioHost - the I/O host to use for printing progress information
|
|
29
29
|
*/
|
|
30
|
-
load(moduleSpec: string, ioHost?: IIoHost): void
|
|
30
|
+
load(moduleSpec: string, ioHost?: IIoHost): Promise<void>;
|
|
31
31
|
/**
|
|
32
32
|
* Allows plug-ins to register new CredentialProviderSources.
|
|
33
33
|
*
|
package/lib/api/plugin/plugin.js
CHANGED
|
@@ -31,11 +31,11 @@ class PluginHost {
|
|
|
31
31
|
* @param moduleSpec - the specification (path or name) of the plug-in module to be loaded.
|
|
32
32
|
* @param ioHost - the I/O host to use for printing progress information
|
|
33
33
|
*/
|
|
34
|
-
load(moduleSpec, ioHost) {
|
|
34
|
+
async load(moduleSpec, ioHost) {
|
|
35
35
|
try {
|
|
36
36
|
const resolved = require.resolve(moduleSpec);
|
|
37
37
|
if (ioHost) {
|
|
38
|
-
|
|
38
|
+
await private_1.IoHelper.fromIoHost(ioHost, 'init').defaults.debug(`Loading plug-in: ${resolved} from ${moduleSpec}`);
|
|
39
39
|
}
|
|
40
40
|
return this._doLoad(resolved);
|
|
41
41
|
}
|
|
@@ -124,4 +124,4 @@ class PluginHost {
|
|
|
124
124
|
}
|
|
125
125
|
}
|
|
126
126
|
exports.PluginHost = PluginHost;
|
|
127
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
127
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"plugin.js","sourceRoot":"","sources":["plugin.ts"],"names":[],"mappings":";;;AAAA,+BAA+B;AAE/B,uEAAgG;AAChG,+DAA2D;AAE3D,2CAAyC;AAEzC;;;;;;GAMG;AACH,MAAa,UAAU;IACrB;;;OAGG;IACa,yBAAyB,GAAG,IAAI,KAAK,EAA4B,CAAC;IAElE,sBAAsB,GAA0C,EAAE,CAAC;IAE5E,MAAM,CAAW;IAEP,aAAa,GAAG,IAAI,GAAG,EAAU,CAAC;IAEnD;;;;;;;;;OASG;IACI,KAAK,CAAC,IAAI,CAAC,UAAkB,EAAE,MAAgB;QACpD,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,OAAO,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;YAC7C,IAAI,MAAM,EAAE,CAAC;gBACX,MAAM,kBAAQ,CAAC,UAAU,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,oBAAoB,QAAQ,SAAS,UAAU,EAAE,CAAC,CAAC;YAC9G,CAAC;YACD,OAAO,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QAChC,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,+EAA+E;YAC/E,yEAAyE;YACzE,gFAAgF;YAChF,wEAAwE;YACxE,MAAM,IAAI,4BAAY,CAAC,kDAAkD,UAAU,MAAM,CAAC,EAAE,CAAC,CAAC;QAChG,CAAC;IACH,CAAC;IAED;;;;OAIG;IACI,OAAO,CAAC,QAAgB;QAC7B,IAAI,CAAC;YACH,IAAI,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC;gBACrC,OAAO;YACT,CAAC;YAED,0DAA0D;YAC1D,MAAM,MAAM,GAAG,OAAO,CAAC,QAAQ,CAAC,CAAC;YACjC,mBAAmB;YACnB,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;gBACtB,MAAM,IAAI,4BAAY,CAAC,UAAU,QAAQ,yDAAyD,CAAC,CAAC;YACtG,CAAC;YACD,IAAI,MAAM,CAAC,IAAI,EAAE,CAAC;gBAChB,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YACpB,CAAC;YAED,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QACnC,CAAC;QAAC,OAAO,CAAM,EAAE,CAAC;YAChB,MAAM,4BAAY,CAAC,SAAS,CAAC,2BAA2B,QAAQ,GAAG,EAAE,CAAC,CAAC,CAAC;QAC1E,CAAC;QAED,SAAS,QAAQ,CAAC,CAAM;YACtB,OAAO,CAAC,IAAI,IAAI,IAAI,CAAC,CAAC,OAAO,KAAK,GAAG,CAAC;QACxC,CAAC;IACH,CAAC;IAED;;;;OAIG;IACI,gCAAgC,CAAC,MAAgC;QACtE,uDAAuD;QACvD,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAC9C,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;OA+BG;IACI,4BAA4B,CAAC,kBAA0B,EAAE,QAA+B;QAC7F,IAAI,CAAC,IAAA,iDAAuB,EAAC,QAAQ,CAAC,EAAE,CAAC;YACvC,MAAM,IAAI,4BAAY,CAAC,kEAAkE,IAAA,cAAO,EAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;QAChH,CAAC;QACD,IAAI,CAAC,sBAAsB,CAAC,kBAAkB,CAAC,GAAG,QAAQ,CAAC;IAC7D,CAAC;CACF;AAtHD,gCAsHC","sourcesContent":["import { inspect } from 'util';\nimport type { CredentialProviderSource, IPluginHost, Plugin } from '@aws-cdk/cli-plugin-contract';\nimport { type ContextProviderPlugin, isContextProviderPlugin } from './context-provider-plugin';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\nimport type { IIoHost } from '../io';\nimport { IoHelper } from '../io/private';\n\n/**\n * Class to manage a plugin collection\n *\n * It provides a `load()` function that loads a JavaScript\n * module from disk, and gives it access to the `IPluginHost` interface\n * to register itself.\n */\nexport class PluginHost implements IPluginHost {\n  /**\n   * Access the currently registered CredentialProviderSources. New sources can\n   * be registered using the +registerCredentialProviderSource+ method.\n   */\n  public readonly credentialProviderSources = new Array<CredentialProviderSource>();\n\n  public readonly contextProviderPlugins: Record<string, ContextProviderPlugin> = {};\n\n  public ioHost?: IIoHost;\n\n  private readonly alreadyLoaded = new Set<string>();\n\n  /**\n   * Loads a plug-in into this PluginHost.\n   *\n   * Will use `require.resolve()` to get the most accurate representation of what\n   * code will get loaded in error messages. As such, it will not work in\n   * unit tests with Jest virtual modules becauase of \\<https://github.com/jestjs/jest/issues/9543\\>.\n   *\n   * @param moduleSpec - the specification (path or name) of the plug-in module to be loaded.\n   * @param ioHost - the I/O host to use for printing progress information\n   */\n  public async load(moduleSpec: string, ioHost?: IIoHost) {\n    try {\n      const resolved = require.resolve(moduleSpec);\n      if (ioHost) {\n        await IoHelper.fromIoHost(ioHost, 'init').defaults.debug(`Loading plug-in: ${resolved} from ${moduleSpec}`);\n      }\n      return this._doLoad(resolved);\n    } catch (e: any) {\n      // according to Node.js docs `MODULE_NOT_FOUND` is the only possible error here\n      // @see https://nodejs.org/api/modules.html#requireresolverequest-options\n      // Not using `withCause()` here, since the node error contains a \"Require Stack\"\n      // as part of the error message that is inherently useless to our users.\n      throw new ToolkitError(`Unable to resolve plug-in: Cannot find module '${moduleSpec}': ${e}`);\n    }\n  }\n\n  /**\n   * Do the loading given an already-resolved module name\n   *\n   * @internal\n   */\n  public _doLoad(resolved: string) {\n    try {\n      if (this.alreadyLoaded.has(resolved)) {\n        return;\n      }\n\n      /* eslint-disable @typescript-eslint/no-require-imports */\n      const plugin = require(resolved);\n      /* eslint-enable */\n      if (!isPlugin(plugin)) {\n        throw new ToolkitError(`Module ${resolved} is not a valid plug-in, or has an unsupported version.`);\n      }\n      if (plugin.init) {\n        plugin.init(this);\n      }\n\n      this.alreadyLoaded.add(resolved);\n    } catch (e: any) {\n      throw ToolkitError.withCause(`Unable to load plug-in '${resolved}'`, e);\n    }\n\n    function isPlugin(x: any): x is Plugin {\n      return x != null && x.version === '1';\n    }\n  }\n\n  /**\n   * Allows plug-ins to register new CredentialProviderSources.\n   *\n   * @param source - a new CredentialProviderSource to register.\n   */\n  public registerCredentialProviderSource(source: CredentialProviderSource) {\n    // Forward to the right credentials-related plugin host\n    this.credentialProviderSources.push(source);\n  }\n\n  /**\n   * (EXPERIMENTAL) Allow plugins to register context providers\n   *\n   * Context providers are objects with the following method:\n   *\n   * ```ts\n   *   getValue(args: {[key: string]: any}): Promise<any>;\n   * ```\n   *\n   * Currently, they cannot reuse the CDK's authentication mechanisms, so they\n   * must be prepared to either not make AWS calls or use their own source of\n   * AWS credentials.\n   *\n   * This feature is experimental, and only intended to be used internally at Amazon\n   * as a trial.\n   *\n   * After registering with 'my-plugin-name', the provider must be addressed as follows:\n   *\n   * ```ts\n   * const value = ContextProvider.getValue(this, {\n   *   providerName: 'plugin',\n   *   props: {\n   *     pluginName: 'my-plugin-name',\n   *     myParameter1: 'xyz',\n   *   },\n   *   includeEnvironment: true | false,\n   *   dummyValue: 'what-to-return-on-the-first-pass',\n   * })\n   * ```\n   *\n   * @experimental\n   */\n  public registerContextProviderAlpha(pluginProviderName: string, provider: ContextProviderPlugin) {\n    if (!isContextProviderPlugin(provider)) {\n      throw new ToolkitError(`Object you gave me does not look like a ContextProviderPlugin: ${inspect(provider)}`);\n    }\n    this.contextProviderPlugins[pluginProviderName] = provider;\n  }\n}\n"]}
|
|
@@ -1,13 +1,15 @@
|
|
|
1
1
|
import type { TypedMapping } from '@aws-cdk/cloudformation-diff';
|
|
2
2
|
import type * as cxapi from '@aws-cdk/cx-api';
|
|
3
|
+
export interface CloudFormationResource {
|
|
4
|
+
Type: string;
|
|
5
|
+
Properties?: any;
|
|
6
|
+
Metadata?: Record<string, any>;
|
|
7
|
+
}
|
|
3
8
|
export interface CloudFormationTemplate {
|
|
4
9
|
Resources?: {
|
|
5
|
-
[logicalId: string]:
|
|
6
|
-
Type: string;
|
|
7
|
-
Properties?: any;
|
|
8
|
-
Metadata?: Record<string, any>;
|
|
9
|
-
};
|
|
10
|
+
[logicalId: string]: CloudFormationResource;
|
|
10
11
|
};
|
|
12
|
+
Outputs?: Record<string, any>;
|
|
11
13
|
}
|
|
12
14
|
export interface CloudFormationStack {
|
|
13
15
|
readonly environment: cxapi.Environment;
|
|
@@ -53,4 +53,4 @@ class ResourceMapping {
|
|
|
53
53
|
}
|
|
54
54
|
}
|
|
55
55
|
exports.ResourceMapping = ResourceMapping;
|
|
56
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
56
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xvdWRmb3JtYXRpb24uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJjbG91ZGZvcm1hdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFzQkE7Ozs7R0FJRztBQUNILE1BQWEsZ0JBQWdCO0lBQ0M7SUFBNEM7SUFBeEUsWUFBNEIsS0FBMEIsRUFBa0IsaUJBQXlCO1FBQXJFLFVBQUssR0FBTCxLQUFLLENBQXFCO1FBQWtCLHNCQUFpQixHQUFqQixpQkFBaUIsQ0FBUTtJQUNqRyxDQUFDO0lBRU0sTUFBTTtRQUNYLE1BQU0sS0FBSyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUM7UUFDekIsTUFBTSxRQUFRLEdBQUcsS0FBSyxDQUFDLFFBQVEsQ0FBQyxTQUFTLEVBQUUsQ0FBQyxJQUFJLENBQUMsaUJBQWlCLENBQUMsQ0FBQztRQUNwRSxNQUFNLE1BQU0sR0FBRyxRQUFRLEVBQUUsUUFBUSxFQUFFLENBQUMsY0FBYyxDQUFDLENBQUM7UUFFcEQsSUFBSSxNQUFNLElBQUksSUFBSSxFQUFFLENBQUM7WUFDbkIsT0FBTyxNQUFNLENBQUM7UUFDaEIsQ0FBQztRQUVELHFFQUFxRTtRQUNyRSxPQUFPLEdBQUcsS0FBSyxDQUFDLFNBQVMsSUFBSSxJQUFJLENBQUMsaUJBQWlCLEVBQUUsQ0FBQztJQUN4RCxDQUFDO0lBRU0sT0FBTztRQUNaLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsUUFBUSxDQUFDLFNBQVMsRUFBRSxDQUFDLElBQUksQ0FBQyxpQkFBaUIsSUFBSSxFQUFFLENBQUMsQ0FBQztRQUMvRSxPQUFPLFFBQVEsRUFBRSxJQUFJLElBQUksU0FBUyxDQUFDO0lBQ3JDLENBQUM7SUFFTSxPQUFPLENBQUMsS0FBdUI7UUFDcEMsT0FBTyxJQUFJLENBQUMsaUJBQWlCLEtBQUssS0FBSyxDQUFDLGlCQUFpQixJQUFJLElBQUksQ0FBQyxLQUFLLENBQUMsU0FBUyxLQUFLLEtBQUssQ0FBQyxLQUFLLENBQUMsU0FBUyxDQUFDO0lBQzlHLENBQUM7Q0FDRjtBQXpCRCw0Q0F5QkM7QUFFRDs7R0FFRztBQUNILE1BQWEsZUFBZTtJQUNFO0lBQTBDO0lBQXRFLFlBQTRCLE1BQXdCLEVBQWtCLFdBQTZCO1FBQXZFLFdBQU0sR0FBTixNQUFNLENBQWtCO1FBQWtCLGdCQUFXLEdBQVgsV0FBVyxDQUFrQjtJQUNuRyxDQUFDO0lBRU0sY0FBYztRQUNuQixPQUFPO1lBQ0wsdURBQXVEO1lBQ3ZELDJCQUEyQjtZQUMzQixJQUFJLEVBQUUsSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLEVBQUU7WUFDM0IsVUFBVSxFQUFFLElBQUksQ0FBQyxNQUFNLENBQUMsTUFBTSxFQUFFO1lBQ2hDLGVBQWUsRUFBRSxJQUFJLENBQUMsV0FBVyxDQUFDLE1BQU0sRUFBRTtTQUMzQyxDQUFDO0lBQ0osQ0FBQztDQUNGO0FBYkQsMENBYUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IFR5cGVkTWFwcGluZyB9IGZyb20gJ0Bhd3MtY2RrL2Nsb3VkZm9ybWF0aW9uLWRpZmYnO1xuaW1wb3J0IHR5cGUgKiBhcyBjeGFwaSBmcm9tICdAYXdzLWNkay9jeC1hcGknO1xuXG5leHBvcnQgaW50ZXJmYWNlIENsb3VkRm9ybWF0aW9uUmVzb3VyY2Uge1xuICBUeXBlOiBzdHJpbmc7XG4gIFByb3BlcnRpZXM/OiBhbnk7XG4gIE1ldGFkYXRhPzogUmVjb3JkPHN0cmluZywgYW55Pjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBDbG91ZEZvcm1hdGlvblRlbXBsYXRlIHtcbiAgUmVzb3VyY2VzPzoge1xuICAgIFtsb2dpY2FsSWQ6IHN0cmluZ106IENsb3VkRm9ybWF0aW9uUmVzb3VyY2U7XG4gIH07XG4gIE91dHB1dHM/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIENsb3VkRm9ybWF0aW9uU3RhY2sge1xuICByZWFkb25seSBlbnZpcm9ubWVudDogY3hhcGkuRW52aXJvbm1lbnQ7XG4gIHJlYWRvbmx5IHN0YWNrTmFtZTogc3RyaW5nO1xuICByZWFkb25seSB0ZW1wbGF0ZTogQ2xvdWRGb3JtYXRpb25UZW1wbGF0ZTtcbn1cblxuLyoqXG4gKiBUaGlzIGNsYXNzIG1pcnJvcnMgdGhlIGBSZXNvdXJjZUxvY2F0aW9uYCBpbnRlcmZhY2UgZnJvbSBDbG91ZEZvcm1hdGlvbixcbiAqIGJ1dCBpcyByaWNoZXIsIHNpbmNlIGl0IGhhcyBhIHJlZmVyZW5jZSB0byB0aGUgc3RhY2sgb2JqZWN0LCByYXRoZXIgdGhhblxuICogbWVyZWx5IHRoZSBzdGFjayBuYW1lLlxuICovXG5leHBvcnQgY2xhc3MgUmVzb3VyY2VMb2NhdGlvbiB7XG4gIGNvbnN0cnVjdG9yKHB1YmxpYyByZWFkb25seSBzdGFjazogQ2xvdWRGb3JtYXRpb25TdGFjaywgcHVibGljIHJlYWRvbmx5IGxvZ2ljYWxSZXNvdXJjZUlkOiBzdHJpbmcpIHtcbiAgfVxuXG4gIHB1YmxpYyB0b1BhdGgoKTogc3RyaW5nIHtcbiAgICBjb25zdCBzdGFjayA9IHRoaXMuc3RhY2s7XG4gICAgY29uc3QgcmVzb3VyY2UgPSBzdGFjay50ZW1wbGF0ZS5SZXNvdXJjZXM/Llt0aGlzLmxvZ2ljYWxSZXNvdXJjZUlkXTtcbiAgICBjb25zdCByZXN1bHQgPSByZXNvdXJjZT8uTWV0YWRhdGE/LlsnYXdzOmNkazpwYXRoJ107XG5cbiAgICBpZiAocmVzdWx0ICE9IG51bGwpIHtcbiAgICAgIHJldHVybiByZXN1bHQ7XG4gICAgfVxuXG4gICAgLy8gSWYgdGhlIHBhdGggaXMgbm90IGF2YWlsYWJsZSwgd2UgY2FuIHVzZSBzdGFjayBuYW1lIGFuZCBsb2dpY2FsIElEXG4gICAgcmV0dXJuIGAke3N0YWNrLnN0YWNrTmFtZX0uJHt0aGlzLmxvZ2ljYWxSZXNvdXJjZUlkfWA7XG4gIH1cblxuICBwdWJsaWMgZ2V0VHlwZSgpOiBzdHJpbmcge1xuICAgIGNvbnN0IHJlc291cmNlID0gdGhpcy5zdGFjay50ZW1wbGF0ZS5SZXNvdXJjZXM/Llt0aGlzLmxvZ2ljYWxSZXNvdXJjZUlkID8/ICcnXTtcbiAgICByZXR1cm4gcmVzb3VyY2U/LlR5cGUgPz8gJ1Vua25vd24nO1xuICB9XG5cbiAgcHVibGljIGVxdWFsVG8ob3RoZXI6IFJlc291cmNlTG9jYXRpb24pOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5sb2dpY2FsUmVzb3VyY2VJZCA9PT0gb3RoZXIubG9naWNhbFJlc291cmNlSWQgJiYgdGhpcy5zdGFjay5zdGFja05hbWUgPT09IG90aGVyLnN0YWNrLnN0YWNrTmFtZTtcbiAgfVxufVxuXG4vKipcbiAqIEEgbWFwcGluZyBiZXR3ZWVuIGEgc291cmNlIGFuZCBhIGRlc3RpbmF0aW9uIGxvY2F0aW9uLlxuICovXG5leHBvcnQgY2xhc3MgUmVzb3VyY2VNYXBwaW5nIHtcbiAgY29uc3RydWN0b3IocHVibGljIHJlYWRvbmx5IHNvdXJjZTogUmVzb3VyY2VMb2NhdGlvbiwgcHVibGljIHJlYWRvbmx5IGRlc3RpbmF0aW9uOiBSZXNvdXJjZUxvY2F0aW9uKSB7XG4gIH1cblxuICBwdWJsaWMgdG9UeXBlZE1hcHBpbmcoKTogVHlwZWRNYXBwaW5nIHtcbiAgICByZXR1cm4ge1xuICAgICAgLy8gdGhlIHR5cGUgaXMgdGhlIHNhbWUgaW4gYm90aCBzb3VyY2UgYW5kIGRlc3RpbmF0aW9uLFxuICAgICAgLy8gc28gd2UgY2FuIHVzZSBlaXRoZXIgb25lXG4gICAgICB0eXBlOiB0aGlzLnNvdXJjZS5nZXRUeXBlKCksXG4gICAgICBzb3VyY2VQYXRoOiB0aGlzLnNvdXJjZS50b1BhdGgoKSxcbiAgICAgIGRlc3RpbmF0aW9uUGF0aDogdGhpcy5kZXN0aW5hdGlvbi50b1BhdGgoKSxcbiAgICB9O1xuICB9XG59XG5cbiJdfQ==
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import type {
|
|
1
|
+
import type { CloudFormationStack } from './cloudformation';
|
|
2
2
|
/**
|
|
3
3
|
* Computes the digest for each resource in the template.
|
|
4
4
|
*
|
|
@@ -22,5 +22,5 @@ import type { CloudFormationTemplate } from './cloudformation';
|
|
|
22
22
|
* CloudFormation template form a directed acyclic graph, this function is
|
|
23
23
|
* well-defined.
|
|
24
24
|
*/
|
|
25
|
-
export declare function computeResourceDigests(
|
|
25
|
+
export declare function computeResourceDigests(stacks: CloudFormationStack[]): Record<string, string>;
|
|
26
26
|
export declare function hashObject(obj: any): string;
|
|
@@ -4,6 +4,7 @@ exports.computeResourceDigests = computeResourceDigests;
|
|
|
4
4
|
exports.hashObject = hashObject;
|
|
5
5
|
const crypto = require("node:crypto");
|
|
6
6
|
const util_1 = require("@aws-cdk/cloudformation-diff/lib/diff/util");
|
|
7
|
+
const graph_1 = require("./graph");
|
|
7
8
|
/**
|
|
8
9
|
* Computes the digest for each resource in the template.
|
|
9
10
|
*
|
|
@@ -27,63 +28,16 @@ const util_1 = require("@aws-cdk/cloudformation-diff/lib/diff/util");
|
|
|
27
28
|
* CloudFormation template form a directed acyclic graph, this function is
|
|
28
29
|
* well-defined.
|
|
29
30
|
*/
|
|
30
|
-
function computeResourceDigests(
|
|
31
|
-
const
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
reverseGraph[id] = new Set();
|
|
38
|
-
}
|
|
39
|
-
// 2. Detect dependencies by searching for Ref/Fn::GetAtt
|
|
40
|
-
const findDependencies = (value) => {
|
|
41
|
-
if (!value || typeof value !== 'object')
|
|
42
|
-
return [];
|
|
43
|
-
if (Array.isArray(value)) {
|
|
44
|
-
return value.flatMap(findDependencies);
|
|
45
|
-
}
|
|
46
|
-
if ('Ref' in value) {
|
|
47
|
-
return [value.Ref];
|
|
48
|
-
}
|
|
49
|
-
if ('Fn::GetAtt' in value) {
|
|
50
|
-
const refTarget = Array.isArray(value['Fn::GetAtt']) ? value['Fn::GetAtt'][0] : value['Fn::GetAtt'].split('.')[0];
|
|
51
|
-
return [refTarget];
|
|
52
|
-
}
|
|
53
|
-
if ('DependsOn' in value) {
|
|
54
|
-
return [value.DependsOn];
|
|
55
|
-
}
|
|
56
|
-
return Object.values(value).flatMap(findDependencies);
|
|
57
|
-
};
|
|
58
|
-
for (const [id, res] of Object.entries(resources)) {
|
|
59
|
-
const deps = findDependencies(res || {});
|
|
60
|
-
for (const dep of deps) {
|
|
61
|
-
if (dep in resources && dep !== id) {
|
|
62
|
-
graph[id].add(dep);
|
|
63
|
-
reverseGraph[dep].add(id);
|
|
64
|
-
}
|
|
65
|
-
}
|
|
66
|
-
}
|
|
67
|
-
// 3. Topological sort
|
|
68
|
-
const outDegree = Object.keys(graph).reduce((acc, k) => {
|
|
69
|
-
acc[k] = graph[k].size;
|
|
70
|
-
return acc;
|
|
71
|
-
}, {});
|
|
72
|
-
const queue = Object.keys(outDegree).filter((k) => outDegree[k] === 0);
|
|
73
|
-
const order = [];
|
|
74
|
-
while (queue.length > 0) {
|
|
75
|
-
const node = queue.shift();
|
|
76
|
-
order.push(node);
|
|
77
|
-
for (const nxt of reverseGraph[node]) {
|
|
78
|
-
outDegree[nxt]--;
|
|
79
|
-
if (outDegree[nxt] === 0) {
|
|
80
|
-
queue.push(nxt);
|
|
81
|
-
}
|
|
82
|
-
}
|
|
83
|
-
}
|
|
31
|
+
function computeResourceDigests(stacks) {
|
|
32
|
+
const exports = Object.fromEntries(stacks.flatMap((s) => Object.values(s.template.Outputs ?? {})
|
|
33
|
+
.filter((o) => o.Export != null && typeof o.Export.Name === 'string')
|
|
34
|
+
.map((o) => [o.Export.Name, { stackName: s.stackName, value: o.Value }])));
|
|
35
|
+
const resources = Object.fromEntries(stacks.flatMap((s) => Object.entries(s.template.Resources ?? {}).map(([id, res]) => [`${s.stackName}.${id}`, res])));
|
|
36
|
+
const graph = new graph_1.ResourceGraph(stacks);
|
|
37
|
+
const nodes = graph.sortedNodes;
|
|
84
38
|
// 4. Compute digests in sorted order
|
|
85
39
|
const result = {};
|
|
86
|
-
for (const id of
|
|
40
|
+
for (const id of nodes) {
|
|
87
41
|
const resource = resources[id];
|
|
88
42
|
const resourceProperties = resource.Properties ?? {};
|
|
89
43
|
const model = (0, util_1.loadResourceModel)(resource.Type);
|
|
@@ -102,8 +56,8 @@ function computeResourceDigests(template) {
|
|
|
102
56
|
else {
|
|
103
57
|
// The resource does not have a physical ID defined, so we need to
|
|
104
58
|
// compute the digest based on its properties and dependencies.
|
|
105
|
-
const depDigests = Array.from(graph
|
|
106
|
-
const propertiesHash = hashObject(stripReferences(stripConstructPath(resource)));
|
|
59
|
+
const depDigests = Array.from(graph.outNeighbors(id)).map((d) => result[d]);
|
|
60
|
+
const propertiesHash = hashObject(stripReferences(stripConstructPath(resource), exports));
|
|
107
61
|
toHash = resource.Type + propertiesHash + depDigests.join('');
|
|
108
62
|
}
|
|
109
63
|
result[id] = crypto.createHash('sha256').update(toHash).digest('hex');
|
|
@@ -140,11 +94,11 @@ function hashObject(obj) {
|
|
|
140
94
|
* Removes sub-properties containing Ref or Fn::GetAtt to avoid hashing
|
|
141
95
|
* references themselves but keeps the property structure.
|
|
142
96
|
*/
|
|
143
|
-
function stripReferences(value) {
|
|
97
|
+
function stripReferences(value, exports) {
|
|
144
98
|
if (!value || typeof value !== 'object')
|
|
145
99
|
return value;
|
|
146
100
|
if (Array.isArray(value)) {
|
|
147
|
-
return value.map(stripReferences);
|
|
101
|
+
return value.map(x => stripReferences(x, exports));
|
|
148
102
|
}
|
|
149
103
|
if ('Ref' in value) {
|
|
150
104
|
return { __cloud_ref__: 'Ref' };
|
|
@@ -155,9 +109,19 @@ function stripReferences(value) {
|
|
|
155
109
|
if ('DependsOn' in value) {
|
|
156
110
|
return { __cloud_ref__: 'DependsOn' };
|
|
157
111
|
}
|
|
112
|
+
if ('Fn::ImportValue' in value) {
|
|
113
|
+
const v = exports[value['Fn::ImportValue']].value;
|
|
114
|
+
// Treat Fn::ImportValue as if it were a reference with the same stack
|
|
115
|
+
if ('Ref' in v) {
|
|
116
|
+
return { __cloud_ref__: 'Ref' };
|
|
117
|
+
}
|
|
118
|
+
else if ('Fn::GetAtt' in v) {
|
|
119
|
+
return { __cloud_ref__: 'Fn::GetAtt' };
|
|
120
|
+
}
|
|
121
|
+
}
|
|
158
122
|
const result = {};
|
|
159
123
|
for (const [k, v] of Object.entries(value)) {
|
|
160
|
-
result[k] = stripReferences(v);
|
|
124
|
+
result[k] = stripReferences(v, exports);
|
|
161
125
|
}
|
|
162
126
|
return result;
|
|
163
127
|
}
|
|
@@ -172,4 +136,4 @@ function stripConstructPath(resource) {
|
|
|
172
136
|
function intersection(a, b) {
|
|
173
137
|
return a.filter((value) => b.includes(value));
|
|
174
138
|
}
|
|
175
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"digest.js","sourceRoot":"","sources":["digest.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'node:crypto';\nimport { loadResourceModel } from '@aws-cdk/cloudformation-diff/lib/diff/util';\nimport type { CloudFormationTemplate } from './cloudformation';\n\n/**\n * Computes the digest for each resource in the template.\n *\n * Conceptually, the digest is computed as:\n *\n *     d(resource) = hash(type + physicalId)                       , if physicalId is defined\n *                 = hash(type + properties + dependencies.map(d)) , otherwise\n *\n * where `hash` is a cryptographic hash function. In other words, if a resource has\n * a physical ID, we use the physical ID plus its type to uniquely identify\n * that resource. In this case, the digest can be computed from these two fields\n * alone. A corollary is that such resources can be renamed and have their\n * properties updated at the same time, and still be considered equivalent.\n *\n * Otherwise, the digest is computed from its type, its own properties (that is,\n * excluding properties that refer to other resources), and the digests of each of\n * its dependencies.\n *\n * The digest of a resource, defined recursively this way, remains stable even if\n * one or more of its dependencies gets renamed. Since the resources in a\n * CloudFormation template form a directed acyclic graph, this function is\n * well-defined.\n */\nexport function computeResourceDigests(template: CloudFormationTemplate): Record<string, string> {\n  const resources = template.Resources || {};\n  const graph: Record<string, Set<string>> = {};\n  const reverseGraph: Record<string, Set<string>> = {};\n\n  // 1. Build adjacency lists\n  for (const id of Object.keys(resources)) {\n    graph[id] = new Set();\n    reverseGraph[id] = new Set();\n  }\n\n  // 2. Detect dependencies by searching for Ref/Fn::GetAtt\n  const findDependencies = (value: any): string[] => {\n    if (!value || typeof value !== 'object') return [];\n    if (Array.isArray(value)) {\n      return value.flatMap(findDependencies);\n    }\n    if ('Ref' in value) {\n      return [value.Ref];\n    }\n    if ('Fn::GetAtt' in value) {\n      const refTarget = Array.isArray(value['Fn::GetAtt']) ? value['Fn::GetAtt'][0] : value['Fn::GetAtt'].split('.')[0];\n      return [refTarget];\n    }\n    if ('DependsOn' in value) {\n      return [value.DependsOn];\n    }\n    return Object.values(value).flatMap(findDependencies);\n  };\n\n  for (const [id, res] of Object.entries(resources)) {\n    const deps = findDependencies(res || {});\n    for (const dep of deps) {\n      if (dep in resources && dep !== id) {\n        graph[id].add(dep);\n        reverseGraph[dep].add(id);\n      }\n    }\n  }\n\n  // 3. Topological sort\n  const outDegree = Object.keys(graph).reduce((acc, k) => {\n    acc[k] = graph[k].size;\n    return acc;\n  }, {} as Record<string, number>);\n\n  const queue = Object.keys(outDegree).filter((k) => outDegree[k] === 0);\n  const order: string[] = [];\n\n  while (queue.length > 0) {\n    const node = queue.shift()!;\n    order.push(node);\n    for (const nxt of reverseGraph[node]) {\n      outDegree[nxt]--;\n      if (outDegree[nxt] === 0) {\n        queue.push(nxt);\n      }\n    }\n  }\n\n  // 4. Compute digests in sorted order\n  const result: Record<string, string> = {};\n  for (const id of order) {\n    const resource = resources[id];\n    const resourceProperties = resource.Properties ?? {};\n    const model = loadResourceModel(resource.Type);\n    const identifier = intersection(Object.keys(resourceProperties), model?.primaryIdentifier ?? []);\n    let toHash: string;\n\n    if (identifier.length === model?.primaryIdentifier?.length) {\n      // The resource has a physical ID defined, so we can\n      // use the ID and the type as the identity of the resource.\n      toHash =\n        resource.Type +\n        identifier\n          .sort()\n          .map((attr) => JSON.stringify(resourceProperties[attr]))\n          .join('');\n    } else {\n      // The resource does not have a physical ID defined, so we need to\n      // compute the digest based on its properties and dependencies.\n      const depDigests = Array.from(graph[id]).map((d) => result[d]);\n      const propertiesHash = hashObject(stripReferences(stripConstructPath(resource)));\n      toHash = resource.Type + propertiesHash + depDigests.join('');\n    }\n\n    result[id] = crypto.createHash('sha256').update(toHash).digest('hex');\n  }\n\n  return result;\n}\n\nexport function hashObject(obj: any): string {\n  const hash = crypto.createHash('sha256');\n\n  function addToHash(value: any) {\n    if (value == null) {\n      addToHash('null');\n    } else if (typeof value === 'object') {\n      if (Array.isArray(value)) {\n        value.forEach(addToHash);\n      } else {\n        Object.keys(value)\n          .sort()\n          .forEach((key) => {\n            hash.update(key);\n            addToHash(value[key]);\n          });\n      }\n    } else {\n      hash.update(typeof value + value.toString());\n    }\n  }\n\n  addToHash(obj);\n  return hash.digest('hex');\n}\n\n/**\n * Removes sub-properties containing Ref or Fn::GetAtt to avoid hashing\n * references themselves but keeps the property structure.\n */\nfunction stripReferences(value: any): any {\n  if (!value || typeof value !== 'object') return value;\n  if (Array.isArray(value)) {\n    return value.map(stripReferences);\n  }\n  if ('Ref' in value) {\n    return { __cloud_ref__: 'Ref' };\n  }\n  if ('Fn::GetAtt' in value) {\n    return { __cloud_ref__: 'Fn::GetAtt' };\n  }\n  if ('DependsOn' in value) {\n    return { __cloud_ref__: 'DependsOn' };\n  }\n  const result: any = {};\n  for (const [k, v] of Object.entries(value)) {\n    result[k] = stripReferences(v);\n  }\n  return result;\n}\n\nfunction stripConstructPath(resource: any): any {\n  if (resource?.Metadata?.['aws:cdk:path'] == null) {\n    return resource;\n  }\n\n  const copy = JSON.parse(JSON.stringify(resource));\n  delete copy.Metadata['aws:cdk:path'];\n  return copy;\n}\n\nfunction intersection<T>(a: T[], b: T[]): T[] {\n  return a.filter((value) => b.includes(value));\n}\n"]}
|
|
139
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"digest.js","sourceRoot":"","sources":["digest.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'node:crypto';\nimport { loadResourceModel } from '@aws-cdk/cloudformation-diff/lib/diff/util';\nimport type { CloudFormationResource, CloudFormationStack } from './cloudformation';\nimport { ResourceGraph } from './graph';\n\n/**\n * Computes the digest for each resource in the template.\n *\n * Conceptually, the digest is computed as:\n *\n *     d(resource) = hash(type + physicalId)                       , if physicalId is defined\n *                 = hash(type + properties + dependencies.map(d)) , otherwise\n *\n * where `hash` is a cryptographic hash function. In other words, if a resource has\n * a physical ID, we use the physical ID plus its type to uniquely identify\n * that resource. In this case, the digest can be computed from these two fields\n * alone. A corollary is that such resources can be renamed and have their\n * properties updated at the same time, and still be considered equivalent.\n *\n * Otherwise, the digest is computed from its type, its own properties (that is,\n * excluding properties that refer to other resources), and the digests of each of\n * its dependencies.\n *\n * The digest of a resource, defined recursively this way, remains stable even if\n * one or more of its dependencies gets renamed. Since the resources in a\n * CloudFormation template form a directed acyclic graph, this function is\n * well-defined.\n */\nexport function computeResourceDigests(stacks: CloudFormationStack[]): Record<string, string> {\n  const exports: { [p: string]: { stackName: string; value: any } } = Object.fromEntries(\n    stacks.flatMap((s) =>\n      Object.values(s.template.Outputs ?? {})\n        .filter((o) => o.Export != null && typeof o.Export.Name === 'string')\n        .map((o) => [o.Export.Name, { stackName: s.stackName, value: o.Value }] as [string, { stackName: string; value: any }]),\n    ),\n  );\n\n  const resources = Object.fromEntries(\n    stacks.flatMap((s) =>\n      Object.entries(s.template.Resources ?? {}).map(\n        ([id, res]) => [`${s.stackName}.${id}`, res] as [string, CloudFormationResource],\n      ),\n    ),\n  );\n\n  const graph = new ResourceGraph(stacks);\n  const nodes = graph.sortedNodes;\n  // 4. Compute digests in sorted order\n  const result: Record<string, string> = {};\n  for (const id of nodes) {\n    const resource = resources[id];\n    const resourceProperties = resource.Properties ?? {};\n    const model = loadResourceModel(resource.Type);\n    const identifier = intersection(Object.keys(resourceProperties), model?.primaryIdentifier ?? []);\n    let toHash: string;\n\n    if (identifier.length === model?.primaryIdentifier?.length) {\n      // The resource has a physical ID defined, so we can\n      // use the ID and the type as the identity of the resource.\n      toHash =\n        resource.Type +\n        identifier\n          .sort()\n          .map((attr) => JSON.stringify(resourceProperties[attr]))\n          .join('');\n    } else {\n      // The resource does not have a physical ID defined, so we need to\n      // compute the digest based on its properties and dependencies.\n      const depDigests = Array.from(graph.outNeighbors(id)).map((d) => result[d]);\n      const propertiesHash = hashObject(stripReferences(stripConstructPath(resource), exports));\n      toHash = resource.Type + propertiesHash + depDigests.join('');\n    }\n\n    result[id] = crypto.createHash('sha256').update(toHash).digest('hex');\n  }\n\n  return result;\n}\n\nexport function hashObject(obj: any): string {\n  const hash = crypto.createHash('sha256');\n\n  function addToHash(value: any) {\n    if (value == null) {\n      addToHash('null');\n    } else if (typeof value === 'object') {\n      if (Array.isArray(value)) {\n        value.forEach(addToHash);\n      } else {\n        Object.keys(value)\n          .sort()\n          .forEach((key) => {\n            hash.update(key);\n            addToHash(value[key]);\n          });\n      }\n    } else {\n      hash.update(typeof value + value.toString());\n    }\n  }\n\n  addToHash(obj);\n  return hash.digest('hex');\n}\n\n/**\n * Removes sub-properties containing Ref or Fn::GetAtt to avoid hashing\n * references themselves but keeps the property structure.\n */\nfunction stripReferences(value: any, exports: { [p: string]: { stackName: string; value: any } }): any {\n  if (!value || typeof value !== 'object') return value;\n  if (Array.isArray(value)) {\n    return value.map(x => stripReferences(x, exports));\n  }\n  if ('Ref' in value) {\n    return { __cloud_ref__: 'Ref' };\n  }\n  if ('Fn::GetAtt' in value) {\n    return { __cloud_ref__: 'Fn::GetAtt' };\n  }\n  if ('DependsOn' in value) {\n    return { __cloud_ref__: 'DependsOn' };\n  }\n  if ('Fn::ImportValue' in value) {\n    const v = exports[value['Fn::ImportValue']].value;\n    // Treat Fn::ImportValue as if it were a reference with the same stack\n    if ('Ref' in v) {\n      return { __cloud_ref__: 'Ref' };\n    } else if ('Fn::GetAtt' in v) {\n      return { __cloud_ref__: 'Fn::GetAtt' };\n    }\n  }\n  const result: any = {};\n  for (const [k, v] of Object.entries(value)) {\n    result[k] = stripReferences(v, exports);\n  }\n  return result;\n}\n\nfunction stripConstructPath(resource: any): any {\n  if (resource?.Metadata?.['aws:cdk:path'] == null) {\n    return resource;\n  }\n\n  const copy = JSON.parse(JSON.stringify(resource));\n  delete copy.Metadata['aws:cdk:path'];\n  return copy;\n}\n\nfunction intersection<T>(a: T[], b: T[]): T[] {\n  return a.filter((value) => b.includes(value));\n}\n"]}
|
|
@@ -0,0 +1,15 @@
|
|
|
1
|
+
import type { CloudFormationStack } from './cloudformation';
|
|
2
|
+
/**
|
|
3
|
+
* An immutable directed graph of resources from multiple CloudFormation stacks.
|
|
4
|
+
*/
|
|
5
|
+
export declare class ResourceGraph {
|
|
6
|
+
private readonly edges;
|
|
7
|
+
private readonly reverseEdges;
|
|
8
|
+
constructor(stacks: Omit<CloudFormationStack, 'environment'>[]);
|
|
9
|
+
/**
|
|
10
|
+
* Returns the sorted nodes in topological order.
|
|
11
|
+
*/
|
|
12
|
+
get sortedNodes(): string[];
|
|
13
|
+
inNeighbors(node: string): string[];
|
|
14
|
+
outNeighbors(node: string): string[];
|
|
15
|
+
}
|
|
@@ -0,0 +1,108 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ResourceGraph = void 0;
|
|
4
|
+
const toolkit_error_1 = require("../../toolkit/toolkit-error");
|
|
5
|
+
/**
|
|
6
|
+
* An immutable directed graph of resources from multiple CloudFormation stacks.
|
|
7
|
+
*/
|
|
8
|
+
class ResourceGraph {
|
|
9
|
+
edges = {};
|
|
10
|
+
reverseEdges = {};
|
|
11
|
+
constructor(stacks) {
|
|
12
|
+
const exports = Object.fromEntries(stacks.flatMap((s) => Object.values(s.template.Outputs ?? {})
|
|
13
|
+
.filter((o) => o.Export != null && typeof o.Export.Name === 'string')
|
|
14
|
+
.map((o) => [o.Export.Name, { stackName: s.stackName, value: o.Value }])));
|
|
15
|
+
const resources = Object.fromEntries(stacks.flatMap((s) => Object.entries(s.template.Resources ?? {}).map(([id, res]) => [`${s.stackName}.${id}`, res])));
|
|
16
|
+
// 1. Build adjacency lists
|
|
17
|
+
for (const id of Object.keys(resources)) {
|
|
18
|
+
this.edges[id] = new Set();
|
|
19
|
+
this.reverseEdges[id] = new Set();
|
|
20
|
+
}
|
|
21
|
+
// 2. Detect dependencies by searching for Ref/Fn::GetAtt
|
|
22
|
+
const findDependencies = (stackName, value) => {
|
|
23
|
+
if (!value || typeof value !== 'object')
|
|
24
|
+
return [];
|
|
25
|
+
if (Array.isArray(value)) {
|
|
26
|
+
return value.flatMap((res) => findDependencies(stackName, res));
|
|
27
|
+
}
|
|
28
|
+
if ('Ref' in value) {
|
|
29
|
+
return [`${stackName}.${value.Ref}`];
|
|
30
|
+
}
|
|
31
|
+
if ('Fn::GetAtt' in value) {
|
|
32
|
+
const refTarget = Array.isArray(value['Fn::GetAtt'])
|
|
33
|
+
? value['Fn::GetAtt'][0]
|
|
34
|
+
: value['Fn::GetAtt'].split('.')[0];
|
|
35
|
+
return [`${stackName}.${refTarget}`];
|
|
36
|
+
}
|
|
37
|
+
if ('Fn::ImportValue' in value) {
|
|
38
|
+
const exp = exports[value['Fn::ImportValue']];
|
|
39
|
+
const v = exp.value;
|
|
40
|
+
if ('Fn::GetAtt' in v) {
|
|
41
|
+
const id = Array.isArray(v['Fn::GetAtt']) ? v['Fn::GetAtt'][0] : v['Fn::GetAtt'].split('.')[0];
|
|
42
|
+
return [`${exp.stackName}.${id}`];
|
|
43
|
+
}
|
|
44
|
+
if ('Ref' in v) {
|
|
45
|
+
return [`${exp.stackName}.${v.Ref}`];
|
|
46
|
+
}
|
|
47
|
+
return [`${exp.stackName}.${v}`];
|
|
48
|
+
}
|
|
49
|
+
const result = [];
|
|
50
|
+
if ('DependsOn' in value) {
|
|
51
|
+
if (Array.isArray(value.DependsOn)) {
|
|
52
|
+
result.push(...value.DependsOn.map((r) => `${stackName}.${r}`));
|
|
53
|
+
}
|
|
54
|
+
else {
|
|
55
|
+
result.push(`${stackName}.${value.DependsOn}`);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
result.push(...Object.values(value).flatMap((res) => findDependencies(stackName, res)));
|
|
59
|
+
return result;
|
|
60
|
+
};
|
|
61
|
+
for (const [id, res] of Object.entries(resources)) {
|
|
62
|
+
const stackName = id.split('.')[0];
|
|
63
|
+
const deps = findDependencies(stackName, res || {});
|
|
64
|
+
for (const dep of deps) {
|
|
65
|
+
if (dep in resources && dep !== id) {
|
|
66
|
+
this.edges[id].add(dep);
|
|
67
|
+
this.reverseEdges[dep].add(id);
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* Returns the sorted nodes in topological order.
|
|
74
|
+
*/
|
|
75
|
+
get sortedNodes() {
|
|
76
|
+
const result = [];
|
|
77
|
+
const outDegree = Object.keys(this.edges).reduce((acc, k) => {
|
|
78
|
+
acc[k] = this.edges[k].size;
|
|
79
|
+
return acc;
|
|
80
|
+
}, {});
|
|
81
|
+
const queue = Object.keys(outDegree).filter((k) => outDegree[k] === 0);
|
|
82
|
+
while (queue.length > 0) {
|
|
83
|
+
const node = queue.shift();
|
|
84
|
+
result.push(node);
|
|
85
|
+
for (const nxt of this.reverseEdges[node]) {
|
|
86
|
+
outDegree[nxt]--;
|
|
87
|
+
if (outDegree[nxt] === 0) {
|
|
88
|
+
queue.push(nxt);
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
return result;
|
|
93
|
+
}
|
|
94
|
+
inNeighbors(node) {
|
|
95
|
+
if (!(node in this.edges)) {
|
|
96
|
+
throw new toolkit_error_1.ToolkitError(`Node ${node} not found in the graph`);
|
|
97
|
+
}
|
|
98
|
+
return Array.from(this.reverseEdges[node] || []);
|
|
99
|
+
}
|
|
100
|
+
outNeighbors(node) {
|
|
101
|
+
if (!(node in this.edges)) {
|
|
102
|
+
throw new toolkit_error_1.ToolkitError(`Node ${node} not found in the graph`);
|
|
103
|
+
}
|
|
104
|
+
return Array.from(this.edges[node] || []);
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
exports.ResourceGraph = ResourceGraph;
|
|
108
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"graph.js","sourceRoot":"","sources":["graph.ts"],"names":[],"mappings":"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","sourcesContent":["import type { CloudFormationResource, CloudFormationStack } from './cloudformation';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\n\n/**\n * An immutable directed graph of resources from multiple CloudFormation stacks.\n */\nexport class ResourceGraph {\n  private readonly edges: Record<string, Set<string>> = {};\n  private readonly reverseEdges: Record<string, Set<string>> = {};\n\n  constructor(stacks: Omit<CloudFormationStack, 'environment'>[]) {\n    const exports: { [p: string]: { stackName: string; value: any } } = Object.fromEntries(\n      stacks.flatMap((s) =>\n        Object.values(s.template.Outputs ?? {})\n          .filter((o) => o.Export != null && typeof o.Export.Name === 'string')\n          .map(\n            (o) =>\n              [o.Export.Name, { stackName: s.stackName, value: o.Value }] as [\n                string,\n                {\n                  stackName: string;\n                  value: any;\n                },\n              ],\n          ),\n      ),\n    );\n\n    const resources = Object.fromEntries(\n      stacks.flatMap((s) =>\n        Object.entries(s.template.Resources ?? {}).map(\n          ([id, res]) => [`${s.stackName}.${id}`, res] as [string, CloudFormationResource],\n        ),\n      ),\n    );\n\n    // 1. Build adjacency lists\n    for (const id of Object.keys(resources)) {\n      this.edges[id] = new Set();\n      this.reverseEdges[id] = new Set();\n    }\n\n    // 2. Detect dependencies by searching for Ref/Fn::GetAtt\n    const findDependencies = (stackName: string, value: any): string[] => {\n      if (!value || typeof value !== 'object') return [];\n      if (Array.isArray(value)) {\n        return value.flatMap((res) => findDependencies(stackName, res));\n      }\n      if ('Ref' in value) {\n        return [`${stackName}.${value.Ref}`];\n      }\n      if ('Fn::GetAtt' in value) {\n        const refTarget = Array.isArray(value['Fn::GetAtt'])\n          ? value['Fn::GetAtt'][0]\n          : value['Fn::GetAtt'].split('.')[0];\n        return [`${stackName}.${refTarget}`];\n      }\n      if ('Fn::ImportValue' in value) {\n        const exp = exports[value['Fn::ImportValue']];\n        const v = exp.value;\n        if ('Fn::GetAtt' in v) {\n          const id = Array.isArray(v['Fn::GetAtt']) ? v['Fn::GetAtt'][0] : v['Fn::GetAtt'].split('.')[0];\n          return [`${exp.stackName}.${id}`];\n        }\n        if ('Ref' in v) {\n          return [`${exp.stackName}.${v.Ref}`];\n        }\n        return [`${exp.stackName}.${v}`];\n      }\n      const result: string[] = [];\n      if ('DependsOn' in value) {\n        if (Array.isArray(value.DependsOn)) {\n          result.push(...value.DependsOn.map((r: string) => `${stackName}.${r}`));\n        } else {\n          result.push(`${stackName}.${value.DependsOn}`);\n        }\n      }\n      result.push(...Object.values(value).flatMap((res) => findDependencies(stackName, res)));\n      return result;\n    };\n\n    for (const [id, res] of Object.entries(resources)) {\n      const stackName = id.split('.')[0];\n      const deps = findDependencies(stackName, res || {});\n      for (const dep of deps) {\n        if (dep in resources && dep !== id) {\n          this.edges[id].add(dep);\n          this.reverseEdges[dep].add(id);\n        }\n      }\n    }\n  }\n\n  /**\n   * Returns the sorted nodes in topological order.\n   */\n  get sortedNodes(): string[] {\n    const result: string[] = [];\n    const outDegree = Object.keys(this.edges).reduce((acc, k) => {\n      acc[k] = this.edges[k].size;\n      return acc;\n    }, {} as Record<string, number>);\n\n    const queue = Object.keys(outDegree).filter((k) => outDegree[k] === 0);\n\n    while (queue.length > 0) {\n      const node = queue.shift()!;\n      result.push(node);\n      for (const nxt of this.reverseEdges[node]) {\n        outDegree[nxt]--;\n        if (outDegree[nxt] === 0) {\n          queue.push(nxt);\n        }\n      }\n    }\n    return result;\n  }\n\n  public inNeighbors(node: string): string[] {\n    if (!(node in this.edges)) {\n      throw new ToolkitError(`Node ${node} not found in the graph`);\n    }\n    return Array.from(this.reverseEdges[node] || []);\n  }\n\n  public outNeighbors(node: string): string[] {\n    if (!(node in this.edges)) {\n      throw new ToolkitError(`Node ${node} not found in the graph`);\n    }\n    return Array.from(this.edges[node] || []);\n  }\n}\n"]}
|