@aws-cdk/toolkit-lib 1.2.3 → 1.3.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.
Files changed (49) hide show
  1. package/build-info.json +2 -2
  2. package/db.json.gz +0 -0
  3. package/lib/actions/diff/index.d.ts +7 -0
  4. package/lib/actions/diff/index.js +1 -1
  5. package/lib/actions/diff/private/helpers.js +7 -1
  6. package/lib/actions/refactor/index.d.ts +15 -34
  7. package/lib/actions/refactor/index.js +1 -54
  8. package/lib/actions/refactor/private/mapping-helpers.d.ts +11 -0
  9. package/lib/actions/refactor/private/mapping-helpers.js +44 -0
  10. package/lib/api/cloud-assembly/private/context-aware-source.js +3 -16
  11. package/lib/api/cloud-assembly/private/exec.js +12 -3
  12. package/lib/api/cloud-assembly/private/prepare-source.d.ts +29 -5
  13. package/lib/api/cloud-assembly/private/prepare-source.js +45 -20
  14. package/lib/api/cloud-assembly/source-builder.d.ts +11 -0
  15. package/lib/api/cloud-assembly/source-builder.js +11 -10
  16. package/lib/api/cloud-assembly/stack-collection.js +2 -1
  17. package/lib/api/diff/diff-formatter.d.ts +8 -0
  18. package/lib/api/diff/diff-formatter.js +29 -7
  19. package/lib/api/io/private/message-maker.d.ts +5 -5
  20. package/lib/api/io/private/messages.d.ts +1 -0
  21. package/lib/api/io/private/messages.js +6 -1
  22. package/lib/api/io/toolkit-action.d.ts +1 -1
  23. package/lib/api/io/toolkit-action.js +1 -1
  24. package/lib/api/refactoring/cloudformation.d.ts +1 -0
  25. package/lib/api/refactoring/cloudformation.js +6 -4
  26. package/lib/api/refactoring/context.d.ts +4 -5
  27. package/lib/api/refactoring/context.js +122 -52
  28. package/lib/api/refactoring/digest.d.ts +7 -12
  29. package/lib/api/refactoring/digest.js +22 -42
  30. package/lib/api/refactoring/graph.d.ts +6 -1
  31. package/lib/api/refactoring/graph.js +21 -8
  32. package/lib/api/refactoring/index.d.ts +13 -4
  33. package/lib/api/refactoring/index.js +38 -18
  34. package/lib/api/tree.js +1 -1
  35. package/lib/context-providers/cc-api-provider.js +23 -14
  36. package/lib/index_bg.wasm +0 -0
  37. package/lib/payloads/refactor.d.ts +1 -1
  38. package/lib/payloads/refactor.js +1 -1
  39. package/lib/payloads/stack-details.d.ts +4 -0
  40. package/lib/payloads/stack-details.js +1 -1
  41. package/lib/toolkit/toolkit.d.ts +6 -2
  42. package/lib/toolkit/toolkit.js +114 -79
  43. package/lib/toolkit/types.d.ts +7 -0
  44. package/lib/toolkit/types.js +1 -1
  45. package/lib/util/arrays.d.ts +1 -0
  46. package/lib/util/arrays.js +16 -1
  47. package/lib/util/sets.d.ts +5 -0
  48. package/lib/util/sets.js +18 -0
  49. package/package.json +9 -9
@@ -4,6 +4,7 @@ exports.RefactoringContext = void 0;
4
4
  const cloudformation_1 = require("./cloudformation");
5
5
  const digest_1 = require("./digest");
6
6
  const toolkit_error_1 = require("../../toolkit/toolkit-error");
7
+ const sets_1 = require("../../util/sets");
7
8
  /**
8
9
  * Encapsulates the information for refactoring resources in a single environment.
9
10
  */
@@ -13,19 +14,12 @@ class RefactoringContext {
13
14
  environment;
14
15
  constructor(props) {
15
16
  this.environment = props.environment;
16
- if (props.mappings != null) {
17
- this._mappings = props.mappings;
18
- }
19
- else {
20
- const moves = resourceMoves(props.deployedStacks, props.localStacks);
21
- this.ambiguousMoves = ambiguousMoves(moves);
22
- if (!this.isAmbiguous) {
23
- this._mappings = resourceMappings(resourceMoves(props.deployedStacks, props.localStacks), props.filteredStacks);
24
- }
25
- }
26
- }
27
- get isAmbiguous() {
28
- return this.ambiguousMoves.length > 0;
17
+ const moves = resourceMoves(props.deployedStacks, props.localStacks, 'direct', props.ignoreModifications);
18
+ const additionalOverrides = structuralOverrides(props.deployedStacks, props.localStacks);
19
+ const overrides = (props.overrides ?? []).concat(additionalOverrides);
20
+ const [nonAmbiguousMoves, ambiguousMoves] = partitionByAmbiguity(overrides, moves);
21
+ this.ambiguousMoves = ambiguousMoves;
22
+ this._mappings = resourceMappings(nonAmbiguousMoves);
29
23
  }
30
24
  get ambiguousPaths() {
31
25
  return this.ambiguousMoves.map(([a, b]) => [convert(a), convert(b)]);
@@ -34,19 +28,61 @@ class RefactoringContext {
34
28
  }
35
29
  }
36
30
  get mappings() {
37
- if (this.isAmbiguous) {
38
- throw new toolkit_error_1.ToolkitError('Cannot access mappings when there are ambiguous resource moves. Please resolve the ambiguity first.');
39
- }
40
31
  return this._mappings;
41
32
  }
42
33
  }
43
34
  exports.RefactoringContext = RefactoringContext;
44
- function resourceMoves(before, after) {
45
- return Object.values(removeUnmovedResources(zip(groupByKey(resourceDigests(before)), groupByKey(resourceDigests(after)))));
35
+ /**
36
+ * Generates an automatic list of overrides that can be deduced from the structure of the opposite resource graph.
37
+ * Suppose we have the following resource graph:
38
+ *
39
+ * A --> B
40
+ * C --> D
41
+ *
42
+ * such that B and D are identical, but A is different from C. Then digest(B) = digest(D). If both resources are moved,
43
+ * we have an ambiguity. But if we reverse the arrows:
44
+ *
45
+ * A <-- B
46
+ * C <-- D
47
+ *
48
+ * then digest(B) ≠ digest(D), because they now have different dependencies. If we compute the mappings from this
49
+ * opposite graph, we can use them as a set of overrides to disambiguate the original moves.
50
+ *
51
+ */
52
+ function structuralOverrides(deployedStacks, localStacks) {
53
+ const moves = resourceMoves(deployedStacks, localStacks, 'opposite', true);
54
+ const [nonAmbiguousMoves] = partitionByAmbiguity([], moves);
55
+ return resourceMappings(nonAmbiguousMoves);
56
+ }
57
+ function resourceMoves(before, after, direction = 'direct', ignoreModifications = false) {
58
+ const digestsBefore = resourceDigests(before, direction);
59
+ const digestsAfter = resourceDigests(after, direction);
60
+ const stackNames = (stacks) => stacks
61
+ .map((s) => s.stackName)
62
+ .sort()
63
+ .join(', ');
64
+ if (!(ignoreModifications || isomorphic(digestsBefore, digestsAfter))) {
65
+ const message = [
66
+ 'A refactor operation cannot add, remove or update resources. Only resource moves and renames are allowed. ',
67
+ "Run 'cdk diff' to compare the local templates to the deployed stacks.\n",
68
+ `Deployed stacks: ${stackNames(before)}`,
69
+ `Local stacks: ${stackNames(after)}`,
70
+ ];
71
+ throw new toolkit_error_1.ToolkitError(message.join('\n'));
72
+ }
73
+ return Object.values(removeUnmovedResources(zip(digestsBefore, digestsAfter)));
46
74
  }
47
- function removeUnmovedResources(m) {
75
+ /**
76
+ * Whether two sets of resources have the same elements (uniquely identified by the digest), and
77
+ * each element is in the same number of locations. The locations themselves may be different.
78
+ */
79
+ function isomorphic(a, b) {
80
+ const sameKeys = (0, sets_1.equalSets)(new Set(Object.keys(a)), new Set(Object.keys(b)));
81
+ return sameKeys && Object.entries(a).every(([digest, locations]) => locations.length === b[digest].length);
82
+ }
83
+ function removeUnmovedResources(moves) {
48
84
  const result = {};
49
- for (const [hash, [before, after]] of Object.entries(m)) {
85
+ for (const [hash, [before, after]] of Object.entries(moves)) {
50
86
  const common = before.filter((b) => after.some((a) => a.equalTo(b)));
51
87
  result[hash] = [
52
88
  before.filter((b) => !common.some((c) => b.equalTo(c))),
@@ -76,54 +112,88 @@ function zip(m1, m2) {
76
112
  }
77
113
  return result;
78
114
  }
79
- function groupByKey(entries) {
80
- const result = {};
81
- for (const [hash, location] of entries) {
82
- if (hash in result) {
83
- result[hash].push(location);
84
- }
85
- else {
86
- result[hash] = [location];
87
- }
88
- }
89
- return result;
90
- }
91
115
  /**
92
116
  * Computes a list of pairs [digest, location] for each resource in the stack.
93
117
  */
94
- function resourceDigests(stacks) {
118
+ function resourceDigests(stacks, direction) {
95
119
  // index stacks by name
96
120
  const stacksByName = new Map();
97
121
  for (const stack of stacks) {
98
122
  stacksByName.set(stack.stackName, stack);
99
123
  }
100
- const digests = (0, digest_1.computeResourceDigests)(stacks);
101
- return Object.entries(digests).map(([loc, digest]) => {
124
+ const digests = (0, digest_1.computeResourceDigests)(stacks, direction);
125
+ return groupByKey(Object.entries(digests).map(([loc, digest]) => {
102
126
  const [stackName, logicalId] = loc.split('.');
103
127
  const location = new cloudformation_1.ResourceLocation(stacksByName.get(stackName), logicalId);
104
128
  return [digest, location];
105
- });
129
+ }));
130
+ function groupByKey(entries) {
131
+ const result = {};
132
+ for (const [key, value] of entries) {
133
+ if (key in result) {
134
+ result[key].push(value);
135
+ }
136
+ else {
137
+ result[key] = [value];
138
+ }
139
+ }
140
+ return result;
141
+ }
106
142
  }
107
- function ambiguousMoves(movements) {
143
+ function isAmbiguousMove(move) {
144
+ const [pre, post] = move;
108
145
  // A move is considered ambiguous if two conditions are met:
109
146
  // 1. Both sides have at least one element (otherwise, it's just addition or deletion)
110
147
  // 2. At least one side has more than one element
111
- return movements
112
- .filter(([pre, post]) => pre.length > 0 && post.length > 0)
113
- .filter(([pre, post]) => pre.length > 1 || post.length > 1);
148
+ return pre.length > 0 && post.length > 0 && (pre.length > 1 || post.length > 1);
114
149
  }
115
- function resourceMappings(movements, stacks) {
116
- const stacksPredicate = stacks == null
117
- ? () => true
118
- : (m) => {
119
- // Any movement that involves one of the selected stacks (either moving from or to)
120
- // is considered a candidate for refactoring.
121
- const stackNames = [m.source.stack.stackName, m.destination.stack.stackName];
122
- return stacks.some((stack) => stackNames.includes(stack.stackName));
123
- };
150
+ function resourceMappings(movements) {
124
151
  return movements
125
152
  .filter(([pre, post]) => pre.length === 1 && post.length === 1 && !pre[0].equalTo(post[0]))
126
- .map(([pre, post]) => new cloudformation_1.ResourceMapping(pre[0], post[0]))
127
- .filter(stacksPredicate);
153
+ .map(([pre, post]) => new cloudformation_1.ResourceMapping(pre[0], post[0]));
154
+ }
155
+ /**
156
+ * Partitions a list of moves into non-ambiguous and ambiguous moves.
157
+ * @param overrides - The list of overrides to disambiguate moves
158
+ * @param moves - a pair of lists of moves. First: non-ambiguous, second: ambiguous
159
+ */
160
+ function partitionByAmbiguity(overrides, moves) {
161
+ const ambiguous = [];
162
+ const nonAmbiguous = [];
163
+ for (let move of moves) {
164
+ if (!isAmbiguousMove(move)) {
165
+ nonAmbiguous.push(move);
166
+ }
167
+ else {
168
+ for (const override of overrides) {
169
+ const resolvedMove = resolve(override, move);
170
+ if (resolvedMove != null) {
171
+ nonAmbiguous.push(resolvedMove);
172
+ move = remove(override, move);
173
+ }
174
+ }
175
+ // One last chance to be non-ambiguous
176
+ if (!isAmbiguousMove(move)) {
177
+ nonAmbiguous.push(move);
178
+ }
179
+ else {
180
+ ambiguous.push(move);
181
+ }
182
+ }
183
+ }
184
+ function resolve(override, move) {
185
+ const [pre, post] = move;
186
+ const source = pre.find((loc) => loc.equalTo(override.source));
187
+ const destination = post.find((loc) => loc.equalTo(override.destination));
188
+ return (source && destination) ? [[source], [destination]] : undefined;
189
+ }
190
+ function remove(override, move) {
191
+ const [pre, post] = move;
192
+ return [
193
+ pre.filter(loc => !loc.equalTo(override.source)),
194
+ post.filter(loc => !loc.equalTo(override.destination)),
195
+ ];
196
+ }
197
+ return [nonAmbiguous, ambiguous];
128
198
  }
129
- //# 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 { computeResourceDigests } from './digest';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\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 RefactorManagerOptions {\n  environment: Environment;\n  localStacks: CloudFormationStack[];\n  deployedStacks: CloudFormationStack[];\n  mappings?: ResourceMapping[];\n  filteredStacks?: CloudFormationStack[];\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: RefactorManagerOptions) {\n    this.environment = props.environment;\n    if (props.mappings != null) {\n      this._mappings = props.mappings;\n    } else {\n      const moves = resourceMoves(props.deployedStacks, props.localStacks);\n      this.ambiguousMoves = ambiguousMoves(moves);\n\n      if (!this.isAmbiguous) {\n        this._mappings = resourceMappings(resourceMoves(props.deployedStacks, props.localStacks), props.filteredStacks);\n      }\n    }\n  }\n\n  public get isAmbiguous(): boolean {\n    return this.ambiguousMoves.length > 0;\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    if (this.isAmbiguous) {\n      throw new ToolkitError(\n        'Cannot access mappings when there are ambiguous resource moves. Please resolve the ambiguity first.',\n      );\n    }\n    return this._mappings;\n  }\n}\n\nfunction resourceMoves(before: CloudFormationStack[], after: CloudFormationStack[]): ResourceMove[] {\n  return Object.values(\n    removeUnmovedResources(zip(groupByKey(resourceDigests(before)), groupByKey(resourceDigests(after)))),\n  );\n}\n\nfunction removeUnmovedResources(m: Record<string, ResourceMove>): Record<string, ResourceMove> {\n  const result: Record<string, ResourceMove> = {};\n  for (const [hash, [before, after]] of Object.entries(m)) {\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\nfunction groupByKey<A>(entries: [string, A][]): Record<string, A[]> {\n  const result: Record<string, A[]> = {};\n\n  for (const [hash, location] of entries) {\n    if (hash in result) {\n      result[hash].push(location);\n    } else {\n      result[hash] = [location];\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[]): [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);\n\n  return 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\nfunction ambiguousMoves(movements: ResourceMove[]) {\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 movements\n    .filter(([pre, post]) => pre.length > 0 && post.length > 0)\n    .filter(([pre, post]) => pre.length > 1 || post.length > 1);\n}\n\nfunction resourceMappings(movements: ResourceMove[], stacks?: CloudFormationStack[]): ResourceMapping[] {\n  const stacksPredicate =\n    stacks == null\n      ? () => true\n      : (m: ResourceMapping) => {\n        // Any movement that involves one of the selected stacks (either moving from or to)\n        // is considered a candidate for refactoring.\n        const stackNames = [m.source.stack.stackName, m.destination.stack.stackName];\n        return stacks.some((stack) => stackNames.includes(stack.stackName));\n      };\n\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    .filter(stacksPredicate);\n}\n"]}
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"]}
@@ -1,27 +1,22 @@
1
1
  import type { CloudFormationStack } from './cloudformation';
2
+ export type GraphDirection = 'direct' | 'opposite';
2
3
  /**
3
4
  * Computes the digest for each resource in the template.
4
5
  *
5
6
  * Conceptually, the digest is computed as:
6
7
  *
7
- * d(resource) = hash(type + physicalId) , if physicalId is defined
8
- * = hash(type + properties + dependencies.map(d)) , otherwise
8
+ * digest(resource) = hash(type + properties + dependencies.map(d))
9
9
  *
10
- * where `hash` is a cryptographic hash function. In other words, if a resource has
11
- * a physical ID, we use the physical ID plus its type to uniquely identify
12
- * that resource. In this case, the digest can be computed from these two fields
13
- * alone. A corollary is that such resources can be renamed and have their
14
- * properties updated at the same time, and still be considered equivalent.
15
- *
16
- * Otherwise, the digest is computed from its type, its own properties (that is,
17
- * excluding properties that refer to other resources), and the digests of each of
18
- * its dependencies.
10
+ * where `hash` is a cryptographic hash function. In other words, the digest of a
11
+ * resource is computed from its type, its own properties (that is, excluding
12
+ * properties that refer to other resources), and the digests of each of its
13
+ * dependencies.
19
14
  *
20
15
  * The digest of a resource, defined recursively this way, remains stable even if
21
16
  * one or more of its dependencies gets renamed. Since the resources in a
22
17
  * CloudFormation template form a directed acyclic graph, this function is
23
18
  * well-defined.
24
19
  */
25
- export declare function computeResourceDigests(stacks: CloudFormationStack[]): Record<string, string>;
20
+ export declare function computeResourceDigests(stacks: CloudFormationStack[], direction?: GraphDirection): Record<string, string>;
26
21
  export declare function hashObject(obj: any): string;
27
22
  //# sourceMappingURL=digest.d.ts.map
@@ -3,63 +3,46 @@ Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.computeResourceDigests = computeResourceDigests;
4
4
  exports.hashObject = hashObject;
5
5
  const crypto = require("node:crypto");
6
- const util_1 = require("@aws-cdk/cloudformation-diff/lib/diff/util");
7
6
  const graph_1 = require("./graph");
8
7
  /**
9
8
  * Computes the digest for each resource in the template.
10
9
  *
11
10
  * Conceptually, the digest is computed as:
12
11
  *
13
- * d(resource) = hash(type + physicalId) , if physicalId is defined
14
- * = hash(type + properties + dependencies.map(d)) , otherwise
12
+ * digest(resource) = hash(type + properties + dependencies.map(d))
15
13
  *
16
- * where `hash` is a cryptographic hash function. In other words, if a resource has
17
- * a physical ID, we use the physical ID plus its type to uniquely identify
18
- * that resource. In this case, the digest can be computed from these two fields
19
- * alone. A corollary is that such resources can be renamed and have their
20
- * properties updated at the same time, and still be considered equivalent.
21
- *
22
- * Otherwise, the digest is computed from its type, its own properties (that is,
23
- * excluding properties that refer to other resources), and the digests of each of
24
- * its dependencies.
14
+ * where `hash` is a cryptographic hash function. In other words, the digest of a
15
+ * resource is computed from its type, its own properties (that is, excluding
16
+ * properties that refer to other resources), and the digests of each of its
17
+ * dependencies.
25
18
  *
26
19
  * The digest of a resource, defined recursively this way, remains stable even if
27
20
  * one or more of its dependencies gets renamed. Since the resources in a
28
21
  * CloudFormation template form a directed acyclic graph, this function is
29
22
  * well-defined.
30
23
  */
31
- function computeResourceDigests(stacks) {
24
+ function computeResourceDigests(stacks, direction = 'direct') {
32
25
  const exports = Object.fromEntries(stacks.flatMap((s) => Object.values(s.template.Outputs ?? {})
33
26
  .filter((o) => o.Export != null && typeof o.Export.Name === 'string')
34
27
  .map((o) => [o.Export.Name, { stackName: s.stackName, value: o.Value }])));
35
- const resources = Object.fromEntries(stacks.flatMap((s) => Object.entries(s.template.Resources ?? {}).map(([id, res]) => [`${s.stackName}.${id}`, res])));
36
- const graph = new graph_1.ResourceGraph(stacks);
37
- const nodes = graph.sortedNodes;
38
- // 4. Compute digests in sorted order
28
+ const resources = Object.fromEntries(stacks.flatMap((s) => {
29
+ return Object.entries(s.template.Resources ?? {})
30
+ .filter(([_, res]) => res.Type !== 'AWS::CDK::Metadata')
31
+ .map(([id, res]) => [`${s.stackName}.${id}`, res]);
32
+ }));
33
+ const graph = direction == 'direct'
34
+ ? graph_1.ResourceGraph.fromStacks(stacks)
35
+ : graph_1.ResourceGraph.fromStacks(stacks).opposite();
36
+ return computeDigestsInTopologicalOrder(graph, resources, exports);
37
+ }
38
+ function computeDigestsInTopologicalOrder(graph, resources, exports) {
39
+ const nodes = graph.sortedNodes.filter(n => resources[n] != null);
39
40
  const result = {};
40
41
  for (const id of nodes) {
41
42
  const resource = resources[id];
42
- const resourceProperties = resource.Properties ?? {};
43
- const model = (0, util_1.loadResourceModel)(resource.Type);
44
- const identifier = intersection(Object.keys(resourceProperties), model?.primaryIdentifier ?? []);
45
- let toHash;
46
- if (identifier.length === model?.primaryIdentifier?.length) {
47
- // The resource has a physical ID defined, so we can
48
- // use the ID and the type as the identity of the resource.
49
- toHash =
50
- resource.Type +
51
- identifier
52
- .sort()
53
- .map((attr) => JSON.stringify(resourceProperties[attr]))
54
- .join('');
55
- }
56
- else {
57
- // The resource does not have a physical ID defined, so we need to
58
- // compute the digest based on its properties and dependencies.
59
- const depDigests = Array.from(graph.outNeighbors(id)).map((d) => result[d]);
60
- const propertiesHash = hashObject(stripReferences(stripConstructPath(resource), exports));
61
- toHash = resource.Type + propertiesHash + depDigests.join('');
62
- }
43
+ const depDigests = Array.from(graph.outNeighbors(id)).map((d) => result[d]);
44
+ const propertiesHash = hashObject(stripReferences(stripConstructPath(resource), exports));
45
+ const toHash = resource.Type + propertiesHash + depDigests.join('');
63
46
  result[id] = crypto.createHash('sha256').update(toHash).digest('hex');
64
47
  }
65
48
  return result;
@@ -133,7 +116,4 @@ function stripConstructPath(resource) {
133
116
  delete copy.Metadata['aws:cdk:path'];
134
117
  return copy;
135
118
  }
136
- function intersection(a, b) {
137
- return a.filter((value) => b.includes(value));
138
- }
139
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"digest.js","sourceRoot":"","sources":["digest.ts"],"names":[],"mappings":"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","sourcesContent":["import * as crypto from 'node:crypto';\nimport { loadResourceModel } from '@aws-cdk/cloudformation-diff/lib/diff/util';\nimport type { CloudFormationResource, CloudFormationStack } from './cloudformation';\nimport { ResourceGraph } from './graph';\n\n/**\n * Computes the digest for each resource in the template.\n *\n * Conceptually, the digest is computed as:\n *\n *     d(resource) = hash(type + physicalId)                       , if physicalId is defined\n *                 = hash(type + properties + dependencies.map(d)) , otherwise\n *\n * where `hash` is a cryptographic hash function. In other words, if a resource has\n * a physical ID, we use the physical ID plus its type to uniquely identify\n * that resource. In this case, the digest can be computed from these two fields\n * alone. A corollary is that such resources can be renamed and have their\n * properties updated at the same time, and still be considered equivalent.\n *\n * Otherwise, the digest is computed from its type, its own properties (that is,\n * excluding properties that refer to other resources), and the digests of each of\n * its dependencies.\n *\n * The digest of a resource, defined recursively this way, remains stable even if\n * one or more of its dependencies gets renamed. Since the resources in a\n * CloudFormation template form a directed acyclic graph, this function is\n * well-defined.\n */\nexport function computeResourceDigests(stacks: CloudFormationStack[]): Record<string, string> {\n  const exports: { [p: string]: { stackName: string; value: any } } = Object.fromEntries(\n    stacks.flatMap((s) =>\n      Object.values(s.template.Outputs ?? {})\n        .filter((o) => o.Export != null && typeof o.Export.Name === 'string')\n        .map((o) => [o.Export.Name, { stackName: s.stackName, value: o.Value }] as [string, { stackName: string; value: any }]),\n    ),\n  );\n\n  const resources = Object.fromEntries(\n    stacks.flatMap((s) =>\n      Object.entries(s.template.Resources ?? {}).map(\n        ([id, res]) => [`${s.stackName}.${id}`, res] as [string, CloudFormationResource],\n      ),\n    ),\n  );\n\n  const graph = new ResourceGraph(stacks);\n  const nodes = graph.sortedNodes;\n  // 4. Compute digests in sorted order\n  const result: Record<string, string> = {};\n  for (const id of nodes) {\n    const resource = resources[id];\n    const resourceProperties = resource.Properties ?? {};\n    const model = loadResourceModel(resource.Type);\n    const identifier = intersection(Object.keys(resourceProperties), model?.primaryIdentifier ?? []);\n    let toHash: string;\n\n    if (identifier.length === model?.primaryIdentifier?.length) {\n      // The resource has a physical ID defined, so we can\n      // use the ID and the type as the identity of the resource.\n      toHash =\n        resource.Type +\n        identifier\n          .sort()\n          .map((attr) => JSON.stringify(resourceProperties[attr]))\n          .join('');\n    } else {\n      // The resource does not have a physical ID defined, so we need to\n      // compute the digest based on its properties and dependencies.\n      const depDigests = Array.from(graph.outNeighbors(id)).map((d) => result[d]);\n      const propertiesHash = hashObject(stripReferences(stripConstructPath(resource), exports));\n      toHash = resource.Type + propertiesHash + depDigests.join('');\n    }\n\n    result[id] = crypto.createHash('sha256').update(toHash).digest('hex');\n  }\n\n  return result;\n}\n\nexport function hashObject(obj: any): string {\n  const hash = crypto.createHash('sha256');\n\n  function addToHash(value: any) {\n    if (value == null) {\n      addToHash('null');\n    } else if (typeof value === 'object') {\n      if (Array.isArray(value)) {\n        value.forEach(addToHash);\n      } else {\n        Object.keys(value)\n          .sort()\n          .forEach((key) => {\n            hash.update(key);\n            addToHash(value[key]);\n          });\n      }\n    } else {\n      hash.update(typeof value + value.toString());\n    }\n  }\n\n  addToHash(obj);\n  return hash.digest('hex');\n}\n\n/**\n * Removes sub-properties containing Ref or Fn::GetAtt to avoid hashing\n * references themselves but keeps the property structure.\n */\nfunction stripReferences(value: any, exports: { [p: string]: { stackName: string; value: any } }): any {\n  if (!value || typeof value !== 'object') return value;\n  if (Array.isArray(value)) {\n    return value.map(x => stripReferences(x, exports));\n  }\n  if ('Ref' in value) {\n    return { __cloud_ref__: 'Ref' };\n  }\n  if ('Fn::GetAtt' in value) {\n    return { __cloud_ref__: 'Fn::GetAtt' };\n  }\n  if ('DependsOn' in value) {\n    return { __cloud_ref__: 'DependsOn' };\n  }\n  if ('Fn::ImportValue' in value) {\n    const v = exports[value['Fn::ImportValue']].value;\n    // Treat Fn::ImportValue as if it were a reference with the same stack\n    if ('Ref' in v) {\n      return { __cloud_ref__: 'Ref' };\n    } else if ('Fn::GetAtt' in v) {\n      return { __cloud_ref__: 'Fn::GetAtt' };\n    }\n  }\n  const result: any = {};\n  for (const [k, v] of Object.entries(value)) {\n    result[k] = stripReferences(v, exports);\n  }\n  return result;\n}\n\nfunction stripConstructPath(resource: any): any {\n  if (resource?.Metadata?.['aws:cdk:path'] == null) {\n    return resource;\n  }\n\n  const copy = JSON.parse(JSON.stringify(resource));\n  delete copy.Metadata['aws:cdk:path'];\n  return copy;\n}\n\nfunction intersection<T>(a: T[], b: T[]): T[] {\n  return a.filter((value) => b.includes(value));\n}\n"]}
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"]}
@@ -3,14 +3,19 @@ import type { CloudFormationStack } from './cloudformation';
3
3
  * An immutable directed graph of resources from multiple CloudFormation stacks.
4
4
  */
5
5
  export declare class ResourceGraph {
6
+ static fromStacks(stacks: Omit<CloudFormationStack, 'environment'>[]): ResourceGraph;
6
7
  private readonly edges;
7
8
  private readonly reverseEdges;
8
- constructor(stacks: Omit<CloudFormationStack, 'environment'>[]);
9
+ private constructor();
9
10
  /**
10
11
  * Returns the sorted nodes in topological order.
11
12
  */
12
13
  get sortedNodes(): string[];
13
14
  inNeighbors(node: string): string[];
14
15
  outNeighbors(node: string): string[];
16
+ /**
17
+ * Returns another graph with the same nodes, but with the edges inverted
18
+ */
19
+ opposite(): ResourceGraph;
15
20
  }
16
21
  //# sourceMappingURL=graph.d.ts.map
@@ -6,17 +6,17 @@ const toolkit_error_1 = require("../../toolkit/toolkit-error");
6
6
  * An immutable directed graph of resources from multiple CloudFormation stacks.
7
7
  */
8
8
  class ResourceGraph {
9
- edges = {};
10
- reverseEdges = {};
11
- constructor(stacks) {
9
+ static fromStacks(stacks) {
12
10
  const exports = Object.fromEntries(stacks.flatMap((s) => Object.values(s.template.Outputs ?? {})
13
11
  .filter((o) => o.Export != null && typeof o.Export.Name === 'string')
14
12
  .map((o) => [o.Export.Name, { stackName: s.stackName, value: o.Value }])));
15
13
  const resources = Object.fromEntries(stacks.flatMap((s) => Object.entries(s.template.Resources ?? {}).map(([id, res]) => [`${s.stackName}.${id}`, res])));
16
14
  // 1. Build adjacency lists
15
+ const edges = {};
16
+ const reverseEdges = {};
17
17
  for (const id of Object.keys(resources)) {
18
- this.edges[id] = new Set();
19
- this.reverseEdges[id] = new Set();
18
+ edges[id] = new Set();
19
+ reverseEdges[id] = new Set();
20
20
  }
21
21
  // 2. Detect dependencies by searching for Ref/Fn::GetAtt
22
22
  const findDependencies = (stackName, value) => {
@@ -63,11 +63,18 @@ class ResourceGraph {
63
63
  const deps = findDependencies(stackName, res || {});
64
64
  for (const dep of deps) {
65
65
  if (dep in resources && dep !== id) {
66
- this.edges[id].add(dep);
67
- this.reverseEdges[dep].add(id);
66
+ edges[id].add(dep);
67
+ reverseEdges[dep].add(id);
68
68
  }
69
69
  }
70
70
  }
71
+ return new ResourceGraph(edges, reverseEdges);
72
+ }
73
+ edges = {};
74
+ reverseEdges = {};
75
+ constructor(edges, reverseEdges) {
76
+ this.edges = edges;
77
+ this.reverseEdges = reverseEdges;
71
78
  }
72
79
  /**
73
80
  * Returns the sorted nodes in topological order.
@@ -103,6 +110,12 @@ class ResourceGraph {
103
110
  }
104
111
  return Array.from(this.edges[node] || []);
105
112
  }
113
+ /**
114
+ * Returns another graph with the same nodes, but with the edges inverted
115
+ */
116
+ opposite() {
117
+ return new ResourceGraph(this.reverseEdges, this.edges);
118
+ }
106
119
  }
107
120
  exports.ResourceGraph = ResourceGraph;
108
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"graph.js","sourceRoot":"","sources":["graph.ts"],"names":[],"mappings":"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","sourcesContent":["import type { CloudFormationResource, CloudFormationStack } from './cloudformation';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\n\n/**\n * An immutable directed graph of resources from multiple CloudFormation stacks.\n */\nexport class ResourceGraph {\n  private readonly edges: Record<string, Set<string>> = {};\n  private readonly reverseEdges: Record<string, Set<string>> = {};\n\n  constructor(stacks: Omit<CloudFormationStack, 'environment'>[]) {\n    const exports: { [p: string]: { stackName: string; value: any } } = Object.fromEntries(\n      stacks.flatMap((s) =>\n        Object.values(s.template.Outputs ?? {})\n          .filter((o) => o.Export != null && typeof o.Export.Name === 'string')\n          .map(\n            (o) =>\n              [o.Export.Name, { stackName: s.stackName, value: o.Value }] as [\n                string,\n                {\n                  stackName: string;\n                  value: any;\n                },\n              ],\n          ),\n      ),\n    );\n\n    const resources = Object.fromEntries(\n      stacks.flatMap((s) =>\n        Object.entries(s.template.Resources ?? {}).map(\n          ([id, res]) => [`${s.stackName}.${id}`, res] as [string, CloudFormationResource],\n        ),\n      ),\n    );\n\n    // 1. Build adjacency lists\n    for (const id of Object.keys(resources)) {\n      this.edges[id] = new Set();\n      this.reverseEdges[id] = new Set();\n    }\n\n    // 2. Detect dependencies by searching for Ref/Fn::GetAtt\n    const findDependencies = (stackName: string, value: any): string[] => {\n      if (!value || typeof value !== 'object') return [];\n      if (Array.isArray(value)) {\n        return value.flatMap((res) => findDependencies(stackName, res));\n      }\n      if ('Ref' in value) {\n        return [`${stackName}.${value.Ref}`];\n      }\n      if ('Fn::GetAtt' in value) {\n        const refTarget = Array.isArray(value['Fn::GetAtt'])\n          ? value['Fn::GetAtt'][0]\n          : value['Fn::GetAtt'].split('.')[0];\n        return [`${stackName}.${refTarget}`];\n      }\n      if ('Fn::ImportValue' in value) {\n        const exp = exports[value['Fn::ImportValue']];\n        const v = exp.value;\n        if ('Fn::GetAtt' in v) {\n          const id = Array.isArray(v['Fn::GetAtt']) ? v['Fn::GetAtt'][0] : v['Fn::GetAtt'].split('.')[0];\n          return [`${exp.stackName}.${id}`];\n        }\n        if ('Ref' in v) {\n          return [`${exp.stackName}.${v.Ref}`];\n        }\n        return [`${exp.stackName}.${v}`];\n      }\n      const result: string[] = [];\n      if ('DependsOn' in value) {\n        if (Array.isArray(value.DependsOn)) {\n          result.push(...value.DependsOn.map((r: string) => `${stackName}.${r}`));\n        } else {\n          result.push(`${stackName}.${value.DependsOn}`);\n        }\n      }\n      result.push(...Object.values(value).flatMap((res) => findDependencies(stackName, res)));\n      return result;\n    };\n\n    for (const [id, res] of Object.entries(resources)) {\n      const stackName = id.split('.')[0];\n      const deps = findDependencies(stackName, res || {});\n      for (const dep of deps) {\n        if (dep in resources && dep !== id) {\n          this.edges[id].add(dep);\n          this.reverseEdges[dep].add(id);\n        }\n      }\n    }\n  }\n\n  /**\n   * Returns the sorted nodes in topological order.\n   */\n  get sortedNodes(): string[] {\n    const result: string[] = [];\n    const outDegree = Object.keys(this.edges).reduce((acc, k) => {\n      acc[k] = this.edges[k].size;\n      return acc;\n    }, {} as Record<string, number>);\n\n    const queue = Object.keys(outDegree).filter((k) => outDegree[k] === 0);\n\n    while (queue.length > 0) {\n      const node = queue.shift()!;\n      result.push(node);\n      for (const nxt of this.reverseEdges[node]) {\n        outDegree[nxt]--;\n        if (outDegree[nxt] === 0) {\n          queue.push(nxt);\n        }\n      }\n    }\n    return result;\n  }\n\n  public inNeighbors(node: string): string[] {\n    if (!(node in this.edges)) {\n      throw new ToolkitError(`Node ${node} not found in the graph`);\n    }\n    return Array.from(this.reverseEdges[node] || []);\n  }\n\n  public outNeighbors(node: string): string[] {\n    if (!(node in this.edges)) {\n      throw new ToolkitError(`Node ${node} not found in the graph`);\n    }\n    return Array.from(this.edges[node] || []);\n  }\n}\n"]}
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"]}