@aws-cdk/toolkit-lib 1.7.0 → 1.8.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/lib/actions/refactor/index.d.ts +8 -0
- package/lib/actions/refactor/index.js +1 -1
- package/lib/api/aws-auth/sdk.d.ts +5 -1
- package/lib/api/aws-auth/sdk.js +23 -1
- package/lib/api/io/private/messages.d.ts +2 -1
- package/lib/api/io/private/messages.js +6 -1
- package/lib/api/notices/notices.d.ts +1 -1
- package/lib/api/notices/notices.js +2 -2
- package/lib/api/refactoring/cloudformation.d.ts +6 -0
- package/lib/api/refactoring/cloudformation.js +16 -1
- package/lib/api/refactoring/context.d.ts +19 -0
- package/lib/api/refactoring/context.js +143 -11
- package/lib/api/refactoring/digest.js +11 -8
- package/lib/api/refactoring/graph.js +11 -9
- package/lib/api/refactoring/index.d.ts +5 -0
- package/lib/api/refactoring/index.js +1 -1
- package/lib/api/refactoring/stack-definitions.js +49 -436
- package/lib/toolkit/toolkit.d.ts +1 -1
- package/lib/toolkit/toolkit.js +94 -28
- package/lib/util/sets.d.ts +1 -0
- package/lib/util/sets.js +5 -1
- package/package.json +3 -2
|
@@ -1,16 +1,23 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.RefactoringContext = void 0;
|
|
4
|
+
exports.replaceAwsPlaceholders = replaceAwsPlaceholders;
|
|
5
|
+
const cdk_assets_lib_1 = require("@aws-cdk/cdk-assets-lib");
|
|
6
|
+
const cx_api_1 = require("@aws-cdk/cx-api");
|
|
4
7
|
const cloudformation_1 = require("./cloudformation");
|
|
5
8
|
const digest_1 = require("./digest");
|
|
6
9
|
const toolkit_error_1 = require("../../toolkit/toolkit-error");
|
|
7
10
|
const sets_1 = require("../../util/sets");
|
|
11
|
+
const environment_1 = require("../environment");
|
|
12
|
+
const plugin_1 = require("../plugin");
|
|
8
13
|
/**
|
|
9
14
|
* Encapsulates the information for refactoring resources in a single environment.
|
|
10
15
|
*/
|
|
11
16
|
class RefactoringContext {
|
|
12
17
|
_mappings = [];
|
|
13
18
|
ambiguousMoves = [];
|
|
19
|
+
localStacks;
|
|
20
|
+
assumeRoleArn;
|
|
14
21
|
environment;
|
|
15
22
|
constructor(props) {
|
|
16
23
|
this.environment = props.environment;
|
|
@@ -19,6 +26,8 @@ class RefactoringContext {
|
|
|
19
26
|
const overrides = (props.overrides ?? []).concat(additionalOverrides);
|
|
20
27
|
const [nonAmbiguousMoves, ambiguousMoves] = partitionByAmbiguity(overrides, moves);
|
|
21
28
|
this.ambiguousMoves = ambiguousMoves;
|
|
29
|
+
this.localStacks = props.localStacks;
|
|
30
|
+
this.assumeRoleArn = props.assumeRoleArn;
|
|
22
31
|
this._mappings = resourceMappings(nonAmbiguousMoves);
|
|
23
32
|
}
|
|
24
33
|
get ambiguousPaths() {
|
|
@@ -30,6 +39,82 @@ class RefactoringContext {
|
|
|
30
39
|
get mappings() {
|
|
31
40
|
return this._mappings;
|
|
32
41
|
}
|
|
42
|
+
async execute(stackDefinitions, sdkProvider, ioHelper) {
|
|
43
|
+
if (this.mappings.length === 0) {
|
|
44
|
+
return;
|
|
45
|
+
}
|
|
46
|
+
const assumeRoleArn = this.assumeRoleArn ?? await this.findRoleToAssume(sdkProvider);
|
|
47
|
+
const sdk = (await sdkProvider.forEnvironment(this.environment, plugin_1.Mode.ForWriting, {
|
|
48
|
+
assumeRoleArn,
|
|
49
|
+
})).sdk;
|
|
50
|
+
await this.checkBootstrapVersion(sdk, ioHelper);
|
|
51
|
+
const cfn = sdk.cloudFormation();
|
|
52
|
+
const mappings = this.mappings;
|
|
53
|
+
const input = {
|
|
54
|
+
ResourceMappings: mappings.map((m) => m.toCloudFormation()),
|
|
55
|
+
StackDefinitions: stackDefinitions,
|
|
56
|
+
};
|
|
57
|
+
const refactor = await cfn.createStackRefactor(input);
|
|
58
|
+
await cfn.waitUntilStackRefactorCreateComplete({
|
|
59
|
+
StackRefactorId: refactor.StackRefactorId,
|
|
60
|
+
});
|
|
61
|
+
await cfn.executeStackRefactor({
|
|
62
|
+
StackRefactorId: refactor.StackRefactorId,
|
|
63
|
+
});
|
|
64
|
+
await cfn.waitUntilStackRefactorExecuteComplete({
|
|
65
|
+
StackRefactorId: refactor.StackRefactorId,
|
|
66
|
+
});
|
|
67
|
+
}
|
|
68
|
+
async checkBootstrapVersion(sdk, ioHelper) {
|
|
69
|
+
const environmentResourcesRegistry = new environment_1.EnvironmentResourcesRegistry();
|
|
70
|
+
const envResources = environmentResourcesRegistry.for(this.environment, sdk, ioHelper);
|
|
71
|
+
let bootstrapVersion = undefined;
|
|
72
|
+
try {
|
|
73
|
+
// Try to get the bootstrap version
|
|
74
|
+
bootstrapVersion = (await envResources.lookupToolkit()).version;
|
|
75
|
+
}
|
|
76
|
+
catch (e) {
|
|
77
|
+
// But if we can't, keep going. Maybe we can still succeed.
|
|
78
|
+
}
|
|
79
|
+
if (bootstrapVersion != null && bootstrapVersion < 28) {
|
|
80
|
+
const environment = `aws://${this.environment.account}/${this.environment.region}`;
|
|
81
|
+
throw new toolkit_error_1.ToolkitError(`The CDK toolkit stack in environment ${environment} doesn't support refactoring. Please run 'cdk bootstrap ${environment}' to update it.`);
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
async findRoleToAssume(sdkProvider) {
|
|
85
|
+
// To execute a refactor, we need the deployment role ARN for the given
|
|
86
|
+
// environment. Most toolkit commands get the information about which roles to
|
|
87
|
+
// assume from the cloud assembly (and ultimately from the CDK framework). Refactor
|
|
88
|
+
// is different because it is not the application/framework that dictates what the
|
|
89
|
+
// toolkit should do, but it is the toolkit itself that has to figure it out.
|
|
90
|
+
//
|
|
91
|
+
// Nevertheless, the cloud assembly is the most reliable source for this kind of
|
|
92
|
+
// information. For the deployment role ARN, in particular, what we do here
|
|
93
|
+
// is look at all the stacks for a given environment in the cloud assembly and
|
|
94
|
+
// extract the deployment role ARN that is common to all of them. If no role is
|
|
95
|
+
// found, we go ahead without assuming a role. If there is more than one role,
|
|
96
|
+
// we consider that an invariant was violated, and throw an error.
|
|
97
|
+
const env = this.environment;
|
|
98
|
+
const roleArns = new Set(this.localStacks
|
|
99
|
+
.filter((s) => s.environment.account === env.account && s.environment.region === env.region)
|
|
100
|
+
.map((s) => s.assumeRoleArn));
|
|
101
|
+
if (roleArns.size === 0) {
|
|
102
|
+
return undefined;
|
|
103
|
+
}
|
|
104
|
+
if (roleArns.size !== 1) {
|
|
105
|
+
// Unlikely to happen. But if it does, we can't proceed
|
|
106
|
+
throw new toolkit_error_1.ToolkitError(`Multiple stacks in environment aws://${env.account}/${env.region} have different deployment role ARNs. Cannot proceed.`);
|
|
107
|
+
}
|
|
108
|
+
const arn = Array.from(roleArns)[0];
|
|
109
|
+
if (arn != null) {
|
|
110
|
+
const resolvedEnv = await sdkProvider.resolveEnvironment(env);
|
|
111
|
+
const region = resolvedEnv.region;
|
|
112
|
+
return (await replaceAwsPlaceholders({ region, assumeRoleArn: arn }, new cdk_assets_lib_1.DefaultAwsClient())).assumeRoleArn;
|
|
113
|
+
}
|
|
114
|
+
// If we couldn't find a role ARN, we can proceed without assuming a role.
|
|
115
|
+
// Maybe the default credentials have permissions to do what we need.
|
|
116
|
+
return undefined;
|
|
117
|
+
}
|
|
33
118
|
}
|
|
34
119
|
exports.RefactoringContext = RefactoringContext;
|
|
35
120
|
/**
|
|
@@ -57,17 +142,35 @@ function structuralOverrides(deployedStacks, localStacks) {
|
|
|
57
142
|
function resourceMoves(before, after, direction = 'direct', ignoreModifications = false) {
|
|
58
143
|
const digestsBefore = resourceDigests(before, direction);
|
|
59
144
|
const digestsAfter = resourceDigests(after, direction);
|
|
60
|
-
const stackNames = (stacks) => stacks
|
|
61
|
-
.map((s) => s.stackName)
|
|
62
|
-
.sort()
|
|
63
|
-
.join(', ');
|
|
64
145
|
if (!(ignoreModifications || isomorphic(digestsBefore, digestsAfter))) {
|
|
65
|
-
const message = [
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
146
|
+
const message = ['A refactor operation cannot add, remove or update resources. Only resource moves and renames are allowed.'];
|
|
147
|
+
const difference = (a, b) => {
|
|
148
|
+
return Array.from((0, sets_1.setDiff)(new Set(Object.keys(a)), new Set(Object.keys(b)))).flatMap(k => a[k])
|
|
149
|
+
.map(x => ` - ${x.toPath()}`)
|
|
150
|
+
.sort()
|
|
151
|
+
.join('\n');
|
|
152
|
+
};
|
|
153
|
+
const stackNames = (stacks) => stacks.length === 0
|
|
154
|
+
? 'NONE'
|
|
155
|
+
: stacks
|
|
156
|
+
.map((s) => s.stackName)
|
|
157
|
+
.sort()
|
|
158
|
+
.join(', ');
|
|
159
|
+
const onlyDeployed = difference(digestsBefore, digestsAfter);
|
|
160
|
+
const onlyLocal = difference(digestsAfter, digestsBefore);
|
|
161
|
+
if (onlyDeployed.length > 0) {
|
|
162
|
+
message.push(`The following resources are present only in the AWS environment:\n${onlyDeployed}`);
|
|
163
|
+
}
|
|
164
|
+
if (onlyLocal.length > 0) {
|
|
165
|
+
message.push(`\nThe following resources are present only in your CDK application:\n${onlyLocal}`);
|
|
166
|
+
}
|
|
167
|
+
message.push('');
|
|
168
|
+
message.push('The following stacks were used in the comparison:');
|
|
169
|
+
message.push(` - Deployed: ${stackNames(before)}`);
|
|
170
|
+
message.push(` - Local: ${stackNames(after)}`);
|
|
171
|
+
message.push('');
|
|
172
|
+
message.push('Hint: by default, only deployed stacks that have the same name as a local stack are included.');
|
|
173
|
+
message.push('If you want to include additional deployed stacks for comparison, re-run the command with the option \'--additional-stack-name=<STACK>\' for each stack.');
|
|
71
174
|
throw new toolkit_error_1.ToolkitError(message.join('\n'));
|
|
72
175
|
}
|
|
73
176
|
return Object.values(removeUnmovedResources(zip(digestsBefore, digestsAfter)));
|
|
@@ -196,4 +299,33 @@ function partitionByAmbiguity(overrides, moves) {
|
|
|
196
299
|
}
|
|
197
300
|
return [nonAmbiguous, ambiguous];
|
|
198
301
|
}
|
|
199
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Environment } from '@aws-cdk/cx-api';\nimport type { CloudFormationStack } from './cloudformation';\nimport { ResourceLocation, ResourceMapping } from './cloudformation';\nimport type { GraphDirection } from './digest';\nimport { computeResourceDigests } from './digest';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\nimport { equalSets } from '../../util/sets';\n\n/**\n * Represents a set of possible moves of a resource from one location\n * to another. In the ideal case, there is only one source and only one\n * destination.\n */\ntype ResourceMove = [ResourceLocation[], ResourceLocation[]];\n\nexport interface RefactoringContextOptions {\n  environment: Environment;\n  localStacks: CloudFormationStack[];\n  deployedStacks: CloudFormationStack[];\n  overrides?: ResourceMapping[];\n  ignoreModifications?: boolean;\n}\n\n/**\n * Encapsulates the information for refactoring resources in a single environment.\n */\nexport class RefactoringContext {\n  private readonly _mappings: ResourceMapping[] = [];\n  private readonly ambiguousMoves: ResourceMove[] = [];\n  public readonly environment: Environment;\n\n  constructor(props: RefactoringContextOptions) {\n    this.environment = props.environment;\n    const moves = resourceMoves(props.deployedStacks, props.localStacks, 'direct', props.ignoreModifications);\n    const additionalOverrides = structuralOverrides(props.deployedStacks, props.localStacks);\n    const overrides = (props.overrides ?? []).concat(additionalOverrides);\n    const [nonAmbiguousMoves, ambiguousMoves] = partitionByAmbiguity(overrides, moves);\n    this.ambiguousMoves = ambiguousMoves;\n\n    this._mappings = resourceMappings(nonAmbiguousMoves);\n  }\n\n  public get ambiguousPaths(): [string[], string[]][] {\n    return this.ambiguousMoves.map(([a, b]) => [convert(a), convert(b)]);\n\n    function convert(locations: ResourceLocation[]): string[] {\n      return locations.map((l) => l.toPath());\n    }\n  }\n\n  public get mappings(): ResourceMapping[] {\n    return this._mappings;\n  }\n}\n\n/**\n * Generates an automatic list of overrides that can be deduced from the structure of the opposite resource graph.\n * Suppose we have the following resource graph:\n *\n *     A --> B\n *     C --> D\n *\n * such that B and D are identical, but A is different from C. Then digest(B) = digest(D). If both resources are moved,\n * we have an ambiguity. But if we reverse the arrows:\n *\n *     A <-- B\n *     C <-- D\n *\n * then digest(B) ≠ digest(D), because they now have different dependencies. If we compute the mappings from this\n * opposite graph, we can use them as a set of overrides to disambiguate the original moves.\n *\n */\nfunction structuralOverrides(deployedStacks: CloudFormationStack[], localStacks: CloudFormationStack[]): ResourceMapping[] {\n  const moves = resourceMoves(deployedStacks, localStacks, 'opposite', true);\n  const [nonAmbiguousMoves] = partitionByAmbiguity([], moves);\n  return resourceMappings(nonAmbiguousMoves);\n}\n\nfunction resourceMoves(\n  before: CloudFormationStack[],\n  after: CloudFormationStack[],\n  direction: GraphDirection = 'direct',\n  ignoreModifications: boolean = false): ResourceMove[] {\n  const digestsBefore = resourceDigests(before, direction);\n  const digestsAfter = resourceDigests(after, direction);\n\n  const stackNames = (stacks: CloudFormationStack[]) =>\n    stacks\n      .map((s) => s.stackName)\n      .sort()\n      .join(', ');\n  if (!(ignoreModifications || isomorphic(digestsBefore, digestsAfter))) {\n    const message = [\n      'A refactor operation cannot add, remove or update resources. Only resource moves and renames are allowed. ',\n      \"Run 'cdk diff' to compare the local templates to the deployed stacks.\\n\",\n      `Deployed stacks: ${stackNames(before)}`,\n      `Local stacks: ${stackNames(after)}`,\n    ];\n\n    throw new ToolkitError(message.join('\\n'));\n  }\n\n  return Object.values(removeUnmovedResources(zip(digestsBefore, digestsAfter)));\n}\n\n/**\n * Whether two sets of resources have the same elements (uniquely identified by the digest), and\n * each element is in the same number of locations. The locations themselves may be different.\n */\nfunction isomorphic(a: Record<string, ResourceLocation[]>, b: Record<string, ResourceLocation[]>): boolean {\n  const sameKeys = equalSets(new Set(Object.keys(a)), new Set(Object.keys(b)));\n  return sameKeys && Object.entries(a).every(([digest, locations]) => locations.length === b[digest].length);\n}\n\nfunction removeUnmovedResources(moves: Record<string, ResourceMove>): Record<string, ResourceMove> {\n  const result: Record<string, ResourceMove> = {};\n  for (const [hash, [before, after]] of Object.entries(moves)) {\n    const common = before.filter((b) => after.some((a) => a.equalTo(b)));\n    result[hash] = [\n      before.filter((b) => !common.some((c) => b.equalTo(c))),\n      after.filter((a) => !common.some((c) => a.equalTo(c))),\n    ];\n  }\n\n  return result;\n}\n\n/**\n * For each hash, identifying a single resource, zip the two lists of locations,\n * producing a resource move\n */\nfunction zip(\n  m1: Record<string, ResourceLocation[]>,\n  m2: Record<string, ResourceLocation[]>,\n): Record<string, ResourceMove> {\n  const result: Record<string, ResourceMove> = {};\n\n  for (const [hash, locations] of Object.entries(m1)) {\n    if (hash in m2) {\n      result[hash] = [locations, m2[hash]];\n    } else {\n      result[hash] = [locations, []];\n    }\n  }\n\n  for (const [hash, locations] of Object.entries(m2)) {\n    if (!(hash in m1)) {\n      result[hash] = [[], locations];\n    }\n  }\n\n  return result;\n}\n\n/**\n * Computes a list of pairs [digest, location] for each resource in the stack.\n */\nfunction resourceDigests(stacks: CloudFormationStack[], direction: GraphDirection): Record<string, ResourceLocation[]> {\n  // index stacks by name\n  const stacksByName = new Map<string, CloudFormationStack>();\n  for (const stack of stacks) {\n    stacksByName.set(stack.stackName, stack);\n  }\n\n  const digests = computeResourceDigests(stacks, direction);\n\n  return groupByKey(\n    Object.entries(digests).map(([loc, digest]) => {\n      const [stackName, logicalId] = loc.split('.');\n      const location: ResourceLocation = new ResourceLocation(stacksByName.get(stackName)!, logicalId);\n      return [digest, location];\n    }),\n  );\n\n  function groupByKey<A>(entries: [string, A][]): Record<string, A[]> {\n    const result: Record<string, A[]> = {};\n\n    for (const [key, value] of entries) {\n      if (key in result) {\n        result[key].push(value);\n      } else {\n        result[key] = [value];\n      }\n    }\n\n    return result;\n  }\n}\n\nfunction isAmbiguousMove(move: ResourceMove): boolean {\n  const [pre, post] = move;\n\n  // A move is considered ambiguous if two conditions are met:\n  //  1. Both sides have at least one element (otherwise, it's just addition or deletion)\n  //  2. At least one side has more than one element\n  return pre.length > 0 && post.length > 0 && (pre.length > 1 || post.length > 1);\n}\n\nfunction resourceMappings(movements: ResourceMove[]): ResourceMapping[] {\n  return movements\n    .filter(([pre, post]) => pre.length === 1 && post.length === 1 && !pre[0].equalTo(post[0]))\n    .map(([pre, post]) => new ResourceMapping(pre[0], post[0]));\n}\n\n/**\n * Partitions a list of moves into non-ambiguous and ambiguous moves.\n * @param overrides - The list of overrides to disambiguate moves\n * @param moves - a pair of lists of moves. First: non-ambiguous, second: ambiguous\n */\nfunction partitionByAmbiguity(overrides: ResourceMapping[], moves: ResourceMove[]): [ResourceMove[], ResourceMove[]] {\n  const ambiguous: ResourceMove[] = [];\n  const nonAmbiguous: ResourceMove[] = [];\n\n  for (let move of moves) {\n    if (!isAmbiguousMove(move)) {\n      nonAmbiguous.push(move);\n    } else {\n      for (const override of overrides) {\n        const resolvedMove = resolve(override, move);\n        if (resolvedMove != null) {\n          nonAmbiguous.push(resolvedMove);\n          move = remove(override, move);\n        }\n      }\n      // One last chance to be non-ambiguous\n      if (!isAmbiguousMove(move)) {\n        nonAmbiguous.push(move);\n      } else {\n        ambiguous.push(move);\n      }\n    }\n  }\n\n  function resolve(override: ResourceMapping, move: ResourceMove): ResourceMove | undefined {\n    const [pre, post] = move;\n    const source = pre.find((loc) => loc.equalTo(override.source));\n    const destination = post.find((loc) => loc.equalTo(override.destination));\n    return (source && destination) ? [[source], [destination]] : undefined;\n  }\n\n  function remove(override: ResourceMapping, move: ResourceMove): ResourceMove {\n    const [pre, post] = move;\n    return [\n      pre.filter(loc => !loc.equalTo(override.source)),\n      post.filter(loc => !loc.equalTo(override.destination)),\n    ];\n  }\n\n  return [nonAmbiguous, ambiguous];\n}\n"]}
|
|
302
|
+
/**
|
|
303
|
+
* Replace the {ACCOUNT} and {REGION} placeholders in all strings found in a complex object.
|
|
304
|
+
*
|
|
305
|
+
* Duplicated between cdk-assets and aws-cdk CLI because we don't have a good single place to put it
|
|
306
|
+
* (they're nominally independent tools).
|
|
307
|
+
*/
|
|
308
|
+
async function replaceAwsPlaceholders(object, aws) {
|
|
309
|
+
let partition = async () => {
|
|
310
|
+
const p = await aws.discoverPartition();
|
|
311
|
+
partition = () => Promise.resolve(p);
|
|
312
|
+
return p;
|
|
313
|
+
};
|
|
314
|
+
let account = async () => {
|
|
315
|
+
const a = await aws.discoverCurrentAccount();
|
|
316
|
+
account = () => Promise.resolve(a);
|
|
317
|
+
return a;
|
|
318
|
+
};
|
|
319
|
+
return cx_api_1.EnvironmentPlaceholders.replaceAsync(object, {
|
|
320
|
+
async region() {
|
|
321
|
+
return object.region ?? aws.discoverDefaultRegion();
|
|
322
|
+
},
|
|
323
|
+
async accountId() {
|
|
324
|
+
return (await account()).accountId;
|
|
325
|
+
},
|
|
326
|
+
async partition() {
|
|
327
|
+
return partition();
|
|
328
|
+
},
|
|
329
|
+
});
|
|
330
|
+
}
|
|
331
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"context.js","sourceRoot":"","sources":["context.ts"],"names":[],"mappings":"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","sourcesContent":["import { DefaultAwsClient, type IAws } from '@aws-cdk/cdk-assets-lib';\nimport type { Environment } from '@aws-cdk/cx-api';\nimport { EnvironmentPlaceholders } from '@aws-cdk/cx-api';\nimport type { StackDefinition } from '@aws-sdk/client-cloudformation';\nimport type { CloudFormationStack } from './cloudformation';\nimport { ResourceLocation, ResourceMapping } from './cloudformation';\nimport type { GraphDirection } from './digest';\nimport { computeResourceDigests } from './digest';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\nimport { equalSets, setDiff } from '../../util/sets';\nimport type { SDK } from '../aws-auth/sdk';\nimport type { SdkProvider } from '../aws-auth/sdk-provider';\nimport { EnvironmentResourcesRegistry } from '../environment';\nimport type { IoHelper } from '../io/private';\nimport { Mode } from '../plugin';\n\n/**\n * Represents a set of possible moves of a resource from one location\n * to another. In the ideal case, there is only one source and only one\n * destination.\n */\ntype ResourceMove = [ResourceLocation[], ResourceLocation[]];\n\nexport interface RefactoringContextOptions {\n  environment: Environment;\n  localStacks: CloudFormationStack[];\n  deployedStacks: CloudFormationStack[];\n  overrides?: ResourceMapping[];\n  assumeRoleArn?: string;\n  ignoreModifications?: boolean;\n}\n\n/**\n * Encapsulates the information for refactoring resources in a single environment.\n */\nexport class RefactoringContext {\n  private readonly _mappings: ResourceMapping[] = [];\n  private readonly ambiguousMoves: ResourceMove[] = [];\n  private readonly localStacks: CloudFormationStack[];\n  private readonly assumeRoleArn?: string;\n  public readonly environment: Environment;\n\n  constructor(props: RefactoringContextOptions) {\n    this.environment = props.environment;\n    const moves = resourceMoves(props.deployedStacks, props.localStacks, 'direct', props.ignoreModifications);\n    const additionalOverrides = structuralOverrides(props.deployedStacks, props.localStacks);\n    const overrides = (props.overrides ?? []).concat(additionalOverrides);\n    const [nonAmbiguousMoves, ambiguousMoves] = partitionByAmbiguity(overrides, moves);\n    this.ambiguousMoves = ambiguousMoves;\n    this.localStacks = props.localStacks;\n    this.assumeRoleArn = props.assumeRoleArn;\n\n    this._mappings = resourceMappings(nonAmbiguousMoves);\n  }\n\n  public get ambiguousPaths(): [string[], string[]][] {\n    return this.ambiguousMoves.map(([a, b]) => [convert(a), convert(b)]);\n\n    function convert(locations: ResourceLocation[]): string[] {\n      return locations.map((l) => l.toPath());\n    }\n  }\n\n  public get mappings(): ResourceMapping[] {\n    return this._mappings;\n  }\n\n  public async execute(stackDefinitions: StackDefinition[], sdkProvider: SdkProvider, ioHelper: IoHelper): Promise<void> {\n    if (this.mappings.length === 0) {\n      return;\n    }\n\n    const assumeRoleArn = this.assumeRoleArn ?? await this.findRoleToAssume(sdkProvider);\n    const sdk = (\n      await sdkProvider.forEnvironment(this.environment, Mode.ForWriting, {\n        assumeRoleArn,\n      })\n    ).sdk;\n\n    await this.checkBootstrapVersion(sdk, ioHelper);\n\n    const cfn = sdk.cloudFormation();\n    const mappings = this.mappings;\n\n    const input = {\n      ResourceMappings: mappings.map((m) => m.toCloudFormation()),\n      StackDefinitions: stackDefinitions,\n    };\n    const refactor = await cfn.createStackRefactor(input);\n\n    await cfn.waitUntilStackRefactorCreateComplete({\n      StackRefactorId: refactor.StackRefactorId,\n    });\n\n    await cfn.executeStackRefactor({\n      StackRefactorId: refactor.StackRefactorId,\n    });\n\n    await cfn.waitUntilStackRefactorExecuteComplete({\n      StackRefactorId: refactor.StackRefactorId,\n    });\n  }\n\n  private async checkBootstrapVersion(sdk: SDK, ioHelper: IoHelper) {\n    const environmentResourcesRegistry = new EnvironmentResourcesRegistry();\n    const envResources = environmentResourcesRegistry.for(this.environment, sdk, ioHelper);\n    let bootstrapVersion: number | undefined = undefined;\n    try {\n      // Try to get the bootstrap version\n      bootstrapVersion = (await envResources.lookupToolkit()).version;\n    } catch (e) {\n      // But if we can't, keep going. Maybe we can still succeed.\n    }\n    if (bootstrapVersion != null && bootstrapVersion < 28) {\n      const environment = `aws://${this.environment.account}/${this.environment.region}`;\n      throw new ToolkitError(\n        `The CDK toolkit stack in environment ${environment} doesn't support refactoring. Please run 'cdk bootstrap ${environment}' to update it.`,\n      );\n    }\n  }\n\n  private async findRoleToAssume(sdkProvider: SdkProvider): Promise < string | undefined > {\n    // To execute a refactor, we need the deployment role ARN for the given\n    // environment. Most toolkit commands get the information about which roles to\n    // assume from the cloud assembly (and ultimately from the CDK framework). Refactor\n    // is different because it is not the application/framework that dictates what the\n    // toolkit should do, but it is the toolkit itself that has to figure it out.\n    //\n    // Nevertheless, the cloud assembly is the most reliable source for this kind of\n    // information. For the deployment role ARN, in particular, what we do here\n    // is look at all the stacks for a given environment in the cloud assembly and\n    // extract the deployment role ARN that is common to all of them. If no role is\n    // found, we go ahead without assuming a role. If there is more than one role,\n    // we consider that an invariant was violated, and throw an error.\n\n    const env = this.environment;\n    const roleArns = new Set(\n      this.localStacks\n        .filter((s) => s.environment.account === env.account && s.environment.region === env.region)\n        .map((s) => s.assumeRoleArn),\n    );\n\n    if (roleArns.size === 0) {\n      return undefined;\n    }\n\n    if (roleArns.size !== 1) {\n      // Unlikely to happen. But if it does, we can't proceed\n      throw new ToolkitError(\n        `Multiple stacks in environment aws://${env.account}/${env.region} have different deployment role ARNs. Cannot proceed.`,\n      );\n    }\n\n    const arn = Array.from(roleArns)[0];\n    if (arn != null) {\n      const resolvedEnv = await sdkProvider.resolveEnvironment(env);\n      const region = resolvedEnv.region;\n      return (await replaceAwsPlaceholders({ region, assumeRoleArn: arn }, new DefaultAwsClient())).assumeRoleArn;\n    }\n\n    // If we couldn't find a role ARN, we can proceed without assuming a role.\n    // Maybe the default credentials have permissions to do what we need.\n    return undefined;\n  }\n}\n\n/**\n * Generates an automatic list of overrides that can be deduced from the structure of the opposite resource graph.\n * Suppose we have the following resource graph:\n *\n *     A --> B\n *     C --> D\n *\n * such that B and D are identical, but A is different from C. Then digest(B) = digest(D). If both resources are moved,\n * we have an ambiguity. But if we reverse the arrows:\n *\n *     A <-- B\n *     C <-- D\n *\n * then digest(B) ≠ digest(D), because they now have different dependencies. If we compute the mappings from this\n * opposite graph, we can use them as a set of overrides to disambiguate the original moves.\n *\n */\nfunction structuralOverrides(deployedStacks: CloudFormationStack[], localStacks: CloudFormationStack[]): ResourceMapping[] {\n  const moves = resourceMoves(deployedStacks, localStacks, 'opposite', true);\n  const [nonAmbiguousMoves] = partitionByAmbiguity([], moves);\n  return resourceMappings(nonAmbiguousMoves);\n}\n\nfunction resourceMoves(\n  before: CloudFormationStack[],\n  after: CloudFormationStack[],\n  direction: GraphDirection = 'direct',\n  ignoreModifications: boolean = false): ResourceMove[] {\n  const digestsBefore = resourceDigests(before, direction);\n  const digestsAfter = resourceDigests(after, direction);\n\n  if (!(ignoreModifications || isomorphic(digestsBefore, digestsAfter))) {\n    const message = ['A refactor operation cannot add, remove or update resources. Only resource moves and renames are allowed.'];\n\n    const difference = (a: Record<string, ResourceLocation[]>, b: Record<string, ResourceLocation[]>) => {\n      return Array.from(setDiff(new Set(Object.keys(a)), new Set(Object.keys(b)))).flatMap(k => a[k]!)\n        .map(x => `  - ${x.toPath()}`)\n        .sort()\n        .join('\\n');\n    };\n\n    const stackNames = (stacks: CloudFormationStack[]) =>\n      stacks.length === 0\n        ? 'NONE'\n        : stacks\n          .map((s) => s.stackName)\n          .sort()\n          .join(', ');\n\n    const onlyDeployed = difference(digestsBefore, digestsAfter);\n    const onlyLocal = difference(digestsAfter, digestsBefore);\n\n    if (onlyDeployed.length > 0) {\n      message.push(`The following resources are present only in the AWS environment:\\n${onlyDeployed}`);\n    }\n\n    if (onlyLocal.length > 0) {\n      message.push(`\\nThe following resources are present only in your CDK application:\\n${onlyLocal}`);\n    }\n\n    message.push('');\n    message.push('The following stacks were used in the comparison:');\n    message.push( `  - Deployed: ${stackNames(before)}`);\n    message.push( `  - Local: ${stackNames(after)}`);\n    message.push('');\n    message.push('Hint: by default, only deployed stacks that have the same name as a local stack are included.');\n    message.push('If you want to include additional deployed stacks for comparison, re-run the command with the option \\'--additional-stack-name=<STACK>\\' for each stack.');\n\n    throw new ToolkitError(message.join('\\n'));\n  }\n\n  return Object.values(removeUnmovedResources(zip(digestsBefore, digestsAfter)));\n}\n\n/**\n * Whether two sets of resources have the same elements (uniquely identified by the digest), and\n * each element is in the same number of locations. The locations themselves may be different.\n */\nfunction isomorphic(a: Record<string, ResourceLocation[]>, b: Record<string, ResourceLocation[]>): boolean {\n  const sameKeys = equalSets(new Set(Object.keys(a)), new Set(Object.keys(b)));\n  return sameKeys && Object.entries(a).every(([digest, locations]) => locations.length === b[digest].length);\n}\n\nfunction removeUnmovedResources(moves: Record<string, ResourceMove>): Record<string, ResourceMove> {\n  const result: Record<string, ResourceMove> = {};\n  for (const [hash, [before, after]] of Object.entries(moves)) {\n    const common = before.filter((b) => after.some((a) => a.equalTo(b)));\n    result[hash] = [\n      before.filter((b) => !common.some((c) => b.equalTo(c))),\n      after.filter((a) => !common.some((c) => a.equalTo(c))),\n    ];\n  }\n\n  return result;\n}\n\n/**\n * For each hash, identifying a single resource, zip the two lists of locations,\n * producing a resource move\n */\nfunction zip(\n  m1: Record<string, ResourceLocation[]>,\n  m2: Record<string, ResourceLocation[]>,\n): Record<string, ResourceMove> {\n  const result: Record<string, ResourceMove> = {};\n\n  for (const [hash, locations] of Object.entries(m1)) {\n    if (hash in m2) {\n      result[hash] = [locations, m2[hash]];\n    } else {\n      result[hash] = [locations, []];\n    }\n  }\n\n  for (const [hash, locations] of Object.entries(m2)) {\n    if (!(hash in m1)) {\n      result[hash] = [[], locations];\n    }\n  }\n\n  return result;\n}\n\n/**\n * Computes a list of pairs [digest, location] for each resource in the stack.\n */\nfunction resourceDigests(stacks: CloudFormationStack[], direction: GraphDirection): Record<string, ResourceLocation[]> {\n  // index stacks by name\n  const stacksByName = new Map<string, CloudFormationStack>();\n  for (const stack of stacks) {\n    stacksByName.set(stack.stackName, stack);\n  }\n\n  const digests = computeResourceDigests(stacks, direction);\n\n  return groupByKey(\n    Object.entries(digests).map(([loc, digest]) => {\n      const [stackName, logicalId] = loc.split('.');\n      const location: ResourceLocation = new ResourceLocation(stacksByName.get(stackName)!, logicalId);\n      return [digest, location];\n    }),\n  );\n\n  function groupByKey<A>(entries: [string, A][]): Record<string, A[]> {\n    const result: Record<string, A[]> = {};\n\n    for (const [key, value] of entries) {\n      if (key in result) {\n        result[key].push(value);\n      } else {\n        result[key] = [value];\n      }\n    }\n\n    return result;\n  }\n}\n\nfunction isAmbiguousMove(move: ResourceMove): boolean {\n  const [pre, post] = move;\n\n  // A move is considered ambiguous if two conditions are met:\n  //  1. Both sides have at least one element (otherwise, it's just addition or deletion)\n  //  2. At least one side has more than one element\n  return pre.length > 0 && post.length > 0 && (pre.length > 1 || post.length > 1);\n}\n\nfunction resourceMappings(movements: ResourceMove[]): ResourceMapping[] {\n  return movements\n    .filter(([pre, post]) => pre.length === 1 && post.length === 1 && !pre[0].equalTo(post[0]))\n    .map(([pre, post]) => new ResourceMapping(pre[0], post[0]));\n}\n\n/**\n * Partitions a list of moves into non-ambiguous and ambiguous moves.\n * @param overrides - The list of overrides to disambiguate moves\n * @param moves - a pair of lists of moves. First: non-ambiguous, second: ambiguous\n */\nfunction partitionByAmbiguity(overrides: ResourceMapping[], moves: ResourceMove[]): [ResourceMove[], ResourceMove[]] {\n  const ambiguous: ResourceMove[] = [];\n  const nonAmbiguous: ResourceMove[] = [];\n\n  for (let move of moves) {\n    if (!isAmbiguousMove(move)) {\n      nonAmbiguous.push(move);\n    } else {\n      for (const override of overrides) {\n        const resolvedMove = resolve(override, move);\n        if (resolvedMove != null) {\n          nonAmbiguous.push(resolvedMove);\n          move = remove(override, move);\n        }\n      }\n      // One last chance to be non-ambiguous\n      if (!isAmbiguousMove(move)) {\n        nonAmbiguous.push(move);\n      } else {\n        ambiguous.push(move);\n      }\n    }\n  }\n\n  function resolve(override: ResourceMapping, move: ResourceMove): ResourceMove | undefined {\n    const [pre, post] = move;\n    const source = pre.find((loc) => loc.equalTo(override.source));\n    const destination = post.find((loc) => loc.equalTo(override.destination));\n    return (source && destination) ? [[source], [destination]] : undefined;\n  }\n\n  function remove(override: ResourceMapping, move: ResourceMove): ResourceMove {\n    const [pre, post] = move;\n    return [\n      pre.filter(loc => !loc.equalTo(override.source)),\n      post.filter(loc => !loc.equalTo(override.destination)),\n    ];\n  }\n\n  return [nonAmbiguous, ambiguous];\n}\n\n/**\n * Replace the {ACCOUNT} and {REGION} placeholders in all strings found in a complex object.\n *\n * Duplicated between cdk-assets and aws-cdk CLI because we don't have a good single place to put it\n * (they're nominally independent tools).\n */\nexport async function replaceAwsPlaceholders<A extends { region?: string }>(\n  object: A,\n  aws: IAws,\n): Promise<A> {\n  let partition = async () => {\n    const p = await aws.discoverPartition();\n    partition = () => Promise.resolve(p);\n    return p;\n  };\n\n  let account = async () => {\n    const a = await aws.discoverCurrentAccount();\n    account = () => Promise.resolve(a);\n    return a;\n  };\n\n  return EnvironmentPlaceholders.replaceAsync(object, {\n    async region() {\n      return object.region ?? aws.discoverDefaultRegion();\n    },\n    async accountId() {\n      return (await account()).accountId;\n    },\n    async partition() {\n      return partition();\n    },\n  });\n}\n\n"]}
|
|
@@ -93,13 +93,16 @@ function stripReferences(value, exports) {
|
|
|
93
93
|
return { __cloud_ref__: 'DependsOn' };
|
|
94
94
|
}
|
|
95
95
|
if ('Fn::ImportValue' in value) {
|
|
96
|
-
const
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
96
|
+
const exp = exports[value['Fn::ImportValue']];
|
|
97
|
+
if (exp != null) {
|
|
98
|
+
const v = exp.value;
|
|
99
|
+
// Treat Fn::ImportValue as if it were a reference with the same stack
|
|
100
|
+
if ('Ref' in v) {
|
|
101
|
+
return { __cloud_ref__: 'Ref' };
|
|
102
|
+
}
|
|
103
|
+
else if ('Fn::GetAtt' in v) {
|
|
104
|
+
return { __cloud_ref__: 'Fn::GetAtt' };
|
|
105
|
+
}
|
|
103
106
|
}
|
|
104
107
|
}
|
|
105
108
|
const result = {};
|
|
@@ -116,4 +119,4 @@ function stripConstructPath(resource) {
|
|
|
116
119
|
delete copy.Metadata['aws:cdk:path'];
|
|
117
120
|
return copy;
|
|
118
121
|
}
|
|
119
|
-
//# 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 type { CloudFormationResource, CloudFormationStack } from './cloudformation';\nimport { ResourceGraph } from './graph';\n\nexport type GraphDirection =\n  'direct' // Edge A -> B mean that A depends on B\n  | 'opposite'; // Edge A -> B mean that B depends on A\n\n/**\n * Computes the digest for each resource in the template.\n *\n * Conceptually, the digest is computed as:\n *\n *     digest(resource) = hash(type + properties + dependencies.map(d))\n *\n * where `hash` is a cryptographic hash function. In other words, the digest of a\n * resource is computed from its type, its own properties (that is, excluding\n * properties that refer to other resources), and the digests of each of its\n * 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[], direction: GraphDirection = 'direct'): 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(\n          (o) =>\n            [o.Export.Name, { stackName: s.stackName, value: o.Value }] as [string, { stackName: string; value: any }],\n        ),\n    ),\n  );\n\n  const resources = Object.fromEntries(\n    stacks.flatMap((s) => {\n      return Object.entries(s.template.Resources ?? {})\n        .filter(([_, res]) => res.Type !== 'AWS::CDK::Metadata')\n        .map(([id, res]) => [`${s.stackName}.${id}`, res] as [string, CloudFormationResource]);\n    }),\n  );\n\n  const graph = direction == 'direct'\n    ? ResourceGraph.fromStacks(stacks)\n    : ResourceGraph.fromStacks(stacks).opposite();\n\n  return computeDigestsInTopologicalOrder(graph, resources, exports);\n}\n\nfunction computeDigestsInTopologicalOrder(\n  graph: ResourceGraph,\n  resources: Record<string, CloudFormationResource>,\n  exports: Record<string, { stackName: string; value: any }>): Record<string, string> {\n  const nodes = graph.sortedNodes.filter(n => resources[n] != null);\n  const result: Record<string, string> = {};\n  for (const id of nodes) {\n    const resource = resources[id];\n    const depDigests = Array.from(graph.outNeighbors(id)).map((d) => result[d]);\n    const propertiesHash = hashObject(stripReferences(stripConstructPath(resource), exports));\n    const toHash = resource.Type + propertiesHash + depDigests.join('');\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"]}
|
|
122
|
+
//# 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 type { CloudFormationResource, CloudFormationStack } from './cloudformation';\nimport { ResourceGraph } from './graph';\n\nexport type GraphDirection =\n  'direct' // Edge A -> B mean that A depends on B\n  | 'opposite'; // Edge A -> B mean that B depends on A\n\n/**\n * Computes the digest for each resource in the template.\n *\n * Conceptually, the digest is computed as:\n *\n *     digest(resource) = hash(type + properties + dependencies.map(d))\n *\n * where `hash` is a cryptographic hash function. In other words, the digest of a\n * resource is computed from its type, its own properties (that is, excluding\n * properties that refer to other resources), and the digests of each of its\n * 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[], direction: GraphDirection = 'direct'): 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(\n          (o) =>\n            [o.Export.Name, { stackName: s.stackName, value: o.Value }] as [string, { stackName: string; value: any }],\n        ),\n    ),\n  );\n\n  const resources = Object.fromEntries(\n    stacks.flatMap((s) => {\n      return Object.entries(s.template.Resources ?? {})\n        .filter(([_, res]) => res.Type !== 'AWS::CDK::Metadata')\n        .map(([id, res]) => [`${s.stackName}.${id}`, res] as [string, CloudFormationResource]);\n    }),\n  );\n\n  const graph = direction == 'direct'\n    ? ResourceGraph.fromStacks(stacks)\n    : ResourceGraph.fromStacks(stacks).opposite();\n\n  return computeDigestsInTopologicalOrder(graph, resources, exports);\n}\n\nfunction computeDigestsInTopologicalOrder(\n  graph: ResourceGraph,\n  resources: Record<string, CloudFormationResource>,\n  exports: Record<string, { stackName: string; value: any }>): Record<string, string> {\n  const nodes = graph.sortedNodes.filter(n => resources[n] != null);\n  const result: Record<string, string> = {};\n  for (const id of nodes) {\n    const resource = resources[id];\n    const depDigests = Array.from(graph.outNeighbors(id)).map((d) => result[d]);\n    const propertiesHash = hashObject(stripReferences(stripConstructPath(resource), exports));\n    const toHash = resource.Type + propertiesHash + depDigests.join('');\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 exp = exports[value['Fn::ImportValue']];\n    if (exp != null) {\n      const v = exp.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  }\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"]}
|
|
@@ -36,15 +36,17 @@ class ResourceGraph {
|
|
|
36
36
|
}
|
|
37
37
|
if ('Fn::ImportValue' in value) {
|
|
38
38
|
const exp = exports[value['Fn::ImportValue']];
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
39
|
+
if (exp != null) {
|
|
40
|
+
const v = exp.value;
|
|
41
|
+
if ('Fn::GetAtt' in v) {
|
|
42
|
+
const id = Array.isArray(v['Fn::GetAtt']) ? v['Fn::GetAtt'][0] : v['Fn::GetAtt'].split('.')[0];
|
|
43
|
+
return [`${exp.stackName}.${id}`];
|
|
44
|
+
}
|
|
45
|
+
if ('Ref' in v) {
|
|
46
|
+
return [`${exp.stackName}.${v.Ref}`];
|
|
47
|
+
}
|
|
48
|
+
return [`${exp.stackName}.${v}`];
|
|
43
49
|
}
|
|
44
|
-
if ('Ref' in v) {
|
|
45
|
-
return [`${exp.stackName}.${v.Ref}`];
|
|
46
|
-
}
|
|
47
|
-
return [`${exp.stackName}.${v}`];
|
|
48
50
|
}
|
|
49
51
|
const result = [];
|
|
50
52
|
if ('DependsOn' in value) {
|
|
@@ -118,4 +120,4 @@ class ResourceGraph {
|
|
|
118
120
|
}
|
|
119
121
|
}
|
|
120
122
|
exports.ResourceGraph = ResourceGraph;
|
|
121
|
-
//# 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  public static fromStacks(stacks: Omit<CloudFormationStack, 'environment'>[]): ResourceGraph {\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    const edges: Record<string, Set<string>> = {};\n    const reverseEdges: Record<string, Set<string>> = {};\n    for (const id of Object.keys(resources)) {\n      edges[id] = new Set();\n      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          edges[id].add(dep);\n          reverseEdges[dep].add(id);\n        }\n      }\n    }\n\n    return new ResourceGraph(edges, reverseEdges);\n  }\n\n  private readonly edges: Record<string, Set<string>> = {};\n  private readonly reverseEdges: Record<string, Set<string>> = {};\n\n  private constructor(edges: Record<string, Set<string>>, reverseEdges: Record<string, Set<string>>) {\n    this.edges = edges;\n    this.reverseEdges = reverseEdges;\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  /**\n   * Returns another graph with the same nodes, but with the edges inverted\n   */\n  public opposite(): ResourceGraph {\n    return new ResourceGraph(this.reverseEdges, this.edges);\n  }\n}\n"]}
|
|
123
|
+
//# 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  public static fromStacks(stacks: Omit<CloudFormationStack, 'environment'>[]): ResourceGraph {\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    const edges: Record<string, Set<string>> = {};\n    const reverseEdges: Record<string, Set<string>> = {};\n    for (const id of Object.keys(resources)) {\n      edges[id] = new Set();\n      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        if (exp != null) {\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      }\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          edges[id].add(dep);\n          reverseEdges[dep].add(id);\n        }\n      }\n    }\n\n    return new ResourceGraph(edges, reverseEdges);\n  }\n\n  private readonly edges: Record<string, Set<string>> = {};\n  private readonly reverseEdges: Record<string, Set<string>> = {};\n\n  private constructor(edges: Record<string, Set<string>>, reverseEdges: Record<string, Set<string>>) {\n    this.edges = edges;\n    this.reverseEdges = reverseEdges;\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  /**\n   * Returns another graph with the same nodes, but with the edges inverted\n   */\n  public opposite(): ResourceGraph {\n    return new ResourceGraph(this.reverseEdges, this.edges);\n  }\n}\n"]}
|
|
@@ -11,6 +11,11 @@ interface StackGroup {
|
|
|
11
11
|
localStacks: CloudFormationStack[];
|
|
12
12
|
deployedStacks: CloudFormationStack[];
|
|
13
13
|
}
|
|
14
|
+
interface StackGroup {
|
|
15
|
+
environment: cxapi.Environment;
|
|
16
|
+
localStacks: CloudFormationStack[];
|
|
17
|
+
deployedStacks: CloudFormationStack[];
|
|
18
|
+
}
|
|
14
19
|
export declare function usePrescribedMappings(mappingGroups: MappingGroup[], sdkProvider: SdkProvider): Promise<ResourceMapping[]>;
|
|
15
20
|
export declare function getDeployedStacks(sdkProvider: SdkProvider, environment: cxapi.Environment): Promise<CloudFormationStack[]>;
|
|
16
21
|
export declare function formatEnvironmentSectionHeader(environment: cxapi.Environment): string;
|
|
@@ -152,4 +152,4 @@ async function groupStacks(sdkProvider, localStacks, additionalStackNames) {
|
|
|
152
152
|
}
|
|
153
153
|
return groups;
|
|
154
154
|
}
|
|
155
|
-
//# 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';\n\nexport * from './exclude';\nexport * from './context';\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  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  return Promise.all(summaries.map(normalize));\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"]}
|
|
155
|
+
//# 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';\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  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism\n  return Promise.all(summaries.map(normalize));\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"]}
|