aws-cdk 2.10.0 → 2.11.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.
@@ -4,15 +4,33 @@ jest.mock('../../lib/api/deploy-stack');
4
4
  const cloudformation_deployments_1 = require("../../lib/api/cloudformation-deployments");
5
5
  const deploy_stack_1 = require("../../lib/api/deploy-stack");
6
6
  const toolkit_info_1 = require("../../lib/api/toolkit-info");
7
+ const cloudformation_1 = require("../../lib/api/util/cloudformation");
7
8
  const util_1 = require("../util");
8
9
  const mock_sdk_1 = require("../util/mock-sdk");
10
+ const fake_cloudformation_stack_1 = require("./fake-cloudformation-stack");
9
11
  let sdkProvider;
10
12
  let deployments;
11
13
  let mockToolkitInfoLookup;
14
+ let currentCfnStackResources;
15
+ let numberOfTimesListStackResourcesWasCalled;
12
16
  beforeEach(() => {
13
17
  jest.resetAllMocks();
14
18
  sdkProvider = new mock_sdk_1.MockSdkProvider();
15
19
  deployments = new cloudformation_deployments_1.CloudFormationDeployments({ sdkProvider });
20
+ numberOfTimesListStackResourcesWasCalled = 0;
21
+ currentCfnStackResources = {};
22
+ sdkProvider.stubCloudFormation({
23
+ listStackResources: ({ StackName: stackName }) => {
24
+ numberOfTimesListStackResourcesWasCalled++;
25
+ const stackResources = currentCfnStackResources[stackName];
26
+ if (!stackResources) {
27
+ throw new Error(`Stack with id ${stackName} does not exist`);
28
+ }
29
+ return {
30
+ StackResourceSummaries: stackResources,
31
+ };
32
+ },
33
+ });
16
34
  toolkit_info_1.ToolkitInfo.lookup = mockToolkitInfoLookup = jest.fn().mockResolvedValue(toolkit_info_1.ToolkitInfo.bootstrapStackNotFoundInfo(sdkProvider.sdk));
17
35
  });
18
36
  function mockSuccessfulBootstrapStackLookup(props) {
@@ -125,4 +143,628 @@ test('if toolkit stack cannot be found but SSM parameter name is present deploym
125
143
  });
126
144
  expect(requestedParameterName).toEqual('/some/parameter');
127
145
  });
128
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudformation-deployments.test.js","sourceRoot":"","sources":["cloudformation-deployments.test.ts"],"names":[],"mappings":"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","sourcesContent":["jest.mock('../../lib/api/deploy-stack');\n\nimport { CloudFormationDeployments } from '../../lib/api/cloudformation-deployments';\nimport { deployStack } from '../../lib/api/deploy-stack';\nimport { ToolkitInfo } from '../../lib/api/toolkit-info';\nimport { testStack } from '../util';\nimport { mockBootstrapStack, MockSdkProvider } from '../util/mock-sdk';\n\nlet sdkProvider: MockSdkProvider;\nlet deployments: CloudFormationDeployments;\nlet mockToolkitInfoLookup: jest.Mock;\nbeforeEach(() => {\n  jest.resetAllMocks();\n  sdkProvider = new MockSdkProvider();\n  deployments = new CloudFormationDeployments({ sdkProvider });\n  ToolkitInfo.lookup = mockToolkitInfoLookup = jest.fn().mockResolvedValue(ToolkitInfo.bootstrapStackNotFoundInfo(sdkProvider.sdk));\n});\n\nfunction mockSuccessfulBootstrapStackLookup(props?: Record<string, any>) {\n  const outputs = {\n    BucketName: 'BUCKET_NAME',\n    BucketDomainName: 'BUCKET_ENDPOINT',\n    BootstrapVersion: '1',\n    ...props,\n  };\n\n  const fakeStack = mockBootstrapStack(sdkProvider.sdk, {\n    Outputs: Object.entries(outputs).map(([k, v]) => ({\n      OutputKey: k,\n      OutputValue: `${v}`,\n    })),\n  });\n\n  mockToolkitInfoLookup.mockResolvedValue(ToolkitInfo.fromStack(fakeStack, sdkProvider.sdk));\n}\n\ntest('passes through hotswap=true to deployStack()', async () => {\n  // WHEN\n  await deployments.deployStack({\n    stack: testStack({\n      stackName: 'boop',\n    }),\n    hotswap: true,\n  });\n\n  // THEN\n  expect(deployStack).toHaveBeenCalledWith(expect.objectContaining({\n    hotswap: true,\n  }));\n});\n\ntest('placeholders are substituted in CloudFormation execution role', async () => {\n  await deployments.deployStack({\n    stack: testStack({\n      stackName: 'boop',\n      properties: {\n        cloudFormationExecutionRoleArn: 'bloop:${AWS::Region}:${AWS::AccountId}',\n      },\n    }),\n  });\n\n  expect(deployStack).toHaveBeenCalledWith(expect.objectContaining({\n    roleArn: 'bloop:here:123456789012',\n  }));\n});\n\ntest('role with placeholders is assumed if assumerole is given', async () => {\n  const mockForEnvironment = jest.fn().mockImplementation(() => { return { sdk: sdkProvider.sdk }; });\n  sdkProvider.forEnvironment = mockForEnvironment;\n\n  await deployments.deployStack({\n    stack: testStack({\n      stackName: 'boop',\n      properties: {\n        assumeRoleArn: 'bloop:${AWS::Region}:${AWS::AccountId}',\n      },\n    }),\n  });\n\n  expect(mockForEnvironment).toHaveBeenCalledWith(expect.anything(), expect.anything(), expect.objectContaining({\n    assumeRoleArn: 'bloop:here:123456789012',\n  }));\n});\n\ntest('deployment fails if bootstrap stack is missing', async () => {\n  await expect(deployments.deployStack({\n    stack: testStack({\n      stackName: 'boop',\n      properties: {\n        assumeRoleArn: 'bloop:${AWS::Region}:${AWS::AccountId}',\n        requiresBootstrapStackVersion: 99,\n      },\n    }),\n  })).rejects.toThrow(/requires a bootstrap stack/);\n});\n\ntest('deployment fails if bootstrap stack is too old', async () => {\n  mockSuccessfulBootstrapStackLookup({\n    BootstrapVersion: 5,\n  });\n\n  await expect(deployments.deployStack({\n    stack: testStack({\n      stackName: 'boop',\n      properties: {\n        assumeRoleArn: 'bloop:${AWS::Region}:${AWS::AccountId}',\n        requiresBootstrapStackVersion: 99,\n      },\n    }),\n  })).rejects.toThrow(/requires bootstrap stack version '99', found '5'/);\n});\n\ntest('if toolkit stack cannot be found but SSM parameter name is present deployment succeeds', async () => {\n  // FIXME: Mocking a successful bootstrap stack lookup here should not be necessary.\n  // This should fail and return a placeholder failure object.\n  mockSuccessfulBootstrapStackLookup({\n    BootstrapVersion: 2,\n  });\n\n  let requestedParameterName: string;\n  sdkProvider.stubSSM({\n    getParameter(request) {\n      requestedParameterName = request.Name;\n      return {\n        Parameter: {\n          Value: '99',\n        },\n      };\n    },\n  });\n\n  await deployments.deployStack({\n    stack: testStack({\n      stackName: 'boop',\n      properties: {\n        assumeRoleArn: 'bloop:${AWS::Region}:${AWS::AccountId}',\n        requiresBootstrapStackVersion: 99,\n        bootstrapStackVersionSsmParameter: '/some/parameter',\n      },\n    }),\n  });\n\n  expect(requestedParameterName!).toEqual('/some/parameter');\n});\n"]}
146
+ test('readCurrentTemplateWithNestedStacks() can handle non-Resources in the template', async () => {
147
+ const cfnStack = new fake_cloudformation_stack_1.FakeCloudformationStack({
148
+ stackName: 'ParentOfStackWithOutputAndParameter',
149
+ stackId: 'StackId',
150
+ });
151
+ cloudformation_1.CloudFormationStack.lookup = (async (_, stackName) => {
152
+ switch (stackName) {
153
+ case 'ParentOfStackWithOutputAndParameter':
154
+ cfnStack.template = async () => ({
155
+ Resources: {
156
+ NestedStack: {
157
+ Type: 'AWS::CloudFormation::Stack',
158
+ Properties: {
159
+ TemplateURL: 'https://www.magic-url.com',
160
+ },
161
+ Metadata: {
162
+ 'aws:asset:path': 'one-output-one-param-stack.nested.template.json',
163
+ },
164
+ },
165
+ },
166
+ });
167
+ break;
168
+ case 'NestedStack':
169
+ cfnStack.template = async () => ({
170
+ Resources: {
171
+ NestedResource: {
172
+ Type: 'AWS::Something',
173
+ Properties: {
174
+ Property: 'old-value',
175
+ },
176
+ },
177
+ },
178
+ Parameters: {
179
+ NestedParam: {
180
+ Type: 'String',
181
+ },
182
+ },
183
+ Outputs: {
184
+ NestedOutput: {
185
+ Value: {
186
+ Ref: 'NestedResource',
187
+ },
188
+ },
189
+ },
190
+ });
191
+ break;
192
+ default:
193
+ throw new Error('unknown stack name ' + stackName + ' found');
194
+ }
195
+ return cfnStack;
196
+ });
197
+ const rootStack = util_1.testStack({
198
+ stackName: 'ParentOfStackWithOutputAndParameter',
199
+ template: {
200
+ Resources: {
201
+ NestedStack: {
202
+ Type: 'AWS::CloudFormation::Stack',
203
+ Properties: {
204
+ TemplateURL: 'https://www.magic-url.com',
205
+ },
206
+ Metadata: {
207
+ 'aws:asset:path': 'one-output-one-param-stack.nested.template.json',
208
+ },
209
+ },
210
+ },
211
+ },
212
+ });
213
+ pushStackResourceSummaries('ParentOfStackWithOutputAndParameter', stackSummaryOf('NestedStack', 'AWS::CloudFormation::Stack', 'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/NestedStack/abcd'));
214
+ // WHEN
215
+ const deployedTemplate = await deployments.readCurrentTemplateWithNestedStacks(rootStack);
216
+ // THEN
217
+ expect(deployedTemplate).toEqual({
218
+ Resources: {
219
+ NestedStack: {
220
+ Type: 'AWS::CloudFormation::Stack',
221
+ Properties: {
222
+ TemplateURL: 'https://www.magic-url.com',
223
+ NestedTemplate: {
224
+ Resources: {
225
+ NestedResource: {
226
+ Type: 'AWS::Something',
227
+ Properties: {
228
+ Property: 'old-value',
229
+ },
230
+ },
231
+ },
232
+ Outputs: {
233
+ NestedOutput: {
234
+ Value: {
235
+ Ref: 'NestedResource',
236
+ },
237
+ },
238
+ },
239
+ Parameters: {
240
+ NestedParam: {
241
+ Type: 'String',
242
+ },
243
+ },
244
+ },
245
+ },
246
+ Metadata: {
247
+ 'aws:asset:path': 'one-output-one-param-stack.nested.template.json',
248
+ },
249
+ },
250
+ },
251
+ });
252
+ expect(rootStack.template).toEqual({
253
+ Resources: {
254
+ NestedStack: {
255
+ Type: 'AWS::CloudFormation::Stack',
256
+ Properties: {
257
+ TemplateURL: 'https://www.magic-url.com',
258
+ NestedTemplate: {
259
+ Resources: {
260
+ NestedResource: {
261
+ Type: 'AWS::Something',
262
+ Properties: {
263
+ Property: 'new-value',
264
+ },
265
+ },
266
+ },
267
+ Outputs: {
268
+ NestedOutput: {
269
+ Value: {
270
+ Ref: 'NestedResource',
271
+ },
272
+ },
273
+ },
274
+ Parameters: {
275
+ NestedParam: {
276
+ Type: 'Number',
277
+ },
278
+ },
279
+ },
280
+ },
281
+ Metadata: {
282
+ 'aws:asset:path': 'one-output-one-param-stack.nested.template.json',
283
+ },
284
+ },
285
+ },
286
+ });
287
+ });
288
+ test('readCurrentTemplateWithNestedStacks() with a 3-level nested + sibling structure works', async () => {
289
+ const cfnStack = new fake_cloudformation_stack_1.FakeCloudformationStack({
290
+ stackName: 'MultiLevelRoot',
291
+ stackId: 'StackId',
292
+ });
293
+ cloudformation_1.CloudFormationStack.lookup = (async (_, stackName) => {
294
+ switch (stackName) {
295
+ case 'MultiLevelRoot':
296
+ cfnStack.template = async () => ({
297
+ Resources: {
298
+ NestedStack: {
299
+ Type: 'AWS::CloudFormation::Stack',
300
+ Properties: {
301
+ TemplateURL: 'https://www.magic-url.com',
302
+ },
303
+ Metadata: {
304
+ 'aws:asset:path': 'one-resource-two-stacks-stack.nested.template.json',
305
+ },
306
+ },
307
+ },
308
+ });
309
+ break;
310
+ case 'NestedStack':
311
+ cfnStack.template = async () => ({
312
+ Resources: {
313
+ SomeResource: {
314
+ Type: 'AWS::Something',
315
+ Properties: {
316
+ Property: 'old-value',
317
+ },
318
+ },
319
+ GrandChildStackA: {
320
+ Type: 'AWS::CloudFormation::Stack',
321
+ Properties: {
322
+ TemplateURL: 'https://www.magic-url.com',
323
+ },
324
+ Metadata: {
325
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
326
+ },
327
+ },
328
+ GrandChildStackB: {
329
+ Type: 'AWS::CloudFormation::Stack',
330
+ Properties: {
331
+ TemplateURL: 'https://www.magic-url.com',
332
+ },
333
+ Metadata: {
334
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
335
+ },
336
+ },
337
+ },
338
+ });
339
+ break;
340
+ case 'GrandChildStackA':
341
+ cfnStack.template = async () => ({
342
+ Resources: {
343
+ SomeResource: {
344
+ Type: 'AWS::Something',
345
+ Properties: {
346
+ Property: 'old-value',
347
+ },
348
+ },
349
+ },
350
+ });
351
+ break;
352
+ case 'GrandChildStackB':
353
+ cfnStack.template = async () => ({
354
+ Resources: {
355
+ SomeResource: {
356
+ Type: 'AWS::Something',
357
+ Properties: {
358
+ Property: 'old-value',
359
+ },
360
+ },
361
+ },
362
+ });
363
+ break;
364
+ default:
365
+ throw new Error('unknown stack name ' + stackName + ' found in cloudformation-deployments.test.ts');
366
+ }
367
+ return cfnStack;
368
+ });
369
+ const rootStack = util_1.testStack({
370
+ stackName: 'MultiLevelRoot',
371
+ template: {
372
+ Resources: {
373
+ NestedStack: {
374
+ Type: 'AWS::CloudFormation::Stack',
375
+ Properties: {
376
+ TemplateURL: 'https://www.magic-url.com',
377
+ },
378
+ Metadata: {
379
+ 'aws:asset:path': 'one-resource-two-stacks-stack.nested.template.json',
380
+ },
381
+ },
382
+ },
383
+ },
384
+ });
385
+ pushStackResourceSummaries('MultiLevelRoot', stackSummaryOf('NestedStack', 'AWS::CloudFormation::Stack', 'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/NestedStack/abcd'));
386
+ pushStackResourceSummaries('NestedStack', stackSummaryOf('GrandChildStackA', 'AWS::CloudFormation::Stack', 'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/GrandChildStackA/abcd'), stackSummaryOf('GrandChildStackB', 'AWS::CloudFormation::Stack', 'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/GrandChildStackB/abcd'));
387
+ pushStackResourceSummaries('GrandChildStackA', stackSummaryOf('NestedStack', 'AWS::CloudFormation::Stack', 'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/GrandChildA/abcd'));
388
+ pushStackResourceSummaries('GrandChildStackB', stackSummaryOf('NestedStack', 'AWS::CloudFormation::Stack', 'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/GrandChildB/abcd'));
389
+ // WHEN
390
+ const deployedTemplate = await deployments.readCurrentTemplateWithNestedStacks(rootStack);
391
+ // THEN
392
+ expect(deployedTemplate).toEqual({
393
+ Resources: {
394
+ NestedStack: {
395
+ Type: 'AWS::CloudFormation::Stack',
396
+ Properties: {
397
+ TemplateURL: 'https://www.magic-url.com',
398
+ NestedTemplate: {
399
+ Resources: {
400
+ GrandChildStackA: {
401
+ Type: 'AWS::CloudFormation::Stack',
402
+ Properties: {
403
+ TemplateURL: 'https://www.magic-url.com',
404
+ NestedTemplate: {
405
+ Resources: {
406
+ SomeResource: {
407
+ Type: 'AWS::Something',
408
+ Properties: {
409
+ Property: 'old-value',
410
+ },
411
+ },
412
+ },
413
+ },
414
+ },
415
+ Metadata: {
416
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
417
+ },
418
+ },
419
+ GrandChildStackB: {
420
+ Type: 'AWS::CloudFormation::Stack',
421
+ Properties: {
422
+ TemplateURL: 'https://www.magic-url.com',
423
+ NestedTemplate: {
424
+ Resources: {
425
+ SomeResource: {
426
+ Type: 'AWS::Something',
427
+ Properties: {
428
+ Property: 'old-value',
429
+ },
430
+ },
431
+ },
432
+ },
433
+ },
434
+ Metadata: {
435
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
436
+ },
437
+ },
438
+ SomeResource: {
439
+ Type: 'AWS::Something',
440
+ Properties: {
441
+ Property: 'old-value',
442
+ },
443
+ },
444
+ },
445
+ },
446
+ },
447
+ Metadata: {
448
+ 'aws:asset:path': 'one-resource-two-stacks-stack.nested.template.json',
449
+ },
450
+ },
451
+ },
452
+ });
453
+ expect(rootStack.template).toEqual({
454
+ Resources: {
455
+ NestedStack: {
456
+ Type: 'AWS::CloudFormation::Stack',
457
+ Properties: {
458
+ TemplateURL: 'https://www.magic-url.com',
459
+ NestedTemplate: {
460
+ Resources: {
461
+ GrandChildStackA: {
462
+ Type: 'AWS::CloudFormation::Stack',
463
+ Properties: {
464
+ TemplateURL: 'https://www.magic-url.com',
465
+ NestedTemplate: {
466
+ Resources: {
467
+ SomeResource: {
468
+ Type: 'AWS::Something',
469
+ Properties: {
470
+ Property: 'new-value',
471
+ },
472
+ },
473
+ },
474
+ },
475
+ },
476
+ Metadata: {
477
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
478
+ },
479
+ },
480
+ GrandChildStackB: {
481
+ Type: 'AWS::CloudFormation::Stack',
482
+ Properties: {
483
+ TemplateURL: 'https://www.magic-url.com',
484
+ NestedTemplate: {
485
+ Resources: {
486
+ SomeResource: {
487
+ Type: 'AWS::Something',
488
+ Properties: {
489
+ Property: 'new-value',
490
+ },
491
+ },
492
+ },
493
+ },
494
+ },
495
+ Metadata: {
496
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
497
+ },
498
+ },
499
+ SomeResource: {
500
+ Type: 'AWS::Something',
501
+ Properties: {
502
+ Property: 'new-value',
503
+ },
504
+ },
505
+ },
506
+ },
507
+ },
508
+ Metadata: {
509
+ 'aws:asset:path': 'one-resource-two-stacks-stack.nested.template.json',
510
+ },
511
+ },
512
+ },
513
+ });
514
+ });
515
+ test('readCurrentTemplateWithNestedStacks() on an undeployed parent stack with an (also undeployed) nested stack works', async () => {
516
+ // GIVEN
517
+ const cfnStack = new fake_cloudformation_stack_1.FakeCloudformationStack({
518
+ stackName: 'UndeployedParent',
519
+ stackId: 'StackId',
520
+ });
521
+ cloudformation_1.CloudFormationStack.lookup = (async (_cfn, _stackName) => {
522
+ cfnStack.template = async () => ({});
523
+ return cfnStack;
524
+ });
525
+ const rootStack = util_1.testStack({
526
+ stackName: 'UndeployedParent',
527
+ template: {
528
+ Resources: {
529
+ NestedStack: {
530
+ Type: 'AWS::CloudFormation::Stack',
531
+ Properties: {
532
+ TemplateURL: 'https://www.magic-url.com',
533
+ },
534
+ Metadata: {
535
+ 'aws:asset:path': 'one-resource-one-stack-stack.nested.template.json',
536
+ },
537
+ },
538
+ },
539
+ },
540
+ });
541
+ // WHEN
542
+ const deployedTemplate = await deployments.readCurrentTemplateWithNestedStacks(rootStack);
543
+ // THEN
544
+ expect(deployedTemplate).toEqual({
545
+ Resources: {
546
+ NestedStack: {
547
+ Type: 'AWS::CloudFormation::Stack',
548
+ Properties: {
549
+ NestedTemplate: {
550
+ Resources: {
551
+ NestedStack: {
552
+ Type: 'AWS::CloudFormation::Stack',
553
+ Properties: {
554
+ NestedTemplate: {},
555
+ },
556
+ },
557
+ },
558
+ },
559
+ },
560
+ },
561
+ },
562
+ });
563
+ });
564
+ test('readCurrentTemplateWithNestedStacks() caches calls to listStackResources()', async () => {
565
+ // GIVEN
566
+ const cfnStack = new fake_cloudformation_stack_1.FakeCloudformationStack({
567
+ stackName: 'CachingRoot',
568
+ stackId: 'StackId',
569
+ });
570
+ cloudformation_1.CloudFormationStack.lookup = (async (_cfn, _stackName) => {
571
+ cfnStack.template = async () => ({
572
+ Resources: {
573
+ NestedStackA: {
574
+ Type: 'AWS::CloudFormation::Stack',
575
+ Properties: {
576
+ TemplateURL: 'https://www.magic-url.com',
577
+ },
578
+ Metadata: {
579
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
580
+ },
581
+ },
582
+ NestedStackB: {
583
+ Type: 'AWS::CloudFormation::Stack',
584
+ Properties: {
585
+ TemplateURL: 'https://www.magic-url.com',
586
+ },
587
+ Metadata: {
588
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
589
+ },
590
+ },
591
+ },
592
+ });
593
+ return cfnStack;
594
+ });
595
+ const rootStack = util_1.testStack({
596
+ stackName: 'CachingRoot',
597
+ template: {
598
+ Resources: {
599
+ NestedStackA: {
600
+ Type: 'AWS::CloudFormation::Stack',
601
+ Properties: {
602
+ TemplateURL: 'https://www.magic-url.com',
603
+ },
604
+ Metadata: {
605
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
606
+ },
607
+ },
608
+ NestedStackB: {
609
+ Type: 'AWS::CloudFormation::Stack',
610
+ Properties: {
611
+ TemplateURL: 'https://www.magic-url.com',
612
+ },
613
+ Metadata: {
614
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
615
+ },
616
+ },
617
+ },
618
+ },
619
+ });
620
+ pushStackResourceSummaries('CachingRoot', stackSummaryOf('NestedStackA', 'AWS::CloudFormation::Stack', 'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/one-resource-stack/abcd'), stackSummaryOf('NestedStackB', 'AWS::CloudFormation::Stack', 'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/one-resource-stack/abcd'));
621
+ // WHEN
622
+ await deployments.readCurrentTemplateWithNestedStacks(rootStack);
623
+ // THEN
624
+ expect(numberOfTimesListStackResourcesWasCalled).toEqual(1);
625
+ });
626
+ test('readCurrentTemplateWithNestedStacks() succesfully ignores stacks without metadata', async () => {
627
+ // GIVEN
628
+ const cfnStack = new fake_cloudformation_stack_1.FakeCloudformationStack({
629
+ stackName: 'MetadataRoot',
630
+ stackId: 'StackId',
631
+ });
632
+ cloudformation_1.CloudFormationStack.lookup = (async (_, stackName) => {
633
+ if (stackName === 'MetadataRoot') {
634
+ cfnStack.template = async () => ({
635
+ Resources: {
636
+ WithMetadata: {
637
+ Type: 'AWS::CloudFormation::Stack',
638
+ Properties: {
639
+ TemplateURL: 'https://www.magic-url.com',
640
+ },
641
+ Metadata: {
642
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
643
+ },
644
+ },
645
+ },
646
+ });
647
+ }
648
+ else {
649
+ cfnStack.template = async () => ({
650
+ Resources: {
651
+ SomeResource: {
652
+ Type: 'AWS::Something',
653
+ Properties: {
654
+ Property: 'old-value',
655
+ },
656
+ },
657
+ },
658
+ });
659
+ }
660
+ return cfnStack;
661
+ });
662
+ const rootStack = util_1.testStack({
663
+ stackName: 'MetadataRoot',
664
+ template: {
665
+ Resources: {
666
+ WithoutMetadata: {
667
+ Properties: {
668
+ TemplateURL: 'https://www.magic-url.com',
669
+ },
670
+ Type: 'AWS::CloudFormation::Stack',
671
+ },
672
+ WithEmptyMetadata: {
673
+ Type: 'AWS::CloudFormation::Stack',
674
+ Properties: {
675
+ TemplateURL: 'https://www.magic-url.com',
676
+ },
677
+ Metadata: {},
678
+ },
679
+ WithMetadata: {
680
+ Type: 'AWS::CloudFormation::Stack',
681
+ Properties: {
682
+ TemplateURL: 'https://www.magic-url.com',
683
+ },
684
+ Metadata: {
685
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
686
+ },
687
+ },
688
+ },
689
+ },
690
+ });
691
+ pushStackResourceSummaries('MetadataRoot', stackSummaryOf('WithMetadata', 'AWS::CloudFormation::Stack', 'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/one-resource-stack/abcd'));
692
+ // WHEN
693
+ const deployedTemplate = await deployments.readCurrentTemplateWithNestedStacks(rootStack);
694
+ // THEN
695
+ expect(deployedTemplate).toEqual({
696
+ Resources: {
697
+ WithMetadata: {
698
+ Type: 'AWS::CloudFormation::Stack',
699
+ Properties: {
700
+ TemplateURL: 'https://www.magic-url.com',
701
+ NestedTemplate: {
702
+ Resources: {
703
+ SomeResource: {
704
+ Type: 'AWS::Something',
705
+ Properties: {
706
+ Property: 'old-value',
707
+ },
708
+ },
709
+ },
710
+ },
711
+ },
712
+ Metadata: {
713
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
714
+ },
715
+ },
716
+ },
717
+ });
718
+ expect(rootStack.template).toEqual({
719
+ Resources: {
720
+ WithoutMetadata: {
721
+ Type: 'AWS::CloudFormation::Stack',
722
+ Properties: {
723
+ TemplateURL: 'https://www.magic-url.com',
724
+ },
725
+ },
726
+ WithEmptyMetadata: {
727
+ Type: 'AWS::CloudFormation::Stack',
728
+ Properties: {
729
+ TemplateURL: 'https://www.magic-url.com',
730
+ },
731
+ Metadata: {},
732
+ },
733
+ WithMetadata: {
734
+ Type: 'AWS::CloudFormation::Stack',
735
+ Properties: {
736
+ TemplateURL: 'https://www.magic-url.com',
737
+ NestedTemplate: {
738
+ Resources: {
739
+ SomeResource: {
740
+ Type: 'AWS::Something',
741
+ Properties: {
742
+ Property: 'new-value',
743
+ },
744
+ },
745
+ },
746
+ },
747
+ },
748
+ Metadata: {
749
+ 'aws:asset:path': 'one-resource-stack.nested.template.json',
750
+ },
751
+ },
752
+ },
753
+ });
754
+ });
755
+ function pushStackResourceSummaries(stackName, ...items) {
756
+ if (!currentCfnStackResources[stackName]) {
757
+ currentCfnStackResources[stackName] = [];
758
+ }
759
+ currentCfnStackResources[stackName].push(...items);
760
+ }
761
+ function stackSummaryOf(logicalId, resourceType, physicalResourceId) {
762
+ return {
763
+ LogicalResourceId: logicalId,
764
+ PhysicalResourceId: physicalResourceId,
765
+ ResourceType: resourceType,
766
+ ResourceStatus: 'CREATE_COMPLETE',
767
+ LastUpdatedTimestamp: new Date(),
768
+ };
769
+ }
770
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloudformation-deployments.test.js","sourceRoot":"","sources":["cloudformation-deployments.test.ts"],"names":[],"mappings":"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","sourcesContent":["jest.mock('../../lib/api/deploy-stack');\n\nimport { CloudFormation } from 'aws-sdk';\nimport { CloudFormationDeployments } from '../../lib/api/cloudformation-deployments';\nimport { deployStack } from '../../lib/api/deploy-stack';\nimport { ToolkitInfo } from '../../lib/api/toolkit-info';\nimport { CloudFormationStack } from '../../lib/api/util/cloudformation';\nimport { testStack } from '../util';\nimport { mockBootstrapStack, MockSdkProvider } from '../util/mock-sdk';\nimport { FakeCloudformationStack } from './fake-cloudformation-stack';\n\nlet sdkProvider: MockSdkProvider;\nlet deployments: CloudFormationDeployments;\nlet mockToolkitInfoLookup: jest.Mock;\nlet currentCfnStackResources: { [key: string]: CloudFormation.StackResourceSummary[] };\nlet numberOfTimesListStackResourcesWasCalled: number;\nbeforeEach(() => {\n  jest.resetAllMocks();\n  sdkProvider = new MockSdkProvider();\n  deployments = new CloudFormationDeployments({ sdkProvider });\n\n  numberOfTimesListStackResourcesWasCalled = 0;\n  currentCfnStackResources = {};\n  sdkProvider.stubCloudFormation({\n    listStackResources: ({ StackName: stackName }) => {\n      numberOfTimesListStackResourcesWasCalled++;\n      const stackResources = currentCfnStackResources[stackName];\n      if (!stackResources) {\n        throw new Error(`Stack with id ${stackName} does not exist`);\n      }\n      return {\n        StackResourceSummaries: stackResources,\n      };\n    },\n  });\n\n  ToolkitInfo.lookup = mockToolkitInfoLookup = jest.fn().mockResolvedValue(ToolkitInfo.bootstrapStackNotFoundInfo(sdkProvider.sdk));\n});\n\nfunction mockSuccessfulBootstrapStackLookup(props?: Record<string, any>) {\n  const outputs = {\n    BucketName: 'BUCKET_NAME',\n    BucketDomainName: 'BUCKET_ENDPOINT',\n    BootstrapVersion: '1',\n    ...props,\n  };\n\n  const fakeStack = mockBootstrapStack(sdkProvider.sdk, {\n    Outputs: Object.entries(outputs).map(([k, v]) => ({\n      OutputKey: k,\n      OutputValue: `${v}`,\n    })),\n  });\n\n  mockToolkitInfoLookup.mockResolvedValue(ToolkitInfo.fromStack(fakeStack, sdkProvider.sdk));\n}\n\ntest('passes through hotswap=true to deployStack()', async () => {\n  // WHEN\n  await deployments.deployStack({\n    stack: testStack({\n      stackName: 'boop',\n    }),\n    hotswap: true,\n  });\n\n  // THEN\n  expect(deployStack).toHaveBeenCalledWith(expect.objectContaining({\n    hotswap: true,\n  }));\n});\n\ntest('placeholders are substituted in CloudFormation execution role', async () => {\n  await deployments.deployStack({\n    stack: testStack({\n      stackName: 'boop',\n      properties: {\n        cloudFormationExecutionRoleArn: 'bloop:${AWS::Region}:${AWS::AccountId}',\n      },\n    }),\n  });\n\n  expect(deployStack).toHaveBeenCalledWith(expect.objectContaining({\n    roleArn: 'bloop:here:123456789012',\n  }));\n});\n\ntest('role with placeholders is assumed if assumerole is given', async () => {\n  const mockForEnvironment = jest.fn().mockImplementation(() => { return { sdk: sdkProvider.sdk }; });\n  sdkProvider.forEnvironment = mockForEnvironment;\n\n  await deployments.deployStack({\n    stack: testStack({\n      stackName: 'boop',\n      properties: {\n        assumeRoleArn: 'bloop:${AWS::Region}:${AWS::AccountId}',\n      },\n    }),\n  });\n\n  expect(mockForEnvironment).toHaveBeenCalledWith(expect.anything(), expect.anything(), expect.objectContaining({\n    assumeRoleArn: 'bloop:here:123456789012',\n  }));\n});\n\ntest('deployment fails if bootstrap stack is missing', async () => {\n  await expect(deployments.deployStack({\n    stack: testStack({\n      stackName: 'boop',\n      properties: {\n        assumeRoleArn: 'bloop:${AWS::Region}:${AWS::AccountId}',\n        requiresBootstrapStackVersion: 99,\n      },\n    }),\n  })).rejects.toThrow(/requires a bootstrap stack/);\n});\n\ntest('deployment fails if bootstrap stack is too old', async () => {\n  mockSuccessfulBootstrapStackLookup({\n    BootstrapVersion: 5,\n  });\n\n  await expect(deployments.deployStack({\n    stack: testStack({\n      stackName: 'boop',\n      properties: {\n        assumeRoleArn: 'bloop:${AWS::Region}:${AWS::AccountId}',\n        requiresBootstrapStackVersion: 99,\n      },\n    }),\n  })).rejects.toThrow(/requires bootstrap stack version '99', found '5'/);\n});\n\ntest('if toolkit stack cannot be found but SSM parameter name is present deployment succeeds', async () => {\n  // FIXME: Mocking a successful bootstrap stack lookup here should not be necessary.\n  // This should fail and return a placeholder failure object.\n  mockSuccessfulBootstrapStackLookup({\n    BootstrapVersion: 2,\n  });\n\n  let requestedParameterName: string;\n  sdkProvider.stubSSM({\n    getParameter(request) {\n      requestedParameterName = request.Name;\n      return {\n        Parameter: {\n          Value: '99',\n        },\n      };\n    },\n  });\n\n  await deployments.deployStack({\n    stack: testStack({\n      stackName: 'boop',\n      properties: {\n        assumeRoleArn: 'bloop:${AWS::Region}:${AWS::AccountId}',\n        requiresBootstrapStackVersion: 99,\n        bootstrapStackVersionSsmParameter: '/some/parameter',\n      },\n    }),\n  });\n\n  expect(requestedParameterName!).toEqual('/some/parameter');\n});\n\ntest('readCurrentTemplateWithNestedStacks() can handle non-Resources in the template', async () => {\n  const cfnStack = new FakeCloudformationStack({\n    stackName: 'ParentOfStackWithOutputAndParameter',\n    stackId: 'StackId',\n  });\n  CloudFormationStack.lookup = (async (_, stackName: string) => {\n    switch (stackName) {\n      case 'ParentOfStackWithOutputAndParameter':\n        cfnStack.template = async () => ({\n          Resources: {\n            NestedStack: {\n              Type: 'AWS::CloudFormation::Stack',\n              Properties: {\n                TemplateURL: 'https://www.magic-url.com',\n              },\n              Metadata: {\n                'aws:asset:path': 'one-output-one-param-stack.nested.template.json',\n              },\n            },\n          },\n        });\n        break;\n\n      case 'NestedStack':\n        cfnStack.template = async () => ({\n          Resources: {\n            NestedResource: {\n              Type: 'AWS::Something',\n              Properties: {\n                Property: 'old-value',\n              },\n            },\n          },\n          Parameters: {\n            NestedParam: {\n              Type: 'String',\n            },\n          },\n          Outputs: {\n            NestedOutput: {\n              Value: {\n                Ref: 'NestedResource',\n              },\n            },\n          },\n        });\n        break;\n\n      default:\n        throw new Error('unknown stack name ' + stackName + ' found');\n    }\n\n    return cfnStack;\n  });\n\n  const rootStack = testStack({\n    stackName: 'ParentOfStackWithOutputAndParameter',\n    template: {\n      Resources: {\n        NestedStack: {\n          Type: 'AWS::CloudFormation::Stack',\n          Properties: {\n            TemplateURL: 'https://www.magic-url.com',\n          },\n          Metadata: {\n            'aws:asset:path': 'one-output-one-param-stack.nested.template.json',\n          },\n        },\n      },\n    },\n  });\n\n  pushStackResourceSummaries('ParentOfStackWithOutputAndParameter',\n    stackSummaryOf('NestedStack', 'AWS::CloudFormation::Stack',\n      'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/NestedStack/abcd',\n    ),\n  );\n\n  // WHEN\n  const deployedTemplate = await deployments.readCurrentTemplateWithNestedStacks(rootStack);\n\n  // THEN\n  expect(deployedTemplate).toEqual({\n    Resources: {\n      NestedStack: {\n        Type: 'AWS::CloudFormation::Stack',\n        Properties: {\n          TemplateURL: 'https://www.magic-url.com',\n          NestedTemplate: {\n            Resources: {\n              NestedResource: {\n                Type: 'AWS::Something',\n                Properties: {\n                  Property: 'old-value',\n                },\n              },\n            },\n            Outputs: {\n              NestedOutput: {\n                Value: {\n                  Ref: 'NestedResource',\n                },\n              },\n            },\n            Parameters: {\n              NestedParam: {\n                Type: 'String',\n              },\n            },\n          },\n        },\n        Metadata: {\n          'aws:asset:path': 'one-output-one-param-stack.nested.template.json',\n        },\n      },\n    },\n  });\n\n  expect(rootStack.template).toEqual({\n    Resources: {\n      NestedStack: {\n        Type: 'AWS::CloudFormation::Stack',\n        Properties: {\n          TemplateURL: 'https://www.magic-url.com',\n          NestedTemplate: {\n            Resources: {\n              NestedResource: {\n                Type: 'AWS::Something',\n                Properties: {\n                  Property: 'new-value',\n                },\n              },\n            },\n            Outputs: {\n              NestedOutput: {\n                Value: {\n                  Ref: 'NestedResource',\n                },\n              },\n            },\n            Parameters: {\n              NestedParam: {\n                Type: 'Number',\n              },\n            },\n          },\n        },\n        Metadata: {\n          'aws:asset:path': 'one-output-one-param-stack.nested.template.json',\n        },\n      },\n    },\n  });\n});\n\ntest('readCurrentTemplateWithNestedStacks() with a 3-level nested + sibling structure works', async () => {\n  const cfnStack = new FakeCloudformationStack({\n    stackName: 'MultiLevelRoot',\n    stackId: 'StackId',\n  });\n  CloudFormationStack.lookup = (async (_, stackName: string) => {\n    switch (stackName) {\n      case 'MultiLevelRoot':\n        cfnStack.template = async () => ({\n          Resources: {\n            NestedStack: {\n              Type: 'AWS::CloudFormation::Stack',\n              Properties: {\n                TemplateURL: 'https://www.magic-url.com',\n              },\n              Metadata: {\n                'aws:asset:path': 'one-resource-two-stacks-stack.nested.template.json',\n              },\n            },\n          },\n        });\n        break;\n\n      case 'NestedStack':\n        cfnStack.template = async () => ({\n          Resources: {\n            SomeResource: {\n              Type: 'AWS::Something',\n              Properties: {\n                Property: 'old-value',\n              },\n            },\n            GrandChildStackA: {\n              Type: 'AWS::CloudFormation::Stack',\n              Properties: {\n                TemplateURL: 'https://www.magic-url.com',\n              },\n              Metadata: {\n                'aws:asset:path': 'one-resource-stack.nested.template.json',\n              },\n            },\n            GrandChildStackB: {\n              Type: 'AWS::CloudFormation::Stack',\n              Properties: {\n                TemplateURL: 'https://www.magic-url.com',\n              },\n              Metadata: {\n                'aws:asset:path': 'one-resource-stack.nested.template.json',\n              },\n            },\n          },\n        });\n        break;\n\n      case 'GrandChildStackA':\n        cfnStack.template = async () => ({\n          Resources: {\n            SomeResource: {\n              Type: 'AWS::Something',\n              Properties: {\n                Property: 'old-value',\n              },\n            },\n          },\n        });\n        break;\n\n      case 'GrandChildStackB':\n        cfnStack.template = async () => ({\n          Resources: {\n            SomeResource: {\n              Type: 'AWS::Something',\n              Properties: {\n                Property: 'old-value',\n              },\n            },\n          },\n        });\n        break;\n\n      default:\n        throw new Error('unknown stack name ' + stackName + ' found in cloudformation-deployments.test.ts');\n    }\n\n    return cfnStack;\n  });\n\n  const rootStack = testStack({\n    stackName: 'MultiLevelRoot',\n    template: {\n      Resources: {\n        NestedStack: {\n          Type: 'AWS::CloudFormation::Stack',\n          Properties: {\n            TemplateURL: 'https://www.magic-url.com',\n          },\n          Metadata: {\n            'aws:asset:path': 'one-resource-two-stacks-stack.nested.template.json',\n          },\n        },\n      },\n    },\n  });\n\n  pushStackResourceSummaries('MultiLevelRoot',\n    stackSummaryOf('NestedStack', 'AWS::CloudFormation::Stack',\n      'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/NestedStack/abcd',\n    ),\n  );\n  pushStackResourceSummaries('NestedStack',\n    stackSummaryOf('GrandChildStackA', 'AWS::CloudFormation::Stack',\n      'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/GrandChildStackA/abcd',\n    ),\n    stackSummaryOf('GrandChildStackB', 'AWS::CloudFormation::Stack',\n      'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/GrandChildStackB/abcd',\n    ),\n  );\n  pushStackResourceSummaries('GrandChildStackA',\n    stackSummaryOf('NestedStack', 'AWS::CloudFormation::Stack',\n      'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/GrandChildA/abcd',\n    ),\n  );\n  pushStackResourceSummaries('GrandChildStackB',\n    stackSummaryOf('NestedStack', 'AWS::CloudFormation::Stack',\n      'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/GrandChildB/abcd',\n    ),\n  );\n\n  // WHEN\n  const deployedTemplate = await deployments.readCurrentTemplateWithNestedStacks(rootStack);\n\n  // THEN\n  expect(deployedTemplate).toEqual({\n    Resources: {\n      NestedStack: {\n        Type: 'AWS::CloudFormation::Stack',\n        Properties: {\n          TemplateURL: 'https://www.magic-url.com',\n          NestedTemplate: {\n            Resources: {\n              GrandChildStackA: {\n                Type: 'AWS::CloudFormation::Stack',\n                Properties: {\n                  TemplateURL: 'https://www.magic-url.com',\n                  NestedTemplate: {\n                    Resources: {\n                      SomeResource: {\n                        Type: 'AWS::Something',\n                        Properties: {\n                          Property: 'old-value',\n                        },\n                      },\n                    },\n                  },\n                },\n                Metadata: {\n                  'aws:asset:path': 'one-resource-stack.nested.template.json',\n                },\n              },\n              GrandChildStackB: {\n                Type: 'AWS::CloudFormation::Stack',\n                Properties: {\n                  TemplateURL: 'https://www.magic-url.com',\n                  NestedTemplate: {\n                    Resources: {\n                      SomeResource: {\n                        Type: 'AWS::Something',\n                        Properties: {\n                          Property: 'old-value',\n                        },\n                      },\n                    },\n                  },\n                },\n                Metadata: {\n                  'aws:asset:path': 'one-resource-stack.nested.template.json',\n                },\n              },\n              SomeResource: {\n                Type: 'AWS::Something',\n                Properties: {\n                  Property: 'old-value',\n                },\n              },\n            },\n          },\n        },\n        Metadata: {\n          'aws:asset:path': 'one-resource-two-stacks-stack.nested.template.json',\n        },\n      },\n    },\n  });\n\n  expect(rootStack.template).toEqual({\n    Resources: {\n      NestedStack: {\n        Type: 'AWS::CloudFormation::Stack',\n        Properties: {\n          TemplateURL: 'https://www.magic-url.com',\n          NestedTemplate: {\n            Resources: {\n              GrandChildStackA: {\n                Type: 'AWS::CloudFormation::Stack',\n                Properties: {\n                  TemplateURL: 'https://www.magic-url.com',\n                  NestedTemplate: {\n                    Resources: {\n                      SomeResource: {\n                        Type: 'AWS::Something',\n                        Properties: {\n                          Property: 'new-value',\n                        },\n                      },\n                    },\n                  },\n                },\n                Metadata: {\n                  'aws:asset:path': 'one-resource-stack.nested.template.json',\n                },\n              },\n              GrandChildStackB: {\n                Type: 'AWS::CloudFormation::Stack',\n                Properties: {\n                  TemplateURL: 'https://www.magic-url.com',\n                  NestedTemplate: {\n                    Resources: {\n                      SomeResource: {\n                        Type: 'AWS::Something',\n                        Properties: {\n                          Property: 'new-value',\n                        },\n                      },\n                    },\n                  },\n                },\n                Metadata: {\n                  'aws:asset:path': 'one-resource-stack.nested.template.json',\n                },\n              },\n              SomeResource: {\n                Type: 'AWS::Something',\n                Properties: {\n                  Property: 'new-value',\n                },\n              },\n            },\n          },\n        },\n        Metadata: {\n          'aws:asset:path': 'one-resource-two-stacks-stack.nested.template.json',\n        },\n      },\n    },\n  });\n});\n\ntest('readCurrentTemplateWithNestedStacks() on an undeployed parent stack with an (also undeployed) nested stack works', async () => {\n  // GIVEN\n  const cfnStack = new FakeCloudformationStack({\n    stackName: 'UndeployedParent',\n    stackId: 'StackId',\n  });\n  CloudFormationStack.lookup = (async (_cfn, _stackName: string) => {\n    cfnStack.template = async () => ({});\n\n    return cfnStack;\n  });\n  const rootStack = testStack({\n    stackName: 'UndeployedParent',\n    template: {\n      Resources: {\n        NestedStack: {\n          Type: 'AWS::CloudFormation::Stack',\n          Properties: {\n            TemplateURL: 'https://www.magic-url.com',\n          },\n          Metadata: {\n            'aws:asset:path': 'one-resource-one-stack-stack.nested.template.json',\n          },\n        },\n      },\n    },\n  });\n\n  // WHEN\n  const deployedTemplate = await deployments.readCurrentTemplateWithNestedStacks(rootStack);\n\n  // THEN\n  expect(deployedTemplate).toEqual({\n    Resources: {\n      NestedStack: {\n        Type: 'AWS::CloudFormation::Stack',\n        Properties: {\n          NestedTemplate: {\n            Resources: {\n              NestedStack: {\n                Type: 'AWS::CloudFormation::Stack',\n                Properties: {\n                  NestedTemplate: {},\n                },\n              },\n            },\n          },\n        },\n      },\n    },\n  });\n});\n\ntest('readCurrentTemplateWithNestedStacks() caches calls to listStackResources()', async () => {\n  // GIVEN\n  const cfnStack = new FakeCloudformationStack({\n    stackName: 'CachingRoot',\n    stackId: 'StackId',\n  });\n  CloudFormationStack.lookup = (async (_cfn, _stackName: string) => {\n    cfnStack.template = async () => ({\n      Resources:\n      {\n        NestedStackA: {\n          Type: 'AWS::CloudFormation::Stack',\n          Properties: {\n            TemplateURL: 'https://www.magic-url.com',\n          },\n          Metadata: {\n            'aws:asset:path': 'one-resource-stack.nested.template.json',\n          },\n        },\n        NestedStackB: {\n          Type: 'AWS::CloudFormation::Stack',\n          Properties: {\n            TemplateURL: 'https://www.magic-url.com',\n          },\n          Metadata: {\n            'aws:asset:path': 'one-resource-stack.nested.template.json',\n          },\n        },\n      },\n    });\n\n    return cfnStack;\n  });\n\n  const rootStack = testStack({\n    stackName: 'CachingRoot',\n    template: {\n      Resources: {\n        NestedStackA: {\n          Type: 'AWS::CloudFormation::Stack',\n          Properties: {\n            TemplateURL: 'https://www.magic-url.com',\n          },\n          Metadata: {\n            'aws:asset:path': 'one-resource-stack.nested.template.json',\n          },\n        },\n        NestedStackB: {\n          Type: 'AWS::CloudFormation::Stack',\n          Properties: {\n            TemplateURL: 'https://www.magic-url.com',\n          },\n          Metadata: {\n            'aws:asset:path': 'one-resource-stack.nested.template.json',\n          },\n        },\n      },\n    },\n  });\n\n  pushStackResourceSummaries('CachingRoot',\n    stackSummaryOf('NestedStackA', 'AWS::CloudFormation::Stack',\n      'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/one-resource-stack/abcd',\n    ),\n    stackSummaryOf('NestedStackB', 'AWS::CloudFormation::Stack',\n      'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/one-resource-stack/abcd',\n    ),\n  );\n\n  // WHEN\n  await deployments.readCurrentTemplateWithNestedStacks(rootStack);\n\n  // THEN\n  expect(numberOfTimesListStackResourcesWasCalled).toEqual(1);\n});\n\ntest('readCurrentTemplateWithNestedStacks() succesfully ignores stacks without metadata', async () => {\n  // GIVEN\n  const cfnStack = new FakeCloudformationStack({\n    stackName: 'MetadataRoot',\n    stackId: 'StackId',\n  });\n  CloudFormationStack.lookup = (async (_, stackName: string) => {\n    if (stackName === 'MetadataRoot') {\n      cfnStack.template = async () => ({\n        Resources: {\n          WithMetadata: {\n            Type: 'AWS::CloudFormation::Stack',\n            Properties: {\n              TemplateURL: 'https://www.magic-url.com',\n            },\n            Metadata: {\n              'aws:asset:path': 'one-resource-stack.nested.template.json',\n            },\n          },\n        },\n      });\n\n    } else {\n      cfnStack.template = async () => ({\n        Resources: {\n          SomeResource: {\n            Type: 'AWS::Something',\n            Properties: {\n              Property: 'old-value',\n            },\n          },\n        },\n      });\n    }\n\n    return cfnStack;\n  });\n\n  const rootStack = testStack({\n    stackName: 'MetadataRoot',\n    template: {\n      Resources: {\n        WithoutMetadata: {\n          Properties: {\n            TemplateURL: 'https://www.magic-url.com',\n          },\n          Type: 'AWS::CloudFormation::Stack',\n        },\n        WithEmptyMetadata: {\n          Type: 'AWS::CloudFormation::Stack',\n          Properties: {\n            TemplateURL: 'https://www.magic-url.com',\n          },\n          Metadata: {},\n        },\n        WithMetadata: {\n          Type: 'AWS::CloudFormation::Stack',\n          Properties: {\n            TemplateURL: 'https://www.magic-url.com',\n          },\n          Metadata: {\n            'aws:asset:path': 'one-resource-stack.nested.template.json',\n          },\n        },\n      },\n    },\n  });\n\n  pushStackResourceSummaries('MetadataRoot', stackSummaryOf('WithMetadata', 'AWS::CloudFormation::Stack',\n    'arn:aws:cloudformation:bermuda-triangle-1337:123456789012:stack/one-resource-stack/abcd',\n  ));\n\n  // WHEN\n  const deployedTemplate = await deployments.readCurrentTemplateWithNestedStacks(rootStack);\n\n  // THEN\n  expect(deployedTemplate).toEqual({\n    Resources: {\n      WithMetadata: {\n        Type: 'AWS::CloudFormation::Stack',\n        Properties: {\n          TemplateURL: 'https://www.magic-url.com',\n          NestedTemplate: {\n            Resources: {\n              SomeResource: {\n                Type: 'AWS::Something',\n                Properties: {\n                  Property: 'old-value',\n                },\n              },\n            },\n          },\n        },\n        Metadata: {\n          'aws:asset:path': 'one-resource-stack.nested.template.json',\n        },\n      },\n    },\n  });\n\n  expect(rootStack.template).toEqual({\n    Resources: {\n      WithoutMetadata: { // Unchanged\n        Type: 'AWS::CloudFormation::Stack',\n        Properties: {\n          TemplateURL: 'https://www.magic-url.com',\n        },\n      },\n      WithEmptyMetadata: { // Unchanged\n        Type: 'AWS::CloudFormation::Stack',\n        Properties: {\n          TemplateURL: 'https://www.magic-url.com',\n        },\n        Metadata: {},\n      },\n      WithMetadata: { // Changed\n        Type: 'AWS::CloudFormation::Stack',\n        Properties: {\n          TemplateURL: 'https://www.magic-url.com',\n          NestedTemplate: {\n            Resources: {\n              SomeResource: {\n                Type: 'AWS::Something',\n                Properties: {\n                  Property: 'new-value',\n                },\n              },\n            },\n          },\n        },\n        Metadata: {\n          'aws:asset:path': 'one-resource-stack.nested.template.json',\n        },\n      },\n    },\n  });\n});\n\nfunction pushStackResourceSummaries(stackName: string, ...items: CloudFormation.StackResourceSummary[]) {\n  if (!currentCfnStackResources[stackName]) {\n    currentCfnStackResources[stackName] = [];\n  }\n\n  currentCfnStackResources[stackName].push(...items);\n}\n\nfunction stackSummaryOf(logicalId: string, resourceType: string, physicalResourceId: string): CloudFormation.StackResourceSummary {\n  return {\n    LogicalResourceId: logicalId,\n    PhysicalResourceId: physicalResourceId,\n    ResourceType: resourceType,\n    ResourceStatus: 'CREATE_COMPLETE',\n    LastUpdatedTimestamp: new Date(),\n  };\n}\n"]}