/ cloudformation-templates / node_modules / aws-cdk / test / api / deploy-stack.test.js
deploy-stack.test.js
  1  "use strict";
  2  Object.defineProperty(exports, "__esModule", { value: true });
  3  const api_1 = require("../../lib/api");
  4  const util_1 = require("../util");
  5  const mock_sdk_1 = require("../util/mock-sdk");
  6  const FAKE_STACK = util_1.testStack({
  7      stackName: 'withouterrors',
  8  });
  9  const FAKE_STACK_WITH_PARAMETERS = util_1.testStack({
 10      stackName: 'withparameters',
 11      template: {
 12          Parameters: {
 13              HasValue: { Type: 'String' },
 14              HasDefault: { Type: 'String', Default: 'TheDefault' },
 15              OtherParameter: { Type: 'String' },
 16          },
 17      },
 18  });
 19  const FAKE_STACK_TERMINATION_PROTECTION = util_1.testStack({
 20      stackName: 'termination-protection',
 21      template: util_1.DEFAULT_FAKE_TEMPLATE,
 22      terminationProtection: true,
 23  });
 24  let sdk;
 25  let sdkProvider;
 26  let cfnMocks;
 27  beforeEach(() => {
 28      sdkProvider = new mock_sdk_1.MockSdkProvider();
 29      sdk = new mock_sdk_1.MockSdk();
 30      cfnMocks = {
 31          describeStackEvents: jest.fn().mockReturnValue({}),
 32          describeStacks: jest.fn()
 33              // First call, no stacks exist
 34              .mockImplementationOnce(() => ({ Stacks: [] }))
 35              // Second call, stack has been created
 36              .mockImplementationOnce(() => ({
 37              Stacks: [
 38                  {
 39                      StackStatus: 'CREATE_COMPLETE',
 40                      StackStatusReason: 'It is magic',
 41                      EnableTerminationProtection: false,
 42                  },
 43              ],
 44          })),
 45          createChangeSet: jest.fn((_o) => ({})),
 46          deleteChangeSet: jest.fn((_o) => ({})),
 47          describeChangeSet: jest.fn((_o) => ({
 48              Status: 'CREATE_COMPLETE',
 49              Changes: [],
 50          })),
 51          executeChangeSet: jest.fn((_o) => ({})),
 52          deleteStack: jest.fn((_o) => ({})),
 53          getTemplate: jest.fn((_o) => ({ TemplateBody: JSON.stringify(util_1.DEFAULT_FAKE_TEMPLATE) })),
 54          updateTerminationProtection: jest.fn((_o) => ({ StackId: 'stack-id' })),
 55      };
 56      sdk.stubCloudFormation(cfnMocks);
 57  });
 58  function standardDeployStackArguments() {
 59      return {
 60          stack: FAKE_STACK,
 61          sdk,
 62          sdkProvider,
 63          resolvedEnvironment: mock_sdk_1.mockResolvedEnvironment(),
 64          toolkitInfo: api_1.ToolkitInfo.bootstraplessDeploymentsOnly(sdk),
 65      };
 66  }
 67  test('do deploy executable change set with 0 changes', async () => {
 68      // WHEN
 69      const ret = await api_1.deployStack({
 70          ...standardDeployStackArguments(),
 71      });
 72      // THEN
 73      expect(ret.noOp).toBeFalsy();
 74      expect(cfnMocks.executeChangeSet).toHaveBeenCalled();
 75  });
 76  test('correctly passes CFN parameters, ignoring ones with empty values', async () => {
 77      // WHEN
 78      await api_1.deployStack({
 79          ...standardDeployStackArguments(),
 80          parameters: {
 81              A: 'A-value',
 82              B: 'B=value',
 83              C: undefined,
 84              D: '',
 85          },
 86      });
 87      // THEN
 88      expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({
 89          Parameters: [
 90              { ParameterKey: 'A', ParameterValue: 'A-value' },
 91              { ParameterKey: 'B', ParameterValue: 'B=value' },
 92          ],
 93      }));
 94  });
 95  test('reuse previous parameters if requested', async () => {
 96      // GIVEN
 97      givenStackExists({
 98          Parameters: [
 99              { ParameterKey: 'HasValue', ParameterValue: 'TheValue' },
100              { ParameterKey: 'HasDefault', ParameterValue: 'TheOldValue' },
101          ],
102      });
103      // WHEN
104      await api_1.deployStack({
105          ...standardDeployStackArguments(),
106          stack: FAKE_STACK_WITH_PARAMETERS,
107          parameters: {
108              OtherParameter: 'SomeValue',
109          },
110          usePreviousParameters: true,
111      });
112      // THEN
113      expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({
114          Parameters: [
115              { ParameterKey: 'HasValue', UsePreviousValue: true },
116              { ParameterKey: 'HasDefault', UsePreviousValue: true },
117              { ParameterKey: 'OtherParameter', ParameterValue: 'SomeValue' },
118          ],
119      }));
120  });
121  test('do not reuse previous parameters if not requested', async () => {
122      // GIVEN
123      givenStackExists({
124          Parameters: [
125              { ParameterKey: 'HasValue', ParameterValue: 'TheValue' },
126              { ParameterKey: 'HasDefault', ParameterValue: 'TheOldValue' },
127          ],
128      });
129      // WHEN
130      await api_1.deployStack({
131          ...standardDeployStackArguments(),
132          stack: FAKE_STACK_WITH_PARAMETERS,
133          parameters: {
134              HasValue: 'SomeValue',
135              OtherParameter: 'SomeValue',
136          },
137      });
138      // THEN
139      expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({
140          Parameters: [
141              { ParameterKey: 'HasValue', ParameterValue: 'SomeValue' },
142              { ParameterKey: 'OtherParameter', ParameterValue: 'SomeValue' },
143          ],
144      }));
145  });
146  test('throw exception if not enough parameters supplied', async () => {
147      // GIVEN
148      givenStackExists({
149          Parameters: [
150              { ParameterKey: 'HasValue', ParameterValue: 'TheValue' },
151              { ParameterKey: 'HasDefault', ParameterValue: 'TheOldValue' },
152          ],
153      });
154      // WHEN
155      await expect(api_1.deployStack({
156          ...standardDeployStackArguments(),
157          stack: FAKE_STACK_WITH_PARAMETERS,
158          parameters: {
159              OtherParameter: 'SomeValue',
160          },
161      })).rejects.toThrow(/CloudFormation Parameters are missing a value/);
162  });
163  test('deploy is skipped if template did not change', async () => {
164      // GIVEN
165      givenStackExists();
166      // WHEN
167      await api_1.deployStack({
168          ...standardDeployStackArguments(),
169      });
170      // THEN
171      expect(cfnMocks.executeChangeSet).not.toBeCalled();
172  });
173  test('deploy is skipped if parameters are the same', async () => {
174      // GIVEN
175      givenTemplateIs(FAKE_STACK_WITH_PARAMETERS.template);
176      givenStackExists({
177          Parameters: [
178              { ParameterKey: 'HasValue', ParameterValue: 'HasValue' },
179              { ParameterKey: 'HasDefault', ParameterValue: 'HasDefault' },
180              { ParameterKey: 'OtherParameter', ParameterValue: 'OtherParameter' },
181          ],
182      });
183      // WHEN
184      await api_1.deployStack({
185          ...standardDeployStackArguments(),
186          stack: FAKE_STACK_WITH_PARAMETERS,
187          parameters: {},
188          usePreviousParameters: true,
189      });
190      // THEN
191      expect(cfnMocks.createChangeSet).not.toHaveBeenCalled();
192  });
193  test('deploy is not skipped if parameters are different', async () => {
194      // GIVEN
195      givenTemplateIs(FAKE_STACK_WITH_PARAMETERS.template);
196      givenStackExists({
197          Parameters: [
198              { ParameterKey: 'HasValue', ParameterValue: 'HasValue' },
199              { ParameterKey: 'HasDefault', ParameterValue: 'HasDefault' },
200              { ParameterKey: 'OtherParameter', ParameterValue: 'OtherParameter' },
201          ],
202      });
203      // WHEN
204      await api_1.deployStack({
205          ...standardDeployStackArguments(),
206          stack: FAKE_STACK_WITH_PARAMETERS,
207          parameters: {
208              HasValue: 'NewValue',
209          },
210          usePreviousParameters: true,
211      });
212      // THEN
213      expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({
214          Parameters: [
215              { ParameterKey: 'HasValue', ParameterValue: 'NewValue' },
216              { ParameterKey: 'HasDefault', UsePreviousValue: true },
217              { ParameterKey: 'OtherParameter', UsePreviousValue: true },
218          ],
219      }));
220  });
221  test('if existing stack failed to create, it is deleted and recreated', async () => {
222      // GIVEN
223      givenStackExists({ StackStatus: 'ROLLBACK_COMPLETE' }, // This is for the initial check
224      { StackStatus: 'DELETE_COMPLETE' }, // Poll the successful deletion
225      { StackStatus: 'CREATE_COMPLETE' });
226      givenTemplateIs({
227          DifferentThan: 'TheDefault',
228      });
229      // WHEN
230      await api_1.deployStack({
231          ...standardDeployStackArguments(),
232      });
233      // THEN
234      expect(cfnMocks.deleteStack).toHaveBeenCalled();
235      expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({
236          ChangeSetType: 'CREATE',
237      }));
238  });
239  test('if existing stack failed to create, it is deleted and recreated even if the template did not change', async () => {
240      // GIVEN
241      givenStackExists({ StackStatus: 'ROLLBACK_COMPLETE' }, // This is for the initial check
242      { StackStatus: 'DELETE_COMPLETE' }, // Poll the successful deletion
243      { StackStatus: 'CREATE_COMPLETE' });
244      // WHEN
245      await api_1.deployStack({
246          ...standardDeployStackArguments(),
247      });
248      // THEN
249      expect(cfnMocks.deleteStack).toHaveBeenCalled();
250      expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({
251          ChangeSetType: 'CREATE',
252      }));
253  });
254  test('deploy not skipped if template did not change and --force is applied', async () => {
255      // GIVEN
256      givenStackExists();
257      // WHEN
258      await api_1.deployStack({
259          ...standardDeployStackArguments(),
260          force: true,
261      });
262      // THEN
263      expect(cfnMocks.executeChangeSet).toHaveBeenCalled();
264  });
265  test('deploy is skipped if template and tags did not change', async () => {
266      // GIVEN
267      givenStackExists({
268          Tags: [
269              { Key: 'Key1', Value: 'Value1' },
270              { Key: 'Key2', Value: 'Value2' },
271          ],
272      });
273      // WHEN
274      await api_1.deployStack({
275          ...standardDeployStackArguments(),
276          tags: [
277              { Key: 'Key1', Value: 'Value1' },
278              { Key: 'Key2', Value: 'Value2' },
279          ],
280      });
281      // THEN
282      expect(cfnMocks.createChangeSet).not.toBeCalled();
283      expect(cfnMocks.executeChangeSet).not.toBeCalled();
284      expect(cfnMocks.describeStacks).toHaveBeenCalledWith({ StackName: 'withouterrors' });
285      expect(cfnMocks.getTemplate).toHaveBeenCalledWith({ StackName: 'withouterrors', TemplateStage: 'Original' });
286  });
287  test('deploy not skipped if template did not change but tags changed', async () => {
288      // GIVEN
289      givenStackExists({
290          Tags: [
291              { Key: 'Key', Value: 'Value' },
292          ],
293      });
294      // WHEN
295      await api_1.deployStack({
296          stack: FAKE_STACK,
297          sdk,
298          sdkProvider,
299          resolvedEnvironment: mock_sdk_1.mockResolvedEnvironment(),
300          tags: [
301              {
302                  Key: 'Key',
303                  Value: 'NewValue',
304              },
305          ],
306          toolkitInfo: api_1.ToolkitInfo.bootstraplessDeploymentsOnly(sdk),
307      });
308      // THEN
309      expect(cfnMocks.createChangeSet).toHaveBeenCalled();
310      expect(cfnMocks.executeChangeSet).toHaveBeenCalled();
311      expect(cfnMocks.describeChangeSet).toHaveBeenCalled();
312      expect(cfnMocks.describeStacks).toHaveBeenCalledWith({ StackName: 'withouterrors' });
313      expect(cfnMocks.getTemplate).toHaveBeenCalledWith({ StackName: 'withouterrors', TemplateStage: 'Original' });
314  });
315  test('deployStack reports no change if describeChangeSet returns specific error', async () => {
316      var _a;
317      (_a = cfnMocks.describeChangeSet) === null || _a === void 0 ? void 0 : _a.mockImplementation(() => ({
318          Status: 'FAILED',
319          StatusReason: 'No updates are to be performed.',
320      }));
321      // WHEN
322      const deployResult = await api_1.deployStack({
323          ...standardDeployStackArguments(),
324      });
325      // THEN
326      expect(deployResult.noOp).toEqual(true);
327  });
328  test('deploy not skipped if template did not change but one tag removed', async () => {
329      // GIVEN
330      givenStackExists({
331          Tags: [
332              { Key: 'Key1', Value: 'Value1' },
333              { Key: 'Key2', Value: 'Value2' },
334          ],
335      });
336      // WHEN
337      await api_1.deployStack({
338          ...standardDeployStackArguments(),
339          tags: [
340              { Key: 'Key1', Value: 'Value1' },
341          ],
342      });
343      // THEN
344      expect(cfnMocks.createChangeSet).toHaveBeenCalled();
345      expect(cfnMocks.executeChangeSet).toHaveBeenCalled();
346      expect(cfnMocks.describeChangeSet).toHaveBeenCalled();
347      expect(cfnMocks.describeStacks).toHaveBeenCalledWith({ StackName: 'withouterrors' });
348      expect(cfnMocks.getTemplate).toHaveBeenCalledWith({ StackName: 'withouterrors', TemplateStage: 'Original' });
349  });
350  test('deploy is not skipped if stack is in a _FAILED state', async () => {
351      // GIVEN
352      givenStackExists({
353          StackStatus: 'DELETE_FAILED',
354      });
355      // WHEN
356      await api_1.deployStack({
357          ...standardDeployStackArguments(),
358          usePreviousParameters: true,
359      }).catch(() => { });
360      // THEN
361      expect(cfnMocks.createChangeSet).toHaveBeenCalled();
362  });
363  test('existing stack in UPDATE_ROLLBACK_COMPLETE state can be updated', async () => {
364      // GIVEN
365      givenStackExists({ StackStatus: 'UPDATE_ROLLBACK_COMPLETE' }, // This is for the initial check
366      { StackStatus: 'UPDATE_COMPLETE' });
367      givenTemplateIs({ changed: 123 });
368      // WHEN
369      await api_1.deployStack({
370          ...standardDeployStackArguments(),
371      });
372      // THEN
373      expect(cfnMocks.deleteStack).not.toHaveBeenCalled();
374      expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({
375          ChangeSetType: 'UPDATE',
376      }));
377  });
378  test('deploy not skipped if template changed', async () => {
379      // GIVEN
380      givenStackExists();
381      givenTemplateIs({ changed: 123 });
382      // WHEN
383      await api_1.deployStack({
384          ...standardDeployStackArguments(),
385      });
386      // THEN
387      expect(cfnMocks.executeChangeSet).toHaveBeenCalled();
388  });
389  test('not executed and no error if --no-execute is given', async () => {
390      // WHEN
391      await api_1.deployStack({
392          ...standardDeployStackArguments(),
393          execute: false,
394      });
395      // THEN
396      expect(cfnMocks.executeChangeSet).not.toHaveBeenCalled();
397  });
398  test('empty change set is deleted if --execute is given', async () => {
399      var _a;
400      (_a = cfnMocks.describeChangeSet) === null || _a === void 0 ? void 0 : _a.mockImplementation(() => ({
401          Status: 'FAILED',
402          StatusReason: 'No updates are to be performed.',
403      }));
404      // GIVEN
405      givenStackExists();
406      // WHEN
407      await api_1.deployStack({
408          ...standardDeployStackArguments(),
409          execute: true,
410          force: true,
411      });
412      // THEN
413      expect(cfnMocks.createChangeSet).toHaveBeenCalled();
414      expect(cfnMocks.executeChangeSet).not.toHaveBeenCalled();
415      //the first deletion is for any existing cdk change sets, the second is for the deleting the new empty change set
416      expect(cfnMocks.deleteChangeSet).toHaveBeenCalledTimes(2);
417  });
418  test('empty change set is not deleted if --no-execute is given', async () => {
419      var _a;
420      (_a = cfnMocks.describeChangeSet) === null || _a === void 0 ? void 0 : _a.mockImplementation(() => ({
421          Status: 'FAILED',
422          StatusReason: 'No updates are to be performed.',
423      }));
424      // GIVEN
425      givenStackExists();
426      // WHEN
427      await api_1.deployStack({
428          ...standardDeployStackArguments(),
429          execute: false,
430      });
431      // THEN
432      expect(cfnMocks.createChangeSet).toHaveBeenCalled();
433      expect(cfnMocks.executeChangeSet).not.toHaveBeenCalled();
434      //the first deletion is for any existing cdk change sets
435      expect(cfnMocks.deleteChangeSet).toHaveBeenCalledTimes(1);
436  });
437  test('use S3 url for stack deployment if present in Stack Artifact', async () => {
438      // WHEN
439      await api_1.deployStack({
440          ...standardDeployStackArguments(),
441          stack: util_1.testStack({
442              stackName: 'withouterrors',
443              properties: {
444                  stackTemplateAssetObjectUrl: 'https://use-me-use-me/',
445              },
446          }),
447      });
448      // THEN
449      expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({
450          TemplateURL: 'https://use-me-use-me/',
451      }));
452      expect(cfnMocks.executeChangeSet).toHaveBeenCalled();
453  });
454  test('use REST API S3 url with substituted placeholders if manifest url starts with s3://', async () => {
455      // WHEN
456      await api_1.deployStack({
457          ...standardDeployStackArguments(),
458          stack: util_1.testStack({
459              stackName: 'withouterrors',
460              properties: {
461                  stackTemplateAssetObjectUrl: 's3://use-me-use-me-${AWS::AccountId}/object',
462              },
463          }),
464      });
465      // THEN
466      expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({
467          TemplateURL: 'https://s3.bermuda-triangle-1337.amazonaws.com/use-me-use-me-123456789/object',
468      }));
469      expect(cfnMocks.executeChangeSet).toHaveBeenCalled();
470  });
471  test('changeset is created when stack exists in REVIEW_IN_PROGRESS status', async () => {
472      // GIVEN
473      givenStackExists({
474          StackStatus: 'REVIEW_IN_PROGRESS',
475          Tags: [
476              { Key: 'Key1', Value: 'Value1' },
477              { Key: 'Key2', Value: 'Value2' },
478          ],
479      });
480      // WHEN
481      await api_1.deployStack({
482          ...standardDeployStackArguments(),
483          execute: false,
484      });
485      // THEN
486      expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({
487          ChangeSetType: 'CREATE',
488          StackName: 'withouterrors',
489      }));
490      expect(cfnMocks.executeChangeSet).not.toHaveBeenCalled();
491  });
492  test('changeset is updated when stack exists in CREATE_COMPLETE status', async () => {
493      // GIVEN
494      givenStackExists({
495          Tags: [
496              { Key: 'Key1', Value: 'Value1' },
497              { Key: 'Key2', Value: 'Value2' },
498          ],
499      });
500      // WHEN
501      await api_1.deployStack({
502          ...standardDeployStackArguments(),
503          execute: false,
504      });
505      // THEN
506      expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({
507          ChangeSetType: 'UPDATE',
508          StackName: 'withouterrors',
509      }));
510      expect(cfnMocks.executeChangeSet).not.toHaveBeenCalled();
511  });
512  test('deploy with termination protection enabled', async () => {
513      // WHEN
514      await api_1.deployStack({
515          ...standardDeployStackArguments(),
516          stack: FAKE_STACK_TERMINATION_PROTECTION,
517      });
518      // THEN
519      expect(cfnMocks.updateTerminationProtection).toHaveBeenCalledWith(expect.objectContaining({
520          EnableTerminationProtection: true,
521      }));
522  });
523  test('updateTerminationProtection not called when termination protection is undefined', async () => {
524      // WHEN
525      await api_1.deployStack({
526          ...standardDeployStackArguments(),
527      });
528      // THEN
529      expect(cfnMocks.updateTerminationProtection).not.toHaveBeenCalled();
530  });
531  test('updateTerminationProtection called when termination protection is undefined and stack has termination protection', async () => {
532      // GIVEN
533      givenStackExists({
534          EnableTerminationProtection: true,
535      });
536      // WHEN
537      await api_1.deployStack({
538          ...standardDeployStackArguments(),
539      });
540      // THEN
541      expect(cfnMocks.updateTerminationProtection).toHaveBeenCalledWith(expect.objectContaining({
542          EnableTerminationProtection: false,
543      }));
544  });
545  describe('disable rollback', () => {
546      test('by default, we do not disable rollback (and also do not pass the flag)', async () => {
547          // WHEN
548          await api_1.deployStack({
549              ...standardDeployStackArguments(),
550          });
551          // THEN
552          expect(cfnMocks.executeChangeSet).toHaveBeenCalledTimes(1);
553          expect(cfnMocks.executeChangeSet).not.toHaveBeenCalledWith(expect.objectContaining({
554              DisableRollback: expect.anything(),
555          }));
556      });
557      test('rollback can be disabled by setting rollback: false', async () => {
558          // WHEN
559          await api_1.deployStack({
560              ...standardDeployStackArguments(),
561              rollback: false,
562          });
563          // THEN
564          expect(cfnMocks.executeChangeSet).toHaveBeenCalledWith(expect.objectContaining({
565              DisableRollback: true,
566          }));
567      });
568  });
569  /**
570   * Set up the mocks so that it looks like the stack exists to start with
571   *
572   * The last element of this array will be continuously repeated.
573   */
574  function givenStackExists(...overrides) {
575      cfnMocks.describeStacks.mockReset();
576      if (overrides.length === 0) {
577          overrides = [{}];
578      }
579      const baseResponse = {
580          StackName: 'mock-stack-name',
581          StackId: 'mock-stack-id',
582          CreationTime: new Date(),
583          StackStatus: 'CREATE_COMPLETE',
584          EnableTerminationProtection: false,
585      };
586      for (const override of overrides.slice(0, overrides.length - 1)) {
587          cfnMocks.describeStacks.mockImplementationOnce(() => ({
588              Stacks: [{ ...baseResponse, ...override }],
589          }));
590      }
591      cfnMocks.describeStacks.mockImplementation(() => ({
592          Stacks: [{ ...baseResponse, ...overrides[overrides.length - 1] }],
593      }));
594  }
595  function givenTemplateIs(template) {
596      cfnMocks.getTemplate.mockReset();
597      cfnMocks.getTemplate.mockReturnValue({
598          TemplateBody: JSON.stringify(template),
599      });
600  }
601  //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"deploy-stack.test.js","sourceRoot":"","sources":["deploy-stack.test.ts"],"names":[],"mappings":"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","sourcesContent":["import { deployStack, ToolkitInfo } from '../../lib/api';\nimport { DEFAULT_FAKE_TEMPLATE, testStack } from '../util';\nimport { MockedObject, mockResolvedEnvironment, MockSdk, MockSdkProvider, SyncHandlerSubsetOf } from '../util/mock-sdk';\n\nconst FAKE_STACK = testStack({\n  stackName: 'withouterrors',\n});\n\nconst FAKE_STACK_WITH_PARAMETERS = testStack({\n  stackName: 'withparameters',\n  template: {\n    Parameters: {\n      HasValue: { Type: 'String' },\n      HasDefault: { Type: 'String', Default: 'TheDefault' },\n      OtherParameter: { Type: 'String' },\n    },\n  },\n});\n\nconst FAKE_STACK_TERMINATION_PROTECTION = testStack({\n  stackName: 'termination-protection',\n  template: DEFAULT_FAKE_TEMPLATE,\n  terminationProtection: true,\n});\n\nlet sdk: MockSdk;\nlet sdkProvider: MockSdkProvider;\nlet cfnMocks: MockedObject<SyncHandlerSubsetOf<AWS.CloudFormation>>;\nbeforeEach(() => {\n  sdkProvider = new MockSdkProvider();\n  sdk = new MockSdk();\n\n  cfnMocks = {\n    describeStackEvents: jest.fn().mockReturnValue({}),\n    describeStacks: jest.fn()\n      // First call, no stacks exist\n      .mockImplementationOnce(() => ({ Stacks: [] }))\n      // Second call, stack has been created\n      .mockImplementationOnce(() => ({\n        Stacks: [\n          {\n            StackStatus: 'CREATE_COMPLETE',\n            StackStatusReason: 'It is magic',\n            EnableTerminationProtection: false,\n          },\n        ],\n      })),\n    createChangeSet: jest.fn((_o) => ({})),\n    deleteChangeSet: jest.fn((_o) => ({})),\n    describeChangeSet: jest.fn((_o) => ({\n      Status: 'CREATE_COMPLETE',\n      Changes: [],\n    })),\n    executeChangeSet: jest.fn((_o) => ({})),\n    deleteStack: jest.fn((_o) => ({})),\n    getTemplate: jest.fn((_o) => ({ TemplateBody: JSON.stringify(DEFAULT_FAKE_TEMPLATE) })),\n    updateTerminationProtection: jest.fn((_o) => ({ StackId: 'stack-id' })),\n  };\n  sdk.stubCloudFormation(cfnMocks as any);\n\n});\n\nfunction standardDeployStackArguments() {\n  return {\n    stack: FAKE_STACK,\n    sdk,\n    sdkProvider,\n    resolvedEnvironment: mockResolvedEnvironment(),\n    toolkitInfo: ToolkitInfo.bootstraplessDeploymentsOnly(sdk),\n  };\n}\n\ntest('do deploy executable change set with 0 changes', async () => {\n  // WHEN\n  const ret = await deployStack({\n    ...standardDeployStackArguments(),\n  });\n\n  // THEN\n  expect(ret.noOp).toBeFalsy();\n  expect(cfnMocks.executeChangeSet).toHaveBeenCalled();\n});\n\ntest('correctly passes CFN parameters, ignoring ones with empty values', async () => {\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    parameters: {\n      A: 'A-value',\n      B: 'B=value',\n      C: undefined,\n      D: '',\n    },\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({\n    Parameters: [\n      { ParameterKey: 'A', ParameterValue: 'A-value' },\n      { ParameterKey: 'B', ParameterValue: 'B=value' },\n    ],\n  }));\n});\n\ntest('reuse previous parameters if requested', async () => {\n  // GIVEN\n  givenStackExists({\n    Parameters: [\n      { ParameterKey: 'HasValue', ParameterValue: 'TheValue' },\n      { ParameterKey: 'HasDefault', ParameterValue: 'TheOldValue' },\n    ],\n  });\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    stack: FAKE_STACK_WITH_PARAMETERS,\n    parameters: {\n      OtherParameter: 'SomeValue',\n    },\n    usePreviousParameters: true,\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({\n    Parameters: [\n      { ParameterKey: 'HasValue', UsePreviousValue: true },\n      { ParameterKey: 'HasDefault', UsePreviousValue: true },\n      { ParameterKey: 'OtherParameter', ParameterValue: 'SomeValue' },\n    ],\n  }));\n});\n\ntest('do not reuse previous parameters if not requested', async () => {\n  // GIVEN\n  givenStackExists({\n    Parameters: [\n      { ParameterKey: 'HasValue', ParameterValue: 'TheValue' },\n      { ParameterKey: 'HasDefault', ParameterValue: 'TheOldValue' },\n    ],\n  });\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    stack: FAKE_STACK_WITH_PARAMETERS,\n    parameters: {\n      HasValue: 'SomeValue',\n      OtherParameter: 'SomeValue',\n    },\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({\n    Parameters: [\n      { ParameterKey: 'HasValue', ParameterValue: 'SomeValue' },\n      { ParameterKey: 'OtherParameter', ParameterValue: 'SomeValue' },\n    ],\n  }));\n});\n\ntest('throw exception if not enough parameters supplied', async () => {\n  // GIVEN\n  givenStackExists({\n    Parameters: [\n      { ParameterKey: 'HasValue', ParameterValue: 'TheValue' },\n      { ParameterKey: 'HasDefault', ParameterValue: 'TheOldValue' },\n    ],\n  });\n\n  // WHEN\n  await expect(deployStack({\n    ...standardDeployStackArguments(),\n    stack: FAKE_STACK_WITH_PARAMETERS,\n    parameters: {\n      OtherParameter: 'SomeValue',\n    },\n  })).rejects.toThrow(/CloudFormation Parameters are missing a value/);\n});\n\ntest('deploy is skipped if template did not change', async () => {\n  // GIVEN\n  givenStackExists();\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n  });\n\n  // THEN\n  expect(cfnMocks.executeChangeSet).not.toBeCalled();\n});\n\ntest('deploy is skipped if parameters are the same', async () => {\n  // GIVEN\n  givenTemplateIs(FAKE_STACK_WITH_PARAMETERS.template);\n  givenStackExists({\n    Parameters: [\n      { ParameterKey: 'HasValue', ParameterValue: 'HasValue' },\n      { ParameterKey: 'HasDefault', ParameterValue: 'HasDefault' },\n      { ParameterKey: 'OtherParameter', ParameterValue: 'OtherParameter' },\n    ],\n  });\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    stack: FAKE_STACK_WITH_PARAMETERS,\n    parameters: {},\n    usePreviousParameters: true,\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).not.toHaveBeenCalled();\n});\n\ntest('deploy is not skipped if parameters are different', async () => {\n  // GIVEN\n  givenTemplateIs(FAKE_STACK_WITH_PARAMETERS.template);\n  givenStackExists({\n    Parameters: [\n      { ParameterKey: 'HasValue', ParameterValue: 'HasValue' },\n      { ParameterKey: 'HasDefault', ParameterValue: 'HasDefault' },\n      { ParameterKey: 'OtherParameter', ParameterValue: 'OtherParameter' },\n    ],\n  });\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    stack: FAKE_STACK_WITH_PARAMETERS,\n    parameters: {\n      HasValue: 'NewValue',\n    },\n    usePreviousParameters: true,\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({\n    Parameters: [\n      { ParameterKey: 'HasValue', ParameterValue: 'NewValue' },\n      { ParameterKey: 'HasDefault', UsePreviousValue: true },\n      { ParameterKey: 'OtherParameter', UsePreviousValue: true },\n    ],\n  }));\n});\n\ntest('if existing stack failed to create, it is deleted and recreated', async () => {\n  // GIVEN\n  givenStackExists(\n    { StackStatus: 'ROLLBACK_COMPLETE' }, // This is for the initial check\n    { StackStatus: 'DELETE_COMPLETE' }, // Poll the successful deletion\n    { StackStatus: 'CREATE_COMPLETE' }, // Poll the recreation\n  );\n  givenTemplateIs({\n    DifferentThan: 'TheDefault',\n  });\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n  });\n\n  // THEN\n  expect(cfnMocks.deleteStack).toHaveBeenCalled();\n  expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({\n    ChangeSetType: 'CREATE',\n  }));\n});\n\ntest('if existing stack failed to create, it is deleted and recreated even if the template did not change', async () => {\n  // GIVEN\n  givenStackExists(\n    { StackStatus: 'ROLLBACK_COMPLETE' }, // This is for the initial check\n    { StackStatus: 'DELETE_COMPLETE' }, // Poll the successful deletion\n    { StackStatus: 'CREATE_COMPLETE' }, // Poll the recreation\n  );\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n  });\n\n  // THEN\n  expect(cfnMocks.deleteStack).toHaveBeenCalled();\n  expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({\n    ChangeSetType: 'CREATE',\n  }));\n});\n\ntest('deploy not skipped if template did not change and --force is applied', async () => {\n  // GIVEN\n  givenStackExists();\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    force: true,\n  });\n\n  // THEN\n  expect(cfnMocks.executeChangeSet).toHaveBeenCalled();\n});\n\ntest('deploy is skipped if template and tags did not change', async () => {\n  // GIVEN\n  givenStackExists({\n    Tags: [\n      { Key: 'Key1', Value: 'Value1' },\n      { Key: 'Key2', Value: 'Value2' },\n    ],\n  });\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    tags: [\n      { Key: 'Key1', Value: 'Value1' },\n      { Key: 'Key2', Value: 'Value2' },\n    ],\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).not.toBeCalled();\n  expect(cfnMocks.executeChangeSet).not.toBeCalled();\n  expect(cfnMocks.describeStacks).toHaveBeenCalledWith({ StackName: 'withouterrors' });\n  expect(cfnMocks.getTemplate).toHaveBeenCalledWith({ StackName: 'withouterrors', TemplateStage: 'Original' });\n});\n\ntest('deploy not skipped if template did not change but tags changed', async () => {\n  // GIVEN\n  givenStackExists({\n    Tags: [\n      { Key: 'Key', Value: 'Value' },\n    ],\n  });\n\n  // WHEN\n  await deployStack({\n    stack: FAKE_STACK,\n    sdk,\n    sdkProvider,\n    resolvedEnvironment: mockResolvedEnvironment(),\n    tags: [\n      {\n        Key: 'Key',\n        Value: 'NewValue',\n      },\n    ],\n    toolkitInfo: ToolkitInfo.bootstraplessDeploymentsOnly(sdk),\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).toHaveBeenCalled();\n  expect(cfnMocks.executeChangeSet).toHaveBeenCalled();\n  expect(cfnMocks.describeChangeSet).toHaveBeenCalled();\n  expect(cfnMocks.describeStacks).toHaveBeenCalledWith({ StackName: 'withouterrors' });\n  expect(cfnMocks.getTemplate).toHaveBeenCalledWith({ StackName: 'withouterrors', TemplateStage: 'Original' });\n});\n\ntest('deployStack reports no change if describeChangeSet returns specific error', async () => {\n  cfnMocks.describeChangeSet?.mockImplementation(() => ({\n    Status: 'FAILED',\n    StatusReason: 'No updates are to be performed.',\n  }));\n\n  // WHEN\n  const deployResult = await deployStack({\n    ...standardDeployStackArguments(),\n  });\n\n  // THEN\n  expect(deployResult.noOp).toEqual(true);\n});\n\ntest('deploy not skipped if template did not change but one tag removed', async () => {\n  // GIVEN\n  givenStackExists({\n    Tags: [\n      { Key: 'Key1', Value: 'Value1' },\n      { Key: 'Key2', Value: 'Value2' },\n    ],\n  });\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    tags: [\n      { Key: 'Key1', Value: 'Value1' },\n    ],\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).toHaveBeenCalled();\n  expect(cfnMocks.executeChangeSet).toHaveBeenCalled();\n  expect(cfnMocks.describeChangeSet).toHaveBeenCalled();\n  expect(cfnMocks.describeStacks).toHaveBeenCalledWith({ StackName: 'withouterrors' });\n  expect(cfnMocks.getTemplate).toHaveBeenCalledWith({ StackName: 'withouterrors', TemplateStage: 'Original' });\n});\n\ntest('deploy is not skipped if stack is in a _FAILED state', async () => {\n  // GIVEN\n  givenStackExists({\n    StackStatus: 'DELETE_FAILED',\n  });\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    usePreviousParameters: true,\n  }).catch(() => {});\n\n  // THEN\n  expect(cfnMocks.createChangeSet).toHaveBeenCalled();\n});\n\ntest('existing stack in UPDATE_ROLLBACK_COMPLETE state can be updated', async () => {\n  // GIVEN\n  givenStackExists(\n    { StackStatus: 'UPDATE_ROLLBACK_COMPLETE' }, // This is for the initial check\n    { StackStatus: 'UPDATE_COMPLETE' }, // Poll the update\n  );\n  givenTemplateIs({ changed: 123 });\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n  });\n\n  // THEN\n  expect(cfnMocks.deleteStack).not.toHaveBeenCalled();\n  expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({\n    ChangeSetType: 'UPDATE',\n  }));\n});\n\ntest('deploy not skipped if template changed', async () => {\n  // GIVEN\n  givenStackExists();\n  givenTemplateIs({ changed: 123 });\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n  });\n\n  // THEN\n  expect(cfnMocks.executeChangeSet).toHaveBeenCalled();\n});\n\ntest('not executed and no error if --no-execute is given', async () => {\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    execute: false,\n  });\n\n  // THEN\n  expect(cfnMocks.executeChangeSet).not.toHaveBeenCalled();\n});\n\ntest('empty change set is deleted if --execute is given', async () => {\n  cfnMocks.describeChangeSet?.mockImplementation(() => ({\n    Status: 'FAILED',\n    StatusReason: 'No updates are to be performed.',\n  }));\n\n  // GIVEN\n  givenStackExists();\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    execute: true,\n    force: true, // Necessary to bypass \"skip deploy\"\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).toHaveBeenCalled();\n  expect(cfnMocks.executeChangeSet).not.toHaveBeenCalled();\n\n  //the first deletion is for any existing cdk change sets, the second is for the deleting the new empty change set\n  expect(cfnMocks.deleteChangeSet).toHaveBeenCalledTimes(2);\n});\n\ntest('empty change set is not deleted if --no-execute is given', async () => {\n  cfnMocks.describeChangeSet?.mockImplementation(() => ({\n    Status: 'FAILED',\n    StatusReason: 'No updates are to be performed.',\n  }));\n\n  // GIVEN\n  givenStackExists();\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    execute: false,\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).toHaveBeenCalled();\n  expect(cfnMocks.executeChangeSet).not.toHaveBeenCalled();\n\n  //the first deletion is for any existing cdk change sets\n  expect(cfnMocks.deleteChangeSet).toHaveBeenCalledTimes(1);\n});\n\ntest('use S3 url for stack deployment if present in Stack Artifact', async () => {\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    stack: testStack({\n      stackName: 'withouterrors',\n      properties: {\n        stackTemplateAssetObjectUrl: 'https://use-me-use-me/',\n      },\n    }),\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({\n    TemplateURL: 'https://use-me-use-me/',\n  }));\n  expect(cfnMocks.executeChangeSet).toHaveBeenCalled();\n});\n\ntest('use REST API S3 url with substituted placeholders if manifest url starts with s3://', async () => {\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    stack: testStack({\n      stackName: 'withouterrors',\n      properties: {\n        stackTemplateAssetObjectUrl: 's3://use-me-use-me-${AWS::AccountId}/object',\n      },\n    }),\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(expect.objectContaining({\n    TemplateURL: 'https://s3.bermuda-triangle-1337.amazonaws.com/use-me-use-me-123456789/object',\n  }));\n  expect(cfnMocks.executeChangeSet).toHaveBeenCalled();\n});\n\ntest('changeset is created when stack exists in REVIEW_IN_PROGRESS status', async () => {\n  // GIVEN\n  givenStackExists({\n    StackStatus: 'REVIEW_IN_PROGRESS',\n    Tags: [\n      { Key: 'Key1', Value: 'Value1' },\n      { Key: 'Key2', Value: 'Value2' },\n    ],\n  });\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    execute: false,\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(\n    expect.objectContaining({\n      ChangeSetType: 'CREATE',\n      StackName: 'withouterrors',\n    }),\n  );\n  expect(cfnMocks.executeChangeSet).not.toHaveBeenCalled();\n});\n\ntest('changeset is updated when stack exists in CREATE_COMPLETE status', async () => {\n  // GIVEN\n  givenStackExists({\n    Tags: [\n      { Key: 'Key1', Value: 'Value1' },\n      { Key: 'Key2', Value: 'Value2' },\n    ],\n  });\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    execute: false,\n  });\n\n  // THEN\n  expect(cfnMocks.createChangeSet).toHaveBeenCalledWith(\n    expect.objectContaining({\n      ChangeSetType: 'UPDATE',\n      StackName: 'withouterrors',\n    }),\n  );\n  expect(cfnMocks.executeChangeSet).not.toHaveBeenCalled();\n});\n\ntest('deploy with termination protection enabled', async () => {\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n    stack: FAKE_STACK_TERMINATION_PROTECTION,\n  });\n\n  // THEN\n  expect(cfnMocks.updateTerminationProtection).toHaveBeenCalledWith(expect.objectContaining({\n    EnableTerminationProtection: true,\n  }));\n});\n\ntest('updateTerminationProtection not called when termination protection is undefined', async () => {\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n  });\n\n  // THEN\n  expect(cfnMocks.updateTerminationProtection).not.toHaveBeenCalled();\n});\n\ntest('updateTerminationProtection called when termination protection is undefined and stack has termination protection', async () => {\n  // GIVEN\n  givenStackExists({\n    EnableTerminationProtection: true,\n  });\n\n  // WHEN\n  await deployStack({\n    ...standardDeployStackArguments(),\n  });\n\n  // THEN\n  expect(cfnMocks.updateTerminationProtection).toHaveBeenCalledWith(expect.objectContaining({\n    EnableTerminationProtection: false,\n  }));\n});\n\ndescribe('disable rollback', () => {\n  test('by default, we do not disable rollback (and also do not pass the flag)', async () => {\n    // WHEN\n    await deployStack({\n      ...standardDeployStackArguments(),\n    });\n\n    // THEN\n    expect(cfnMocks.executeChangeSet).toHaveBeenCalledTimes(1);\n    expect(cfnMocks.executeChangeSet).not.toHaveBeenCalledWith(expect.objectContaining({\n      DisableRollback: expect.anything(),\n    }));\n  });\n\n  test('rollback can be disabled by setting rollback: false', async () => {\n    // WHEN\n    await deployStack({\n      ...standardDeployStackArguments(),\n      rollback: false,\n    });\n\n    // THEN\n    expect(cfnMocks.executeChangeSet).toHaveBeenCalledWith(expect.objectContaining({\n      DisableRollback: true,\n    }));\n  });\n\n});\n\n/**\n * Set up the mocks so that it looks like the stack exists to start with\n *\n * The last element of this array will be continuously repeated.\n */\nfunction givenStackExists(...overrides: Array<Partial<AWS.CloudFormation.Stack>>) {\n  cfnMocks.describeStacks!.mockReset();\n\n  if (overrides.length === 0) {\n    overrides = [{}];\n  }\n\n  const baseResponse = {\n    StackName: 'mock-stack-name',\n    StackId: 'mock-stack-id',\n    CreationTime: new Date(),\n    StackStatus: 'CREATE_COMPLETE',\n    EnableTerminationProtection: false,\n  };\n\n  for (const override of overrides.slice(0, overrides.length - 1)) {\n    cfnMocks.describeStacks!.mockImplementationOnce(() => ({\n      Stacks: [{ ...baseResponse, ...override }],\n    }));\n  }\n  cfnMocks.describeStacks!.mockImplementation(() => ({\n    Stacks: [{ ...baseResponse, ...overrides[overrides.length - 1] }],\n  }));\n}\n\nfunction givenTemplateIs(template: any) {\n  cfnMocks.getTemplate!.mockReset();\n  cfnMocks.getTemplate!.mockReturnValue({\n    TemplateBody: JSON.stringify(template),\n  });\n}"]}