aws-cdk 2.19.0 → 2.20.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 (47) hide show
  1. package/README.md +59 -11
  2. package/build-info.json +2 -2
  3. package/does-not-exist.json +1 -1
  4. package/lib/api/bootstrap/bootstrap-template.yaml +3 -1
  5. package/lib/api/cloudformation-deployments.d.ts +12 -1
  6. package/lib/api/cloudformation-deployments.js +31 -8
  7. package/lib/api/deploy-stack.d.ts +24 -0
  8. package/lib/api/deploy-stack.js +35 -8
  9. package/lib/api/evaluate-cloudformation-template.js +3 -2
  10. package/lib/api/hotswap/common.js +2 -2
  11. package/lib/api/util/cloudformation/stack-activity-monitor.js +7 -6
  12. package/lib/api/util/cloudformation/stack-status.js +2 -2
  13. package/lib/api/util/cloudformation.d.ts +3 -0
  14. package/lib/api/util/cloudformation.js +1 -1
  15. package/lib/api/util/display.d.ts +2 -0
  16. package/lib/api/util/display.js +15 -2
  17. package/lib/cdk-toolkit.d.ts +44 -21
  18. package/lib/cdk-toolkit.js +68 -1
  19. package/lib/cli.js +46 -3
  20. package/lib/commands/doctor.js +2 -2
  21. package/lib/import.d.ts +151 -0
  22. package/lib/import.js +245 -0
  23. package/lib/index.js +2296 -815
  24. package/lib/settings.js +3 -2
  25. package/lib/util/asset-publishing.d.ts +7 -1
  26. package/lib/util/asset-publishing.js +13 -6
  27. package/lib/version.js +2 -2
  28. package/package.json +10 -10
  29. package/test/api/hotswap/state-machine-hotswap-deployments.test.js +75 -1
  30. package/test/api/util/display.test.d.ts +1 -0
  31. package/test/api/util/display.test.js +34 -0
  32. package/test/bockfs.js +2 -2
  33. package/test/cdk-toolkit.test.js +4 -2
  34. package/test/import.test.d.ts +1 -0
  35. package/test/import.test.js +189 -0
  36. package/test/integ/cli/app/app.js +25 -0
  37. package/test/integ/cli/app/docker/Dockerfile +1 -1
  38. package/test/integ/cli/app/docker/Dockerfile.Custom +1 -1
  39. package/test/integ/cli/cli.integtest.js +33 -1
  40. package/test/integ/cli-regression-patches/v1.64.0/cdk-helpers.js +1 -1
  41. package/test/integ/cli-regression-patches/v1.64.1/cdk-helpers.js +1 -1
  42. package/test/integ/cli-regression-patches/v1.67.0/cdk-helpers.js +1 -1
  43. package/test/integ/helpers/cdk.js +4 -5
  44. package/test/integ/helpers/test-helpers.js +1 -3
  45. package/test/settings.test.js +10 -1
  46. package/test/util/mock-sdk.d.ts +2 -0
  47. package/test/util/mock-sdk.js +8 -2
@@ -0,0 +1,151 @@
1
+ import { ResourceDifference } from '@aws-cdk/cloudformation-diff';
2
+ import * as cxapi from '@aws-cdk/cx-api';
3
+ import { CloudFormationDeployments, DeployStackOptions } from './api/cloudformation-deployments';
4
+ import { ResourceIdentifierProperties } from './api/util/cloudformation';
5
+ /**
6
+ * Parameters that uniquely identify a physical resource of a given type
7
+ * for the import operation, example:
8
+ *
9
+ * ```
10
+ * {
11
+ * "AWS::S3::Bucket": ["BucketName"],
12
+ * "AWS::IAM::Role": ["RoleName"],
13
+ * "AWS::EC2::VPC": ["VpcId"]
14
+ * }
15
+ * ```
16
+ */
17
+ export declare type ResourceIdentifiers = {
18
+ [resourceType: string]: string[];
19
+ };
20
+ /**
21
+ * Mapping of CDK resources (L1 constructs) to physical resources to be imported
22
+ * in their place, example:
23
+ *
24
+ * ```
25
+ * {
26
+ * "MyStack/MyS3Bucket/Resource": {
27
+ * "BucketName": "my-manually-created-s3-bucket"
28
+ * },
29
+ * "MyStack/MyVpc/Resource": {
30
+ * "VpcId": "vpc-123456789"
31
+ * }
32
+ * }
33
+ * ```
34
+ */
35
+ export declare type ResourceMap = {
36
+ [logicalResource: string]: ResourceIdentifierProperties;
37
+ };
38
+ export interface ResourceImporterOptions {
39
+ /**
40
+ * Name of toolkit stack if non-default
41
+ *
42
+ * @default - Default toolkit stack name
43
+ */
44
+ readonly toolkitStackName?: string;
45
+ }
46
+ /**
47
+ * Resource importing utility class
48
+ *
49
+ * - Determines the resources added to a template (compared to the deployed version)
50
+ * - Look up the identification information
51
+ * - Load them from a file, or
52
+ * - Ask the user, based on information supplied to us by CloudFormation's GetTemplateSummary
53
+ * - Translate the input to a structure expected by CloudFormation, update the template to add the
54
+ * importable resources, then run an IMPORT changeset.
55
+ */
56
+ export declare class ResourceImporter {
57
+ private readonly stack;
58
+ private readonly cfn;
59
+ private readonly options;
60
+ private _currentTemplate;
61
+ constructor(stack: cxapi.CloudFormationStackArtifact, cfn: CloudFormationDeployments, options?: ResourceImporterOptions);
62
+ /**
63
+ * Ask the user for resources to import
64
+ */
65
+ askForResourceIdentifiers(available: ImportableResource[]): Promise<ImportMap>;
66
+ /**
67
+ * Load the resources to import from a file
68
+ */
69
+ loadResourceIdentifiers(available: ImportableResource[], filename: string): Promise<ImportMap>;
70
+ /**
71
+ * Based on the provided resource mapping, prepare CFN structures for import (template,
72
+ * ResourcesToImport structure) and perform the import operation (CloudFormation deployment)
73
+ *
74
+ * @param resourceMap Mapping from CDK construct tree path to physical resource import identifiers
75
+ * @param options Options to pass to CloudFormation deploy operation
76
+ */
77
+ importResources(importMap: ImportMap, options: DeployStackOptions): Promise<void>;
78
+ /**
79
+ * Perform a diff between the currently running and the new template, enusre that it is valid
80
+ * for importing and return a list of resources that are being added in the new version
81
+ *
82
+ * @return mapping logicalResourceId -> resourceDifference
83
+ */
84
+ discoverImportableResources(allowNonAdditions?: boolean): Promise<DiscoverImportableResourcesResult>;
85
+ /**
86
+ * Get currently deployed template of the given stack (SINGLETON)
87
+ *
88
+ * @returns Currently deployed CloudFormation template
89
+ */
90
+ private currentTemplate;
91
+ /**
92
+ * Return teh current template, with the given resources added to it
93
+ */
94
+ private currentTemplateWithAdditions;
95
+ /**
96
+ * Get a list of import identifiers for all resource types used in the given
97
+ * template that do support the import operation (SINGLETON)
98
+ *
99
+ * @returns a mapping from a resource type to a list of property names that together identify the resource for import
100
+ */
101
+ private resourceIdentifiers;
102
+ private askForResourceIdentifier;
103
+ /**
104
+ * Convert the internal "resource mapping" structure to CloudFormation accepted "ResourcesToImport" structure
105
+ */
106
+ private makeResourcesToImport;
107
+ /**
108
+ * Convert CloudFormation logical resource ID to CDK construct tree path
109
+ *
110
+ * @param logicalId CloudFormation logical ID of the resource (the key in the template's Resources section)
111
+ * @returns Forward-slash separated path of the resource in CDK construct tree, e.g. MyStack/MyBucket/Resource
112
+ */
113
+ private describeResource;
114
+ }
115
+ /**
116
+ * Information about a resource in the template that is importable
117
+ */
118
+ export interface ImportableResource {
119
+ /**
120
+ * The logical ID of the resource
121
+ */
122
+ readonly logicalId: string;
123
+ /**
124
+ * The resource definition in the new template
125
+ */
126
+ readonly resourceDefinition: any;
127
+ /**
128
+ * The diff as reported by `cloudformation-diff`.
129
+ */
130
+ readonly resourceDiff: ResourceDifference;
131
+ }
132
+ /**
133
+ * The information necessary to execute an import operation
134
+ */
135
+ export interface ImportMap {
136
+ /**
137
+ * Mapping logical IDs to physical names
138
+ */
139
+ readonly resourceMap: ResourceMap;
140
+ /**
141
+ * The selection of resources we are actually importing
142
+ *
143
+ * For each of the resources in this list, there is a corresponding entry in
144
+ * the `resourceMap` map.
145
+ */
146
+ readonly importResources: ImportableResource[];
147
+ }
148
+ export interface DiscoverImportableResourcesResult {
149
+ readonly additions: ImportableResource[];
150
+ readonly hasNonAdditions: boolean;
151
+ }
package/lib/import.js ADDED
@@ -0,0 +1,245 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.ResourceImporter = void 0;
4
+ const cfnDiff = require("@aws-cdk/cloudformation-diff");
5
+ const chalk = require("chalk");
6
+ const fs = require("fs-extra");
7
+ const promptly = require("promptly");
8
+ const logging_1 = require("./logging");
9
+ /**
10
+ * Resource importing utility class
11
+ *
12
+ * - Determines the resources added to a template (compared to the deployed version)
13
+ * - Look up the identification information
14
+ * - Load them from a file, or
15
+ * - Ask the user, based on information supplied to us by CloudFormation's GetTemplateSummary
16
+ * - Translate the input to a structure expected by CloudFormation, update the template to add the
17
+ * importable resources, then run an IMPORT changeset.
18
+ */
19
+ class ResourceImporter {
20
+ constructor(stack, cfn, options = {}) {
21
+ this.stack = stack;
22
+ this.cfn = cfn;
23
+ this.options = options;
24
+ }
25
+ /**
26
+ * Ask the user for resources to import
27
+ */
28
+ async askForResourceIdentifiers(available) {
29
+ const ret = { importResources: [], resourceMap: {} };
30
+ const resourceIdentifiers = await this.resourceIdentifiers();
31
+ for (const resource of available) {
32
+ const identifier = await this.askForResourceIdentifier(resourceIdentifiers, resource);
33
+ if (!identifier) {
34
+ continue;
35
+ }
36
+ ret.importResources.push(resource);
37
+ ret.resourceMap[resource.logicalId] = identifier;
38
+ }
39
+ return ret;
40
+ }
41
+ /**
42
+ * Load the resources to import from a file
43
+ */
44
+ async loadResourceIdentifiers(available, filename) {
45
+ const contents = await fs.readJson(filename);
46
+ const ret = { importResources: [], resourceMap: {} };
47
+ for (const resource of available) {
48
+ const descr = this.describeResource(resource.logicalId);
49
+ const idProps = contents[resource.logicalId];
50
+ if (idProps) {
51
+ logging_1.print('%s: importing using %s', chalk.blue(descr), chalk.blue(fmtdict(idProps)));
52
+ ret.importResources.push(resource);
53
+ ret.resourceMap[resource.logicalId] = idProps;
54
+ delete contents[resource.logicalId];
55
+ }
56
+ else {
57
+ logging_1.print('%s: skipping', chalk.blue(descr));
58
+ }
59
+ }
60
+ const unknown = Object.keys(contents);
61
+ if (unknown.length > 0) {
62
+ logging_1.warning(`Unrecognized resource identifiers in mapping file: ${unknown.join(', ')}`);
63
+ }
64
+ return ret;
65
+ }
66
+ /**
67
+ * Based on the provided resource mapping, prepare CFN structures for import (template,
68
+ * ResourcesToImport structure) and perform the import operation (CloudFormation deployment)
69
+ *
70
+ * @param resourceMap Mapping from CDK construct tree path to physical resource import identifiers
71
+ * @param options Options to pass to CloudFormation deploy operation
72
+ */
73
+ async importResources(importMap, options) {
74
+ const resourcesToImport = await this.makeResourcesToImport(importMap);
75
+ const updatedTemplate = await this.currentTemplateWithAdditions(importMap.importResources);
76
+ try {
77
+ const result = await this.cfn.deployStack({
78
+ ...options,
79
+ overrideTemplate: updatedTemplate,
80
+ resourcesToImport,
81
+ });
82
+ const message = result.noOp
83
+ ? ' ✅ %s (no changes)'
84
+ : ' ✅ %s';
85
+ logging_1.success('\n' + message, options.stack.displayName);
86
+ }
87
+ catch (e) {
88
+ logging_1.error('\n ❌ %s failed: %s', chalk.bold(options.stack.displayName), e);
89
+ throw e;
90
+ }
91
+ }
92
+ /**
93
+ * Perform a diff between the currently running and the new template, enusre that it is valid
94
+ * for importing and return a list of resources that are being added in the new version
95
+ *
96
+ * @return mapping logicalResourceId -> resourceDifference
97
+ */
98
+ async discoverImportableResources(allowNonAdditions = false) {
99
+ const currentTemplate = await this.currentTemplate();
100
+ const diff = cfnDiff.diffTemplate(currentTemplate, this.stack.template);
101
+ // Ignore changes to CDKMetadata
102
+ const resourceChanges = Object.entries(diff.resources.changes)
103
+ .filter(([logicalId, _]) => logicalId !== 'CDKMetadata');
104
+ // Split the changes into additions and non-additions. Imports only make sense
105
+ // for newly-added resources.
106
+ const nonAdditions = resourceChanges.filter(([_, dif]) => !dif.isAddition);
107
+ const additions = resourceChanges.filter(([_, dif]) => dif.isAddition);
108
+ if (nonAdditions.length) {
109
+ const offendingResources = nonAdditions.map(([logId, _]) => this.describeResource(logId));
110
+ if (allowNonAdditions) {
111
+ logging_1.warning(`Ignoring updated/deleted resources (--force): ${offendingResources.join(', ')}`);
112
+ }
113
+ else {
114
+ throw new Error('No resource updates or deletes are allowed on import operation. Make sure to resolve pending changes ' +
115
+ `to existing resources, before attempting an import. Updated/deleted resources: ${offendingResources.join(', ')} (--force to override)`);
116
+ }
117
+ }
118
+ // Resources in the new template, that are not present in the current template, are a potential import candidates
119
+ return {
120
+ additions: additions.map(([logicalId, resourceDiff]) => {
121
+ var _a, _b, _c;
122
+ return ({
123
+ logicalId,
124
+ resourceDiff,
125
+ resourceDefinition: addDefaultDeletionPolicy((_c = (_b = (_a = this.stack.template) === null || _a === void 0 ? void 0 : _a.Resources) === null || _b === void 0 ? void 0 : _b[logicalId]) !== null && _c !== void 0 ? _c : {}),
126
+ });
127
+ }),
128
+ hasNonAdditions: nonAdditions.length > 0,
129
+ };
130
+ }
131
+ /**
132
+ * Get currently deployed template of the given stack (SINGLETON)
133
+ *
134
+ * @returns Currently deployed CloudFormation template
135
+ */
136
+ async currentTemplate() {
137
+ if (!this._currentTemplate) {
138
+ this._currentTemplate = await this.cfn.readCurrentTemplate(this.stack);
139
+ }
140
+ return this._currentTemplate;
141
+ }
142
+ /**
143
+ * Return teh current template, with the given resources added to it
144
+ */
145
+ async currentTemplateWithAdditions(additions) {
146
+ const template = await this.currentTemplate();
147
+ if (!template.Resources) {
148
+ template.Resources = {};
149
+ }
150
+ for (const add of additions) {
151
+ template.Resources[add.logicalId] = add.resourceDefinition;
152
+ }
153
+ return template;
154
+ }
155
+ /**
156
+ * Get a list of import identifiers for all resource types used in the given
157
+ * template that do support the import operation (SINGLETON)
158
+ *
159
+ * @returns a mapping from a resource type to a list of property names that together identify the resource for import
160
+ */
161
+ async resourceIdentifiers() {
162
+ const ret = {};
163
+ const resourceIdentifierSummaries = await this.cfn.resourceIdentifierSummaries(this.stack, this.options.toolkitStackName);
164
+ for (const summary of resourceIdentifierSummaries) {
165
+ if ('ResourceType' in summary && summary.ResourceType && 'ResourceIdentifiers' in summary && summary.ResourceIdentifiers) {
166
+ ret[summary.ResourceType] = summary.ResourceIdentifiers;
167
+ }
168
+ }
169
+ return ret;
170
+ }
171
+ async askForResourceIdentifier(resourceIdentifiers, chg) {
172
+ var _a;
173
+ const resourceName = this.describeResource(chg.logicalId);
174
+ // Skip resources that do not support importing
175
+ const resourceType = chg.resourceDiff.newResourceType;
176
+ if (resourceType === undefined || !(resourceType in resourceIdentifiers)) {
177
+ logging_1.warning(`${resourceName}: unsupported resource type ${resourceType}, skipping import.`);
178
+ return undefined;
179
+ }
180
+ const idProps = resourceIdentifiers[resourceType];
181
+ const resourceProps = (_a = chg.resourceDefinition.Properties) !== null && _a !== void 0 ? _a : {};
182
+ const fixedIdProps = idProps.filter(p => resourceProps[p]);
183
+ const fixedIdInput = Object.fromEntries(fixedIdProps.map(p => [p, resourceProps[p]]));
184
+ const missingIdProps = idProps.filter(p => !resourceProps[p]);
185
+ if (missingIdProps.length === 0) {
186
+ // We can auto-import this, but ask the user to confirm
187
+ const props = fmtdict(fixedIdInput);
188
+ if (!await promptly.confirm(`${chalk.blue(resourceName)} (${resourceType}): import with ${chalk.yellow(props)} (yes/no) [default: yes]? `, { default: 'yes' })) {
189
+ logging_1.print(chalk.grey(`Skipping import of ${resourceName}`));
190
+ return undefined;
191
+ }
192
+ }
193
+ // Ask the user to provide missing props
194
+ const userInput = {};
195
+ for (const missingIdProp of missingIdProps) {
196
+ const response = (await promptly.prompt(`${chalk.blue(resourceName)} (${resourceType}): enter ${chalk.blue(missingIdProp)} to import (empty to skip):`, { default: '', trim: true }));
197
+ if (!response) {
198
+ logging_1.print(chalk.grey(`Skipping import of ${resourceName}`));
199
+ return undefined;
200
+ }
201
+ userInput[missingIdProp] = response;
202
+ }
203
+ return {
204
+ ...fixedIdInput,
205
+ ...userInput,
206
+ };
207
+ }
208
+ /**
209
+ * Convert the internal "resource mapping" structure to CloudFormation accepted "ResourcesToImport" structure
210
+ */
211
+ async makeResourcesToImport(resourceMap) {
212
+ return resourceMap.importResources.map(res => ({
213
+ LogicalResourceId: res.logicalId,
214
+ ResourceType: res.resourceDiff.newResourceType,
215
+ ResourceIdentifier: resourceMap.resourceMap[res.logicalId],
216
+ }));
217
+ }
218
+ /**
219
+ * Convert CloudFormation logical resource ID to CDK construct tree path
220
+ *
221
+ * @param logicalId CloudFormation logical ID of the resource (the key in the template's Resources section)
222
+ * @returns Forward-slash separated path of the resource in CDK construct tree, e.g. MyStack/MyBucket/Resource
223
+ */
224
+ describeResource(logicalId) {
225
+ var _a, _b, _c, _d, _e;
226
+ return (_e = (_d = (_c = (_b = (_a = this.stack.template) === null || _a === void 0 ? void 0 : _a.Resources) === null || _b === void 0 ? void 0 : _b[logicalId]) === null || _c === void 0 ? void 0 : _c.Metadata) === null || _d === void 0 ? void 0 : _d['aws:cdk:path']) !== null && _e !== void 0 ? _e : logicalId;
227
+ }
228
+ }
229
+ exports.ResourceImporter = ResourceImporter;
230
+ function fmtdict(xs) {
231
+ return Object.entries(xs).map(([k, v]) => `${k}=${v}`).join(', ');
232
+ }
233
+ /**
234
+ * Add a default 'Delete' policy, which is required to make the import succeed
235
+ */
236
+ function addDefaultDeletionPolicy(resource) {
237
+ if (resource.DeletionPolicy) {
238
+ return resource;
239
+ }
240
+ return {
241
+ ...resource,
242
+ DeletionPolicy: 'Delete',
243
+ };
244
+ }
245
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"import.js","sourceRoot":"","sources":["import.ts"],"names":[],"mappings":"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","sourcesContent":["import * as cfnDiff from '@aws-cdk/cloudformation-diff';\nimport { ResourceDifference } from '@aws-cdk/cloudformation-diff';\nimport * as cxapi from '@aws-cdk/cx-api';\nimport * as chalk from 'chalk';\nimport * as fs from 'fs-extra';\nimport * as promptly from 'promptly';\nimport { CloudFormationDeployments, DeployStackOptions } from './api/cloudformation-deployments';\nimport { ResourceIdentifierProperties, ResourcesToImport } from './api/util/cloudformation';\nimport { error, print, success, warning } from './logging';\n\n/**\n * Parameters that uniquely identify a physical resource of a given type\n * for the import operation, example:\n *\n * ```\n * {\n *   \"AWS::S3::Bucket\": [\"BucketName\"],\n *   \"AWS::IAM::Role\": [\"RoleName\"],\n *   \"AWS::EC2::VPC\": [\"VpcId\"]\n * }\n * ```\n */\nexport type ResourceIdentifiers = { [resourceType: string]: string[] };\n\n/**\n * Mapping of CDK resources (L1 constructs) to physical resources to be imported\n * in their place, example:\n *\n * ```\n * {\n *   \"MyStack/MyS3Bucket/Resource\": {\n *     \"BucketName\": \"my-manually-created-s3-bucket\"\n *   },\n *   \"MyStack/MyVpc/Resource\": {\n *     \"VpcId\": \"vpc-123456789\"\n *   }\n * }\n * ```\n */\nexport type ResourceMap = { [logicalResource: string]: ResourceIdentifierProperties };\n\nexport interface ResourceImporterOptions {\n  /**\n   * Name of toolkit stack if non-default\n   *\n   * @default - Default toolkit stack name\n   */\n  readonly toolkitStackName?: string;\n}\n\n/**\n * Resource importing utility class\n *\n * - Determines the resources added to a template (compared to the deployed version)\n * - Look up the identification information\n *   - Load them from a file, or\n *   - Ask the user, based on information supplied to us by CloudFormation's GetTemplateSummary\n * - Translate the input to a structure expected by CloudFormation, update the template to add the\n *   importable resources, then run an IMPORT changeset.\n */\nexport class ResourceImporter {\n  private _currentTemplate: any;\n\n  constructor(\n    private readonly stack: cxapi.CloudFormationStackArtifact,\n    private readonly cfn: CloudFormationDeployments,\n    private readonly options: ResourceImporterOptions = {}) { }\n\n  /**\n   * Ask the user for resources to import\n   */\n  public async askForResourceIdentifiers(available: ImportableResource[]): Promise<ImportMap> {\n    const ret: ImportMap = { importResources: [], resourceMap: {} };\n    const resourceIdentifiers = await this.resourceIdentifiers();\n\n    for (const resource of available) {\n      const identifier = await this.askForResourceIdentifier(resourceIdentifiers, resource);\n      if (!identifier) {\n        continue;\n      }\n\n      ret.importResources.push(resource);\n      ret.resourceMap[resource.logicalId] = identifier;\n    }\n\n    return ret;\n  }\n\n  /**\n   * Load the resources to import from a file\n   */\n  public async loadResourceIdentifiers(available: ImportableResource[], filename: string): Promise<ImportMap> {\n    const contents = await fs.readJson(filename);\n\n    const ret: ImportMap = { importResources: [], resourceMap: {} };\n    for (const resource of available) {\n      const descr = this.describeResource(resource.logicalId);\n      const idProps = contents[resource.logicalId];\n      if (idProps) {\n        print('%s: importing using %s', chalk.blue(descr), chalk.blue(fmtdict(idProps)));\n\n        ret.importResources.push(resource);\n        ret.resourceMap[resource.logicalId] = idProps;\n        delete contents[resource.logicalId];\n      } else {\n        print('%s: skipping', chalk.blue(descr));\n      }\n    }\n\n    const unknown = Object.keys(contents);\n    if (unknown.length > 0) {\n      warning(`Unrecognized resource identifiers in mapping file: ${unknown.join(', ')}`);\n    }\n\n    return ret;\n  }\n\n  /**\n   * Based on the provided resource mapping, prepare CFN structures for import (template,\n   * ResourcesToImport structure) and perform the import operation (CloudFormation deployment)\n   *\n   * @param resourceMap Mapping from CDK construct tree path to physical resource import identifiers\n   * @param options Options to pass to CloudFormation deploy operation\n   */\n  public async importResources(importMap: ImportMap, options: DeployStackOptions) {\n    const resourcesToImport: ResourcesToImport = await this.makeResourcesToImport(importMap);\n    const updatedTemplate = await this.currentTemplateWithAdditions(importMap.importResources);\n\n    try {\n      const result = await this.cfn.deployStack({\n        ...options,\n        overrideTemplate: updatedTemplate,\n        resourcesToImport,\n      });\n\n      const message = result.noOp\n        ? ' ✅  %s (no changes)'\n        : ' ✅  %s';\n\n      success('\\n' + message, options.stack.displayName);\n    } catch (e) {\n      error('\\n ❌  %s failed: %s', chalk.bold(options.stack.displayName), e);\n      throw e;\n    }\n  }\n\n  /**\n   * Perform a diff between the currently running and the new template, enusre that it is valid\n   * for importing and return a list of resources that are being added in the new version\n   *\n   * @return mapping logicalResourceId -> resourceDifference\n   */\n  public async discoverImportableResources(allowNonAdditions = false): Promise<DiscoverImportableResourcesResult> {\n    const currentTemplate = await this.currentTemplate();\n\n    const diff = cfnDiff.diffTemplate(currentTemplate, this.stack.template);\n\n    // Ignore changes to CDKMetadata\n    const resourceChanges = Object.entries(diff.resources.changes)\n      .filter(([logicalId, _]) => logicalId !== 'CDKMetadata');\n\n    // Split the changes into additions and non-additions. Imports only make sense\n    // for newly-added resources.\n    const nonAdditions = resourceChanges.filter(([_, dif]) => !dif.isAddition);\n    const additions = resourceChanges.filter(([_, dif]) => dif.isAddition);\n\n    if (nonAdditions.length) {\n      const offendingResources = nonAdditions.map(([logId, _]) => this.describeResource(logId));\n\n      if (allowNonAdditions) {\n        warning(`Ignoring updated/deleted resources (--force): ${offendingResources.join(', ')}`);\n      } else {\n        throw new Error('No resource updates or deletes are allowed on import operation. Make sure to resolve pending changes ' +\n                        `to existing resources, before attempting an import. Updated/deleted resources: ${offendingResources.join(', ')} (--force to override)`);\n      }\n    }\n\n    // Resources in the new template, that are not present in the current template, are a potential import candidates\n    return {\n      additions: additions.map(([logicalId, resourceDiff]) => ({\n        logicalId,\n        resourceDiff,\n        resourceDefinition: addDefaultDeletionPolicy(this.stack.template?.Resources?.[logicalId] ?? {}),\n      })),\n      hasNonAdditions: nonAdditions.length > 0,\n    };\n  }\n\n  /**\n   * Get currently deployed template of the given stack (SINGLETON)\n   *\n   * @returns Currently deployed CloudFormation template\n   */\n  private async currentTemplate(): Promise<any> {\n    if (!this._currentTemplate) {\n      this._currentTemplate = await this.cfn.readCurrentTemplate(this.stack);\n    }\n    return this._currentTemplate;\n  }\n\n  /**\n   * Return teh current template, with the given resources added to it\n   */\n  private async currentTemplateWithAdditions(additions: ImportableResource[]): Promise<any> {\n    const template = await this.currentTemplate();\n    if (!template.Resources) {\n      template.Resources = {};\n    }\n\n    for (const add of additions) {\n      template.Resources[add.logicalId] = add.resourceDefinition;\n    }\n\n    return template;\n  }\n\n  /**\n   * Get a list of import identifiers for all resource types used in the given\n   * template that do support the import operation (SINGLETON)\n   *\n   * @returns a mapping from a resource type to a list of property names that together identify the resource for import\n   */\n  private async resourceIdentifiers(): Promise<ResourceIdentifiers> {\n    const ret: ResourceIdentifiers = {};\n    const resourceIdentifierSummaries = await this.cfn.resourceIdentifierSummaries(this.stack, this.options.toolkitStackName);\n    for (const summary of resourceIdentifierSummaries) {\n      if ('ResourceType' in summary && summary.ResourceType && 'ResourceIdentifiers' in summary && summary.ResourceIdentifiers) {\n        ret[summary.ResourceType] = summary.ResourceIdentifiers;\n      }\n    }\n    return ret;\n  }\n\n  private async askForResourceIdentifier(\n    resourceIdentifiers: ResourceIdentifiers,\n    chg: ImportableResource,\n  ): Promise<ResourceIdentifierProperties | undefined> {\n    const resourceName = this.describeResource(chg.logicalId);\n\n    // Skip resources that do not support importing\n    const resourceType = chg.resourceDiff.newResourceType;\n    if (resourceType === undefined || !(resourceType in resourceIdentifiers)) {\n      warning(`${resourceName}: unsupported resource type ${resourceType}, skipping import.`);\n      return undefined;\n    }\n\n    const idProps = resourceIdentifiers[resourceType];\n    const resourceProps = chg.resourceDefinition.Properties ?? {};\n\n    const fixedIdProps = idProps.filter(p => resourceProps[p]);\n    const fixedIdInput: ResourceIdentifierProperties = Object.fromEntries(fixedIdProps.map(p => [p, resourceProps[p]]));\n\n    const missingIdProps = idProps.filter(p => !resourceProps[p]);\n\n    if (missingIdProps.length === 0) {\n      // We can auto-import this, but ask the user to confirm\n      const props = fmtdict(fixedIdInput);\n\n      if (!await promptly.confirm(\n        `${chalk.blue(resourceName)} (${resourceType}): import with ${chalk.yellow(props)} (yes/no) [default: yes]? `,\n        { default: 'yes' },\n      )) {\n        print(chalk.grey(`Skipping import of ${resourceName}`));\n        return undefined;\n      }\n    }\n\n    // Ask the user to provide missing props\n    const userInput: ResourceIdentifierProperties = {};\n    for (const missingIdProp of missingIdProps) {\n      const response = (await promptly.prompt(\n        `${chalk.blue(resourceName)} (${resourceType}): enter ${chalk.blue(missingIdProp)} to import (empty to skip):`,\n        { default: '', trim: true },\n      ));\n      if (!response) {\n        print(chalk.grey(`Skipping import of ${resourceName}`));\n        return undefined;\n      }\n      userInput[missingIdProp] = response;\n    }\n\n    return {\n      ...fixedIdInput,\n      ...userInput,\n    };\n  }\n\n  /**\n   * Convert the internal \"resource mapping\" structure to CloudFormation accepted \"ResourcesToImport\" structure\n   */\n  private async makeResourcesToImport(resourceMap: ImportMap): Promise<ResourcesToImport> {\n    return resourceMap.importResources.map(res => ({\n      LogicalResourceId: res.logicalId,\n      ResourceType: res.resourceDiff.newResourceType!,\n      ResourceIdentifier: resourceMap.resourceMap[res.logicalId],\n    }));\n  }\n\n  /**\n   * Convert CloudFormation logical resource ID to CDK construct tree path\n   *\n   * @param logicalId CloudFormation logical ID of the resource (the key in the template's Resources section)\n   * @returns Forward-slash separated path of the resource in CDK construct tree, e.g. MyStack/MyBucket/Resource\n   */\n  private describeResource(logicalId: string): string {\n    return this.stack.template?.Resources?.[logicalId]?.Metadata?.['aws:cdk:path'] ?? logicalId;\n  }\n}\n\n/**\n * Information about a resource in the template that is importable\n */\nexport interface ImportableResource {\n  /**\n   * The logical ID of the resource\n   */\n  readonly logicalId: string;\n\n  /**\n   * The resource definition in the new template\n   */\n  readonly resourceDefinition: any;\n\n  /**\n   * The diff as reported by `cloudformation-diff`.\n   */\n  readonly resourceDiff: ResourceDifference;\n}\n\n/**\n * The information necessary to execute an import operation\n */\nexport interface ImportMap {\n  /**\n   * Mapping logical IDs to physical names\n   */\n  readonly resourceMap: ResourceMap;\n\n  /**\n   * The selection of resources we are actually importing\n   *\n   * For each of the resources in this list, there is a corresponding entry in\n   * the `resourceMap` map.\n   */\n  readonly importResources: ImportableResource[];\n}\n\nfunction fmtdict<A>(xs: Record<string, A>) {\n  return Object.entries(xs).map(([k, v]) => `${k}=${v}`).join(', ');\n}\n\n/**\n * Add a default 'Delete' policy, which is required to make the import succeed\n */\nfunction addDefaultDeletionPolicy(resource: any): any {\n  if (resource.DeletionPolicy) { return resource; }\n\n  return {\n    ...resource,\n    DeletionPolicy: 'Delete',\n  };\n}\n\nexport interface DiscoverImportableResourcesResult {\n  readonly additions: ImportableResource[];\n  readonly hasNonAdditions: boolean;\n}"]}