@aws-cdk/toolkit-lib 1.11.0 → 1.12.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/build-info.json +2 -2
- package/db.json.gz +0 -0
- package/lib/api/aws-auth/sdk.d.ts +4 -1
- package/lib/api/aws-auth/sdk.js +10 -1
- package/lib/api/bootstrap/bootstrap-template.yaml +3 -2
- package/lib/api/deployments/cfn-api.d.ts +5 -1
- package/lib/api/deployments/cfn-api.js +10 -2
- package/lib/api/deployments/deploy-stack.js +7 -1
- package/lib/api/deployments/early-validation.d.ts +17 -0
- package/lib/api/deployments/early-validation.js +53 -0
- package/lib/api/refactoring/index.d.ts +1 -1
- package/lib/api/refactoring/index.js +31 -22
- package/package.json +11 -11
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.EarlyValidationReporter = void 0;
|
|
4
|
+
/**
|
|
5
|
+
* A ValidationReporter that checks for early validation errors right after
|
|
6
|
+
* creating the change set. If any are found, it throws an error listing all validation failures.
|
|
7
|
+
* If the DescribeEvents API call fails (for example, due to insufficient permissions),
|
|
8
|
+
* it logs a warning instead.
|
|
9
|
+
*/
|
|
10
|
+
class EarlyValidationReporter {
|
|
11
|
+
sdk;
|
|
12
|
+
envResources;
|
|
13
|
+
constructor(sdk, envResources) {
|
|
14
|
+
this.sdk = sdk;
|
|
15
|
+
this.envResources = envResources;
|
|
16
|
+
}
|
|
17
|
+
async fetchDetails(changeSetName, stackName) {
|
|
18
|
+
let operationEvents = [];
|
|
19
|
+
try {
|
|
20
|
+
operationEvents = await this.getFailedEvents(stackName, changeSetName);
|
|
21
|
+
}
|
|
22
|
+
catch (error) {
|
|
23
|
+
let currentVersion = undefined;
|
|
24
|
+
try {
|
|
25
|
+
currentVersion = (await this.envResources.lookupToolkit()).version;
|
|
26
|
+
}
|
|
27
|
+
catch (e) {
|
|
28
|
+
}
|
|
29
|
+
return `The template cannot be deployed because of early validation errors, but retrieving more details about those
|
|
30
|
+
errors failed (${error}). Make sure you have permissions to call the DescribeEvents API, or re-bootstrap
|
|
31
|
+
your environment with the latest version of the CLI (need at least version 30, current version ${currentVersion ?? 'unknown'}).`;
|
|
32
|
+
}
|
|
33
|
+
let message = `ChangeSet '${changeSetName}' on stack '${stackName}' failed early validation`;
|
|
34
|
+
if (operationEvents.length > 0) {
|
|
35
|
+
const failures = operationEvents
|
|
36
|
+
.map((event) => ` - ${event.ValidationStatusReason} (at ${event.ValidationPath})`)
|
|
37
|
+
.join('\n');
|
|
38
|
+
message += `:\n${failures}\n`;
|
|
39
|
+
}
|
|
40
|
+
return message;
|
|
41
|
+
}
|
|
42
|
+
async getFailedEvents(stackName, changeSetName) {
|
|
43
|
+
return this.sdk.cloudFormation().paginatedDescribeEvents({
|
|
44
|
+
StackName: stackName,
|
|
45
|
+
ChangeSetName: changeSetName,
|
|
46
|
+
Filters: {
|
|
47
|
+
FailedEvents: true,
|
|
48
|
+
},
|
|
49
|
+
});
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
exports.EarlyValidationReporter = EarlyValidationReporter;
|
|
53
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZWFybHktdmFsaWRhdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbImVhcmx5LXZhbGlkYXRpb24udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBS0E7Ozs7O0dBS0c7QUFDSCxNQUFhLHVCQUF1QjtJQUNMO0lBQTJCO0lBQXhELFlBQTZCLEdBQVEsRUFBbUIsWUFBa0M7UUFBN0QsUUFBRyxHQUFILEdBQUcsQ0FBSztRQUFtQixpQkFBWSxHQUFaLFlBQVksQ0FBc0I7SUFDMUYsQ0FBQztJQUVNLEtBQUssQ0FBQyxZQUFZLENBQUMsYUFBcUIsRUFBRSxTQUFpQjtRQUNoRSxJQUFJLGVBQWUsR0FBcUIsRUFBRSxDQUFDO1FBQzNDLElBQUksQ0FBQztZQUNILGVBQWUsR0FBRyxNQUFNLElBQUksQ0FBQyxlQUFlLENBQUMsU0FBUyxFQUFFLGFBQWEsQ0FBQyxDQUFDO1FBQ3pFLENBQUM7UUFBQyxPQUFPLEtBQUssRUFBRSxDQUFDO1lBQ2YsSUFBSSxjQUFjLEdBQXVCLFNBQVMsQ0FBQztZQUNuRCxJQUFJLENBQUM7Z0JBQ0gsY0FBYyxHQUFHLENBQUMsTUFBTSxJQUFJLENBQUMsWUFBWSxDQUFDLGFBQWEsRUFBRSxDQUFDLENBQUMsT0FBTyxDQUFDO1lBQ3JFLENBQUM7WUFBQyxPQUFPLENBQUMsRUFBRSxDQUFDO1lBQ2IsQ0FBQztZQUVELE9BQU87aUJBQ0ksS0FBSztpR0FDMkUsY0FBYyxJQUFJLFNBQVMsSUFBSSxDQUFDO1FBQzdILENBQUM7UUFFRCxJQUFJLE9BQU8sR0FBRyxjQUFjLGFBQWEsZUFBZSxTQUFTLDJCQUEyQixDQUFDO1FBQzdGLElBQUksZUFBZSxDQUFDLE1BQU0sR0FBRyxDQUFDLEVBQUUsQ0FBQztZQUMvQixNQUFNLFFBQVEsR0FBRyxlQUFlO2lCQUM3QixHQUFHLENBQUMsQ0FBQyxLQUFLLEVBQUUsRUFBRSxDQUFDLE9BQU8sS0FBSyxDQUFDLHNCQUFzQixRQUFRLEtBQUssQ0FBQyxjQUFjLEdBQUcsQ0FBQztpQkFDbEYsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO1lBRWQsT0FBTyxJQUFJLE1BQU0sUUFBUSxJQUFJLENBQUM7UUFDaEMsQ0FBQztRQUNELE9BQU8sT0FBTyxDQUFDO0lBQ2pCLENBQUM7SUFFTyxLQUFLLENBQUMsZUFBZSxDQUFDLFNBQWlCLEVBQUUsYUFBcUI7UUFDcEUsT0FBTyxJQUFJLENBQUMsR0FBRyxDQUFDLGNBQWMsRUFBRSxDQUFDLHVCQUF1QixDQUFDO1lBQ3ZELFNBQVMsRUFBRSxTQUFTO1lBQ3BCLGFBQWEsRUFBRSxhQUFhO1lBQzVCLE9BQU8sRUFBRTtnQkFDUCxZQUFZLEVBQUUsSUFBSTthQUNuQjtTQUNGLENBQUMsQ0FBQztJQUNMLENBQUM7Q0FDRjtBQXhDRCwwREF3Q0MiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSB7IE9wZXJhdGlvbkV2ZW50IH0gZnJvbSAnQGF3cy1zZGsvY2xpZW50LWNsb3VkZm9ybWF0aW9uJztcbmltcG9ydCB0eXBlIHsgVmFsaWRhdGlvblJlcG9ydGVyIH0gZnJvbSAnLi9jZm4tYXBpJztcbmltcG9ydCB0eXBlIHsgU0RLIH0gZnJvbSAnLi4vYXdzLWF1dGgvc2RrJztcbmltcG9ydCB0eXBlIHsgRW52aXJvbm1lbnRSZXNvdXJjZXMgfSBmcm9tICcuLi9lbnZpcm9ubWVudCc7XG5cbi8qKlxuICogQSBWYWxpZGF0aW9uUmVwb3J0ZXIgdGhhdCBjaGVja3MgZm9yIGVhcmx5IHZhbGlkYXRpb24gZXJyb3JzIHJpZ2h0IGFmdGVyXG4gKiBjcmVhdGluZyB0aGUgY2hhbmdlIHNldC4gSWYgYW55IGFyZSBmb3VuZCwgaXQgdGhyb3dzIGFuIGVycm9yIGxpc3RpbmcgYWxsIHZhbGlkYXRpb24gZmFpbHVyZXMuXG4gKiBJZiB0aGUgRGVzY3JpYmVFdmVudHMgQVBJIGNhbGwgZmFpbHMgKGZvciBleGFtcGxlLCBkdWUgdG8gaW5zdWZmaWNpZW50IHBlcm1pc3Npb25zKSxcbiAqIGl0IGxvZ3MgYSB3YXJuaW5nIGluc3RlYWQuXG4gKi9cbmV4cG9ydCBjbGFzcyBFYXJseVZhbGlkYXRpb25SZXBvcnRlciBpbXBsZW1lbnRzIFZhbGlkYXRpb25SZXBvcnRlciB7XG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgcmVhZG9ubHkgc2RrOiBTREssIHByaXZhdGUgcmVhZG9ubHkgZW52UmVzb3VyY2VzOiBFbnZpcm9ubWVudFJlc291cmNlcykge1xuICB9XG5cbiAgcHVibGljIGFzeW5jIGZldGNoRGV0YWlscyhjaGFuZ2VTZXROYW1lOiBzdHJpbmcsIHN0YWNrTmFtZTogc3RyaW5nKTogUHJvbWlzZTxzdHJpbmc+IHtcbiAgICBsZXQgb3BlcmF0aW9uRXZlbnRzOiBPcGVyYXRpb25FdmVudFtdID0gW107XG4gICAgdHJ5IHtcbiAgICAgIG9wZXJhdGlvbkV2ZW50cyA9IGF3YWl0IHRoaXMuZ2V0RmFpbGVkRXZlbnRzKHN0YWNrTmFtZSwgY2hhbmdlU2V0TmFtZSk7XG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIGxldCBjdXJyZW50VmVyc2lvbjogbnVtYmVyIHwgdW5kZWZpbmVkID0gdW5kZWZpbmVkO1xuICAgICAgdHJ5IHtcbiAgICAgICAgY3VycmVudFZlcnNpb24gPSAoYXdhaXQgdGhpcy5lbnZSZXNvdXJjZXMubG9va3VwVG9vbGtpdCgpKS52ZXJzaW9uO1xuICAgICAgfSBjYXRjaCAoZSkge1xuICAgICAgfVxuXG4gICAgICByZXR1cm4gYFRoZSB0ZW1wbGF0ZSBjYW5ub3QgYmUgZGVwbG95ZWQgYmVjYXVzZSBvZiBlYXJseSB2YWxpZGF0aW9uIGVycm9ycywgYnV0IHJldHJpZXZpbmcgbW9yZSBkZXRhaWxzIGFib3V0IHRob3NlXG5lcnJvcnMgZmFpbGVkICgke2Vycm9yfSkuIE1ha2Ugc3VyZSB5b3UgaGF2ZSBwZXJtaXNzaW9ucyB0byBjYWxsIHRoZSBEZXNjcmliZUV2ZW50cyBBUEksIG9yIHJlLWJvb3RzdHJhcFxueW91ciBlbnZpcm9ubWVudCB3aXRoIHRoZSBsYXRlc3QgdmVyc2lvbiBvZiB0aGUgQ0xJIChuZWVkIGF0IGxlYXN0IHZlcnNpb24gMzAsIGN1cnJlbnQgdmVyc2lvbiAke2N1cnJlbnRWZXJzaW9uID8/ICd1bmtub3duJ30pLmA7XG4gICAgfVxuXG4gICAgbGV0IG1lc3NhZ2UgPSBgQ2hhbmdlU2V0ICcke2NoYW5nZVNldE5hbWV9JyBvbiBzdGFjayAnJHtzdGFja05hbWV9JyBmYWlsZWQgZWFybHkgdmFsaWRhdGlvbmA7XG4gICAgaWYgKG9wZXJhdGlvbkV2ZW50cy5sZW5ndGggPiAwKSB7XG4gICAgICBjb25zdCBmYWlsdXJlcyA9IG9wZXJhdGlvbkV2ZW50c1xuICAgICAgICAubWFwKChldmVudCkgPT4gYCAgLSAke2V2ZW50LlZhbGlkYXRpb25TdGF0dXNSZWFzb259IChhdCAke2V2ZW50LlZhbGlkYXRpb25QYXRofSlgKVxuICAgICAgICAuam9pbignXFxuJyk7XG5cbiAgICAgIG1lc3NhZ2UgKz0gYDpcXG4ke2ZhaWx1cmVzfVxcbmA7XG4gICAgfVxuICAgIHJldHVybiBtZXNzYWdlO1xuICB9XG5cbiAgcHJpdmF0ZSBhc3luYyBnZXRGYWlsZWRFdmVudHMoc3RhY2tOYW1lOiBzdHJpbmcsIGNoYW5nZVNldE5hbWU6IHN0cmluZykge1xuICAgIHJldHVybiB0aGlzLnNkay5jbG91ZEZvcm1hdGlvbigpLnBhZ2luYXRlZERlc2NyaWJlRXZlbnRzKHtcbiAgICAgIFN0YWNrTmFtZTogc3RhY2tOYW1lLFxuICAgICAgQ2hhbmdlU2V0TmFtZTogY2hhbmdlU2V0TmFtZSxcbiAgICAgIEZpbHRlcnM6IHtcbiAgICAgICAgRmFpbGVkRXZlbnRzOiB0cnVlLFxuICAgICAgfSxcbiAgICB9KTtcbiAgfVxufVxuIl19
|
|
@@ -17,7 +17,7 @@ interface StackGroup {
|
|
|
17
17
|
deployedStacks: CloudFormationStack[];
|
|
18
18
|
}
|
|
19
19
|
export declare function usePrescribedMappings(mappingGroups: MappingGroup[], sdkProvider: SdkProvider): Promise<ResourceMapping[]>;
|
|
20
|
-
export declare function
|
|
20
|
+
export declare function getDeployedStacksByNames(sdkProvider: SdkProvider, environment: cxapi.Environment, stackNames: string[]): Promise<CloudFormationStack[]>;
|
|
21
21
|
export declare function formatEnvironmentSectionHeader(environment: cxapi.Environment): string;
|
|
22
22
|
export declare function formatTypedMappings(mappings: TypedMapping[]): string;
|
|
23
23
|
export declare function formatAmbiguousMappings(paths: [string[], string[]][]): string;
|
|
@@ -15,7 +15,7 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
|
15
15
|
};
|
|
16
16
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
17
|
exports.usePrescribedMappings = usePrescribedMappings;
|
|
18
|
-
exports.
|
|
18
|
+
exports.getDeployedStacksByNames = getDeployedStacksByNames;
|
|
19
19
|
exports.formatEnvironmentSectionHeader = formatEnvironmentSectionHeader;
|
|
20
20
|
exports.formatTypedMappings = formatTypedMappings;
|
|
21
21
|
exports.formatAmbiguousMappings = formatAmbiguousMappings;
|
|
@@ -33,9 +33,11 @@ __exportStar(require("./context"), exports);
|
|
|
33
33
|
async function usePrescribedMappings(mappingGroups, sdkProvider) {
|
|
34
34
|
const stackGroups = [];
|
|
35
35
|
for (const group of mappingGroups) {
|
|
36
|
+
const summaries = await listStacks(sdkProvider, environmentOf(group));
|
|
37
|
+
const stacks = await getDeployedStacksByNames(sdkProvider, environmentOf(group), summaries.map(s => s.StackName));
|
|
36
38
|
stackGroups.push({
|
|
37
39
|
...group,
|
|
38
|
-
stacks
|
|
40
|
+
stacks,
|
|
39
41
|
});
|
|
40
42
|
}
|
|
41
43
|
// Validate that there are no duplicate destinations
|
|
@@ -89,29 +91,32 @@ async function usePrescribedMappings(mappingGroups, sdkProvider) {
|
|
|
89
91
|
}, logicalId);
|
|
90
92
|
}
|
|
91
93
|
}
|
|
92
|
-
async function
|
|
94
|
+
async function getDeployedStacksByNames(sdkProvider, environment, stackNames) {
|
|
93
95
|
const cfn = (await sdkProvider.forEnvironment(environment, plugin_1.Mode.ForReading)).sdk.cloudFormation();
|
|
94
|
-
const
|
|
95
|
-
|
|
96
|
-
'CREATE_COMPLETE',
|
|
97
|
-
'UPDATE_COMPLETE',
|
|
98
|
-
'UPDATE_ROLLBACK_COMPLETE',
|
|
99
|
-
'IMPORT_COMPLETE',
|
|
100
|
-
'ROLLBACK_COMPLETE',
|
|
101
|
-
],
|
|
102
|
-
});
|
|
103
|
-
const normalize = async (summary) => {
|
|
104
|
-
const templateCommandOutput = await cfn.getTemplate({ StackName: summary.StackName });
|
|
96
|
+
const normalize = async (stackName) => {
|
|
97
|
+
const templateCommandOutput = await cfn.getTemplate({ StackName: stackName });
|
|
105
98
|
const template = (0, util_1.deserializeStructure)(templateCommandOutput.TemplateBody ?? '{}');
|
|
106
99
|
return {
|
|
107
100
|
environment,
|
|
108
|
-
stackName:
|
|
101
|
+
stackName: stackName,
|
|
109
102
|
template,
|
|
110
103
|
};
|
|
111
104
|
};
|
|
112
105
|
const limit = (0, concurrency_1.pLimit)(20);
|
|
113
106
|
// eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism
|
|
114
|
-
return Promise.all(
|
|
107
|
+
return Promise.all(stackNames.map(s => limit(() => normalize(s))));
|
|
108
|
+
}
|
|
109
|
+
async function listStacks(sdkProvider, environment) {
|
|
110
|
+
const cfn = (await sdkProvider.forEnvironment(environment, plugin_1.Mode.ForReading)).sdk.cloudFormation();
|
|
111
|
+
return cfn.paginatedListStacks({
|
|
112
|
+
StackStatusFilter: [
|
|
113
|
+
'CREATE_COMPLETE',
|
|
114
|
+
'UPDATE_COMPLETE',
|
|
115
|
+
'UPDATE_ROLLBACK_COMPLETE',
|
|
116
|
+
'IMPORT_COMPLETE',
|
|
117
|
+
'ROLLBACK_COMPLETE',
|
|
118
|
+
],
|
|
119
|
+
});
|
|
115
120
|
}
|
|
116
121
|
function formatEnvironmentSectionHeader(environment) {
|
|
117
122
|
const env = `aws://${environment.account}/${environment.region}`;
|
|
@@ -141,17 +146,21 @@ async function groupStacks(sdkProvider, localStacks, additionalStackNames) {
|
|
|
141
146
|
const localByEnvironment = await (0, util_1.indexBy)(localStacks, async (s) => (0, digest_1.hashObject)(await sdkProvider.resolveEnvironment(s.environment)));
|
|
142
147
|
const groups = [];
|
|
143
148
|
for (let key of localByEnvironment.keys()) {
|
|
144
|
-
const environment = environments.get(key);
|
|
145
|
-
const allDeployedStacks = await getDeployedStacks(sdkProvider, environment);
|
|
146
149
|
const local = localByEnvironment.get(key);
|
|
147
|
-
const hasLocalCounterpart = (
|
|
148
|
-
const wasExplicitlyProvided = (
|
|
150
|
+
const hasLocalCounterpart = (stackName) => local.some((l) => l.stackName === stackName);
|
|
151
|
+
const wasExplicitlyProvided = (stackName) => additionalStackNames.includes(stackName);
|
|
152
|
+
const environment = environments.get(key);
|
|
153
|
+
const stackSummaries = await listStacks(sdkProvider, environment);
|
|
154
|
+
const stackNames = stackSummaries
|
|
155
|
+
.map(s => s.StackName)
|
|
156
|
+
.filter(s => hasLocalCounterpart(s) || wasExplicitlyProvided(s));
|
|
157
|
+
const deployedStacks = await getDeployedStacksByNames(sdkProvider, environment, stackNames);
|
|
149
158
|
groups.push({
|
|
150
159
|
environment,
|
|
151
|
-
deployedStacks
|
|
160
|
+
deployedStacks,
|
|
152
161
|
localStacks: local,
|
|
153
162
|
});
|
|
154
163
|
}
|
|
155
164
|
return groups;
|
|
156
165
|
}
|
|
157
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import type { TypedMapping } from '@aws-cdk/cloudformation-diff';\nimport {\n  formatAmbiguousMappings as fmtAmbiguousMappings,\n  formatEnvironmentSectionHeader as fmtEnvironmentSectionHeader,\n  formatTypedMappings as fmtTypedMappings,\n} from '@aws-cdk/cloudformation-diff';\nimport type * as cxapi from '@aws-cdk/cx-api';\nimport type { StackSummary } from '@aws-sdk/client-cloudformation';\nimport { deserializeStructure, indexBy } from '../../util';\nimport type { SdkProvider } from '../aws-auth/private';\nimport { Mode } from '../plugin';\nimport { StringWriteStream } from '../streams';\nimport type { CloudFormationStack } from './cloudformation';\nimport { ResourceLocation, ResourceMapping } from './cloudformation';\nimport { hashObject } from './digest';\nimport type { MappingGroup } from '../../actions';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\nimport { pLimit } from '../../util/concurrency';\n\nexport * from './exclude';\nexport * from './context';\n\ninterface StackGroup {\n  environment: cxapi.Environment;\n  localStacks: CloudFormationStack[];\n  deployedStacks: CloudFormationStack[];\n}\n\ninterface StackGroup {\n  environment: cxapi.Environment;\n  localStacks: CloudFormationStack[];\n  deployedStacks: CloudFormationStack[];\n}\n\nexport async function usePrescribedMappings(\n  mappingGroups: MappingGroup[],\n  sdkProvider: SdkProvider,\n): Promise<ResourceMapping[]> {\n  interface MappingGroupWithStacks extends MappingGroup {\n    stacks: CloudFormationStack[];\n  }\n\n  const stackGroups: MappingGroupWithStacks[] = [];\n  for (const group of mappingGroups) {\n    stackGroups.push({\n      ...group,\n      stacks: await getDeployedStacks(sdkProvider, environmentOf(group)),\n    });\n  }\n\n  // Validate that there are no duplicate destinations\n  for (let group of stackGroups) {\n    const destinations = new Set<string>();\n\n    for (const destination of Object.values(group.resources)) {\n      if (destinations.has(destination)) {\n        throw new ToolkitError(\n          `Duplicate destination resource '${destination}' in environment ${group.account}/${group.region}`,\n        );\n      }\n      destinations.add(destination);\n    }\n  }\n\n  const result: ResourceMapping[] = [];\n  for (const group of stackGroups) {\n    for (const [source, destination] of Object.entries(group.resources)) {\n      if (!inUse(source, group.stacks)) {\n        throw new ToolkitError(`Source resource '${source}' does not exist in environment ${group.account}/${group.region}`);\n      }\n\n      if (inUse(destination, group.stacks)) {\n        throw new ToolkitError(\n          `Destination resource '${destination}' already in use in environment ${group.account}/${group.region}`,\n        );\n      }\n\n      const environment = environmentOf(group);\n      const src = makeLocation(source, environment, group.stacks);\n      const dst = makeLocation(destination, environment);\n      result.push(new ResourceMapping(src, dst));\n    }\n  }\n  return result;\n\n  function inUse(location: string, stacks: CloudFormationStack[]): boolean {\n    const [stackName, logicalId] = location.split('.');\n    if (stackName == null || logicalId == null) {\n      throw new ToolkitError(`Invalid location '${location}'`);\n    }\n    const stack = stacks.find((s) => s.stackName === stackName);\n    return stack != null && stack.template.Resources?.[logicalId] != null;\n  }\n\n  function environmentOf(group: MappingGroup) {\n    return {\n      account: group.account,\n      region: group.region,\n      name: '',\n    };\n  }\n\n  function makeLocation(\n    loc: string,\n    environment: cxapi.Environment,\n    stacks: CloudFormationStack[] = [],\n  ): ResourceLocation {\n    const [stackName, logicalId] = loc.split('.');\n    const stack = stacks.find((s) => s.stackName === stackName);\n\n    return new ResourceLocation(\n      {\n        stackName,\n        environment,\n        template: stack?.template ?? {},\n      },\n      logicalId,\n    );\n  }\n}\n\nexport async function getDeployedStacks(\n  sdkProvider: SdkProvider,\n  environment: cxapi.Environment,\n): Promise<CloudFormationStack[]> {\n  const cfn = (await sdkProvider.forEnvironment(environment, Mode.ForReading)).sdk.cloudFormation();\n\n  const summaries = await cfn.paginatedListStacks({\n    StackStatusFilter: [\n      'CREATE_COMPLETE',\n      'UPDATE_COMPLETE',\n      'UPDATE_ROLLBACK_COMPLETE',\n      'IMPORT_COMPLETE',\n      'ROLLBACK_COMPLETE',\n    ],\n  });\n\n  const normalize = async (summary: StackSummary) => {\n    const templateCommandOutput = await cfn.getTemplate({ StackName: summary.StackName! });\n    const template = deserializeStructure(templateCommandOutput.TemplateBody ?? '{}');\n    return {\n      environment,\n      stackName: summary.StackName!,\n      template,\n    };\n  };\n\n  const limit = pLimit(20);\n\n  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  return Promise.all(summaries.map(s => limit(() => normalize(s))));\n}\n\nexport function formatEnvironmentSectionHeader(environment: cxapi.Environment) {\n  const env = `aws://${environment.account}/${environment.region}`;\n  return formatToStream(stream => fmtEnvironmentSectionHeader(stream, env));\n}\n\nexport function formatTypedMappings(mappings: TypedMapping[]): string {\n  return formatToStream((stream) => fmtTypedMappings(stream, mappings));\n}\n\nexport function formatAmbiguousMappings(paths: [string[], string[]][]): string {\n  return formatToStream((stream) => fmtAmbiguousMappings(stream, paths));\n}\n\nfunction formatToStream(cb: (stream: NodeJS.WritableStream) => void): string {\n  const stream = new StringWriteStream();\n  cb(stream);\n  return stream.toString();\n}\n\n/**\n * Returns a list of stack groups, each containing the local stacks and the deployed stacks that match the given patterns.\n */\nexport async function groupStacks(sdkProvider: SdkProvider, localStacks: CloudFormationStack[], additionalStackNames: string[]) {\n  const environments: Map<string, cxapi.Environment> = new Map();\n\n  for (const stack of localStacks) {\n    const environment = await sdkProvider.resolveEnvironment(stack.environment);\n    const key = hashObject(environment);\n    environments.set(key, environment);\n  }\n\n  const localByEnvironment = await indexBy(localStacks,\n    async (s) => hashObject(await sdkProvider.resolveEnvironment(s.environment)),\n  );\n\n  const groups: StackGroup[] = [];\n  for (let key of localByEnvironment.keys()) {\n    const environment = environments.get(key)!;\n    const allDeployedStacks = await getDeployedStacks(sdkProvider, environment);\n    const local = localByEnvironment.get(key)!;\n    const hasLocalCounterpart = (s: CloudFormationStack) => local.some((l) => l.stackName === s.stackName);\n    const wasExplicitlyProvided = (s: CloudFormationStack) => additionalStackNames.includes(s.stackName);\n\n    groups.push({\n      environment,\n      deployedStacks: allDeployedStacks.filter(s => hasLocalCounterpart(s) || wasExplicitlyProvided(s)),\n      localStacks: local,\n    });\n  }\n\n  return groups;\n}\n"]}
|
|
166
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"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","sourcesContent":["import type { TypedMapping } from '@aws-cdk/cloudformation-diff';\nimport {\n  formatAmbiguousMappings as fmtAmbiguousMappings,\n  formatEnvironmentSectionHeader as fmtEnvironmentSectionHeader,\n  formatTypedMappings as fmtTypedMappings,\n} from '@aws-cdk/cloudformation-diff';\nimport type * as cxapi from '@aws-cdk/cx-api';\nimport type { StackSummary } from '@aws-sdk/client-cloudformation';\nimport { deserializeStructure, indexBy } from '../../util';\nimport type { SdkProvider } from '../aws-auth/private';\nimport { Mode } from '../plugin';\nimport { StringWriteStream } from '../streams';\nimport type { CloudFormationStack } from './cloudformation';\nimport { ResourceLocation, ResourceMapping } from './cloudformation';\nimport { hashObject } from './digest';\nimport type { MappingGroup } from '../../actions';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\nimport { pLimit } from '../../util/concurrency';\n\nexport * from './exclude';\nexport * from './context';\n\ninterface StackGroup {\n  environment: cxapi.Environment;\n  localStacks: CloudFormationStack[];\n  deployedStacks: CloudFormationStack[];\n}\n\ninterface StackGroup {\n  environment: cxapi.Environment;\n  localStacks: CloudFormationStack[];\n  deployedStacks: CloudFormationStack[];\n}\n\nexport async function usePrescribedMappings(\n  mappingGroups: MappingGroup[],\n  sdkProvider: SdkProvider,\n): Promise<ResourceMapping[]> {\n  interface MappingGroupWithStacks extends MappingGroup {\n    stacks: CloudFormationStack[];\n  }\n\n  const stackGroups: MappingGroupWithStacks[] = [];\n  for (const group of mappingGroups) {\n    const summaries = await listStacks(sdkProvider, environmentOf(group));\n    const stacks = await getDeployedStacksByNames(sdkProvider, environmentOf(group), summaries.map(s => s.StackName!));\n    stackGroups.push({\n      ...group,\n      stacks,\n    });\n  }\n\n  // Validate that there are no duplicate destinations\n  for (let group of stackGroups) {\n    const destinations = new Set<string>();\n\n    for (const destination of Object.values(group.resources)) {\n      if (destinations.has(destination)) {\n        throw new ToolkitError(\n          `Duplicate destination resource '${destination}' in environment ${group.account}/${group.region}`,\n        );\n      }\n      destinations.add(destination);\n    }\n  }\n\n  const result: ResourceMapping[] = [];\n  for (const group of stackGroups) {\n    for (const [source, destination] of Object.entries(group.resources)) {\n      if (!inUse(source, group.stacks)) {\n        throw new ToolkitError(`Source resource '${source}' does not exist in environment ${group.account}/${group.region}`);\n      }\n\n      if (inUse(destination, group.stacks)) {\n        throw new ToolkitError(\n          `Destination resource '${destination}' already in use in environment ${group.account}/${group.region}`,\n        );\n      }\n\n      const environment = environmentOf(group);\n      const src = makeLocation(source, environment, group.stacks);\n      const dst = makeLocation(destination, environment);\n      result.push(new ResourceMapping(src, dst));\n    }\n  }\n  return result;\n\n  function inUse(location: string, stacks: CloudFormationStack[]): boolean {\n    const [stackName, logicalId] = location.split('.');\n    if (stackName == null || logicalId == null) {\n      throw new ToolkitError(`Invalid location '${location}'`);\n    }\n    const stack = stacks.find((s) => s.stackName === stackName);\n    return stack != null && stack.template.Resources?.[logicalId] != null;\n  }\n\n  function environmentOf(group: MappingGroup) {\n    return {\n      account: group.account,\n      region: group.region,\n      name: '',\n    };\n  }\n\n  function makeLocation(\n    loc: string,\n    environment: cxapi.Environment,\n    stacks: CloudFormationStack[] = [],\n  ): ResourceLocation {\n    const [stackName, logicalId] = loc.split('.');\n    const stack = stacks.find((s) => s.stackName === stackName);\n\n    return new ResourceLocation(\n      {\n        stackName,\n        environment,\n        template: stack?.template ?? {},\n      },\n      logicalId,\n    );\n  }\n}\n\nexport async function getDeployedStacksByNames(\n  sdkProvider: SdkProvider,\n  environment: cxapi.Environment,\n  stackNames: string[],\n): Promise<CloudFormationStack[]> {\n  const cfn = (await sdkProvider.forEnvironment(environment, Mode.ForReading)).sdk.cloudFormation();\n\n  const normalize = async (stackName: string) => {\n    const templateCommandOutput = await cfn.getTemplate({ StackName: stackName });\n    const template = deserializeStructure(templateCommandOutput.TemplateBody ?? '{}');\n    return {\n      environment,\n      stackName: stackName,\n      template,\n    };\n  };\n\n  const limit = pLimit(20);\n\n  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  return Promise.all(stackNames.map(s => limit(() => normalize(s))));\n}\n\nasync function listStacks(\n  sdkProvider: SdkProvider,\n  environment: cxapi.Environment,\n): Promise<StackSummary[]> {\n  const cfn = (await sdkProvider.forEnvironment(environment, Mode.ForReading)).sdk.cloudFormation();\n\n  return cfn.paginatedListStacks({\n    StackStatusFilter: [\n      'CREATE_COMPLETE',\n      'UPDATE_COMPLETE',\n      'UPDATE_ROLLBACK_COMPLETE',\n      'IMPORT_COMPLETE',\n      'ROLLBACK_COMPLETE',\n    ],\n  });\n}\n\nexport function formatEnvironmentSectionHeader(environment: cxapi.Environment) {\n  const env = `aws://${environment.account}/${environment.region}`;\n  return formatToStream(stream => fmtEnvironmentSectionHeader(stream, env));\n}\n\nexport function formatTypedMappings(mappings: TypedMapping[]): string {\n  return formatToStream((stream) => fmtTypedMappings(stream, mappings));\n}\n\nexport function formatAmbiguousMappings(paths: [string[], string[]][]): string {\n  return formatToStream((stream) => fmtAmbiguousMappings(stream, paths));\n}\n\nfunction formatToStream(cb: (stream: NodeJS.WritableStream) => void): string {\n  const stream = new StringWriteStream();\n  cb(stream);\n  return stream.toString();\n}\n\n/**\n * Returns a list of stack groups, each containing the local stacks and the deployed stacks that match the given patterns.\n */\nexport async function groupStacks(sdkProvider: SdkProvider, localStacks: CloudFormationStack[], additionalStackNames: string[]) {\n  const environments: Map<string, cxapi.Environment> = new Map();\n\n  for (const stack of localStacks) {\n    const environment = await sdkProvider.resolveEnvironment(stack.environment);\n    const key = hashObject(environment);\n    environments.set(key, environment);\n  }\n\n  const localByEnvironment = await indexBy(localStacks,\n    async (s) => hashObject(await sdkProvider.resolveEnvironment(s.environment)),\n  );\n\n  const groups: StackGroup[] = [];\n  for (let key of localByEnvironment.keys()) {\n    const local = localByEnvironment.get(key)!;\n    const hasLocalCounterpart = (stackName: string) => local.some((l) => l.stackName === stackName);\n    const wasExplicitlyProvided = (stackName: string) => additionalStackNames.includes(stackName);\n\n    const environment = environments.get(key)!;\n    const stackSummaries = await listStacks(sdkProvider, environment);\n    const stackNames = stackSummaries\n      .map(s => s.StackName!)\n      .filter(s => hasLocalCounterpart(s) || wasExplicitlyProvided(s));\n    const deployedStacks = await getDeployedStacksByNames(sdkProvider, environment, stackNames);\n\n    groups.push({\n      environment,\n      deployedStacks,\n      localStacks: local,\n    });\n  }\n\n  return groups;\n}\n"]}
|
package/package.json
CHANGED
|
@@ -34,14 +34,14 @@
|
|
|
34
34
|
"organization": true
|
|
35
35
|
},
|
|
36
36
|
"devDependencies": {
|
|
37
|
-
"@aws-cdk/aws-service-spec": "^0.1.
|
|
38
|
-
"@aws-cdk/cli-plugin-contract": "2.
|
|
37
|
+
"@aws-cdk/aws-service-spec": "^0.1.130",
|
|
38
|
+
"@aws-cdk/cli-plugin-contract": "2.182.0",
|
|
39
39
|
"@cdklabs/eslint-plugin": "^1.3.5",
|
|
40
40
|
"@jest/environment": "^29.7.0",
|
|
41
41
|
"@jest/globals": "^29.7.0",
|
|
42
42
|
"@jest/types": "^29.6.3",
|
|
43
|
-
"@microsoft/api-extractor": "^7.
|
|
44
|
-
"@smithy/util-stream": "^4.
|
|
43
|
+
"@microsoft/api-extractor": "^7.55.2",
|
|
44
|
+
"@smithy/util-stream": "^4.5.6",
|
|
45
45
|
"@stylistic/eslint-plugin": "^3",
|
|
46
46
|
"@types/fs-extra": "^11.0.4",
|
|
47
47
|
"@types/jest": "^29.5.14",
|
|
@@ -50,7 +50,7 @@
|
|
|
50
50
|
"@types/split2": "^4.2.3",
|
|
51
51
|
"@typescript-eslint/eslint-plugin": "^8",
|
|
52
52
|
"@typescript-eslint/parser": "^8",
|
|
53
|
-
"aws-cdk-lib": "2.
|
|
53
|
+
"aws-cdk-lib": "2.232.1",
|
|
54
54
|
"aws-sdk-client-mock": "^4.1.0",
|
|
55
55
|
"aws-sdk-client-mock-jest": "^4.1.0",
|
|
56
56
|
"commit-and-tag-version": "^12",
|
|
@@ -62,7 +62,7 @@
|
|
|
62
62
|
"eslint-plugin-jest": "^28.14.0",
|
|
63
63
|
"eslint-plugin-jsdoc": "^50.8.0",
|
|
64
64
|
"eslint-plugin-prettier": "^5.5.4",
|
|
65
|
-
"fast-check": "^4.
|
|
65
|
+
"fast-check": "^4.4.0",
|
|
66
66
|
"jest": "^29.7.0",
|
|
67
67
|
"jest-environment-node": "^29.7.0",
|
|
68
68
|
"jest-junit": "^16",
|
|
@@ -70,7 +70,7 @@
|
|
|
70
70
|
"license-checker": "^25.0.1",
|
|
71
71
|
"nock": "13",
|
|
72
72
|
"prettier": "^2.8",
|
|
73
|
-
"ts-jest": "^29.4.
|
|
73
|
+
"ts-jest": "^29.4.6",
|
|
74
74
|
"typescript": "5.9",
|
|
75
75
|
"xml-js": "^1.6.11"
|
|
76
76
|
},
|
|
@@ -79,7 +79,7 @@
|
|
|
79
79
|
},
|
|
80
80
|
"dependencies": {
|
|
81
81
|
"@aws-cdk/cdk-assets-lib": "^1",
|
|
82
|
-
"@aws-cdk/cloud-assembly-schema": ">=
|
|
82
|
+
"@aws-cdk/cloud-assembly-schema": ">=49.0.0",
|
|
83
83
|
"@aws-cdk/cloudformation-diff": "^2",
|
|
84
84
|
"@aws-cdk/cx-api": "^2",
|
|
85
85
|
"@aws-sdk/client-appsync": "^3",
|
|
@@ -115,9 +115,9 @@
|
|
|
115
115
|
"fast-deep-equal": "^3.1.3",
|
|
116
116
|
"fs-extra": "^9",
|
|
117
117
|
"glob": "^11.1.0",
|
|
118
|
-
"minimatch": "10.0.
|
|
118
|
+
"minimatch": "10.0.1",
|
|
119
119
|
"p-limit": "^3",
|
|
120
|
-
"semver": "^7.7.
|
|
120
|
+
"semver": "^7.7.3",
|
|
121
121
|
"split2": "^4.2.0",
|
|
122
122
|
"uuid": "^11.1.0",
|
|
123
123
|
"wrap-ansi": "^7",
|
|
@@ -136,7 +136,7 @@
|
|
|
136
136
|
"publishConfig": {
|
|
137
137
|
"access": "public"
|
|
138
138
|
},
|
|
139
|
-
"version": "1.
|
|
139
|
+
"version": "1.12.0",
|
|
140
140
|
"types": "lib/index.d.ts",
|
|
141
141
|
"exports": {
|
|
142
142
|
".": {
|