@aws-cdk/toolkit-lib 0.4.0 → 1.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +65 -51
- package/build-info.json +2 -2
- package/db.json.gz +0 -0
- package/lib/actions/diff/private/helpers.d.ts +2 -2
- package/lib/actions/diff/private/helpers.js +3 -3
- package/lib/actions/drift/index.d.ts +4 -2
- package/lib/actions/drift/index.js +1 -1
- package/lib/actions/refactor/index.d.ts +25 -21
- package/lib/actions/refactor/index.js +54 -1
- package/lib/actions/watch/index.d.ts +26 -3
- package/lib/actions/watch/index.js +1 -1
- package/lib/actions/watch/private/helpers.d.ts +6 -4
- package/lib/actions/watch/private/helpers.js +37 -6
- package/lib/api/aws-auth/account-cache.d.ts +1 -1
- package/lib/api/aws-auth/account-cache.js +2 -2
- package/lib/api/aws-auth/awscli-compatible.d.ts +1 -1
- package/lib/api/aws-auth/awscli-compatible.js +2 -2
- package/lib/api/cloud-assembly/environment.d.ts +1 -1
- package/lib/api/cloud-assembly/environment.js +2 -2
- package/lib/api/cloud-assembly/private/prepare-source.js +2 -2
- package/lib/api/cloudformation/template-body-parameter.d.ts +2 -2
- package/lib/api/cloudformation/template-body-parameter.js +3 -3
- package/lib/api/deployments/cfn-api.d.ts +8 -8
- package/lib/api/deployments/cfn-api.js +15 -15
- package/lib/api/drift/drift-formatter.d.ts +4 -4
- package/lib/api/drift/drift-formatter.js +5 -5
- package/lib/api/hotswap/hotswap-deployments.js +2 -2
- package/lib/api/io/private/level-priority.d.ts +2 -2
- package/lib/api/io/private/level-priority.js +3 -3
- package/lib/api/io/toolkit-action.d.ts +1 -1
- package/lib/api/io/toolkit-action.js +1 -1
- package/lib/api/logs-monitor/logs-monitor.d.ts +1 -0
- package/lib/api/logs-monitor/logs-monitor.js +5 -3
- package/lib/api/notices/notices.d.ts +1 -1
- package/lib/api/notices/notices.js +2 -2
- package/lib/api/refactoring/context.d.ts +22 -0
- package/lib/api/refactoring/context.js +129 -0
- package/lib/api/refactoring/exclude.d.ts +11 -6
- package/lib/api/refactoring/exclude.js +14 -10
- package/lib/api/refactoring/execution.d.ts +1 -1
- package/lib/api/refactoring/execution.js +50 -25
- package/lib/api/refactoring/index.d.ts +7 -28
- package/lib/api/refactoring/index.js +23 -146
- package/lib/api/resource-import/importer.d.ts +5 -5
- package/lib/api/resource-import/importer.js +6 -6
- package/lib/api/resource-metadata/resource-metadata.d.ts +2 -2
- package/lib/api/resource-metadata/resource-metadata.js +3 -3
- package/lib/context-providers/cc-api-provider.js +23 -14
- package/lib/context-providers/ssm-parameters.d.ts +4 -4
- package/lib/context-providers/ssm-parameters.js +5 -5
- package/lib/index_bg.wasm +0 -0
- package/lib/toolkit/private/index.d.ts +2 -2
- package/lib/toolkit/private/index.js +3 -3
- package/lib/toolkit/toolkit-error.d.ts +16 -2
- package/lib/toolkit/toolkit-error.js +33 -9
- package/lib/toolkit/toolkit.d.ts +15 -22
- package/lib/toolkit/toolkit.js +94 -68
- package/lib/util/bool.d.ts +1 -1
- package/lib/util/bool.js +2 -2
- package/lib/util/bytes.d.ts +2 -2
- package/lib/util/bytes.js +3 -3
- package/lib/util/cloudformation.d.ts +2 -2
- package/lib/util/cloudformation.js +3 -3
- package/lib/util/format-error.d.ts +1 -1
- package/lib/util/format-error.js +6 -2
- package/lib/util/net.d.ts +1 -1
- package/lib/util/net.js +2 -2
- package/lib/util/objects.d.ts +3 -3
- package/lib/util/objects.js +4 -4
- package/lib/util/yaml-cfn.d.ts +2 -2
- package/lib/util/yaml-cfn.js +3 -3
- package/package.json +16 -16
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.RefactoringContext = void 0;
|
|
4
|
+
const cloudformation_1 = require("./cloudformation");
|
|
5
|
+
const digest_1 = require("./digest");
|
|
6
|
+
const toolkit_error_1 = require("../../toolkit/toolkit-error");
|
|
7
|
+
/**
|
|
8
|
+
* Encapsulates the information for refactoring resources in a single environment.
|
|
9
|
+
*/
|
|
10
|
+
class RefactoringContext {
|
|
11
|
+
_mappings = [];
|
|
12
|
+
ambiguousMoves = [];
|
|
13
|
+
environment;
|
|
14
|
+
constructor(props) {
|
|
15
|
+
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;
|
|
29
|
+
}
|
|
30
|
+
get ambiguousPaths() {
|
|
31
|
+
return this.ambiguousMoves.map(([a, b]) => [convert(a), convert(b)]);
|
|
32
|
+
function convert(locations) {
|
|
33
|
+
return locations.map((l) => l.toPath());
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
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
|
+
return this._mappings;
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
exports.RefactoringContext = RefactoringContext;
|
|
44
|
+
function resourceMoves(before, after) {
|
|
45
|
+
return Object.values(removeUnmovedResources(zip(groupByKey(resourceDigests(before)), groupByKey(resourceDigests(after)))));
|
|
46
|
+
}
|
|
47
|
+
function removeUnmovedResources(m) {
|
|
48
|
+
const result = {};
|
|
49
|
+
for (const [hash, [before, after]] of Object.entries(m)) {
|
|
50
|
+
const common = before.filter((b) => after.some((a) => a.equalTo(b)));
|
|
51
|
+
result[hash] = [
|
|
52
|
+
before.filter((b) => !common.some((c) => b.equalTo(c))),
|
|
53
|
+
after.filter((a) => !common.some((c) => a.equalTo(c))),
|
|
54
|
+
];
|
|
55
|
+
}
|
|
56
|
+
return result;
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* For each hash, identifying a single resource, zip the two lists of locations,
|
|
60
|
+
* producing a resource move
|
|
61
|
+
*/
|
|
62
|
+
function zip(m1, m2) {
|
|
63
|
+
const result = {};
|
|
64
|
+
for (const [hash, locations] of Object.entries(m1)) {
|
|
65
|
+
if (hash in m2) {
|
|
66
|
+
result[hash] = [locations, m2[hash]];
|
|
67
|
+
}
|
|
68
|
+
else {
|
|
69
|
+
result[hash] = [locations, []];
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
for (const [hash, locations] of Object.entries(m2)) {
|
|
73
|
+
if (!(hash in m1)) {
|
|
74
|
+
result[hash] = [[], locations];
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
return result;
|
|
78
|
+
}
|
|
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
|
+
/**
|
|
92
|
+
* Computes a list of pairs [digest, location] for each resource in the stack.
|
|
93
|
+
*/
|
|
94
|
+
function resourceDigests(stacks) {
|
|
95
|
+
// index stacks by name
|
|
96
|
+
const stacksByName = new Map();
|
|
97
|
+
for (const stack of stacks) {
|
|
98
|
+
stacksByName.set(stack.stackName, stack);
|
|
99
|
+
}
|
|
100
|
+
const digests = (0, digest_1.computeResourceDigests)(stacks);
|
|
101
|
+
return Object.entries(digests).map(([loc, digest]) => {
|
|
102
|
+
const [stackName, logicalId] = loc.split('.');
|
|
103
|
+
const location = new cloudformation_1.ResourceLocation(stacksByName.get(stackName), logicalId);
|
|
104
|
+
return [digest, location];
|
|
105
|
+
});
|
|
106
|
+
}
|
|
107
|
+
function ambiguousMoves(movements) {
|
|
108
|
+
// A move is considered ambiguous if two conditions are met:
|
|
109
|
+
// 1. Both sides have at least one element (otherwise, it's just addition or deletion)
|
|
110
|
+
// 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);
|
|
114
|
+
}
|
|
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
|
+
};
|
|
124
|
+
return movements
|
|
125
|
+
.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);
|
|
128
|
+
}
|
|
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"]}
|
|
@@ -2,28 +2,33 @@ import type { AssemblyManifest } from '@aws-cdk/cloud-assembly-schema';
|
|
|
2
2
|
import type { ResourceLocation } from './cloudformation';
|
|
3
3
|
export interface ExcludeList {
|
|
4
4
|
isExcluded(location: ResourceLocation): boolean;
|
|
5
|
+
union(other: ExcludeList): ExcludeList;
|
|
5
6
|
}
|
|
6
|
-
|
|
7
|
+
declare abstract class AbstractExcludeList implements ExcludeList {
|
|
8
|
+
abstract isExcluded(location: ResourceLocation): boolean;
|
|
9
|
+
union(other: ExcludeList): ExcludeList;
|
|
10
|
+
}
|
|
11
|
+
export declare class ManifestExcludeList extends AbstractExcludeList {
|
|
7
12
|
private readonly excludedLocations;
|
|
8
13
|
constructor(manifest: AssemblyManifest);
|
|
9
14
|
private getExcludedLocations;
|
|
10
15
|
isExcluded(location: ResourceLocation): boolean;
|
|
11
16
|
}
|
|
12
|
-
export declare class InMemoryExcludeList
|
|
17
|
+
export declare class InMemoryExcludeList extends AbstractExcludeList {
|
|
13
18
|
private readonly excludedLocations;
|
|
14
19
|
private readonly excludedPaths;
|
|
15
20
|
constructor(items: string[]);
|
|
16
21
|
isExcluded(location: ResourceLocation): boolean;
|
|
17
22
|
}
|
|
18
|
-
export declare class UnionExcludeList
|
|
23
|
+
export declare class UnionExcludeList extends AbstractExcludeList {
|
|
19
24
|
private readonly excludeLists;
|
|
20
25
|
constructor(excludeLists: ExcludeList[]);
|
|
21
26
|
isExcluded(location: ResourceLocation): boolean;
|
|
22
27
|
}
|
|
23
|
-
export declare class NeverExclude
|
|
28
|
+
export declare class NeverExclude extends AbstractExcludeList {
|
|
24
29
|
isExcluded(_location: ResourceLocation): boolean;
|
|
25
30
|
}
|
|
26
|
-
export declare class AlwaysExclude
|
|
31
|
+
export declare class AlwaysExclude extends AbstractExcludeList {
|
|
27
32
|
isExcluded(_location: ResourceLocation): boolean;
|
|
28
33
|
}
|
|
29
|
-
export
|
|
34
|
+
export {};
|
|
@@ -1,11 +1,16 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
3
|
exports.AlwaysExclude = exports.NeverExclude = exports.UnionExcludeList = exports.InMemoryExcludeList = exports.ManifestExcludeList = void 0;
|
|
4
|
-
exports.fromManifestAndExclusionList = fromManifestAndExclusionList;
|
|
5
4
|
const cloud_assembly_schema_1 = require("@aws-cdk/cloud-assembly-schema");
|
|
6
|
-
class
|
|
5
|
+
class AbstractExcludeList {
|
|
6
|
+
union(other) {
|
|
7
|
+
return new UnionExcludeList([this, other]);
|
|
8
|
+
}
|
|
9
|
+
}
|
|
10
|
+
class ManifestExcludeList extends AbstractExcludeList {
|
|
7
11
|
excludedLocations;
|
|
8
12
|
constructor(manifest) {
|
|
13
|
+
super();
|
|
9
14
|
this.excludedLocations = this.getExcludedLocations(manifest);
|
|
10
15
|
}
|
|
11
16
|
getExcludedLocations(asmManifest) {
|
|
@@ -34,10 +39,11 @@ class ManifestExcludeList {
|
|
|
34
39
|
}
|
|
35
40
|
}
|
|
36
41
|
exports.ManifestExcludeList = ManifestExcludeList;
|
|
37
|
-
class InMemoryExcludeList {
|
|
42
|
+
class InMemoryExcludeList extends AbstractExcludeList {
|
|
38
43
|
excludedLocations;
|
|
39
44
|
excludedPaths;
|
|
40
45
|
constructor(items) {
|
|
46
|
+
super();
|
|
41
47
|
this.excludedLocations = [];
|
|
42
48
|
this.excludedPaths = [];
|
|
43
49
|
if (items.length === 0) {
|
|
@@ -66,9 +72,10 @@ class InMemoryExcludeList {
|
|
|
66
72
|
}
|
|
67
73
|
}
|
|
68
74
|
exports.InMemoryExcludeList = InMemoryExcludeList;
|
|
69
|
-
class UnionExcludeList {
|
|
75
|
+
class UnionExcludeList extends AbstractExcludeList {
|
|
70
76
|
excludeLists;
|
|
71
77
|
constructor(excludeLists) {
|
|
78
|
+
super();
|
|
72
79
|
this.excludeLists = excludeLists;
|
|
73
80
|
}
|
|
74
81
|
isExcluded(location) {
|
|
@@ -76,19 +83,16 @@ class UnionExcludeList {
|
|
|
76
83
|
}
|
|
77
84
|
}
|
|
78
85
|
exports.UnionExcludeList = UnionExcludeList;
|
|
79
|
-
class NeverExclude {
|
|
86
|
+
class NeverExclude extends AbstractExcludeList {
|
|
80
87
|
isExcluded(_location) {
|
|
81
88
|
return false;
|
|
82
89
|
}
|
|
83
90
|
}
|
|
84
91
|
exports.NeverExclude = NeverExclude;
|
|
85
|
-
class AlwaysExclude {
|
|
92
|
+
class AlwaysExclude extends AbstractExcludeList {
|
|
86
93
|
isExcluded(_location) {
|
|
87
94
|
return true;
|
|
88
95
|
}
|
|
89
96
|
}
|
|
90
97
|
exports.AlwaysExclude = AlwaysExclude;
|
|
91
|
-
|
|
92
|
-
return new UnionExcludeList([new ManifestExcludeList(manifest), new InMemoryExcludeList(exclude ?? [])]);
|
|
93
|
-
}
|
|
94
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"exclude.js","sourceRoot":"","sources":["exclude.ts"],"names":[],"mappings":"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","sourcesContent":["import type { AssemblyManifest } from '@aws-cdk/cloud-assembly-schema';\nimport { ArtifactMetadataEntryType, ArtifactType } from '@aws-cdk/cloud-assembly-schema';\nimport type { ResourceLocation as CfnResourceLocation } from '@aws-sdk/client-cloudformation';\nimport type { ResourceLocation } from './cloudformation';\n\nexport interface ExcludeList {\n  isExcluded(location: ResourceLocation): boolean;\n}\n\nexport class ManifestExcludeList implements ExcludeList {\n  private readonly excludedLocations: CfnResourceLocation[];\n\n  constructor(manifest: AssemblyManifest) {\n    this.excludedLocations = this.getExcludedLocations(manifest);\n  }\n\n  private getExcludedLocations(asmManifest: AssemblyManifest): CfnResourceLocation[] {\n    // First, we need to filter the artifacts to only include CloudFormation stacks\n    const stackManifests = Object.entries(asmManifest.artifacts ?? {}).filter(\n      ([_, manifest]) => manifest.type === ArtifactType.AWS_CLOUDFORMATION_STACK,\n    );\n\n    const result: CfnResourceLocation[] = [];\n    for (let [stackName, manifest] of stackManifests) {\n      const locations = Object.values(manifest.metadata ?? {})\n        // Then pick only the resources in each stack marked with DO_NOT_REFACTOR\n        .filter((entries) =>\n          entries.some((entry) => entry.type === ArtifactMetadataEntryType.DO_NOT_REFACTOR && entry.data === true),\n        )\n        // Finally, get the logical ID of each resource\n        .map((entries) => {\n          const logicalIdEntry = entries.find((entry) => entry.type === ArtifactMetadataEntryType.LOGICAL_ID);\n          const location: CfnResourceLocation = {\n            StackName: stackName,\n            LogicalResourceId: logicalIdEntry!.data! as string,\n          };\n          return location;\n        });\n      result.push(...locations);\n    }\n    return result;\n  }\n\n  isExcluded(location: ResourceLocation): boolean {\n    return this.excludedLocations.some(\n      (loc) => loc.StackName === location.stack.stackName && loc.LogicalResourceId === location.logicalResourceId,\n    );\n  }\n}\n\nexport class InMemoryExcludeList implements ExcludeList {\n  private readonly excludedLocations: CfnResourceLocation[];\n  private readonly excludedPaths: string[];\n\n  constructor(items: string[]) {\n    this.excludedLocations = [];\n    this.excludedPaths = [];\n\n    if (items.length === 0) {\n      return;\n    }\n\n    const locationRegex = /^[A-Za-z0-9]+\\.[A-Za-z0-9]+$/;\n\n    items.forEach((item: string) => {\n      if (locationRegex.test(item)) {\n        const [stackName, logicalId] = item.split('.');\n        this.excludedLocations.push({\n          StackName: stackName,\n          LogicalResourceId: logicalId,\n        });\n      } else {\n        this.excludedPaths.push(item);\n      }\n    });\n  }\n\n  isExcluded(location: ResourceLocation): boolean {\n    const containsLocation = this.excludedLocations.some((loc) => {\n      return loc.StackName === location.stack.stackName && loc.LogicalResourceId === location.logicalResourceId;\n    });\n\n    const containsPath = this.excludedPaths.some((path) => location.toPath() === path);\n    return containsLocation || containsPath;\n  }\n}\n\nexport class UnionExcludeList implements ExcludeList {\n  constructor(private readonly excludeLists: ExcludeList[]) {\n  }\n\n  isExcluded(location: ResourceLocation): boolean {\n    return this.excludeLists.some((excludeList) => excludeList.isExcluded(location));\n  }\n}\n\nexport class NeverExclude implements ExcludeList {\n  isExcluded(_location: ResourceLocation): boolean {\n    return false;\n  }\n}\n\nexport class AlwaysExclude implements ExcludeList {\n  isExcluded(_location: ResourceLocation): boolean {\n    return true;\n  }\n}\n\nexport function fromManifestAndExclusionList(manifest: AssemblyManifest, exclude?: string[]): ExcludeList {\n  return new UnionExcludeList([new ManifestExcludeList(manifest), new InMemoryExcludeList(exclude ?? [])]);\n}\n\n"]}
|
|
98
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"exclude.js","sourceRoot":"","sources":["exclude.ts"],"names":[],"mappings":"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","sourcesContent":["import type { AssemblyManifest } from '@aws-cdk/cloud-assembly-schema';\nimport { ArtifactMetadataEntryType, ArtifactType } from '@aws-cdk/cloud-assembly-schema';\nimport type { ResourceLocation as CfnResourceLocation } from '@aws-sdk/client-cloudformation';\nimport type { ResourceLocation } from './cloudformation';\n\nexport interface ExcludeList {\n  isExcluded(location: ResourceLocation): boolean;\n\n  union(other: ExcludeList): ExcludeList;\n}\n\nabstract class AbstractExcludeList implements ExcludeList {\n  abstract isExcluded(location: ResourceLocation): boolean;\n\n  union(other: ExcludeList): ExcludeList {\n    return new UnionExcludeList([this, other]);\n  }\n}\n\nexport class ManifestExcludeList extends AbstractExcludeList {\n  private readonly excludedLocations: CfnResourceLocation[];\n\n  constructor(manifest: AssemblyManifest) {\n    super();\n    this.excludedLocations = this.getExcludedLocations(manifest);\n  }\n\n  private getExcludedLocations(asmManifest: AssemblyManifest): CfnResourceLocation[] {\n    // First, we need to filter the artifacts to only include CloudFormation stacks\n    const stackManifests = Object.entries(asmManifest.artifacts ?? {}).filter(\n      ([_, manifest]) => manifest.type === ArtifactType.AWS_CLOUDFORMATION_STACK,\n    );\n\n    const result: CfnResourceLocation[] = [];\n    for (let [stackName, manifest] of stackManifests) {\n      const locations = Object.values(manifest.metadata ?? {})\n        // Then pick only the resources in each stack marked with DO_NOT_REFACTOR\n        .filter((entries) =>\n          entries.some((entry) => entry.type === ArtifactMetadataEntryType.DO_NOT_REFACTOR && entry.data === true),\n        )\n        // Finally, get the logical ID of each resource\n        .map((entries) => {\n          const logicalIdEntry = entries.find((entry) => entry.type === ArtifactMetadataEntryType.LOGICAL_ID);\n          const location: CfnResourceLocation = {\n            StackName: stackName,\n            LogicalResourceId: logicalIdEntry!.data! as string,\n          };\n          return location;\n        });\n      result.push(...locations);\n    }\n    return result;\n  }\n\n  isExcluded(location: ResourceLocation): boolean {\n    return this.excludedLocations.some(\n      (loc) => loc.StackName === location.stack.stackName && loc.LogicalResourceId === location.logicalResourceId,\n    );\n  }\n}\n\nexport class InMemoryExcludeList extends AbstractExcludeList {\n  private readonly excludedLocations: CfnResourceLocation[];\n  private readonly excludedPaths: string[];\n\n  constructor(items: string[]) {\n    super();\n    this.excludedLocations = [];\n    this.excludedPaths = [];\n\n    if (items.length === 0) {\n      return;\n    }\n\n    const locationRegex = /^[A-Za-z0-9]+\\.[A-Za-z0-9]+$/;\n\n    items.forEach((item: string) => {\n      if (locationRegex.test(item)) {\n        const [stackName, logicalId] = item.split('.');\n        this.excludedLocations.push({\n          StackName: stackName,\n          LogicalResourceId: logicalId,\n        });\n      } else {\n        this.excludedPaths.push(item);\n      }\n    });\n  }\n\n  isExcluded(location: ResourceLocation): boolean {\n    const containsLocation = this.excludedLocations.some((loc) => {\n      return loc.StackName === location.stack.stackName && loc.LogicalResourceId === location.logicalResourceId;\n    });\n\n    const containsPath = this.excludedPaths.some((path) => location.toPath() === path);\n    return containsLocation || containsPath;\n  }\n}\n\nexport class UnionExcludeList extends AbstractExcludeList {\n  constructor(private readonly excludeLists: ExcludeList[]) {\n    super();\n  }\n\n  isExcluded(location: ResourceLocation): boolean {\n    return this.excludeLists.some((excludeList) => excludeList.isExcluded(location));\n  }\n}\n\nexport class NeverExclude extends AbstractExcludeList {\n  isExcluded(_location: ResourceLocation): boolean {\n    return false;\n  }\n}\n\nexport class AlwaysExclude extends AbstractExcludeList {\n  isExcluded(_location: ResourceLocation): boolean {\n    return true;\n  }\n}\n"]}
|
|
@@ -4,4 +4,4 @@ import type { CloudFormationStack, ResourceMapping } from './cloudformation';
|
|
|
4
4
|
* Generates a list of stack definitions to be sent to the CloudFormation API
|
|
5
5
|
* by applying each mapping to the corresponding stack template(s).
|
|
6
6
|
*/
|
|
7
|
-
export declare function generateStackDefinitions(mappings: ResourceMapping[], deployedStacks: CloudFormationStack[]): StackDefinition[];
|
|
7
|
+
export declare function generateStackDefinitions(mappings: ResourceMapping[], deployedStacks: CloudFormationStack[], localStacks: CloudFormationStack[]): StackDefinition[];
|
|
@@ -6,38 +6,63 @@ const toolkit_error_1 = require("../../toolkit/toolkit-error");
|
|
|
6
6
|
* Generates a list of stack definitions to be sent to the CloudFormation API
|
|
7
7
|
* by applying each mapping to the corresponding stack template(s).
|
|
8
8
|
*/
|
|
9
|
-
function generateStackDefinitions(mappings, deployedStacks) {
|
|
10
|
-
const
|
|
11
|
-
|
|
12
|
-
//
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
const destinationStackName = mapping.destination.stack.stackName;
|
|
20
|
-
const destinationLogicalId = mapping.destination.logicalResourceId;
|
|
21
|
-
if (templates[destinationStackName] == null) {
|
|
22
|
-
// The API doesn't allow anything in the template other than the resources
|
|
23
|
-
// that are part of the mappings. So we need to create an empty template
|
|
24
|
-
// to start adding resources to.
|
|
25
|
-
templates[destinationStackName] = { Resources: {} };
|
|
9
|
+
function generateStackDefinitions(mappings, deployedStacks, localStacks) {
|
|
10
|
+
const localTemplates = Object.fromEntries(localStacks.map((s) => [s.stackName, JSON.parse(JSON.stringify(s.template))]));
|
|
11
|
+
const deployedTemplates = Object.fromEntries(deployedStacks.map((s) => [s.stackName, JSON.parse(JSON.stringify(s.template))]));
|
|
12
|
+
// First, remove from the local templates any resources that are not in the deployed templates
|
|
13
|
+
iterate(localTemplates, (stackName, logicalResourceId) => {
|
|
14
|
+
const location = searchLocation(stackName, logicalResourceId, 'destination', 'source');
|
|
15
|
+
const deployedResource = deployedStacks.find((s) => s.stackName === location.stackName)?.template
|
|
16
|
+
.Resources?.[location.logicalResourceId];
|
|
17
|
+
if (deployedResource == null) {
|
|
18
|
+
delete localTemplates[stackName].Resources?.[logicalResourceId];
|
|
26
19
|
}
|
|
27
|
-
const destinationTemplate = templates[destinationStackName];
|
|
28
|
-
// Do the move
|
|
29
|
-
destinationTemplate.Resources[destinationLogicalId] = sourceTemplate.Resources[sourceLogicalId];
|
|
30
|
-
delete sourceTemplate.Resources[sourceLogicalId];
|
|
31
20
|
});
|
|
32
|
-
//
|
|
33
|
-
|
|
21
|
+
// Now do the opposite: add to the local templates any resources that are in the deployed templates
|
|
22
|
+
iterate(deployedTemplates, (stackName, logicalResourceId, deployedResource) => {
|
|
23
|
+
const location = searchLocation(stackName, logicalResourceId, 'source', 'destination');
|
|
24
|
+
const resources = Object
|
|
25
|
+
.entries(localTemplates)
|
|
26
|
+
.find(([name, _]) => name === location.stackName)?.[1].Resources;
|
|
27
|
+
const localResource = resources?.[location.logicalResourceId];
|
|
28
|
+
if (localResource == null) {
|
|
29
|
+
if (localTemplates[stackName]?.Resources) {
|
|
30
|
+
localTemplates[stackName].Resources[logicalResourceId] = deployedResource;
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
else {
|
|
34
|
+
// This is temporary, until CloudFormation supports CDK construct path updates in the refactor API
|
|
35
|
+
if (localResource.Metadata != null) {
|
|
36
|
+
localResource.Metadata['aws:cdk:path'] = deployedResource.Metadata?.['aws:cdk:path'];
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
});
|
|
40
|
+
function searchLocation(stackName, logicalResourceId, from, to) {
|
|
41
|
+
const mapping = mappings.find((m) => m[from].stack.stackName === stackName && m[from].logicalResourceId === logicalResourceId);
|
|
42
|
+
return mapping != null
|
|
43
|
+
? { stackName: mapping[to].stack.stackName, logicalResourceId: mapping[to].logicalResourceId }
|
|
44
|
+
: { stackName, logicalResourceId };
|
|
45
|
+
}
|
|
46
|
+
function iterate(templates, cb) {
|
|
47
|
+
Object.entries(templates).forEach(([stackName, template]) => {
|
|
48
|
+
Object.entries(template.Resources ?? {}).forEach(([logicalResourceId, resource]) => {
|
|
49
|
+
cb(stackName, logicalResourceId, resource);
|
|
50
|
+
});
|
|
51
|
+
});
|
|
52
|
+
}
|
|
53
|
+
for (const [stackName, template] of Object.entries(localTemplates)) {
|
|
34
54
|
if (Object.keys(template.Resources ?? {}).length === 0) {
|
|
35
55
|
throw new toolkit_error_1.ToolkitError(`Stack ${stackName} has no resources after refactor. You must add a resource to this stack. This resource can be a simple one, like a waitCondition resource type.`);
|
|
36
56
|
}
|
|
37
57
|
}
|
|
38
|
-
return Object.entries(
|
|
58
|
+
return Object.entries(localTemplates)
|
|
59
|
+
.filter(([stackName, _]) => mappings.some((m) => {
|
|
60
|
+
// Only send templates for stacks that are part of the mappings
|
|
61
|
+
return m.source.stack.stackName === stackName || m.destination.stack.stackName === stackName;
|
|
62
|
+
}))
|
|
63
|
+
.map(([stackName, template]) => ({
|
|
39
64
|
StackName: stackName,
|
|
40
65
|
TemplateBody: JSON.stringify(template),
|
|
41
66
|
}));
|
|
42
67
|
}
|
|
43
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
68
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"execution.js","sourceRoot":"","sources":["execution.ts"],"names":[],"mappings":";;AAaA,4DAoFC;AA1FD,+DAA2D;AAE3D;;;GAGG;AACH,SAAgB,wBAAwB,CACtC,QAA2B,EAC3B,cAAqC,EACrC,WAAkC;IAElC,MAAM,cAAc,GAAG,MAAM,CAAC,WAAW,CACvC,WAAW,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,SAAS,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,QAAQ,CAAC,CAA2B,CAAC,CAAC,CACxG,CAAC;IACF,MAAM,iBAAiB,GAAG,MAAM,CAAC,WAAW,CAC1C,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,SAAS,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,QAAQ,CAAC,CAA2B,CAAC,CAAC,CAC3G,CAAC;IAEF,8FAA8F;IAC9F,OAAO,CAAC,cAAc,EAAE,CAAC,SAAS,EAAE,iBAAiB,EAAE,EAAE;QACvD,MAAM,QAAQ,GAAG,cAAc,CAAC,SAAS,EAAE,iBAAiB,EAAE,aAAa,EAAE,QAAQ,CAAC,CAAC;QAEvF,MAAM,gBAAgB,GAAG,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,SAAS,KAAK,QAAQ,CAAC,SAAS,CAAC,EAAE,QAAQ;aAC9F,SAAS,EAAE,CAAC,QAAQ,CAAC,iBAAiB,CAAC,CAAC;QAE3C,IAAI,gBAAgB,IAAI,IAAI,EAAE,CAAC;YAC7B,OAAO,cAAc,CAAC,SAAS,CAAC,CAAC,SAAS,EAAE,CAAC,iBAAiB,CAAC,CAAC;QAClE,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,mGAAmG;IACnG,OAAO,CAAC,iBAAiB,EAAE,CAAC,SAAS,EAAE,iBAAiB,EAAE,gBAAgB,EAAE,EAAE;QAC5E,MAAM,QAAQ,GAAG,cAAc,CAAC,SAAS,EAAE,iBAAiB,EAAE,QAAQ,EAAE,aAAa,CAAC,CAAC;QAEvF,MAAM,SAAS,GAAG,MAAM;aACrB,OAAO,CAAC,cAAc,CAAC;aACvB,IAAI,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,IAAI,KAAK,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QACnE,MAAM,aAAa,GAAG,SAAS,EAAE,CAAC,QAAQ,CAAC,iBAAiB,CAAC,CAAC;QAE9D,IAAI,aAAa,IAAI,IAAI,EAAE,CAAC;YAC1B,IAAI,cAAc,CAAC,SAAS,CAAC,EAAE,SAAS,EAAE,CAAC;gBACzC,cAAc,CAAC,SAAS,CAAC,CAAC,SAAS,CAAC,iBAAiB,CAAC,GAAG,gBAAgB,CAAC;YAC5E,CAAC;QACH,CAAC;aAAM,CAAC;YACN,kGAAkG;YAClG,IAAI,aAAa,CAAC,QAAQ,IAAI,IAAI,EAAE,CAAC;gBACnC,aAAa,CAAC,QAAQ,CAAC,cAAc,CAAC,GAAG,gBAAgB,CAAC,QAAQ,EAAE,CAAC,cAAc,CAAC,CAAC;YACvF,CAAC;QACH,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,SAAS,cAAc,CAAC,SAAiB,EAAE,iBAAyB,EAAE,IAA8B,EAAE,EAA4B;QAChI,MAAM,OAAO,GAAG,QAAQ,CAAC,IAAI,CAC3B,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,KAAK,CAAC,SAAS,KAAK,SAAS,IAAI,CAAC,CAAC,IAAI,CAAC,CAAC,iBAAiB,KAAK,iBAAiB,CAChG,CAAC;QACF,OAAO,OAAO,IAAI,IAAI;YACpB,CAAC,CAAC,EAAE,SAAS,EAAE,OAAO,CAAC,EAAE,CAAC,CAAC,KAAK,CAAC,SAAS,EAAE,iBAAiB,EAAE,OAAO,CAAC,EAAE,CAAC,CAAC,iBAAiB,EAAE;YAC9F,CAAC,CAAC,EAAE,SAAS,EAAE,iBAAiB,EAAE,CAAC;IACvC,CAAC;IAED,SAAS,OAAO,CACd,SAAiD,EACjD,EAA4F;QAE5F,MAAM,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,SAAS,EAAE,QAAQ,CAAC,EAAE,EAAE;YAC1D,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,SAAS,IAAI,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,iBAAiB,EAAE,QAAQ,CAAC,EAAE,EAAE;gBACjF,EAAE,CAAC,SAAS,EAAE,iBAAiB,EAAE,QAAQ,CAAC,CAAC;YAC7C,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED,KAAK,MAAM,CAAC,SAAS,EAAE,QAAQ,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,cAAc,CAAC,EAAE,CAAC;QACnE,IAAI,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,IAAI,EAAE,CAAC,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACvD,MAAM,IAAI,4BAAY,CACpB,SAAS,SAAS,iJAAiJ,CACpK,CAAC;QACJ,CAAC;IACH,CAAC;IAED,OAAO,MAAM,CAAC,OAAO,CAAC,cAAc,CAAC;SAClC,MAAM,CAAC,CAAC,CAAC,SAAS,EAAE,CAAC,CAAC,EAAE,EAAE,CACzB,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE;QAClB,+DAA+D;QAC/D,OAAO,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,SAAS,KAAK,SAAS,IAAI,CAAC,CAAC,WAAW,CAAC,KAAK,CAAC,SAAS,KAAK,SAAS,CAAC;IAC/F,CAAC,CAAC,CACH;SACA,GAAG,CAAC,CAAC,CAAC,SAAS,EAAE,QAAQ,CAAC,EAAE,EAAE,CAAC,CAAC;QAC/B,SAAS,EAAE,SAAS;QACpB,YAAY,EAAE,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC;KACvC,CAAC,CAAC,CAAC;AACR,CAAC","sourcesContent":["import type { StackDefinition } from '@aws-sdk/client-cloudformation';\nimport type {\n  CloudFormationResource,\n  CloudFormationStack,\n  CloudFormationTemplate,\n  ResourceMapping,\n} from './cloudformation';\nimport { ToolkitError } from '../../toolkit/toolkit-error';\n\n/**\n * Generates a list of stack definitions to be sent to the CloudFormation API\n * by applying each mapping to the corresponding stack template(s).\n */\nexport function generateStackDefinitions(\n  mappings: ResourceMapping[],\n  deployedStacks: CloudFormationStack[],\n  localStacks: CloudFormationStack[],\n): StackDefinition[] {\n  const localTemplates = Object.fromEntries(\n    localStacks.map((s) => [s.stackName, JSON.parse(JSON.stringify(s.template)) as CloudFormationTemplate]),\n  );\n  const deployedTemplates = Object.fromEntries(\n    deployedStacks.map((s) => [s.stackName, JSON.parse(JSON.stringify(s.template)) as CloudFormationTemplate]),\n  );\n\n  // First, remove from the local templates any resources that are not in the deployed templates\n  iterate(localTemplates, (stackName, logicalResourceId) => {\n    const location = searchLocation(stackName, logicalResourceId, 'destination', 'source');\n\n    const deployedResource = deployedStacks.find((s) => s.stackName === location.stackName)?.template\n      .Resources?.[location.logicalResourceId];\n\n    if (deployedResource == null) {\n      delete localTemplates[stackName].Resources?.[logicalResourceId];\n    }\n  });\n\n  // Now do the opposite: add to the local templates any resources that are in the deployed templates\n  iterate(deployedTemplates, (stackName, logicalResourceId, deployedResource) => {\n    const location = searchLocation(stackName, logicalResourceId, 'source', 'destination');\n\n    const resources = Object\n      .entries(localTemplates)\n      .find(([name, _]) => name === location.stackName)?.[1].Resources;\n    const localResource = resources?.[location.logicalResourceId];\n\n    if (localResource == null) {\n      if (localTemplates[stackName]?.Resources) {\n        localTemplates[stackName].Resources[logicalResourceId] = deployedResource;\n      }\n    } else {\n      // This is temporary, until CloudFormation supports CDK construct path updates in the refactor API\n      if (localResource.Metadata != null) {\n        localResource.Metadata['aws:cdk:path'] = deployedResource.Metadata?.['aws:cdk:path'];\n      }\n    }\n  });\n\n  function searchLocation(stackName: string, logicalResourceId: string, from: 'source' | 'destination', to: 'source' | 'destination') {\n    const mapping = mappings.find(\n      (m) => m[from].stack.stackName === stackName && m[from].logicalResourceId === logicalResourceId,\n    );\n    return mapping != null\n      ? { stackName: mapping[to].stack.stackName, logicalResourceId: mapping[to].logicalResourceId }\n      : { stackName, logicalResourceId };\n  }\n\n  function iterate(\n    templates: Record<string, CloudFormationTemplate>,\n    cb: (stackName: string, logicalResourceId: string, resource: CloudFormationResource) => void,\n  ) {\n    Object.entries(templates).forEach(([stackName, template]) => {\n      Object.entries(template.Resources ?? {}).forEach(([logicalResourceId, resource]) => {\n        cb(stackName, logicalResourceId, resource);\n      });\n    });\n  }\n\n  for (const [stackName, template] of Object.entries(localTemplates)) {\n    if (Object.keys(template.Resources ?? {}).length === 0) {\n      throw new ToolkitError(\n        `Stack ${stackName} has no resources after refactor. You must add a resource to this stack. This resource can be a simple one, like a waitCondition resource type.`,\n      );\n    }\n  }\n\n  return Object.entries(localTemplates)\n    .filter(([stackName, _]) =>\n      mappings.some((m) => {\n        // Only send templates for stacks that are part of the mappings\n        return m.source.stack.stackName === stackName || m.destination.stack.stackName === stackName;\n      }),\n    )\n    .map(([stackName, template]) => ({\n      StackName: stackName,\n      TemplateBody: JSON.stringify(template),\n    }));\n}\n"]}
|
|
@@ -1,34 +1,13 @@
|
|
|
1
1
|
import type { TypedMapping } from '@aws-cdk/cloudformation-diff';
|
|
2
|
+
import type * as cxapi from '@aws-cdk/cx-api';
|
|
2
3
|
import type { SdkProvider } from '../aws-auth/private';
|
|
3
4
|
import type { CloudFormationStack } from './cloudformation';
|
|
4
|
-
import {
|
|
5
|
-
import { type ExcludeList } from './exclude';
|
|
5
|
+
import { ResourceMapping } from './cloudformation';
|
|
6
6
|
import type { MappingGroup } from '../../actions';
|
|
7
7
|
export * from './exclude';
|
|
8
|
-
/**
|
|
9
|
-
* Represents a set of possible movements of a resource from one location
|
|
10
|
-
* to another. In the ideal case, there is only one source and only one
|
|
11
|
-
* destination.
|
|
12
|
-
*/
|
|
13
|
-
export type ResourceMovement = [ResourceLocation[], ResourceLocation[]];
|
|
14
|
-
export declare class AmbiguityError extends Error {
|
|
15
|
-
readonly movements: ResourceMovement[];
|
|
16
|
-
constructor(movements: ResourceMovement[]);
|
|
17
|
-
paths(): [string[], string[]][];
|
|
18
|
-
}
|
|
19
8
|
export declare function usePrescribedMappings(mappingGroups: MappingGroup[], sdkProvider: SdkProvider): Promise<ResourceMapping[]>;
|
|
20
|
-
export declare function
|
|
21
|
-
export declare function
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
*/
|
|
26
|
-
export declare function resourceMappings(movements: ResourceMovement[], stacks?: CloudFormationStack[]): ResourceMapping[];
|
|
27
|
-
/**
|
|
28
|
-
* Compares the deployed state to the cloud assembly state, and finds all resources
|
|
29
|
-
* that were moved from one location (stack + logical ID) to another. The comparison
|
|
30
|
-
* is done per environment.
|
|
31
|
-
*/
|
|
32
|
-
export declare function findResourceMovements(stacks: CloudFormationStack[], sdkProvider: SdkProvider, exclude?: ExcludeList): Promise<ResourceMovement[]>;
|
|
33
|
-
export declare function formatTypedMappings(mappings: TypedMapping[]): string;
|
|
34
|
-
export declare function formatAmbiguousMappings(paths: [string[], string[]][]): string;
|
|
9
|
+
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;
|