aws-cdk 2.178.2 → 2.179.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 (53) hide show
  1. package/THIRD_PARTY_LICENSES +256 -234
  2. package/build-info.json +2 -2
  3. package/lib/api/cxapp/cloud-assembly.d.ts +18 -1
  4. package/lib/api/cxapp/cloud-assembly.js +38 -5
  5. package/lib/api/deployments/asset-publishing.d.ts +4 -27
  6. package/lib/api/deployments/asset-publishing.js +8 -34
  7. package/lib/api/deployments/cloudformation.d.ts +1 -0
  8. package/lib/api/deployments/cloudformation.js +9 -1
  9. package/lib/api/deployments/deployments.d.ts +0 -22
  10. package/lib/api/deployments/deployments.js +1 -29
  11. package/lib/api/garbage-collection/garbage-collector.js +3 -3
  12. package/lib/api/garbage-collection/progress-printer.js +8 -1
  13. package/lib/{import.d.ts → api/resource-import/importer.d.ts} +55 -12
  14. package/lib/api/resource-import/importer.js +332 -0
  15. package/lib/api/resource-import/index.d.ts +2 -0
  16. package/lib/api/resource-import/index.js +19 -0
  17. package/lib/{migrator.d.ts → api/resource-import/migrator.d.ts} +9 -6
  18. package/lib/api/resource-import/migrator.js +74 -0
  19. package/lib/cli/cdk-toolkit.d.ts +7 -0
  20. package/lib/cli/cdk-toolkit.js +20 -9
  21. package/lib/cli/cli.js +4 -3
  22. package/lib/cli/convert-to-user-input.js +1 -1
  23. package/lib/cli/messages.d.ts +30 -0
  24. package/lib/cli/messages.js +112 -0
  25. package/lib/cli/parse-command-line-arguments.js +1 -1
  26. package/lib/cli/user-configuration.js +2 -1
  27. package/lib/cli/user-input.js +1 -1
  28. package/lib/cli/util/yargs-helpers.js +2 -2
  29. package/lib/cli/version.d.ts +1 -1
  30. package/lib/cli/version.js +6 -3
  31. package/lib/commands/doctor.js +2 -2
  32. package/lib/index.js +126 -197
  33. package/lib/init-templates/.init-version.json +1 -1
  34. package/lib/init-templates/app/javascript/package.json +1 -1
  35. package/lib/init-templates/app/typescript/package.json +1 -1
  36. package/lib/init-templates/sample-app/javascript/package.json +1 -1
  37. package/lib/init-templates/sample-app/typescript/package.json +1 -1
  38. package/lib/init.d.ts +13 -1
  39. package/lib/init.js +37 -32
  40. package/lib/list-stacks.d.ts +1 -17
  41. package/lib/list-stacks.js +2 -37
  42. package/package.json +10 -10
  43. package/test/api/deployments/cloudformation-deployments.test.js +27 -3
  44. package/test/api/garbage-collection.test.js +23 -1
  45. package/test/api/resource-import/import.test.js +373 -0
  46. package/test/init.test.js +14 -1
  47. package/test/toolkit/cli-io-host.test.js +18 -10
  48. package/lib/import.js +0 -329
  49. package/lib/migrator.js +0 -67
  50. package/test/_helpers/prompts.d.ts +0 -11
  51. package/test/_helpers/prompts.js +0 -22
  52. package/test/import.test.js +0 -364
  53. /package/test/{import.test.d.ts → api/resource-import/import.test.d.ts} +0 -0
package/lib/migrator.js DELETED
@@ -1,67 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.ResourceMigrator = void 0;
4
- const chalk = require("chalk");
5
- const fs = require("fs-extra");
6
- const string_manipulation_1 = require("./api/util/string-manipulation");
7
- const import_1 = require("./import");
8
- const logging_1 = require("./logging");
9
- class ResourceMigrator {
10
- constructor(props) {
11
- this.props = props;
12
- }
13
- /**
14
- * Checks to see if a migrate.json file exists. If it does and the source is either `filepath` or
15
- * is in the same environment as the stack deployment, a new stack is created and the resources are
16
- * migrated to the stack using an IMPORT changeset. The normal deployment will resume after this is complete
17
- * to add back in any outputs and the CDKMetadata.
18
- */
19
- async tryMigrateResources(stacks, options) {
20
- const stack = stacks.stackArtifacts[0];
21
- const migrateDeployment = new import_1.ResourceImporter(stack, this.props.deployments);
22
- const resourcesToImport = await this.tryGetResources(await migrateDeployment.resolveEnvironment());
23
- if (resourcesToImport) {
24
- (0, logging_1.info)('%s: creating stack for resource migration...', chalk.bold(stack.displayName));
25
- (0, logging_1.info)('%s: importing resources into stack...', chalk.bold(stack.displayName));
26
- await this.performResourceMigration(migrateDeployment, resourcesToImport, options);
27
- fs.rmSync('migrate.json');
28
- (0, logging_1.info)('%s: applying CDKMetadata and Outputs to stack (if applicable)...', chalk.bold(stack.displayName));
29
- }
30
- }
31
- /**
32
- * Creates a new stack with just the resources to be migrated
33
- */
34
- async performResourceMigration(migrateDeployment, resourcesToImport, options) {
35
- const startDeployTime = new Date().getTime();
36
- let elapsedDeployTime = 0;
37
- // Initial Deployment
38
- await migrateDeployment.importResourcesFromMigrate(resourcesToImport, {
39
- roleArn: options.roleArn,
40
- toolkitStackName: options.toolkitStackName,
41
- deploymentMethod: options.deploymentMethod,
42
- usePreviousParameters: true,
43
- progress: options.progress,
44
- rollback: options.rollback,
45
- });
46
- elapsedDeployTime = new Date().getTime() - startDeployTime;
47
- (0, logging_1.info)('\n✨ Resource migration time: %ss\n', (0, string_manipulation_1.formatTime)(elapsedDeployTime));
48
- }
49
- async tryGetResources(environment) {
50
- try {
51
- const migrateFile = fs.readJsonSync('migrate.json', {
52
- encoding: 'utf-8',
53
- });
54
- const sourceEnv = migrateFile.Source.split(':');
55
- if (sourceEnv[0] === 'localfile' ||
56
- (sourceEnv[4] === environment.account && sourceEnv[3] === environment.region)) {
57
- return migrateFile.Resources;
58
- }
59
- }
60
- catch (e) {
61
- // Nothing to do
62
- }
63
- return undefined;
64
- }
65
- }
66
- exports.ResourceMigrator = ResourceMigrator;
67
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWlncmF0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJtaWdyYXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFDQSwrQkFBK0I7QUFDL0IsK0JBQStCO0FBRy9CLHdFQUE0RDtBQUU1RCxxQ0FBNEM7QUFDNUMsdUNBQWlDO0FBUWpDLE1BQWEsZ0JBQWdCO0lBQzNCLFlBQW9DLEtBQTRCO1FBQTVCLFVBQUssR0FBTCxLQUFLLENBQXVCO0lBQUcsQ0FBQztJQUVwRTs7Ozs7T0FLRztJQUNJLEtBQUssQ0FBQyxtQkFBbUIsQ0FBQyxNQUF1QixFQUFFLE9BQWdDO1FBQ3hGLE1BQU0sS0FBSyxHQUFHLE1BQU0sQ0FBQyxjQUFjLENBQUMsQ0FBQyxDQUFDLENBQUM7UUFDdkMsTUFBTSxpQkFBaUIsR0FBRyxJQUFJLHlCQUFnQixDQUFDLEtBQUssRUFBRSxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxDQUFDO1FBQzlFLE1BQU0saUJBQWlCLEdBQUcsTUFBTSxJQUFJLENBQUMsZUFBZSxDQUFDLE1BQU0saUJBQWlCLENBQUMsa0JBQWtCLEVBQUUsQ0FBQyxDQUFDO1FBRW5HLElBQUksaUJBQWlCLEVBQUUsQ0FBQztZQUN0QixJQUFBLGNBQUksRUFBQyw4Q0FBOEMsRUFBRSxLQUFLLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDO1lBQ3BGLElBQUEsY0FBSSxFQUFDLHVDQUF1QyxFQUFFLEtBQUssQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUM7WUFFN0UsTUFBTSxJQUFJLENBQUMsd0JBQXdCLENBQUMsaUJBQWlCLEVBQUUsaUJBQWlCLEVBQUUsT0FBTyxDQUFDLENBQUM7WUFFbkYsRUFBRSxDQUFDLE1BQU0sQ0FBQyxjQUFjLENBQUMsQ0FBQztZQUMxQixJQUFBLGNBQUksRUFBQyxrRUFBa0UsRUFBRSxLQUFLLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxXQUFXLENBQUMsQ0FBQyxDQUFDO1FBQzFHLENBQUM7SUFDSCxDQUFDO0lBRUQ7O09BRUc7SUFDSyxLQUFLLENBQUMsd0JBQXdCLENBQ3BDLGlCQUFtQyxFQUNuQyxpQkFBb0MsRUFDcEMsT0FBZ0M7UUFFaEMsTUFBTSxlQUFlLEdBQUcsSUFBSSxJQUFJLEVBQUUsQ0FBQyxPQUFPLEVBQUUsQ0FBQztRQUM3QyxJQUFJLGlCQUFpQixHQUFHLENBQUMsQ0FBQztRQUUxQixxQkFBcUI7UUFDckIsTUFBTSxpQkFBaUIsQ0FBQywwQkFBMEIsQ0FBQyxpQkFBaUIsRUFBRTtZQUNwRSxPQUFPLEVBQUUsT0FBTyxDQUFDLE9BQU87WUFDeEIsZ0JBQWdCLEVBQUUsT0FBTyxDQUFDLGdCQUFnQjtZQUMxQyxnQkFBZ0IsRUFBRSxPQUFPLENBQUMsZ0JBQWdCO1lBQzFDLHFCQUFxQixFQUFFLElBQUk7WUFDM0IsUUFBUSxFQUFFLE9BQU8sQ0FBQyxRQUFRO1lBQzFCLFFBQVEsRUFBRSxPQUFPLENBQUMsUUFBUTtTQUMzQixDQUFDLENBQUM7UUFFSCxpQkFBaUIsR0FBRyxJQUFJLElBQUksRUFBRSxDQUFDLE9BQU8sRUFBRSxHQUFHLGVBQWUsQ0FBQztRQUMzRCxJQUFBLGNBQUksRUFBQyxxQ0FBcUMsRUFBRSxJQUFBLGdDQUFVLEVBQUMsaUJBQWlCLENBQUMsQ0FBQyxDQUFDO0lBQzdFLENBQUM7SUFFTSxLQUFLLENBQUMsZUFBZSxDQUFDLFdBQThCO1FBQ3pELElBQUksQ0FBQztZQUNILE1BQU0sV0FBVyxHQUFHLEVBQUUsQ0FBQyxZQUFZLENBQUMsY0FBYyxFQUFFO2dCQUNsRCxRQUFRLEVBQUUsT0FBTzthQUNsQixDQUFDLENBQUM7WUFDSCxNQUFNLFNBQVMsR0FBSSxXQUFXLENBQUMsTUFBaUIsQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLENBQUM7WUFDNUQsSUFDRSxTQUFTLENBQUMsQ0FBQyxDQUFDLEtBQUssV0FBVztnQkFDNUIsQ0FBQyxTQUFTLENBQUMsQ0FBQyxDQUFDLEtBQUssV0FBVyxDQUFDLE9BQU8sSUFBSSxTQUFTLENBQUMsQ0FBQyxDQUFDLEtBQUssV0FBVyxDQUFDLE1BQU0sQ0FBQyxFQUM3RSxDQUFDO2dCQUNELE9BQU8sV0FBVyxDQUFDLFNBQVMsQ0FBQztZQUMvQixDQUFDO1FBQ0gsQ0FBQztRQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUM7WUFDWCxnQkFBZ0I7UUFDbEIsQ0FBQztRQUVELE9BQU8sU0FBUyxDQUFDO0lBQ25CLENBQUM7Q0FDRjtBQXBFRCw0Q0FvRUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgdHlwZSAqIGFzIGN4YXBpIGZyb20gJ0Bhd3MtY2RrL2N4LWFwaSc7XG5pbXBvcnQgKiBhcyBjaGFsayBmcm9tICdjaGFsayc7XG5pbXBvcnQgKiBhcyBmcyBmcm9tICdmcy1leHRyYSc7XG5pbXBvcnQgeyBTdGFja0NvbGxlY3Rpb24gfSBmcm9tICcuL2FwaS9jeGFwcC9jbG91ZC1hc3NlbWJseSc7XG5pbXBvcnQgeyBEZXBsb3ltZW50cywgUmVzb3VyY2VzVG9JbXBvcnQgfSBmcm9tICcuL2FwaS9kZXBsb3ltZW50cyc7XG5pbXBvcnQgeyBmb3JtYXRUaW1lIH0gZnJvbSAnLi9hcGkvdXRpbC9zdHJpbmctbWFuaXB1bGF0aW9uJztcbmltcG9ydCB7IERlcGxveU9wdGlvbnMgfSBmcm9tICcuL2NsaS9jZGstdG9vbGtpdCc7XG5pbXBvcnQgeyBSZXNvdXJjZUltcG9ydGVyIH0gZnJvbSAnLi9pbXBvcnQnO1xuaW1wb3J0IHsgaW5mbyB9IGZyb20gJy4vbG9nZ2luZyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgUmVzb3VyY2VNaWdyYXRvclByb3BzIHtcbiAgZGVwbG95bWVudHM6IERlcGxveW1lbnRzO1xufVxuXG50eXBlIFJlc291cmNlTWlncmF0b3JPcHRpb25zID0gUGljazxEZXBsb3lPcHRpb25zLCAncm9sZUFybicgfCAndG9vbGtpdFN0YWNrTmFtZScgfCAnZGVwbG95bWVudE1ldGhvZCcgfCAncHJvZ3Jlc3MnIHwgJ3JvbGxiYWNrJz5cblxuZXhwb3J0IGNsYXNzIFJlc291cmNlTWlncmF0b3Ige1xuICBwdWJsaWMgY29uc3RydWN0b3IocHJpdmF0ZSByZWFkb25seSBwcm9wczogUmVzb3VyY2VNaWdyYXRvclByb3BzKSB7fVxuXG4gIC8qKlxuICAgKiBDaGVja3MgdG8gc2VlIGlmIGEgbWlncmF0ZS5qc29uIGZpbGUgZXhpc3RzLiBJZiBpdCBkb2VzIGFuZCB0aGUgc291cmNlIGlzIGVpdGhlciBgZmlsZXBhdGhgIG9yXG4gICAqIGlzIGluIHRoZSBzYW1lIGVudmlyb25tZW50IGFzIHRoZSBzdGFjayBkZXBsb3ltZW50LCBhIG5ldyBzdGFjayBpcyBjcmVhdGVkIGFuZCB0aGUgcmVzb3VyY2VzIGFyZVxuICAgKiBtaWdyYXRlZCB0byB0aGUgc3RhY2sgdXNpbmcgYW4gSU1QT1JUIGNoYW5nZXNldC4gVGhlIG5vcm1hbCBkZXBsb3ltZW50IHdpbGwgcmVzdW1lIGFmdGVyIHRoaXMgaXMgY29tcGxldGVcbiAgICogdG8gYWRkIGJhY2sgaW4gYW55IG91dHB1dHMgYW5kIHRoZSBDREtNZXRhZGF0YS5cbiAgICovXG4gIHB1YmxpYyBhc3luYyB0cnlNaWdyYXRlUmVzb3VyY2VzKHN0YWNrczogU3RhY2tDb2xsZWN0aW9uLCBvcHRpb25zOiBSZXNvdXJjZU1pZ3JhdG9yT3B0aW9ucyk6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IHN0YWNrID0gc3RhY2tzLnN0YWNrQXJ0aWZhY3RzWzBdO1xuICAgIGNvbnN0IG1pZ3JhdGVEZXBsb3ltZW50ID0gbmV3IFJlc291cmNlSW1wb3J0ZXIoc3RhY2ssIHRoaXMucHJvcHMuZGVwbG95bWVudHMpO1xuICAgIGNvbnN0IHJlc291cmNlc1RvSW1wb3J0ID0gYXdhaXQgdGhpcy50cnlHZXRSZXNvdXJjZXMoYXdhaXQgbWlncmF0ZURlcGxveW1lbnQucmVzb2x2ZUVudmlyb25tZW50KCkpO1xuXG4gICAgaWYgKHJlc291cmNlc1RvSW1wb3J0KSB7XG4gICAgICBpbmZvKCclczogY3JlYXRpbmcgc3RhY2sgZm9yIHJlc291cmNlIG1pZ3JhdGlvbi4uLicsIGNoYWxrLmJvbGQoc3RhY2suZGlzcGxheU5hbWUpKTtcbiAgICAgIGluZm8oJyVzOiBpbXBvcnRpbmcgcmVzb3VyY2VzIGludG8gc3RhY2suLi4nLCBjaGFsay5ib2xkKHN0YWNrLmRpc3BsYXlOYW1lKSk7XG5cbiAgICAgIGF3YWl0IHRoaXMucGVyZm9ybVJlc291cmNlTWlncmF0aW9uKG1pZ3JhdGVEZXBsb3ltZW50LCByZXNvdXJjZXNUb0ltcG9ydCwgb3B0aW9ucyk7XG5cbiAgICAgIGZzLnJtU3luYygnbWlncmF0ZS5qc29uJyk7XG4gICAgICBpbmZvKCclczogYXBwbHlpbmcgQ0RLTWV0YWRhdGEgYW5kIE91dHB1dHMgdG8gc3RhY2sgKGlmIGFwcGxpY2FibGUpLi4uJywgY2hhbGsuYm9sZChzdGFjay5kaXNwbGF5TmFtZSkpO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBDcmVhdGVzIGEgbmV3IHN0YWNrIHdpdGgganVzdCB0aGUgcmVzb3VyY2VzIHRvIGJlIG1pZ3JhdGVkXG4gICAqL1xuICBwcml2YXRlIGFzeW5jIHBlcmZvcm1SZXNvdXJjZU1pZ3JhdGlvbihcbiAgICBtaWdyYXRlRGVwbG95bWVudDogUmVzb3VyY2VJbXBvcnRlcixcbiAgICByZXNvdXJjZXNUb0ltcG9ydDogUmVzb3VyY2VzVG9JbXBvcnQsXG4gICAgb3B0aW9uczogUmVzb3VyY2VNaWdyYXRvck9wdGlvbnMsXG4gICkge1xuICAgIGNvbnN0IHN0YXJ0RGVwbG95VGltZSA9IG5ldyBEYXRlKCkuZ2V0VGltZSgpO1xuICAgIGxldCBlbGFwc2VkRGVwbG95VGltZSA9IDA7XG5cbiAgICAvLyBJbml0aWFsIERlcGxveW1lbnRcbiAgICBhd2FpdCBtaWdyYXRlRGVwbG95bWVudC5pbXBvcnRSZXNvdXJjZXNGcm9tTWlncmF0ZShyZXNvdXJjZXNUb0ltcG9ydCwge1xuICAgICAgcm9sZUFybjogb3B0aW9ucy5yb2xlQXJuLFxuICAgICAgdG9vbGtpdFN0YWNrTmFtZTogb3B0aW9ucy50b29sa2l0U3RhY2tOYW1lLFxuICAgICAgZGVwbG95bWVudE1ldGhvZDogb3B0aW9ucy5kZXBsb3ltZW50TWV0aG9kLFxuICAgICAgdXNlUHJldmlvdXNQYXJhbWV0ZXJzOiB0cnVlLFxuICAgICAgcHJvZ3Jlc3M6IG9wdGlvbnMucHJvZ3Jlc3MsXG4gICAgICByb2xsYmFjazogb3B0aW9ucy5yb2xsYmFjayxcbiAgICB9KTtcblxuICAgIGVsYXBzZWREZXBsb3lUaW1lID0gbmV3IERhdGUoKS5nZXRUaW1lKCkgLSBzdGFydERlcGxveVRpbWU7XG4gICAgaW5mbygnXFxu4pyoICBSZXNvdXJjZSBtaWdyYXRpb24gdGltZTogJXNzXFxuJywgZm9ybWF0VGltZShlbGFwc2VkRGVwbG95VGltZSkpO1xuICB9XG5cbiAgcHVibGljIGFzeW5jIHRyeUdldFJlc291cmNlcyhlbnZpcm9ubWVudDogY3hhcGkuRW52aXJvbm1lbnQpOiBQcm9taXNlPFJlc291cmNlc1RvSW1wb3J0IHwgdW5kZWZpbmVkPiB7XG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IG1pZ3JhdGVGaWxlID0gZnMucmVhZEpzb25TeW5jKCdtaWdyYXRlLmpzb24nLCB7XG4gICAgICAgIGVuY29kaW5nOiAndXRmLTgnLFxuICAgICAgfSk7XG4gICAgICBjb25zdCBzb3VyY2VFbnYgPSAobWlncmF0ZUZpbGUuU291cmNlIGFzIHN0cmluZykuc3BsaXQoJzonKTtcbiAgICAgIGlmIChcbiAgICAgICAgc291cmNlRW52WzBdID09PSAnbG9jYWxmaWxlJyB8fFxuICAgICAgICAoc291cmNlRW52WzRdID09PSBlbnZpcm9ubWVudC5hY2NvdW50ICYmIHNvdXJjZUVudlszXSA9PT0gZW52aXJvbm1lbnQucmVnaW9uKVxuICAgICAgKSB7XG4gICAgICAgIHJldHVybiBtaWdyYXRlRmlsZS5SZXNvdXJjZXM7XG4gICAgICB9XG4gICAgfSBjYXRjaCAoZSkge1xuICAgICAgLy8gTm90aGluZyB0byBkb1xuICAgIH1cblxuICAgIHJldHVybiB1bmRlZmluZWQ7XG4gIH1cbn1cblxuIl19
@@ -1,11 +0,0 @@
1
- /**
2
- * Sends a response to a prompt to stdin
3
- * When using this in tests, call just before the prompt runs.
4
- *
5
- * @example
6
- * ```ts
7
- * sendResponse('y');
8
- * await prompt('Confirm (y/n)?');
9
- * ```
10
- */
11
- export declare function sendResponse(res: string, delay?: number): void;
@@ -1,22 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.sendResponse = sendResponse;
4
- /**
5
- * Sends a response to a prompt to stdin
6
- * When using this in tests, call just before the prompt runs.
7
- *
8
- * @example
9
- * ```ts
10
- * sendResponse('y');
11
- * await prompt('Confirm (y/n)?');
12
- * ```
13
- */
14
- function sendResponse(res, delay = 0) {
15
- if (!delay) {
16
- setImmediate(() => process.stdin.emit('data', `${res}\n`));
17
- }
18
- else {
19
- setTimeout(() => process.stdin.emit('data', `${res}\n`), delay);
20
- }
21
- }
22
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvbXB0cy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbInByb21wdHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFVQSxvQ0FNQztBQWhCRDs7Ozs7Ozs7O0dBU0c7QUFDSCxTQUFnQixZQUFZLENBQUMsR0FBVyxFQUFFLEtBQUssR0FBRyxDQUFDO0lBQ2pELElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQztRQUNYLFlBQVksQ0FBQyxHQUFHLEVBQUUsQ0FBQyxPQUFPLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxHQUFHLElBQUksQ0FBQyxDQUFDLENBQUM7SUFDN0QsQ0FBQztTQUFNLENBQUM7UUFDTixVQUFVLENBQUMsR0FBRyxFQUFFLENBQUMsT0FBTyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLEdBQUcsR0FBRyxJQUFJLENBQUMsRUFBRSxLQUFLLENBQUMsQ0FBQztJQUNsRSxDQUFDO0FBQ0gsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogU2VuZHMgYSByZXNwb25zZSB0byBhIHByb21wdCB0byBzdGRpblxuICogV2hlbiB1c2luZyB0aGlzIGluIHRlc3RzLCBjYWxsIGp1c3QgYmVmb3JlIHRoZSBwcm9tcHQgcnVucy5cbiAqXG4gKiBAZXhhbXBsZVxuICogYGBgdHNcbiAqIHNlbmRSZXNwb25zZSgneScpO1xuICogYXdhaXQgcHJvbXB0KCdDb25maXJtICh5L24pPycpO1xuICogYGBgXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBzZW5kUmVzcG9uc2UocmVzOiBzdHJpbmcsIGRlbGF5ID0gMCkge1xuICBpZiAoIWRlbGF5KSB7XG4gICAgc2V0SW1tZWRpYXRlKCgpID0+IHByb2Nlc3Muc3RkaW4uZW1pdCgnZGF0YScsIGAke3Jlc31cXG5gKSk7XG4gIH0gZWxzZSB7XG4gICAgc2V0VGltZW91dCgoKSA9PiBwcm9jZXNzLnN0ZGluLmVtaXQoJ2RhdGEnLCBgJHtyZXN9XFxuYCksIGRlbGF5KTtcbiAgfVxufVxuIl19
@@ -1,364 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- jest.mock('promptly', () => {
4
- return {
5
- ...jest.requireActual('promptly'),
6
- confirm: jest.fn(),
7
- prompt: jest.fn(),
8
- };
9
- });
10
- const client_cloudformation_1 = require("@aws-sdk/client-cloudformation");
11
- const promptly = require("promptly");
12
- const util_1 = require("./util");
13
- const mock_sdk_1 = require("./util/mock-sdk");
14
- const deployments_1 = require("../lib/api/deployments");
15
- const import_1 = require("../lib/import");
16
- const promptlyConfirm = promptly.confirm;
17
- const promptlyPrompt = promptly.prompt;
18
- function stackWithQueue(props) {
19
- return (0, util_1.testStack)({
20
- stackName: 'StackWithQueue',
21
- template: {
22
- Resources: {
23
- MyQueue: {
24
- Type: 'AWS::SQS::Queue',
25
- Properties: props,
26
- },
27
- },
28
- },
29
- });
30
- }
31
- const STACK_WITH_QUEUE = stackWithQueue({});
32
- const STACK_WITH_NAMED_QUEUE = stackWithQueue({
33
- QueueName: 'TheQueueName',
34
- });
35
- function stackWithGlobalTable(props) {
36
- return (0, util_1.testStack)({
37
- stackName: 'StackWithTable',
38
- template: {
39
- Resources: {
40
- MyTable: {
41
- Type: 'AWS::DynamoDB::GlobalTable',
42
- Properties: props,
43
- },
44
- },
45
- },
46
- });
47
- }
48
- function stackWithKeySigningKey(props) {
49
- return (0, util_1.testStack)({
50
- stackName: 'StackWithKSK',
51
- template: {
52
- Resources: {
53
- MyKSK: {
54
- Type: 'AWS::Route53::KeySigningKey',
55
- Properties: props,
56
- },
57
- },
58
- },
59
- });
60
- }
61
- let sdkProvider;
62
- let deployments;
63
- beforeEach(() => {
64
- (0, mock_sdk_1.restoreSdkMocksToDefault)();
65
- jest.resetAllMocks();
66
- sdkProvider = new mock_sdk_1.MockSdkProvider();
67
- deployments = new deployments_1.Deployments({ sdkProvider });
68
- });
69
- test('discovers importable resources', async () => {
70
- givenCurrentStack(STACK_WITH_QUEUE.stackName, {
71
- Resources: {},
72
- });
73
- const importer = new import_1.ResourceImporter(STACK_WITH_QUEUE, deployments);
74
- const { additions } = await importer.discoverImportableResources();
75
- expect(additions).toEqual([
76
- expect.objectContaining({
77
- logicalId: 'MyQueue',
78
- }),
79
- ]);
80
- });
81
- test('by default, its an error if there are non-addition changes in the template', async () => {
82
- givenCurrentStack(STACK_WITH_QUEUE.stackName, {
83
- Resources: {
84
- SomethingThatDisappeared: {
85
- Type: 'AWS::S3::Bucket',
86
- },
87
- },
88
- });
89
- const importer = new import_1.ResourceImporter(STACK_WITH_QUEUE, deployments);
90
- await expect(importer.discoverImportableResources()).rejects.toThrow(/No resource updates or deletes/);
91
- // But the error can be silenced
92
- await expect(importer.discoverImportableResources(true)).resolves.toBeTruthy();
93
- });
94
- test('asks human for resource identifiers', async () => {
95
- // GIVEN
96
- givenCurrentStack(STACK_WITH_QUEUE.stackName, { Resources: {} });
97
- const importer = new import_1.ResourceImporter(STACK_WITH_QUEUE, deployments);
98
- const { additions } = await importer.discoverImportableResources();
99
- // WHEN
100
- promptlyPrompt.mockResolvedValue('TheQueueName');
101
- const importable = await importer.askForResourceIdentifiers(additions);
102
- // THEN
103
- expect(importable.resourceMap).toEqual({
104
- MyQueue: {
105
- QueueName: 'TheQueueName',
106
- },
107
- });
108
- expect(importable.importResources).toEqual([
109
- expect.objectContaining({
110
- logicalId: 'MyQueue',
111
- }),
112
- ]);
113
- });
114
- test('asks human to confirm automic import if identifier is in template', async () => {
115
- // GIVEN
116
- givenCurrentStack(STACK_WITH_NAMED_QUEUE.stackName, { Resources: {} });
117
- const importer = new import_1.ResourceImporter(STACK_WITH_NAMED_QUEUE, deployments);
118
- const { additions } = await importer.discoverImportableResources();
119
- // WHEN
120
- promptlyConfirm.mockResolvedValue(true);
121
- const importable = await importer.askForResourceIdentifiers(additions);
122
- // THEN
123
- expect(importable.resourceMap).toEqual({
124
- MyQueue: {
125
- QueueName: 'TheQueueName',
126
- },
127
- });
128
- expect(importable.importResources).toEqual([
129
- expect.objectContaining({
130
- logicalId: 'MyQueue',
131
- }),
132
- ]);
133
- });
134
- test('asks human to confirm automic import if identifier is in template', async () => {
135
- // GIVEN
136
- givenCurrentStack(STACK_WITH_QUEUE.stackName, { Resources: {} });
137
- const importer = new import_1.ResourceImporter(STACK_WITH_QUEUE, deployments);
138
- const { additions } = await importer.discoverImportableResources();
139
- const importMap = {
140
- importResources: additions,
141
- resourceMap: {
142
- MyQueue: { QueueName: 'TheQueueName' },
143
- },
144
- };
145
- // WHEN
146
- await importer.importResourcesFromMap(importMap, {});
147
- expect(mock_sdk_1.mockCloudFormationClient).toHaveReceivedCommandWith(client_cloudformation_1.CreateChangeSetCommand, {
148
- ChangeSetName: expect.any(String),
149
- StackName: STACK_WITH_QUEUE.stackName,
150
- TemplateBody: expect.any(String),
151
- ChangeSetType: 'IMPORT',
152
- ResourcesToImport: [
153
- {
154
- LogicalResourceId: 'MyQueue',
155
- ResourceIdentifier: { QueueName: 'TheQueueName' },
156
- ResourceType: 'AWS::SQS::Queue',
157
- },
158
- ],
159
- });
160
- });
161
- test('importing resources from migrate strips cdk metadata and outputs', async () => {
162
- // GIVEN
163
- const MyQueue = {
164
- Type: 'AWS::SQS::Queue',
165
- Properties: {},
166
- };
167
- const stack = {
168
- stackName: 'StackWithQueue',
169
- template: {
170
- Resources: {
171
- MyQueue,
172
- CDKMetadata: {
173
- Type: 'AWS::CDK::Metadata',
174
- Properties: {
175
- Analytics: 'exists',
176
- },
177
- },
178
- },
179
- Outputs: {
180
- Output: {
181
- Description: 'There is an output',
182
- Value: 'OutputValue',
183
- },
184
- },
185
- },
186
- };
187
- givenCurrentStack(stack.stackName, stack);
188
- const importer = new import_1.ResourceImporter((0, util_1.testStack)(stack), deployments);
189
- const migrateMap = [
190
- {
191
- LogicalResourceId: 'MyQueue',
192
- ResourceIdentifier: { QueueName: 'TheQueueName' },
193
- ResourceType: 'AWS::SQS::Queue',
194
- },
195
- ];
196
- // WHEN
197
- await importer.importResourcesFromMigrate(migrateMap, STACK_WITH_QUEUE.template);
198
- // THEN
199
- expect(mock_sdk_1.mockCloudFormationClient).toHaveReceivedCommandWith(client_cloudformation_1.CreateChangeSetCommand, {
200
- ChangeSetName: expect.any(String),
201
- StackName: STACK_WITH_QUEUE.stackName,
202
- TemplateBody: expect.any(String),
203
- ChangeSetType: 'IMPORT',
204
- ResourcesToImport: [
205
- {
206
- LogicalResourceId: 'MyQueue',
207
- ResourceIdentifier: { QueueName: 'TheQueueName' },
208
- ResourceType: 'AWS::SQS::Queue',
209
- },
210
- ],
211
- });
212
- });
213
- test('only use one identifier if multiple are in template', async () => {
214
- // GIVEN
215
- const stack = stackWithGlobalTable({
216
- TableName: 'TheTableName',
217
- TableArn: 'ThisFieldDoesntExistInReality',
218
- TableStreamArn: 'NorDoesThisOne',
219
- });
220
- // WHEN
221
- promptlyConfirm.mockResolvedValue(true); // Confirm yes/no
222
- await importTemplateFromClean(stack);
223
- // THEN
224
- expect(mock_sdk_1.mockCloudFormationClient).toHaveReceivedCommandWith(client_cloudformation_1.CreateChangeSetCommand, {
225
- ChangeSetName: expect.any(String),
226
- StackName: stack.stackName,
227
- TemplateBody: expect.any(String),
228
- ChangeSetType: 'IMPORT',
229
- ResourcesToImport: [
230
- {
231
- LogicalResourceId: 'MyTable',
232
- ResourceIdentifier: { TableName: 'TheTableName' },
233
- ResourceType: 'AWS::DynamoDB::GlobalTable',
234
- },
235
- ],
236
- });
237
- });
238
- test('only ask user for one identifier if multiple possible ones are possible', async () => {
239
- // GIVEN -- no identifiers in template, so ask user
240
- const stack = stackWithGlobalTable({});
241
- // WHEN
242
- promptlyPrompt.mockResolvedValue('Banana');
243
- const importable = await importTemplateFromClean(stack);
244
- // THEN -- only asked once
245
- expect(promptlyPrompt).toHaveBeenCalledTimes(1);
246
- expect(importable.resourceMap).toEqual({
247
- MyTable: { TableName: 'Banana' },
248
- });
249
- });
250
- test('ask identifier if the value in the template is a CFN intrinsic', async () => {
251
- // GIVEN -- identifier in template is a CFN intrinsic so it doesn't count
252
- const stack = stackWithQueue({
253
- QueueName: { Ref: 'SomeParam' },
254
- });
255
- // WHEN
256
- promptlyPrompt.mockResolvedValue('Banana');
257
- const importable = await importTemplateFromClean(stack);
258
- // THEN
259
- expect(importable.resourceMap).toEqual({
260
- MyQueue: { QueueName: 'Banana' },
261
- });
262
- });
263
- test('take compound identifiers from the template if found', async () => {
264
- // GIVEN
265
- const stack = stackWithKeySigningKey({
266
- HostedZoneId: 'z-123',
267
- Name: 'KeyName',
268
- });
269
- // WHEN
270
- promptlyConfirm.mockResolvedValue(true);
271
- await importTemplateFromClean(stack);
272
- // THEN
273
- expect(mock_sdk_1.mockCloudFormationClient).toHaveReceivedCommandWith(client_cloudformation_1.CreateChangeSetCommand, {
274
- ChangeSetName: expect.any(String),
275
- StackName: stack.stackName,
276
- TemplateBody: expect.any(String),
277
- ChangeSetType: 'IMPORT',
278
- ResourcesToImport: [
279
- {
280
- LogicalResourceId: 'MyKSK',
281
- ResourceIdentifier: { HostedZoneId: 'z-123', Name: 'KeyName' },
282
- ResourceType: 'AWS::Route53::KeySigningKey',
283
- },
284
- ],
285
- });
286
- });
287
- test('ask user for compound identifiers if not found', async () => {
288
- // GIVEN
289
- const stack = stackWithKeySigningKey({});
290
- // WHEN
291
- promptlyPrompt.mockReturnValue('Banana');
292
- await importTemplateFromClean(stack);
293
- // THEN
294
- expect(mock_sdk_1.mockCloudFormationClient).toHaveReceivedCommandWith(client_cloudformation_1.CreateChangeSetCommand, {
295
- ChangeSetName: expect.any(String),
296
- StackName: stack.stackName,
297
- TemplateBody: expect.any(String),
298
- ChangeSetType: 'IMPORT',
299
- ResourcesToImport: [
300
- {
301
- LogicalResourceId: 'MyKSK',
302
- ResourceIdentifier: { HostedZoneId: 'Banana', Name: 'Banana' },
303
- ResourceType: 'AWS::Route53::KeySigningKey',
304
- },
305
- ],
306
- });
307
- });
308
- test('do not ask for second part of compound identifier if the user skips the first', async () => {
309
- // GIVEN
310
- const stack = stackWithKeySigningKey({});
311
- // WHEN
312
- promptlyPrompt.mockReturnValue('');
313
- const importMap = await importTemplateFromClean(stack);
314
- // THEN
315
- expect(importMap.resourceMap).toEqual({});
316
- });
317
- /**
318
- * Do a full import cycle with the given stack template
319
- */
320
- async function importTemplateFromClean(stack) {
321
- givenCurrentStack(stack.stackName, { Resources: {} });
322
- const importer = new import_1.ResourceImporter(stack, deployments);
323
- const { additions } = await importer.discoverImportableResources();
324
- const importable = await importer.askForResourceIdentifiers(additions);
325
- await importer.importResourcesFromMap(importable, {});
326
- return importable;
327
- }
328
- function givenCurrentStack(stackName, template) {
329
- mock_sdk_1.mockCloudFormationClient.on(client_cloudformation_1.DescribeStacksCommand).resolves({
330
- Stacks: [
331
- {
332
- StackName: stackName,
333
- CreationTime: new Date(),
334
- StackStatus: client_cloudformation_1.StackStatus.UPDATE_COMPLETE,
335
- StackStatusReason: 'It is magic',
336
- Outputs: [],
337
- },
338
- ],
339
- });
340
- mock_sdk_1.mockCloudFormationClient.on(client_cloudformation_1.GetTemplateCommand).resolves({
341
- TemplateBody: JSON.stringify(template),
342
- });
343
- mock_sdk_1.mockCloudFormationClient.on(client_cloudformation_1.GetTemplateSummaryCommand).resolves({
344
- ResourceIdentifierSummaries: [
345
- {
346
- ResourceType: 'AWS::SQS::Queue',
347
- ResourceIdentifiers: ['QueueName'],
348
- },
349
- {
350
- ResourceType: 'AWS::DynamoDB::GlobalTable',
351
- ResourceIdentifiers: ['TableName', 'TableArn', 'TableStreamArn'],
352
- },
353
- {
354
- ResourceType: 'AWS::Route53::KeySigningKey',
355
- ResourceIdentifiers: ['HostedZoneId,Name'],
356
- },
357
- ],
358
- });
359
- mock_sdk_1.mockCloudFormationClient.on(client_cloudformation_1.DescribeChangeSetCommand).resolves({
360
- Status: client_cloudformation_1.StackStatus.CREATE_COMPLETE,
361
- Changes: [],
362
- });
363
- }
364
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"import.test.js","sourceRoot":"","sources":["import.test.ts"],"names":[],"mappings":"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","sourcesContent":["jest.mock('promptly', () => {\n  return {\n    ...jest.requireActual('promptly'),\n    confirm: jest.fn(),\n    prompt: jest.fn(),\n  };\n});\n\nimport {\n  CreateChangeSetCommand,\n  DescribeChangeSetCommand,\n  DescribeStacksCommand,\n  GetTemplateCommand,\n  GetTemplateSummaryCommand,\n  StackStatus,\n} from '@aws-sdk/client-cloudformation';\nimport * as promptly from 'promptly';\nimport { testStack } from './util';\nimport { MockSdkProvider, mockCloudFormationClient, restoreSdkMocksToDefault } from './util/mock-sdk';\nimport { Deployments } from '../lib/api/deployments';\nimport { ResourceImporter, ImportMap } from '../lib/import';\n\nconst promptlyConfirm = promptly.confirm as jest.Mock;\nconst promptlyPrompt = promptly.prompt as jest.Mock;\n\nfunction stackWithQueue(props: Record<string, unknown>) {\n  return testStack({\n    stackName: 'StackWithQueue',\n    template: {\n      Resources: {\n        MyQueue: {\n          Type: 'AWS::SQS::Queue',\n          Properties: props,\n        },\n      },\n    },\n  });\n}\n\nconst STACK_WITH_QUEUE = stackWithQueue({});\n\nconst STACK_WITH_NAMED_QUEUE = stackWithQueue({\n  QueueName: 'TheQueueName',\n});\n\nfunction stackWithGlobalTable(props: Record<string, unknown>) {\n  return testStack({\n    stackName: 'StackWithTable',\n    template: {\n      Resources: {\n        MyTable: {\n          Type: 'AWS::DynamoDB::GlobalTable',\n          Properties: props,\n        },\n      },\n    },\n  });\n}\n\nfunction stackWithKeySigningKey(props: Record<string, unknown>) {\n  return testStack({\n    stackName: 'StackWithKSK',\n    template: {\n      Resources: {\n        MyKSK: {\n          Type: 'AWS::Route53::KeySigningKey',\n          Properties: props,\n        },\n      },\n    },\n  });\n}\n\nlet sdkProvider: MockSdkProvider;\nlet deployments: Deployments;\nbeforeEach(() => {\n  restoreSdkMocksToDefault();\n  jest.resetAllMocks();\n  sdkProvider = new MockSdkProvider();\n  deployments = new Deployments({ sdkProvider });\n});\n\ntest('discovers importable resources', async () => {\n  givenCurrentStack(STACK_WITH_QUEUE.stackName, {\n    Resources: {},\n  });\n\n  const importer = new ResourceImporter(STACK_WITH_QUEUE, deployments);\n  const { additions } = await importer.discoverImportableResources();\n  expect(additions).toEqual([\n    expect.objectContaining({\n      logicalId: 'MyQueue',\n    }),\n  ]);\n});\n\ntest('by default, its an error if there are non-addition changes in the template', async () => {\n  givenCurrentStack(STACK_WITH_QUEUE.stackName, {\n    Resources: {\n      SomethingThatDisappeared: {\n        Type: 'AWS::S3::Bucket',\n      },\n    },\n  });\n\n  const importer = new ResourceImporter(STACK_WITH_QUEUE, deployments);\n  await expect(importer.discoverImportableResources()).rejects.toThrow(/No resource updates or deletes/);\n\n  // But the error can be silenced\n  await expect(importer.discoverImportableResources(true)).resolves.toBeTruthy();\n});\n\ntest('asks human for resource identifiers', async () => {\n  // GIVEN\n  givenCurrentStack(STACK_WITH_QUEUE.stackName, { Resources: {} });\n  const importer = new ResourceImporter(STACK_WITH_QUEUE, deployments);\n  const { additions } = await importer.discoverImportableResources();\n\n  // WHEN\n  promptlyPrompt.mockResolvedValue('TheQueueName');\n  const importable = await importer.askForResourceIdentifiers(additions);\n\n  // THEN\n  expect(importable.resourceMap).toEqual({\n    MyQueue: {\n      QueueName: 'TheQueueName',\n    },\n  });\n  expect(importable.importResources).toEqual([\n    expect.objectContaining({\n      logicalId: 'MyQueue',\n    }),\n  ]);\n});\n\ntest('asks human to confirm automic import if identifier is in template', async () => {\n  // GIVEN\n  givenCurrentStack(STACK_WITH_NAMED_QUEUE.stackName, { Resources: {} });\n  const importer = new ResourceImporter(STACK_WITH_NAMED_QUEUE, deployments);\n  const { additions } = await importer.discoverImportableResources();\n\n  // WHEN\n  promptlyConfirm.mockResolvedValue(true);\n  const importable = await importer.askForResourceIdentifiers(additions);\n\n  // THEN\n  expect(importable.resourceMap).toEqual({\n    MyQueue: {\n      QueueName: 'TheQueueName',\n    },\n  });\n  expect(importable.importResources).toEqual([\n    expect.objectContaining({\n      logicalId: 'MyQueue',\n    }),\n  ]);\n});\n\ntest('asks human to confirm automic import if identifier is in template', async () => {\n  // GIVEN\n  givenCurrentStack(STACK_WITH_QUEUE.stackName, { Resources: {} });\n  const importer = new ResourceImporter(STACK_WITH_QUEUE, deployments);\n  const { additions } = await importer.discoverImportableResources();\n  const importMap: ImportMap = {\n    importResources: additions,\n    resourceMap: {\n      MyQueue: { QueueName: 'TheQueueName' },\n    },\n  };\n\n  // WHEN\n  await importer.importResourcesFromMap(importMap, {});\n\n  expect(mockCloudFormationClient).toHaveReceivedCommandWith(CreateChangeSetCommand, {\n    ChangeSetName: expect.any(String),\n    StackName: STACK_WITH_QUEUE.stackName,\n    TemplateBody: expect.any(String),\n    ChangeSetType: 'IMPORT',\n    ResourcesToImport: [\n      {\n        LogicalResourceId: 'MyQueue',\n        ResourceIdentifier: { QueueName: 'TheQueueName' },\n        ResourceType: 'AWS::SQS::Queue',\n      },\n    ],\n  });\n});\n\ntest('importing resources from migrate strips cdk metadata and outputs', async () => {\n  // GIVEN\n\n  const MyQueue = {\n    Type: 'AWS::SQS::Queue',\n    Properties: {},\n  };\n  const stack = {\n    stackName: 'StackWithQueue',\n    template: {\n      Resources: {\n        MyQueue,\n        CDKMetadata: {\n          Type: 'AWS::CDK::Metadata',\n          Properties: {\n            Analytics: 'exists',\n          },\n        },\n      },\n      Outputs: {\n        Output: {\n          Description: 'There is an output',\n          Value: 'OutputValue',\n        },\n      },\n    },\n  };\n\n  givenCurrentStack(stack.stackName, stack);\n  const importer = new ResourceImporter(testStack(stack), deployments);\n  const migrateMap = [\n    {\n      LogicalResourceId: 'MyQueue',\n      ResourceIdentifier: { QueueName: 'TheQueueName' },\n      ResourceType: 'AWS::SQS::Queue',\n    },\n  ];\n\n  // WHEN\n  await importer.importResourcesFromMigrate(migrateMap, STACK_WITH_QUEUE.template);\n\n  // THEN\n  expect(mockCloudFormationClient).toHaveReceivedCommandWith(CreateChangeSetCommand, {\n    ChangeSetName: expect.any(String),\n    StackName: STACK_WITH_QUEUE.stackName,\n    TemplateBody: expect.any(String),\n    ChangeSetType: 'IMPORT',\n    ResourcesToImport: [\n      {\n        LogicalResourceId: 'MyQueue',\n        ResourceIdentifier: { QueueName: 'TheQueueName' },\n        ResourceType: 'AWS::SQS::Queue',\n      },\n    ],\n  });\n});\n\ntest('only use one identifier if multiple are in template', async () => {\n  // GIVEN\n  const stack = stackWithGlobalTable({\n    TableName: 'TheTableName',\n    TableArn: 'ThisFieldDoesntExistInReality',\n    TableStreamArn: 'NorDoesThisOne',\n  });\n\n  // WHEN\n  promptlyConfirm.mockResolvedValue(true); // Confirm yes/no\n  await importTemplateFromClean(stack);\n\n  // THEN\n  expect(mockCloudFormationClient).toHaveReceivedCommandWith(CreateChangeSetCommand, {\n    ChangeSetName: expect.any(String),\n    StackName: stack.stackName,\n    TemplateBody: expect.any(String),\n    ChangeSetType: 'IMPORT',\n    ResourcesToImport: [\n      {\n        LogicalResourceId: 'MyTable',\n        ResourceIdentifier: { TableName: 'TheTableName' },\n        ResourceType: 'AWS::DynamoDB::GlobalTable',\n      },\n    ],\n  });\n});\n\ntest('only ask user for one identifier if multiple possible ones are possible', async () => {\n  // GIVEN -- no identifiers in template, so ask user\n  const stack = stackWithGlobalTable({});\n\n  // WHEN\n  promptlyPrompt.mockResolvedValue('Banana');\n  const importable = await importTemplateFromClean(stack);\n\n  // THEN -- only asked once\n  expect(promptlyPrompt).toHaveBeenCalledTimes(1);\n  expect(importable.resourceMap).toEqual({\n    MyTable: { TableName: 'Banana' },\n  });\n});\n\ntest('ask identifier if the value in the template is a CFN intrinsic', async () => {\n  // GIVEN -- identifier in template is a CFN intrinsic so it doesn't count\n  const stack = stackWithQueue({\n    QueueName: { Ref: 'SomeParam' },\n  });\n\n  // WHEN\n  promptlyPrompt.mockResolvedValue('Banana');\n  const importable = await importTemplateFromClean(stack);\n\n  // THEN\n  expect(importable.resourceMap).toEqual({\n    MyQueue: { QueueName: 'Banana' },\n  });\n});\n\ntest('take compound identifiers from the template if found', async () => {\n  // GIVEN\n  const stack = stackWithKeySigningKey({\n    HostedZoneId: 'z-123',\n    Name: 'KeyName',\n  });\n\n  // WHEN\n  promptlyConfirm.mockResolvedValue(true);\n  await importTemplateFromClean(stack);\n\n  // THEN\n  expect(mockCloudFormationClient).toHaveReceivedCommandWith(CreateChangeSetCommand, {\n    ChangeSetName: expect.any(String),\n    StackName: stack.stackName,\n    TemplateBody: expect.any(String),\n    ChangeSetType: 'IMPORT',\n    ResourcesToImport: [\n      {\n        LogicalResourceId: 'MyKSK',\n        ResourceIdentifier: { HostedZoneId: 'z-123', Name: 'KeyName' },\n        ResourceType: 'AWS::Route53::KeySigningKey',\n      },\n    ],\n  });\n});\n\ntest('ask user for compound identifiers if not found', async () => {\n  // GIVEN\n  const stack = stackWithKeySigningKey({});\n\n  // WHEN\n  promptlyPrompt.mockReturnValue('Banana');\n  await importTemplateFromClean(stack);\n\n  // THEN\n  expect(mockCloudFormationClient).toHaveReceivedCommandWith(CreateChangeSetCommand, {\n    ChangeSetName: expect.any(String),\n    StackName: stack.stackName,\n    TemplateBody: expect.any(String),\n    ChangeSetType: 'IMPORT',\n    ResourcesToImport: [\n      {\n        LogicalResourceId: 'MyKSK',\n        ResourceIdentifier: { HostedZoneId: 'Banana', Name: 'Banana' },\n        ResourceType: 'AWS::Route53::KeySigningKey',\n      },\n    ],\n  });\n});\n\ntest('do not ask for second part of compound identifier if the user skips the first', async () => {\n  // GIVEN\n  const stack = stackWithKeySigningKey({});\n\n  // WHEN\n  promptlyPrompt.mockReturnValue('');\n  const importMap = await importTemplateFromClean(stack);\n\n  // THEN\n  expect(importMap.resourceMap).toEqual({});\n});\n\n/**\n * Do a full import cycle with the given stack template\n */\nasync function importTemplateFromClean(stack: ReturnType<typeof testStack>) {\n  givenCurrentStack(stack.stackName, { Resources: {} });\n  const importer = new ResourceImporter(stack, deployments);\n  const { additions } = await importer.discoverImportableResources();\n  const importable = await importer.askForResourceIdentifiers(additions);\n  await importer.importResourcesFromMap(importable, {});\n  return importable;\n}\n\nfunction givenCurrentStack(stackName: string, template: any) {\n  mockCloudFormationClient.on(DescribeStacksCommand).resolves({\n    Stacks: [\n      {\n        StackName: stackName,\n        CreationTime: new Date(),\n        StackStatus: StackStatus.UPDATE_COMPLETE,\n        StackStatusReason: 'It is magic',\n        Outputs: [],\n      },\n    ],\n  });\n  mockCloudFormationClient.on(GetTemplateCommand).resolves({\n    TemplateBody: JSON.stringify(template),\n  });\n  mockCloudFormationClient.on(GetTemplateSummaryCommand).resolves({\n    ResourceIdentifierSummaries: [\n      {\n        ResourceType: 'AWS::SQS::Queue',\n        ResourceIdentifiers: ['QueueName'],\n      },\n      {\n        ResourceType: 'AWS::DynamoDB::GlobalTable',\n        ResourceIdentifiers: ['TableName', 'TableArn', 'TableStreamArn'],\n      },\n      {\n        ResourceType: 'AWS::Route53::KeySigningKey',\n        ResourceIdentifiers: ['HostedZoneId,Name'],\n      },\n    ],\n  });\n\n  mockCloudFormationClient.on(DescribeChangeSetCommand).resolves({\n    Status: StackStatus.CREATE_COMPLETE,\n    Changes: [],\n  });\n}\n"]}