@aws-cdk/toolkit-lib 1.2.4 → 1.3.1

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 (50) 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/plugin/plugin.d.ts +4 -0
  25. package/lib/api/plugin/plugin.js +12 -6
  26. package/lib/api/refactoring/cloudformation.d.ts +1 -0
  27. package/lib/api/refactoring/cloudformation.js +6 -4
  28. package/lib/api/refactoring/context.d.ts +4 -5
  29. package/lib/api/refactoring/context.js +122 -52
  30. package/lib/api/refactoring/digest.d.ts +7 -12
  31. package/lib/api/refactoring/digest.js +22 -42
  32. package/lib/api/refactoring/graph.d.ts +6 -1
  33. package/lib/api/refactoring/graph.js +21 -8
  34. package/lib/api/refactoring/index.d.ts +13 -4
  35. package/lib/api/refactoring/index.js +38 -18
  36. package/lib/api/tree.js +1 -1
  37. package/lib/index_bg.wasm +0 -0
  38. package/lib/payloads/refactor.d.ts +1 -1
  39. package/lib/payloads/refactor.js +1 -1
  40. package/lib/payloads/stack-details.d.ts +4 -0
  41. package/lib/payloads/stack-details.js +1 -1
  42. package/lib/toolkit/toolkit.d.ts +6 -2
  43. package/lib/toolkit/toolkit.js +114 -79
  44. package/lib/toolkit/types.d.ts +7 -0
  45. package/lib/toolkit/types.js +1 -1
  46. package/lib/util/arrays.d.ts +1 -0
  47. package/lib/util/arrays.js +16 -1
  48. package/lib/util/sets.d.ts +5 -0
  49. package/lib/util/sets.js +18 -0
  50. package/package.json +11 -11
@@ -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"]}
@@ -5,10 +5,19 @@ import type { CloudFormationStack } from './cloudformation';
5
5
  import { ResourceMapping } from './cloudformation';
6
6
  import type { MappingGroup } from '../../actions';
7
7
  export * from './exclude';
8
+ export * from './context';
9
+ interface StackGroup {
10
+ environment: cxapi.Environment;
11
+ localStacks: CloudFormationStack[];
12
+ deployedStacks: CloudFormationStack[];
13
+ }
8
14
  export declare function usePrescribedMappings(mappingGroups: MappingGroup[], sdkProvider: SdkProvider): Promise<ResourceMapping[]>;
9
15
  export declare function getDeployedStacks(sdkProvider: SdkProvider, environment: cxapi.Environment): Promise<CloudFormationStack[]>;
10
- export declare function formatMappingsHeader(): string;
11
- export declare function formatTypedMappings(environment: cxapi.Environment, mappings: TypedMapping[]): string;
12
- export declare function formatAmbiguitySectionHeader(): string;
13
- export declare function formatAmbiguousMappings(environment: cxapi.Environment, paths: [string[], string[]][]): string;
16
+ export declare function formatEnvironmentSectionHeader(environment: cxapi.Environment): string;
17
+ export declare function formatTypedMappings(mappings: TypedMapping[]): string;
18
+ export declare function formatAmbiguousMappings(paths: [string[], string[]][]): string;
19
+ /**
20
+ * Returns a list of stack groups, each containing the local stacks and the deployed stacks that match the given patterns.
21
+ */
22
+ export declare function groupStacks(sdkProvider: SdkProvider, localStacks: CloudFormationStack[], additionalStackNames: string[]): Promise<StackGroup[]>;
14
23
  //# sourceMappingURL=index.d.ts.map
@@ -16,17 +16,19 @@ var __exportStar = (this && this.__exportStar) || function(m, exports) {
16
16
  Object.defineProperty(exports, "__esModule", { value: true });
17
17
  exports.usePrescribedMappings = usePrescribedMappings;
18
18
  exports.getDeployedStacks = getDeployedStacks;
19
- exports.formatMappingsHeader = formatMappingsHeader;
19
+ exports.formatEnvironmentSectionHeader = formatEnvironmentSectionHeader;
20
20
  exports.formatTypedMappings = formatTypedMappings;
21
- exports.formatAmbiguitySectionHeader = formatAmbiguitySectionHeader;
22
21
  exports.formatAmbiguousMappings = formatAmbiguousMappings;
22
+ exports.groupStacks = groupStacks;
23
23
  const cloudformation_diff_1 = require("@aws-cdk/cloudformation-diff");
24
24
  const util_1 = require("../../util");
25
25
  const plugin_1 = require("../plugin");
26
26
  const streams_1 = require("../streams");
27
27
  const cloudformation_1 = require("./cloudformation");
28
+ const digest_1 = require("./digest");
28
29
  const toolkit_error_1 = require("../../toolkit/toolkit-error");
29
30
  __exportStar(require("./exclude"), exports);
31
+ __exportStar(require("./context"), exports);
30
32
  async function usePrescribedMappings(mappingGroups, sdkProvider) {
31
33
  const stackGroups = [];
32
34
  for (const group of mappingGroups) {
@@ -109,27 +111,45 @@ async function getDeployedStacks(sdkProvider, environment) {
109
111
  // eslint-disable-next-line @cdklabs/promiseall-no-unbounded-parallelism
110
112
  return Promise.all(summaries.map(normalize));
111
113
  }
112
- function formatMappingsHeader() {
113
- return formatToStream(cloudformation_diff_1.formatMappingsHeader);
114
+ function formatEnvironmentSectionHeader(environment) {
115
+ const env = `aws://${environment.account}/${environment.region}`;
116
+ return formatToStream(stream => (0, cloudformation_diff_1.formatEnvironmentSectionHeader)(stream, env));
114
117
  }
115
- function formatTypedMappings(environment, mappings) {
116
- return formatToStream((stream) => {
117
- const env = `aws://${environment.account}/${environment.region}`;
118
- (0, cloudformation_diff_1.formatTypedMappings)(stream, mappings, env);
119
- });
120
- }
121
- function formatAmbiguitySectionHeader() {
122
- return formatToStream(cloudformation_diff_1.formatAmbiguitySectionHeader);
118
+ function formatTypedMappings(mappings) {
119
+ return formatToStream((stream) => (0, cloudformation_diff_1.formatTypedMappings)(stream, mappings));
123
120
  }
124
- function formatAmbiguousMappings(environment, paths) {
125
- return formatToStream((stream) => {
126
- const env = `aws://${environment.account}/${environment.region}`;
127
- (0, cloudformation_diff_1.formatAmbiguousMappings)(stream, paths, env);
128
- });
121
+ function formatAmbiguousMappings(paths) {
122
+ return formatToStream((stream) => (0, cloudformation_diff_1.formatAmbiguousMappings)(stream, paths));
129
123
  }
130
124
  function formatToStream(cb) {
131
125
  const stream = new streams_1.StringWriteStream();
132
126
  cb(stream);
133
127
  return stream.toString();
134
128
  }
135
- //# 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  formatMappingsHeader as fmtMappingsHeader,\n  formatTypedMappings as fmtTypedMappings,\n  formatAmbiguitySectionHeader as fmtAmbiguitySectionHeader,\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 } 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 type { MappingGroup } from '../../actions';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\n\nexport * from './exclude';\n\nexport async function usePrescribedMappings(\n  mappingGroups: MappingGroup[],\n  sdkProvider: SdkProvider,\n): Promise<ResourceMapping[]> {\n  interface StackGroup extends MappingGroup {\n    stacks: CloudFormationStack[];\n  }\n\n  const stackGroups: StackGroup[] = [];\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 formatMappingsHeader(): string {\n  return formatToStream(fmtMappingsHeader);\n}\n\nexport function formatTypedMappings(environment: cxapi.Environment, mappings: TypedMapping[]): string {\n  return formatToStream((stream) => {\n    const env = `aws://${environment.account}/${environment.region}`;\n    fmtTypedMappings(stream, mappings, env);\n  });\n}\n\nexport function formatAmbiguitySectionHeader(): string {\n  return formatToStream(fmtAmbiguitySectionHeader);\n}\n\nexport function formatAmbiguousMappings(environment: cxapi.Environment, paths: [string[], string[]][]): string {\n  return formatToStream((stream) => {\n    const env = `aws://${environment.account}/${environment.region}`;\n    fmtAmbiguousMappings(stream, paths, env);\n  });\n}\n\nfunction formatToStream(cb: (stream: NodeJS.WritableStream) => void): string {\n  const stream = new StringWriteStream();\n  cb(stream);\n  return stream.toString();\n}\n"]}
129
+ /**
130
+ * Returns a list of stack groups, each containing the local stacks and the deployed stacks that match the given patterns.
131
+ */
132
+ async function groupStacks(sdkProvider, localStacks, additionalStackNames) {
133
+ const environments = new Map();
134
+ for (const stack of localStacks) {
135
+ const environment = await sdkProvider.resolveEnvironment(stack.environment);
136
+ const key = (0, digest_1.hashObject)(environment);
137
+ environments.set(key, environment);
138
+ }
139
+ const localByEnvironment = await (0, util_1.indexBy)(localStacks, async (s) => (0, digest_1.hashObject)(await sdkProvider.resolveEnvironment(s.environment)));
140
+ const groups = [];
141
+ for (let key of localByEnvironment.keys()) {
142
+ const environment = environments.get(key);
143
+ const allDeployedStacks = await getDeployedStacks(sdkProvider, environment);
144
+ const local = localByEnvironment.get(key);
145
+ const hasLocalCounterpart = (s) => local.some((l) => l.stackName === s.stackName);
146
+ const wasExplicitlyProvided = (s) => additionalStackNames.includes(s.stackName);
147
+ groups.push({
148
+ environment,
149
+ deployedStacks: allDeployedStacks.filter(s => hasLocalCounterpart(s) || wasExplicitlyProvided(s)),
150
+ localStacks: local,
151
+ });
152
+ }
153
+ return groups;
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"]}
package/lib/api/tree.js CHANGED
@@ -34,4 +34,4 @@ async function loadTreeFromDir(outdir, trace) {
34
34
  return undefined;
35
35
  }
36
36
  }
37
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJlZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInRyZWUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUE4QkEsb0JBTUM7QUFFRCw0QkFTQztBQUVELDBDQU9DO0FBeERELGtDQUFrQztBQUVsQywrQkFBK0I7QUF5Qi9COztHQUVHO0FBQ0gsU0FBZ0IsSUFBSSxDQUFDLElBQW1DLEVBQUUsU0FBNEM7SUFDcEcsT0FBTyxJQUFJLElBQUksSUFBSSxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLGNBQWMsRUFBRSxDQUFDLENBQUM7SUFFN0QsU0FBUyxjQUFjO1FBQ3JCLE9BQU8sTUFBTSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsUUFBUSxJQUFJLEVBQUUsQ0FBQyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsU0FBUyxDQUFDLENBQUMsQ0FBQztJQUNuRixDQUFDO0FBQ0gsQ0FBQztBQUVNLEtBQUssVUFBVSxRQUFRLENBQUMsUUFBdUIsRUFBRSxLQUFxQztJQUMzRixJQUFJLENBQUM7UUFDSCxNQUFNLE1BQU0sR0FBRyxRQUFRLENBQUMsU0FBUyxDQUFDO1FBQ2xDLE1BQU0sUUFBUSxHQUFHLFFBQVEsQ0FBQyxJQUFJLEVBQUUsRUFBRSxJQUFJLENBQUM7UUFDdkMsT0FBTyxRQUFRLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsUUFBUSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFFLEVBQXdCLENBQUM7SUFDbEcsQ0FBQztJQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7UUFDWCxNQUFNLEtBQUssQ0FBQyxpQ0FBaUMsQ0FBQywrQkFBK0IsQ0FBQyxDQUFDO1FBQy9FLE9BQU8sU0FBUyxDQUFDO0lBQ25CLENBQUM7QUFDSCxDQUFDO0FBRU0sS0FBSyxVQUFVLGVBQWUsQ0FBQyxNQUFjLEVBQUUsS0FBcUM7SUFDekYsSUFBSSxDQUFDO1FBQ0gsT0FBTyxFQUFFLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLFdBQVcsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDO0lBQzlELENBQUM7SUFBQyxPQUFPLENBQUMsRUFBRSxDQUFDO1FBQ1gsTUFBTSxLQUFLLENBQUMsaUNBQWlDLENBQUMsK0JBQStCLENBQUMsQ0FBQztRQUMvRSxPQUFPLFNBQVMsQ0FBQztJQUNuQixDQUFDO0FBQ0gsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIHBhdGggZnJvbSAnbm9kZTpwYXRoJztcbmltcG9ydCB0eXBlIHsgQ2xvdWRBc3NlbWJseSB9IGZyb20gJ0Bhd3MtY2RrL2N4LWFwaSc7XG5pbXBvcnQgKiBhcyBmcyBmcm9tICdmcy1leHRyYSc7XG5cbi8qKlxuICogU291cmNlIGluZm9ybWF0aW9uIG9uIGEgY29uc3RydWN0IChjbGFzcyBmcW4gYW5kIHZlcnNpb24pXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ29uc3RydWN0SW5mbyB7XG4gIHJlYWRvbmx5IGZxbjogc3RyaW5nO1xuICByZWFkb25seSB2ZXJzaW9uOiBzdHJpbmc7XG59XG5cbi8qKlxuICogQSBub2RlIGluIHRoZSBjb25zdHJ1Y3QgdHJlZS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBDb25zdHJ1Y3RUcmVlTm9kZSB7XG4gIHJlYWRvbmx5IGlkOiBzdHJpbmc7XG4gIHJlYWRvbmx5IHBhdGg6IHN0cmluZztcbiAgcmVhZG9ubHkgY2hpbGRyZW4/OiB7IFtrZXk6IHN0cmluZ106IENvbnN0cnVjdFRyZWVOb2RlIH07XG4gIHJlYWRvbmx5IGF0dHJpYnV0ZXM/OiB7IFtrZXk6IHN0cmluZ106IGFueSB9O1xuXG4gIC8qKlxuICAgKiBJbmZvcm1hdGlvbiBvbiB0aGUgY29uc3RydWN0IGNsYXNzIHRoYXQgbGVkIHRvIHRoaXMgbm9kZSwgaWYgYXZhaWxhYmxlXG4gICAqL1xuICByZWFkb25seSBjb25zdHJ1Y3RJbmZvPzogQ29uc3RydWN0SW5mbztcbn1cblxuLyoqXG4gKiBXaGV0aGVyIHRoZSBwcm92aWRlZCBwcmVkaWNhdGUgaXMgdHJ1ZSBmb3IgYXQgbGVhc3Qgb25lIGVsZW1lbnQgaW4gdGhlIGNvbnN0cnVjdCAoc3ViLSl0cmVlLlxuICovXG5leHBvcnQgZnVuY3Rpb24gc29tZShub2RlOiBDb25zdHJ1Y3RUcmVlTm9kZSB8IHVuZGVmaW5lZCwgcHJlZGljYXRlOiAobjogQ29uc3RydWN0VHJlZU5vZGUpID0+IGJvb2xlYW4pOiBib29sZWFuIHtcbiAgcmV0dXJuIG5vZGUgIT0gbnVsbCAmJiAocHJlZGljYXRlKG5vZGUpIHx8IGZpbmRJbkNoaWxkcmVuKCkpO1xuXG4gIGZ1bmN0aW9uIGZpbmRJbkNoaWxkcmVuKCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiBPYmplY3QudmFsdWVzKG5vZGU/LmNoaWxkcmVuID8/IHt9KS5zb21lKGNoaWxkID0+IHNvbWUoY2hpbGQsIHByZWRpY2F0ZSkpO1xuICB9XG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBsb2FkVHJlZShhc3NlbWJseTogQ2xvdWRBc3NlbWJseSwgdHJhY2U6IChtc2c6IHN0cmluZykgPT4gUHJvbWlzZTx2b2lkPik6IFByb21pc2U8Q29uc3RydWN0VHJlZU5vZGUgfCB1bmRlZmluZWQgPiB7XG4gIHRyeSB7XG4gICAgY29uc3Qgb3V0ZGlyID0gYXNzZW1ibHkuZGlyZWN0b3J5O1xuICAgIGNvbnN0IGZpbGVOYW1lID0gYXNzZW1ibHkudHJlZSgpPy5maWxlO1xuICAgIHJldHVybiBmaWxlTmFtZSA/IGZzLnJlYWRKU09OU3luYyhwYXRoLmpvaW4ob3V0ZGlyLCBmaWxlTmFtZSkpLnRyZWUgOiAoe30gYXMgQ29uc3RydWN0VHJlZU5vZGUpO1xuICB9IGNhdGNoIChlKSB7XG4gICAgYXdhaXQgdHJhY2UoYEZhaWxlZCB0byBnZXQgdHJlZS5qc29uIGZpbGU6ICR7ZX0uIFByb2NlZWRpbmcgd2l0aCBlbXB0eSB0cmVlLmApO1xuICAgIHJldHVybiB1bmRlZmluZWQ7XG4gIH1cbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGxvYWRUcmVlRnJvbURpcihvdXRkaXI6IHN0cmluZywgdHJhY2U6IChtc2c6IHN0cmluZykgPT4gUHJvbWlzZTx2b2lkPik6IFByb21pc2U8Q29uc3RydWN0VHJlZU5vZGUgfCB1bmRlZmluZWQ+IHtcbiAgdHJ5IHtcbiAgICByZXR1cm4gZnMucmVhZEpTT05TeW5jKHBhdGguam9pbihvdXRkaXIsICd0cmVlLmpzb24nKSkudHJlZTtcbiAgfSBjYXRjaCAoZSkge1xuICAgIGF3YWl0IHRyYWNlKGBGYWlsZWQgdG8gZ2V0IHRyZWUuanNvbiBmaWxlOiAke2V9LiBQcm9jZWVkaW5nIHdpdGggZW1wdHkgdHJlZS5gKTtcbiAgICByZXR1cm4gdW5kZWZpbmVkO1xuICB9XG59XG4iXX0=
37
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHJlZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInRyZWUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUE4QkEsb0JBTUM7QUFFRCw0QkFTQztBQUVELDBDQU9DO0FBeERELGtDQUFrQztBQUVsQywrQkFBK0I7QUF5Qi9COztHQUVHO0FBQ0gsU0FBZ0IsSUFBSSxDQUFDLElBQW1DLEVBQUUsU0FBNEM7SUFDcEcsT0FBTyxJQUFJLElBQUksSUFBSSxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxJQUFJLGNBQWMsRUFBRSxDQUFDLENBQUM7SUFFN0QsU0FBUyxjQUFjO1FBQ3JCLE9BQU8sTUFBTSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEVBQUUsUUFBUSxJQUFJLEVBQUUsQ0FBQyxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsRUFBRSxDQUFDLElBQUksQ0FBQyxLQUFLLEVBQUUsU0FBUyxDQUFDLENBQUMsQ0FBQztJQUNuRixDQUFDO0FBQ0gsQ0FBQztBQUVNLEtBQUssVUFBVSxRQUFRLENBQUMsUUFBdUIsRUFBRSxLQUFxQztJQUMzRixJQUFJLENBQUM7UUFDSCxNQUFNLE1BQU0sR0FBRyxRQUFRLENBQUMsU0FBUyxDQUFDO1FBQ2xDLE1BQU0sUUFBUSxHQUFHLFFBQVEsQ0FBQyxJQUFJLEVBQUUsRUFBRSxJQUFJLENBQUM7UUFDdkMsT0FBTyxRQUFRLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsUUFBUSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFFLEVBQXdCLENBQUM7SUFDbEcsQ0FBQztJQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7UUFDWCxNQUFNLEtBQUssQ0FBQyxpQ0FBaUMsQ0FBQywrQkFBK0IsQ0FBQyxDQUFDO1FBQy9FLE9BQU8sU0FBUyxDQUFDO0lBQ25CLENBQUM7QUFDSCxDQUFDO0FBRU0sS0FBSyxVQUFVLGVBQWUsQ0FBQyxNQUFjLEVBQUUsS0FBcUM7SUFDekYsSUFBSSxDQUFDO1FBQ0gsT0FBTyxFQUFFLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLFdBQVcsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDO0lBQzlELENBQUM7SUFBQyxPQUFPLENBQUMsRUFBRSxDQUFDO1FBQ1gsTUFBTSxLQUFLLENBQUMsaUNBQWlDLENBQUMsK0JBQStCLENBQUMsQ0FBQztRQUMvRSxPQUFPLFNBQVMsQ0FBQztJQUNuQixDQUFDO0FBQ0gsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCAqIGFzIHBhdGggZnJvbSAnbm9kZTpwYXRoJztcbmltcG9ydCB0eXBlIHsgQ2xvdWRBc3NlbWJseSB9IGZyb20gJ0Bhd3MtY2RrL2N4LWFwaSc7XG5pbXBvcnQgKiBhcyBmcyBmcm9tICdmcy1leHRyYSc7XG5cbi8qKlxuICogU291cmNlIGluZm9ybWF0aW9uIG9uIGEgY29uc3RydWN0IChjbGFzcyBmcW4gYW5kIHZlcnNpb24pXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgQ29uc3RydWN0SW5mbyB7XG4gIHJlYWRvbmx5IGZxbjogc3RyaW5nO1xuICByZWFkb25seSB2ZXJzaW9uOiBzdHJpbmc7XG59XG5cbi8qKlxuICogQSBub2RlIGluIHRoZSBjb25zdHJ1Y3QgdHJlZS5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBDb25zdHJ1Y3RUcmVlTm9kZSB7XG4gIHJlYWRvbmx5IGlkOiBzdHJpbmc7XG4gIHJlYWRvbmx5IHBhdGg6IHN0cmluZztcbiAgcmVhZG9ubHkgY2hpbGRyZW4/OiB7IFtrZXk6IHN0cmluZ106IENvbnN0cnVjdFRyZWVOb2RlIH07XG4gIHJlYWRvbmx5IGF0dHJpYnV0ZXM/OiB7IFtrZXk6IHN0cmluZ106IGFueSB9O1xuXG4gIC8qKlxuICAgKiBJbmZvcm1hdGlvbiBvbiB0aGUgY29uc3RydWN0IGNsYXNzIHRoYXQgbGVkIHRvIHRoaXMgbm9kZSwgaWYgYXZhaWxhYmxlXG4gICAqL1xuICByZWFkb25seSBjb25zdHJ1Y3RJbmZvPzogQ29uc3RydWN0SW5mbztcbn1cblxuLyoqXG4gKiBXaGV0aGVyIHRoZSBwcm92aWRlZCBwcmVkaWNhdGUgaXMgdHJ1ZSBmb3IgYXQgbGVhc3Qgb25lIGVsZW1lbnQgaW4gdGhlIGNvbnN0cnVjdCAoc3ViLSl0cmVlLlxuICovXG5leHBvcnQgZnVuY3Rpb24gc29tZShub2RlOiBDb25zdHJ1Y3RUcmVlTm9kZSB8IHVuZGVmaW5lZCwgcHJlZGljYXRlOiAobjogQ29uc3RydWN0VHJlZU5vZGUpID0+IGJvb2xlYW4pOiBib29sZWFuIHtcbiAgcmV0dXJuIG5vZGUgIT0gbnVsbCAmJiAocHJlZGljYXRlKG5vZGUpIHx8IGZpbmRJbkNoaWxkcmVuKCkpO1xuXG4gIGZ1bmN0aW9uIGZpbmRJbkNoaWxkcmVuKCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiBPYmplY3QudmFsdWVzKG5vZGU/LmNoaWxkcmVuID8/IHt9KS5zb21lKGNoaWxkID0+IHNvbWUoY2hpbGQsIHByZWRpY2F0ZSkpO1xuICB9XG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBsb2FkVHJlZShhc3NlbWJseTogQ2xvdWRBc3NlbWJseSwgdHJhY2U6IChtc2c6IHN0cmluZykgPT4gUHJvbWlzZTx2b2lkPik6IFByb21pc2U8Q29uc3RydWN0VHJlZU5vZGUgfCB1bmRlZmluZWQ+IHtcbiAgdHJ5IHtcbiAgICBjb25zdCBvdXRkaXIgPSBhc3NlbWJseS5kaXJlY3Rvcnk7XG4gICAgY29uc3QgZmlsZU5hbWUgPSBhc3NlbWJseS50cmVlKCk/LmZpbGU7XG4gICAgcmV0dXJuIGZpbGVOYW1lID8gZnMucmVhZEpTT05TeW5jKHBhdGguam9pbihvdXRkaXIsIGZpbGVOYW1lKSkudHJlZSA6ICh7fSBhcyBDb25zdHJ1Y3RUcmVlTm9kZSk7XG4gIH0gY2F0Y2ggKGUpIHtcbiAgICBhd2FpdCB0cmFjZShgRmFpbGVkIHRvIGdldCB0cmVlLmpzb24gZmlsZTogJHtlfS4gUHJvY2VlZGluZyB3aXRoIGVtcHR5IHRyZWUuYCk7XG4gICAgcmV0dXJuIHVuZGVmaW5lZDtcbiAgfVxufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gbG9hZFRyZWVGcm9tRGlyKG91dGRpcjogc3RyaW5nLCB0cmFjZTogKG1zZzogc3RyaW5nKSA9PiBQcm9taXNlPHZvaWQ+KTogUHJvbWlzZTxDb25zdHJ1Y3RUcmVlTm9kZSB8IHVuZGVmaW5lZD4ge1xuICB0cnkge1xuICAgIHJldHVybiBmcy5yZWFkSlNPTlN5bmMocGF0aC5qb2luKG91dGRpciwgJ3RyZWUuanNvbicpKS50cmVlO1xuICB9IGNhdGNoIChlKSB7XG4gICAgYXdhaXQgdHJhY2UoYEZhaWxlZCB0byBnZXQgdHJlZS5qc29uIGZpbGU6ICR7ZX0uIFByb2NlZWRpbmcgd2l0aCBlbXB0eSB0cmVlLmApO1xuICAgIHJldHVybiB1bmRlZmluZWQ7XG4gIH1cbn1cbiJdfQ==
package/lib/index_bg.wasm CHANGED
Binary file
@@ -10,6 +10,6 @@ export interface RefactorResult {
10
10
  /**
11
11
  * Ambiguous path correspondences, if any
12
12
  */
13
- readonly ambiguousPaths?: [string[], string[]][];
13
+ ambiguousPaths?: [string[], string[]][];
14
14
  }
15
15
  //# sourceMappingURL=refactor.d.ts.map
@@ -1,3 +1,3 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVmYWN0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJyZWZhY3Rvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBUeXBlZE1hcHBpbmcgfSBmcm9tICdAYXdzLWNkay9jbG91ZGZvcm1hdGlvbi1kaWZmJztcblxuLyoqXG4gKiBPdXRwdXQgb2YgdGhlIHJlZmFjdG9yIGNvbW1hbmRcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBSZWZhY3RvclJlc3VsdCB7XG4gIC8qKlxuICAgKiBNYXBwaW5ncyBhbG9uZyB3aXRoIHRoZSByZXNvdXJjZSB0eXBlXG4gICAqL1xuICByZWFkb25seSB0eXBlZE1hcHBpbmdzPzogVHlwZWRNYXBwaW5nW107XG5cbiAgLyoqXG4gICAqIEFtYmlndW91cyBwYXRoIGNvcnJlc3BvbmRlbmNlcywgaWYgYW55XG4gICAqL1xuICByZWFkb25seSBhbWJpZ3VvdXNQYXRocz86IFtzdHJpbmdbXSwgc3RyaW5nW11dW107XG59XG4iXX0=
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVmYWN0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJyZWZhY3Rvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHR5cGUgeyBUeXBlZE1hcHBpbmcgfSBmcm9tICdAYXdzLWNkay9jbG91ZGZvcm1hdGlvbi1kaWZmJztcblxuLyoqXG4gKiBPdXRwdXQgb2YgdGhlIHJlZmFjdG9yIGNvbW1hbmRcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBSZWZhY3RvclJlc3VsdCB7XG4gIC8qKlxuICAgKiBNYXBwaW5ncyBhbG9uZyB3aXRoIHRoZSByZXNvdXJjZSB0eXBlXG4gICAqL1xuICByZWFkb25seSB0eXBlZE1hcHBpbmdzPzogVHlwZWRNYXBwaW5nW107XG5cbiAgLyoqXG4gICAqIEFtYmlndW91cyBwYXRoIGNvcnJlc3BvbmRlbmNlcywgaWYgYW55XG4gICAqL1xuICBhbWJpZ3VvdXNQYXRocz86IFtzdHJpbmdbXSwgc3RyaW5nW11dW107XG59XG4iXX0=
@@ -1,3 +1,4 @@
1
+ import type * as cxschema from '@aws-cdk/cloud-assembly-schema';
1
2
  import type * as cxapi from '@aws-cdk/cx-api';
2
3
  /**
3
4
  * The dependencies of a stack.
@@ -13,6 +14,9 @@ export interface StackDetails {
13
14
  id: string;
14
15
  name: string;
15
16
  environment: cxapi.Environment;
17
+ metadata?: {
18
+ [path: string]: cxschema.MetadataEntry[];
19
+ };
16
20
  dependencies: StackDependency[];
17
21
  }
18
22
  //# sourceMappingURL=stack-details.d.ts.map
@@ -1,3 +1,3 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhY2stZGV0YWlscy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInN0YWNrLWRldGFpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlICogYXMgY3hhcGkgZnJvbSAnQGF3cy1jZGsvY3gtYXBpJztcblxuLyoqXG4gKiBUaGUgZGVwZW5kZW5jaWVzIG9mIGEgc3RhY2suXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU3RhY2tEZXBlbmRlbmN5IHtcbiAgaWQ6IHN0cmluZztcbiAgZGVwZW5kZW5jaWVzOiBTdGFja0RlcGVuZGVuY3lbXTtcbn1cblxuLyoqXG4gKiBEZXRhaWxzIG9mIGEgc3RhY2suXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU3RhY2tEZXRhaWxzIHtcbiAgaWQ6IHN0cmluZztcbiAgbmFtZTogc3RyaW5nO1xuICBlbnZpcm9ubWVudDogY3hhcGkuRW52aXJvbm1lbnQ7XG4gIGRlcGVuZGVuY2llczogU3RhY2tEZXBlbmRlbmN5W107XG59XG5cbiJdfQ==
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhY2stZGV0YWlscy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInN0YWNrLWRldGFpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlICogYXMgY3hzY2hlbWEgZnJvbSAnQGF3cy1jZGsvY2xvdWQtYXNzZW1ibHktc2NoZW1hJztcbmltcG9ydCB0eXBlICogYXMgY3hhcGkgZnJvbSAnQGF3cy1jZGsvY3gtYXBpJztcblxuLyoqXG4gKiBUaGUgZGVwZW5kZW5jaWVzIG9mIGEgc3RhY2suXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU3RhY2tEZXBlbmRlbmN5IHtcbiAgaWQ6IHN0cmluZztcbiAgZGVwZW5kZW5jaWVzOiBTdGFja0RlcGVuZGVuY3lbXTtcbn1cblxuLyoqXG4gKiBEZXRhaWxzIG9mIGEgc3RhY2suXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU3RhY2tEZXRhaWxzIHtcbiAgaWQ6IHN0cmluZztcbiAgbmFtZTogc3RyaW5nO1xuICBlbnZpcm9ubWVudDogY3hhcGkuRW52aXJvbm1lbnQ7XG4gIG1ldGFkYXRhPzogeyBbcGF0aDogc3RyaW5nXTogY3hzY2hlbWEuTWV0YWRhdGFFbnRyeVtdIH07XG4gIGRlcGVuZGVuY2llczogU3RhY2tEZXBlbmRlbmN5W107XG59XG5cbiJdfQ==
@@ -1,6 +1,6 @@
1
1
  import '../private/dispose-polyfill';
2
2
  import type { TemplateDiff } from '@aws-cdk/cloudformation-diff';
3
- import type { DeployResult, DestroyResult, RollbackResult } from './types';
3
+ import type { FeatureFlag, DeployResult, DestroyResult, RollbackResult } from './types';
4
4
  import type { BootstrapEnvironments, BootstrapOptions, BootstrapResult } from '../actions/bootstrap';
5
5
  import { type DeployOptions } from '../actions/deploy';
6
6
  import { type DestroyOptions } from '../actions/destroy';
@@ -75,7 +75,7 @@ export interface ToolkitOptions {
75
75
  * Names of toolkit features that are still under development, and may change in
76
76
  * the future.
77
77
  */
78
- export type UnstableFeature = 'refactor';
78
+ export type UnstableFeature = 'refactor' | 'flags';
79
79
  /**
80
80
  * The AWS CDK Programmatic Toolkit
81
81
  */
@@ -182,6 +182,10 @@ export declare class Toolkit extends CloudAssemblySourceBuilder {
182
182
  */
183
183
  private deploymentsForAction;
184
184
  private invokeDeployFromWatch;
185
+ /**
186
+ * Retrieve feature flag information from the cloud assembly
187
+ */
188
+ flags(cx: ICloudAssemblySource): Promise<FeatureFlag[]>;
185
189
  private requireUnstableFeature;
186
190
  }
187
191
  //# sourceMappingURL=toolkit.d.ts.map