@aws-cdk/integ-runner 2.49.1 → 2.51.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.
@@ -16,6 +16,15 @@ export declare class IntegSnapshotRunner extends IntegRunner {
16
16
  diagnostics: Diagnostic[];
17
17
  destructiveChanges: DestructiveChange[];
18
18
  };
19
+ /**
20
+ * For a given cloud assembly return a collection of all templates
21
+ * that should be part of the snapshot and any required meta data.
22
+ *
23
+ * @param cloudAssemblyDir The directory of the cloud assembly to look for snapshots
24
+ * @param pickStacks Pick only these stacks from the cloud assembly
25
+ * @returns A SnapshotAssembly, the collection of all templates in this snapshot and required meta data
26
+ */
27
+ private getSnapshotAssembly;
19
28
  /**
20
29
  * For a given stack return all resource types that are allowed to be destroyed
21
30
  * as part of a stack update
@@ -26,15 +26,7 @@ class IntegSnapshotRunner extends runner_base_1.IntegRunner {
26
26
  var _a;
27
27
  let doClean = true;
28
28
  try {
29
- // read the existing snapshot
30
- const expectedStacks = this.readAssembly(this.snapshotDir);
31
- // only diff stacks that are part of the test case
32
- const expectedStacksToDiff = {};
33
- for (const [stackName, template] of Object.entries(expectedStacks)) {
34
- if ((_a = this.expectedTestSuite) === null || _a === void 0 ? void 0 : _a.stacks.includes(stackName)) {
35
- expectedStacksToDiff[stackName] = template;
36
- }
37
- }
29
+ const expectedSnapshotAssembly = this.getSnapshotAssembly(this.snapshotDir, (_a = this.expectedTestSuite) === null || _a === void 0 ? void 0 : _a.stacks);
38
30
  // synth the integration test
39
31
  // FIXME: ideally we should not need to run this again if
40
32
  // the cdkOutDir exists already, but for some reason generateActualSnapshot
@@ -52,22 +44,14 @@ class IntegSnapshotRunner extends runner_base_1.IntegRunner {
52
44
  output: path.relative(this.directory, this.cdkOutDir),
53
45
  });
54
46
  // read the "actual" snapshot
55
- const actualDir = this.cdkOutDir;
56
- const actualStacks = this.readAssembly(actualDir);
57
- // only diff stacks that are part of the test case
58
- const actualStacksToDiff = {};
59
- for (const [stackName, template] of Object.entries(actualStacks)) {
60
- if (this.actualTestSuite.stacks.includes(stackName)) {
61
- actualStacksToDiff[stackName] = template;
62
- }
63
- }
47
+ const actualSnapshotAssembly = this.getSnapshotAssembly(this.cdkOutDir, this.actualTestSuite.stacks);
64
48
  // diff the existing snapshot (expected) with the integration test (actual)
65
- const diagnostics = this.diffAssembly(expectedStacksToDiff, actualStacksToDiff);
49
+ const diagnostics = this.diffAssembly(expectedSnapshotAssembly, actualSnapshotAssembly);
66
50
  if (diagnostics.diagnostics.length) {
67
51
  // Attach additional messages to the first diagnostic
68
52
  const additionalMessages = [];
69
53
  if (options.retain) {
70
- additionalMessages.push(`(Failure retained) Expected: ${path.relative(process.cwd(), this.snapshotDir)}`, ` Actual: ${path.relative(process.cwd(), actualDir)}`),
54
+ additionalMessages.push(`(Failure retained) Expected: ${path.relative(process.cwd(), this.snapshotDir)}`, ` Actual: ${path.relative(process.cwd(), this.cdkOutDir)}`),
71
55
  doClean = false;
72
56
  }
73
57
  if (options.verbose) {
@@ -94,6 +78,33 @@ class IntegSnapshotRunner extends runner_base_1.IntegRunner {
94
78
  }
95
79
  }
96
80
  }
81
+ /**
82
+ * For a given cloud assembly return a collection of all templates
83
+ * that should be part of the snapshot and any required meta data.
84
+ *
85
+ * @param cloudAssemblyDir The directory of the cloud assembly to look for snapshots
86
+ * @param pickStacks Pick only these stacks from the cloud assembly
87
+ * @returns A SnapshotAssembly, the collection of all templates in this snapshot and required meta data
88
+ */
89
+ getSnapshotAssembly(cloudAssemblyDir, pickStacks = []) {
90
+ const assembly = this.readAssembly(cloudAssemblyDir);
91
+ const stacks = assembly.stacks;
92
+ const snapshots = {};
93
+ for (const [stackName, stackTemplate] of Object.entries(stacks)) {
94
+ if (pickStacks.includes(stackName)) {
95
+ const manifest = cloud_assembly_1.AssemblyManifestReader.fromPath(cloudAssemblyDir);
96
+ const assets = manifest.getAssetIdsForStack(stackName);
97
+ snapshots[stackName] = {
98
+ templates: {
99
+ [stackName]: stackTemplate,
100
+ ...assembly.getNestedStacksForStack(stackName),
101
+ },
102
+ assets,
103
+ };
104
+ }
105
+ }
106
+ return snapshots;
107
+ }
97
108
  /**
98
109
  * For a given stack return all resource types that are allowed to be destroyed
99
110
  * as part of a stack update
@@ -118,84 +129,95 @@ class IntegSnapshotRunner extends runner_base_1.IntegRunner {
118
129
  * @returns any diagnostics and any destructive changes
119
130
  */
120
131
  diffAssembly(expected, actual) {
121
- var _a, _b;
132
+ var _a, _b, _c, _d;
122
133
  const failures = [];
123
134
  const destructiveChanges = [];
124
135
  // check if there is a CFN template in the current snapshot
125
136
  // that does not exist in the "actual" snapshot
126
- for (const templateId of Object.keys(expected)) {
127
- if (!actual.hasOwnProperty(templateId)) {
128
- failures.push({
129
- testName: this.testName,
130
- reason: common_1.DiagnosticReason.SNAPSHOT_FAILED,
131
- message: `${templateId} exists in snapshot, but not in actual`,
132
- });
137
+ for (const [stackId, stack] of Object.entries(expected)) {
138
+ for (const templateId of Object.keys(stack.templates)) {
139
+ if (!((_a = actual[stackId]) === null || _a === void 0 ? void 0 : _a.templates[templateId])) {
140
+ failures.push({
141
+ testName: this.testName,
142
+ stackName: templateId,
143
+ reason: common_1.DiagnosticReason.SNAPSHOT_FAILED,
144
+ message: `${templateId} exists in snapshot, but not in actual`,
145
+ });
146
+ }
133
147
  }
134
148
  }
135
- for (const templateId of Object.keys(actual)) {
136
- // check if there is a CFN template in the "actual" snapshot
137
- // that does not exist in the current snapshot
138
- if (!expected.hasOwnProperty(templateId)) {
139
- failures.push({
140
- testName: this.testName,
141
- reason: common_1.DiagnosticReason.SNAPSHOT_FAILED,
142
- message: `${templateId} does not exist in snapshot, but does in actual`,
143
- });
144
- continue;
145
- }
146
- else {
147
- let actualTemplate = actual[templateId];
148
- let expectedTemplate = expected[templateId];
149
- const allowedDestroyTypes = (_a = this.getAllowedDestroyTypesForStack(templateId)) !== null && _a !== void 0 ? _a : [];
150
- // if we are not verifying asset hashes then remove the specific
151
- // asset hashes from the templates so they are not part of the diff
152
- // comparison
153
- if (!((_b = this.actualTestSuite.getOptionsForStack(templateId)) === null || _b === void 0 ? void 0 : _b.diffAssets)) {
154
- actualTemplate = this.canonicalizeTemplate(actualTemplate, templateId, this.cdkOutDir);
155
- expectedTemplate = this.canonicalizeTemplate(expectedTemplate, templateId, this.snapshotDir);
156
- }
157
- const templateDiff = cloudformation_diff_1.diffTemplate(expectedTemplate, actualTemplate);
158
- if (!templateDiff.isEmpty) {
159
- // go through all the resource differences and check for any
160
- // "destructive" changes
161
- templateDiff.resources.forEachDifference((logicalId, change) => {
162
- var _a, _b, _c;
163
- // if the change is a removal it will not show up as a 'changeImpact'
164
- // so need to check for it separately, unless it is a resourceType that
165
- // has been "allowed" to be destroyed
166
- const resourceType = (_b = (_a = change.oldValue) === null || _a === void 0 ? void 0 : _a.Type) !== null && _b !== void 0 ? _b : (_c = change.newValue) === null || _c === void 0 ? void 0 : _c.Type;
167
- if (resourceType && allowedDestroyTypes.includes(resourceType)) {
168
- return;
169
- }
170
- if (change.isRemoval) {
171
- destructiveChanges.push({
172
- impact: cloudformation_diff_1.ResourceImpact.WILL_DESTROY,
173
- logicalId,
174
- stackName: templateId,
175
- });
176
- }
177
- else {
178
- switch (change.changeImpact) {
179
- case cloudformation_diff_1.ResourceImpact.MAY_REPLACE:
180
- case cloudformation_diff_1.ResourceImpact.WILL_ORPHAN:
181
- case cloudformation_diff_1.ResourceImpact.WILL_DESTROY:
182
- case cloudformation_diff_1.ResourceImpact.WILL_REPLACE:
183
- destructiveChanges.push({
184
- impact: change.changeImpact,
185
- logicalId,
186
- stackName: templateId,
187
- });
188
- break;
189
- }
190
- }
191
- });
192
- const writable = new StringWritable({});
193
- cloudformation_diff_1.formatDifferences(writable, templateDiff);
149
+ for (const [stackId, stack] of Object.entries(actual)) {
150
+ for (const templateId of Object.keys(stack.templates)) {
151
+ // check if there is a CFN template in the "actual" snapshot
152
+ // that does not exist in the current snapshot
153
+ if (!((_b = expected[stackId]) === null || _b === void 0 ? void 0 : _b.templates[templateId])) {
194
154
  failures.push({
195
- reason: common_1.DiagnosticReason.SNAPSHOT_FAILED,
196
- message: writable.data,
197
155
  testName: this.testName,
156
+ stackName: templateId,
157
+ reason: common_1.DiagnosticReason.SNAPSHOT_FAILED,
158
+ message: `${templateId} does not exist in snapshot, but does in actual`,
198
159
  });
160
+ continue;
161
+ }
162
+ else {
163
+ const config = {
164
+ diffAssets: (_c = this.actualTestSuite.getOptionsForStack(stackId)) === null || _c === void 0 ? void 0 : _c.diffAssets,
165
+ };
166
+ let actualTemplate = actual[stackId].templates[templateId];
167
+ let expectedTemplate = expected[stackId].templates[templateId];
168
+ // if we are not verifying asset hashes then remove the specific
169
+ // asset hashes from the templates so they are not part of the diff
170
+ // comparison
171
+ if (!config.diffAssets) {
172
+ actualTemplate = this.canonicalizeTemplate(actualTemplate, actual[stackId].assets);
173
+ expectedTemplate = this.canonicalizeTemplate(expectedTemplate, expected[stackId].assets);
174
+ }
175
+ const templateDiff = cloudformation_diff_1.diffTemplate(expectedTemplate, actualTemplate);
176
+ if (!templateDiff.isEmpty) {
177
+ const allowedDestroyTypes = (_d = this.getAllowedDestroyTypesForStack(stackId)) !== null && _d !== void 0 ? _d : [];
178
+ // go through all the resource differences and check for any
179
+ // "destructive" changes
180
+ templateDiff.resources.forEachDifference((logicalId, change) => {
181
+ var _a, _b, _c;
182
+ // if the change is a removal it will not show up as a 'changeImpact'
183
+ // so need to check for it separately, unless it is a resourceType that
184
+ // has been "allowed" to be destroyed
185
+ const resourceType = (_b = (_a = change.oldValue) === null || _a === void 0 ? void 0 : _a.Type) !== null && _b !== void 0 ? _b : (_c = change.newValue) === null || _c === void 0 ? void 0 : _c.Type;
186
+ if (resourceType && allowedDestroyTypes.includes(resourceType)) {
187
+ return;
188
+ }
189
+ if (change.isRemoval) {
190
+ destructiveChanges.push({
191
+ impact: cloudformation_diff_1.ResourceImpact.WILL_DESTROY,
192
+ logicalId,
193
+ stackName: templateId,
194
+ });
195
+ }
196
+ else {
197
+ switch (change.changeImpact) {
198
+ case cloudformation_diff_1.ResourceImpact.MAY_REPLACE:
199
+ case cloudformation_diff_1.ResourceImpact.WILL_ORPHAN:
200
+ case cloudformation_diff_1.ResourceImpact.WILL_DESTROY:
201
+ case cloudformation_diff_1.ResourceImpact.WILL_REPLACE:
202
+ destructiveChanges.push({
203
+ impact: change.changeImpact,
204
+ logicalId,
205
+ stackName: templateId,
206
+ });
207
+ break;
208
+ }
209
+ }
210
+ });
211
+ const writable = new StringWritable({});
212
+ cloudformation_diff_1.formatDifferences(writable, templateDiff);
213
+ failures.push({
214
+ reason: common_1.DiagnosticReason.SNAPSHOT_FAILED,
215
+ message: writable.data,
216
+ stackName: templateId,
217
+ testName: this.testName,
218
+ config,
219
+ });
220
+ }
199
221
  }
200
222
  }
201
223
  }
@@ -205,9 +227,7 @@ class IntegSnapshotRunner extends runner_base_1.IntegRunner {
205
227
  };
206
228
  }
207
229
  readAssembly(dir) {
208
- const assembly = cloud_assembly_1.AssemblyManifestReader.fromPath(dir);
209
- const stacks = assembly.stacks;
210
- return stacks;
230
+ return cloud_assembly_1.AssemblyManifestReader.fromPath(dir);
211
231
  }
212
232
  /**
213
233
  * Reduce template to a normal form where asset references have been normalized
@@ -215,7 +235,7 @@ class IntegSnapshotRunner extends runner_base_1.IntegRunner {
215
235
  * This makes it possible to compare templates if all that's different between
216
236
  * them is the hashes of the asset values.
217
237
  */
218
- canonicalizeTemplate(template, stackName, manifestDir) {
238
+ canonicalizeTemplate(template, assets) {
219
239
  const assetsSeen = new Set();
220
240
  const stringSubstitutions = new Array();
221
241
  // Find assets via parameters (for LegacyStackSynthesizer)
@@ -243,8 +263,6 @@ class IntegSnapshotRunner extends runner_base_1.IntegRunner {
243
263
  }
244
264
  // find assets defined in the asset manifest
245
265
  try {
246
- const manifest = cloud_assembly_1.AssemblyManifestReader.fromPath(manifestDir);
247
- const assets = manifest.getAssetIdsForStack(stackName);
248
266
  assets.forEach(asset => {
249
267
  if (!assetsSeen.has(asset)) {
250
268
  assetsSeen.add(asset);
@@ -304,4 +322,4 @@ class StringWritable extends stream_1.Writable {
304
322
  callback();
305
323
  }
306
324
  }
307
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snapshot-test-runner.js","sourceRoot":"","sources":["snapshot-test-runner.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { Writable, WritableOptions } from 'stream';\nimport { StringDecoder } from 'string_decoder';\nimport { diffTemplate, formatDifferences, ResourceDifference, ResourceImpact } from '@aws-cdk/cloudformation-diff';\nimport { Diagnostic, DiagnosticReason, DestructiveChange, SnapshotVerificationOptions } from '../workers/common';\nimport { AssemblyManifestReader } from './private/cloud-assembly';\nimport { IntegRunnerOptions, IntegRunner, DEFAULT_SYNTH_OPTIONS } from './runner-base';\n\n/**\n * Runner for snapshot tests. This handles orchestrating\n * the validation of the integration test snapshots\n */\nexport class IntegSnapshotRunner extends IntegRunner {\n  constructor(options: IntegRunnerOptions) {\n    super(options);\n  }\n\n  /**\n   * Synth the integration tests and compare the templates\n   * to the existing snapshot.\n   *\n   * @returns any diagnostics and any destructive changes\n   */\n  public testSnapshot(options: SnapshotVerificationOptions = {}): { diagnostics: Diagnostic[], destructiveChanges: DestructiveChange[] } {\n    let doClean = true;\n    try {\n      // read the existing snapshot\n      const expectedStacks = this.readAssembly(this.snapshotDir);\n      // only diff stacks that are part of the test case\n      const expectedStacksToDiff: Record<string, any> = {};\n      for (const [stackName, template] of Object.entries(expectedStacks)) {\n        if (this.expectedTestSuite?.stacks.includes(stackName)) {\n          expectedStacksToDiff[stackName] = template;\n        }\n      }\n\n      // synth the integration test\n      // FIXME: ideally we should not need to run this again if\n      // the cdkOutDir exists already, but for some reason generateActualSnapshot\n      // generates an incorrect snapshot and I have no idea why so synth again here\n      // to produce the \"correct\" snapshot\n      const env = {\n        ...DEFAULT_SYNTH_OPTIONS.env,\n        CDK_CONTEXT_JSON: JSON.stringify(this.getContext({\n          ...this.actualTestSuite.enableLookups ? DEFAULT_SYNTH_OPTIONS.context : {},\n        })),\n      };\n      this.cdk.synthFast({\n        execCmd: this.cdkApp.split(' '),\n        env,\n        output: path.relative(this.directory, this.cdkOutDir),\n      });\n\n      // read the \"actual\" snapshot\n      const actualDir = this.cdkOutDir;\n      const actualStacks = this.readAssembly(actualDir);\n      // only diff stacks that are part of the test case\n      const actualStacksToDiff: Record<string, any> = {};\n      for (const [stackName, template] of Object.entries(actualStacks)) {\n        if (this.actualTestSuite.stacks.includes(stackName)) {\n          actualStacksToDiff[stackName] = template;\n        }\n      }\n\n      // diff the existing snapshot (expected) with the integration test (actual)\n      const diagnostics = this.diffAssembly(expectedStacksToDiff, actualStacksToDiff);\n\n      if (diagnostics.diagnostics.length) {\n        // Attach additional messages to the first diagnostic\n        const additionalMessages: string[] = [];\n\n        if (options.retain) {\n          additionalMessages.push(\n            `(Failure retained) Expected: ${path.relative(process.cwd(), this.snapshotDir)}`,\n            `                   Actual:   ${path.relative(process.cwd(), actualDir)}`,\n          ),\n          doClean = false;\n        }\n\n        if (options.verbose) {\n          // Show the command necessary to repro this\n          const envSet = Object.entries(env)\n            .filter(([k, _]) => k !== 'CDK_CONTEXT_JSON')\n            .map(([k, v]) => `${k}='${v}'`);\n          const envCmd = envSet.length > 0 ? ['env', ...envSet] : [];\n\n          additionalMessages.push(\n            'Repro:',\n            `  ${[...envCmd, 'cdk synth', `-a '${this.cdkApp}'`, `-o '${this.cdkOutDir}'`, ...Object.entries(this.getContext()).flatMap(([k, v]) => typeof v !== 'object' ? [`-c '${k}=${v}'`] : [])].join(' ')}`,\n\n          );\n        }\n\n        diagnostics.diagnostics[0] = {\n          ...diagnostics.diagnostics[0],\n          additionalMessages,\n        };\n      }\n\n      return diagnostics;\n    } catch (e) {\n      throw e;\n    } finally {\n      if (doClean) {\n        this.cleanup();\n      }\n    }\n  }\n\n  /**\n   * For a given stack return all resource types that are allowed to be destroyed\n   * as part of a stack update\n   *\n   * @param stackId the stack id\n   * @returns a list of resource types or undefined if none are found\n   */\n  private getAllowedDestroyTypesForStack(stackId: string): string[] | undefined {\n    for (const testCase of Object.values(this.actualTests() ?? {})) {\n      if (testCase.stacks.includes(stackId)) {\n        return testCase.allowDestroy;\n      }\n    }\n    return undefined;\n  }\n\n  /**\n   * Find any differences between the existing and expected snapshots\n   *\n   * @param existing - the existing (expected) snapshot\n   * @param actual - the new (actual) snapshot\n   * @returns any diagnostics and any destructive changes\n   */\n  private diffAssembly(\n    expected: Record<string, any>,\n    actual: Record<string, any>,\n  ): { diagnostics: Diagnostic[], destructiveChanges: DestructiveChange[] } {\n    const failures: Diagnostic[] = [];\n    const destructiveChanges: DestructiveChange[] = [];\n\n    // check if there is a CFN template in the current snapshot\n    // that does not exist in the \"actual\" snapshot\n    for (const templateId of Object.keys(expected)) {\n      if (!actual.hasOwnProperty(templateId)) {\n        failures.push({\n          testName: this.testName,\n          reason: DiagnosticReason.SNAPSHOT_FAILED,\n          message: `${templateId} exists in snapshot, but not in actual`,\n        });\n      }\n    }\n\n    for (const templateId of Object.keys(actual)) {\n      // check if there is a CFN template in the \"actual\" snapshot\n      // that does not exist in the current snapshot\n      if (!expected.hasOwnProperty(templateId)) {\n        failures.push({\n          testName: this.testName,\n          reason: DiagnosticReason.SNAPSHOT_FAILED,\n          message: `${templateId} does not exist in snapshot, but does in actual`,\n        });\n        continue;\n      } else {\n        let actualTemplate = actual[templateId];\n        let expectedTemplate = expected[templateId];\n        const allowedDestroyTypes = this.getAllowedDestroyTypesForStack(templateId) ?? [];\n\n        // if we are not verifying asset hashes then remove the specific\n        // asset hashes from the templates so they are not part of the diff\n        // comparison\n        if (!this.actualTestSuite.getOptionsForStack(templateId)?.diffAssets) {\n          actualTemplate = this.canonicalizeTemplate(actualTemplate, templateId, this.cdkOutDir);\n          expectedTemplate = this.canonicalizeTemplate(expectedTemplate, templateId, this.snapshotDir);\n        }\n        const templateDiff = diffTemplate(expectedTemplate, actualTemplate);\n        if (!templateDiff.isEmpty) {\n          // go through all the resource differences and check for any\n          // \"destructive\" changes\n          templateDiff.resources.forEachDifference((logicalId: string, change: ResourceDifference) => {\n            // if the change is a removal it will not show up as a 'changeImpact'\n            // so need to check for it separately, unless it is a resourceType that\n            // has been \"allowed\" to be destroyed\n            const resourceType = change.oldValue?.Type ?? change.newValue?.Type;\n            if (resourceType && allowedDestroyTypes.includes(resourceType)) {\n              return;\n            }\n            if (change.isRemoval) {\n              destructiveChanges.push({\n                impact: ResourceImpact.WILL_DESTROY,\n                logicalId,\n                stackName: templateId,\n              });\n            } else {\n              switch (change.changeImpact) {\n                case ResourceImpact.MAY_REPLACE:\n                case ResourceImpact.WILL_ORPHAN:\n                case ResourceImpact.WILL_DESTROY:\n                case ResourceImpact.WILL_REPLACE:\n                  destructiveChanges.push({\n                    impact: change.changeImpact,\n                    logicalId,\n                    stackName: templateId,\n                  });\n                  break;\n              }\n            }\n          });\n          const writable = new StringWritable({});\n          formatDifferences(writable, templateDiff);\n          failures.push({\n            reason: DiagnosticReason.SNAPSHOT_FAILED,\n            message: writable.data,\n            testName: this.testName,\n          });\n        }\n      }\n    }\n\n    return {\n      diagnostics: failures,\n      destructiveChanges,\n    };\n  }\n\n  private readAssembly(dir: string): Record<string, any> {\n    const assembly = AssemblyManifestReader.fromPath(dir);\n    const stacks = assembly.stacks;\n\n    return stacks;\n  }\n\n  /**\n  * Reduce template to a normal form where asset references have been normalized\n  *\n  * This makes it possible to compare templates if all that's different between\n  * them is the hashes of the asset values.\n  */\n  private canonicalizeTemplate(template: any, stackName: string, manifestDir: string): any {\n    const assetsSeen = new Set<string>();\n    const stringSubstitutions = new Array<[RegExp, string]>();\n\n    // Find assets via parameters (for LegacyStackSynthesizer)\n    const paramRe = /^AssetParameters([a-zA-Z0-9]{64})(S3Bucket|S3VersionKey|ArtifactHash)([a-zA-Z0-9]{8})$/;\n    for (const paramName of Object.keys(template?.Parameters || {})) {\n      const m = paramRe.exec(paramName);\n      if (!m) { continue; }\n      if (assetsSeen.has(m[1])) { continue; }\n\n      assetsSeen.add(m[1]);\n      const ix = assetsSeen.size;\n\n      // Full parameter reference\n      stringSubstitutions.push([\n        new RegExp(`AssetParameters${m[1]}(S3Bucket|S3VersionKey|ArtifactHash)([a-zA-Z0-9]{8})`),\n        `Asset${ix}$1`,\n      ]);\n      // Substring asset hash reference\n      stringSubstitutions.push([\n        new RegExp(`${m[1]}`),\n        `Asset${ix}Hash`,\n      ]);\n    }\n\n    // find assets defined in the asset manifest\n    try {\n      const manifest = AssemblyManifestReader.fromPath(manifestDir);\n      const assets = manifest.getAssetIdsForStack(stackName);\n      assets.forEach(asset => {\n        if (!assetsSeen.has(asset)) {\n          assetsSeen.add(asset);\n          const ix = assetsSeen.size;\n          stringSubstitutions.push([\n            new RegExp(asset),\n            `Asset${ix}$1`,\n          ]);\n        }\n      });\n    } catch {\n      // if there is no asset manifest that is fine.\n    }\n\n    // Substitute them out\n    return substitute(template);\n\n    function substitute(what: any): any {\n      if (Array.isArray(what)) {\n        return what.map(substitute);\n      }\n\n      if (typeof what === 'object' && what !== null) {\n        const ret: any = {};\n        for (const [k, v] of Object.entries(what)) {\n          ret[stringSub(k)] = substitute(v);\n        }\n        return ret;\n      }\n\n      if (typeof what === 'string') {\n        return stringSub(what);\n      }\n\n      return what;\n    }\n\n    function stringSub(x: string) {\n      for (const [re, replacement] of stringSubstitutions) {\n        x = x.replace(re, replacement);\n      }\n      return x;\n    }\n  }\n}\n\nclass StringWritable extends Writable {\n  public data: string;\n  private _decoder: StringDecoder;\n  constructor(options: WritableOptions) {\n    super(options);\n    this._decoder = new StringDecoder();\n    this.data = '';\n  }\n\n  _write(chunk: any, encoding: string, callback: (error?: Error | null) => void): void {\n    if (encoding === 'buffer') {\n      chunk = this._decoder.write(chunk);\n    }\n\n    this.data += chunk;\n    callback();\n  }\n\n  _final(callback: (error?: Error | null) => void): void {\n    this.data += this._decoder.end();\n    callback();\n  }\n}\n"]}
325
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snapshot-test-runner.js","sourceRoot":"","sources":["snapshot-test-runner.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { Writable, WritableOptions } from 'stream';\nimport { StringDecoder } from 'string_decoder';\nimport { diffTemplate, formatDifferences, ResourceDifference, ResourceImpact } from '@aws-cdk/cloudformation-diff';\nimport { Diagnostic, DiagnosticReason, DestructiveChange, SnapshotVerificationOptions } from '../workers/common';\nimport { AssemblyManifestReader } from './private/cloud-assembly';\nimport { IntegRunnerOptions, IntegRunner, DEFAULT_SYNTH_OPTIONS } from './runner-base';\n\ninterface SnapshotAssembly {\n  /**\n   * Map of stacks that are part of this assembly\n   */\n  [stackName: string]: {\n    /**\n     * All templates for this stack, including nested stacks\n     */\n    templates: {\n      [templateId: string]: any\n    },\n\n    /**\n     * List of asset Ids that are used by this assembly\n     */\n    assets: string[]\n  }\n}\n\n/**\n * Runner for snapshot tests. This handles orchestrating\n * the validation of the integration test snapshots\n */\nexport class IntegSnapshotRunner extends IntegRunner {\n  constructor(options: IntegRunnerOptions) {\n    super(options);\n  }\n\n  /**\n   * Synth the integration tests and compare the templates\n   * to the existing snapshot.\n   *\n   * @returns any diagnostics and any destructive changes\n   */\n  public testSnapshot(options: SnapshotVerificationOptions = {}): { diagnostics: Diagnostic[], destructiveChanges: DestructiveChange[] } {\n    let doClean = true;\n    try {\n      const expectedSnapshotAssembly = this.getSnapshotAssembly(this.snapshotDir, this.expectedTestSuite?.stacks);\n\n      // synth the integration test\n      // FIXME: ideally we should not need to run this again if\n      // the cdkOutDir exists already, but for some reason generateActualSnapshot\n      // generates an incorrect snapshot and I have no idea why so synth again here\n      // to produce the \"correct\" snapshot\n      const env = {\n        ...DEFAULT_SYNTH_OPTIONS.env,\n        CDK_CONTEXT_JSON: JSON.stringify(this.getContext({\n          ...this.actualTestSuite.enableLookups ? DEFAULT_SYNTH_OPTIONS.context : {},\n        })),\n      };\n      this.cdk.synthFast({\n        execCmd: this.cdkApp.split(' '),\n        env,\n        output: path.relative(this.directory, this.cdkOutDir),\n      });\n\n      // read the \"actual\" snapshot\n      const actualSnapshotAssembly = this.getSnapshotAssembly(this.cdkOutDir, this.actualTestSuite.stacks);\n\n      // diff the existing snapshot (expected) with the integration test (actual)\n      const diagnostics = this.diffAssembly(expectedSnapshotAssembly, actualSnapshotAssembly);\n\n      if (diagnostics.diagnostics.length) {\n        // Attach additional messages to the first diagnostic\n        const additionalMessages: string[] = [];\n\n        if (options.retain) {\n          additionalMessages.push(\n            `(Failure retained) Expected: ${path.relative(process.cwd(), this.snapshotDir)}`,\n            `                   Actual:   ${path.relative(process.cwd(), this.cdkOutDir)}`,\n          ),\n          doClean = false;\n        }\n\n        if (options.verbose) {\n          // Show the command necessary to repro this\n          const envSet = Object.entries(env)\n            .filter(([k, _]) => k !== 'CDK_CONTEXT_JSON')\n            .map(([k, v]) => `${k}='${v}'`);\n          const envCmd = envSet.length > 0 ? ['env', ...envSet] : [];\n\n          additionalMessages.push(\n            'Repro:',\n            `  ${[...envCmd, 'cdk synth', `-a '${this.cdkApp}'`, `-o '${this.cdkOutDir}'`, ...Object.entries(this.getContext()).flatMap(([k, v]) => typeof v !== 'object' ? [`-c '${k}=${v}'`] : [])].join(' ')}`,\n\n          );\n        }\n\n        diagnostics.diagnostics[0] = {\n          ...diagnostics.diagnostics[0],\n          additionalMessages,\n        };\n      }\n\n      return diagnostics;\n    } catch (e) {\n      throw e;\n    } finally {\n      if (doClean) {\n        this.cleanup();\n      }\n    }\n  }\n\n  /**\n   * For a given cloud assembly return a collection of all templates\n   * that should be part of the snapshot and any required meta data.\n   *\n   * @param cloudAssemblyDir The directory of the cloud assembly to look for snapshots\n   * @param pickStacks Pick only these stacks from the cloud assembly\n   * @returns A SnapshotAssembly, the collection of all templates in this snapshot and required meta data\n   */\n  private getSnapshotAssembly(cloudAssemblyDir: string, pickStacks: string[] = []): SnapshotAssembly {\n    const assembly = this.readAssembly(cloudAssemblyDir);\n    const stacks = assembly.stacks;\n    const snapshots: SnapshotAssembly = {};\n    for (const [stackName, stackTemplate] of Object.entries(stacks)) {\n      if (pickStacks.includes(stackName)) {\n        const manifest = AssemblyManifestReader.fromPath(cloudAssemblyDir);\n        const assets = manifest.getAssetIdsForStack(stackName);\n\n        snapshots[stackName] = {\n          templates: {\n            [stackName]: stackTemplate,\n            ...assembly.getNestedStacksForStack(stackName),\n          },\n          assets,\n        };\n      }\n    }\n\n    return snapshots;\n  }\n\n  /**\n   * For a given stack return all resource types that are allowed to be destroyed\n   * as part of a stack update\n   *\n   * @param stackId the stack id\n   * @returns a list of resource types or undefined if none are found\n   */\n  private getAllowedDestroyTypesForStack(stackId: string): string[] | undefined {\n    for (const testCase of Object.values(this.actualTests() ?? {})) {\n      if (testCase.stacks.includes(stackId)) {\n        return testCase.allowDestroy;\n      }\n    }\n    return undefined;\n  }\n\n  /**\n   * Find any differences between the existing and expected snapshots\n   *\n   * @param existing - the existing (expected) snapshot\n   * @param actual - the new (actual) snapshot\n   * @returns any diagnostics and any destructive changes\n   */\n  private diffAssembly(\n    expected: SnapshotAssembly,\n    actual: SnapshotAssembly,\n  ): { diagnostics: Diagnostic[], destructiveChanges: DestructiveChange[] } {\n    const failures: Diagnostic[] = [];\n    const destructiveChanges: DestructiveChange[] = [];\n\n    // check if there is a CFN template in the current snapshot\n    // that does not exist in the \"actual\" snapshot\n    for (const [stackId, stack] of Object.entries(expected)) {\n      for (const templateId of Object.keys(stack.templates)) {\n        if (!actual[stackId]?.templates[templateId]) {\n          failures.push({\n            testName: this.testName,\n            stackName: templateId,\n            reason: DiagnosticReason.SNAPSHOT_FAILED,\n            message: `${templateId} exists in snapshot, but not in actual`,\n          });\n        }\n      }\n    }\n\n    for (const [stackId, stack] of Object.entries(actual)) {\n      for (const templateId of Object.keys(stack.templates)) {\n      // check if there is a CFN template in the \"actual\" snapshot\n      // that does not exist in the current snapshot\n        if (!expected[stackId]?.templates[templateId]) {\n          failures.push({\n            testName: this.testName,\n            stackName: templateId,\n            reason: DiagnosticReason.SNAPSHOT_FAILED,\n            message: `${templateId} does not exist in snapshot, but does in actual`,\n          });\n          continue;\n        } else {\n          const config = {\n            diffAssets: this.actualTestSuite.getOptionsForStack(stackId)?.diffAssets,\n          };\n          let actualTemplate = actual[stackId].templates[templateId];\n          let expectedTemplate = expected[stackId].templates[templateId];\n\n          // if we are not verifying asset hashes then remove the specific\n          // asset hashes from the templates so they are not part of the diff\n          // comparison\n          if (!config.diffAssets) {\n            actualTemplate = this.canonicalizeTemplate(actualTemplate, actual[stackId].assets);\n            expectedTemplate = this.canonicalizeTemplate(expectedTemplate, expected[stackId].assets);\n          }\n          const templateDiff = diffTemplate(expectedTemplate, actualTemplate);\n          if (!templateDiff.isEmpty) {\n            const allowedDestroyTypes = this.getAllowedDestroyTypesForStack(stackId) ?? [];\n\n            // go through all the resource differences and check for any\n            // \"destructive\" changes\n            templateDiff.resources.forEachDifference((logicalId: string, change: ResourceDifference) => {\n            // if the change is a removal it will not show up as a 'changeImpact'\n            // so need to check for it separately, unless it is a resourceType that\n            // has been \"allowed\" to be destroyed\n              const resourceType = change.oldValue?.Type ?? change.newValue?.Type;\n              if (resourceType && allowedDestroyTypes.includes(resourceType)) {\n                return;\n              }\n              if (change.isRemoval) {\n                destructiveChanges.push({\n                  impact: ResourceImpact.WILL_DESTROY,\n                  logicalId,\n                  stackName: templateId,\n                });\n              } else {\n                switch (change.changeImpact) {\n                  case ResourceImpact.MAY_REPLACE:\n                  case ResourceImpact.WILL_ORPHAN:\n                  case ResourceImpact.WILL_DESTROY:\n                  case ResourceImpact.WILL_REPLACE:\n                    destructiveChanges.push({\n                      impact: change.changeImpact,\n                      logicalId,\n                      stackName: templateId,\n                    });\n                    break;\n                }\n              }\n            });\n            const writable = new StringWritable({});\n            formatDifferences(writable, templateDiff);\n            failures.push({\n              reason: DiagnosticReason.SNAPSHOT_FAILED,\n              message: writable.data,\n              stackName: templateId,\n              testName: this.testName,\n              config,\n            });\n          }\n        }\n      }\n    }\n\n    return {\n      diagnostics: failures,\n      destructiveChanges,\n    };\n  }\n\n  private readAssembly(dir: string): AssemblyManifestReader {\n    return AssemblyManifestReader.fromPath(dir);\n  }\n\n  /**\n  * Reduce template to a normal form where asset references have been normalized\n  *\n  * This makes it possible to compare templates if all that's different between\n  * them is the hashes of the asset values.\n  */\n  private canonicalizeTemplate(template: any, assets: string[]): any {\n    const assetsSeen = new Set<string>();\n    const stringSubstitutions = new Array<[RegExp, string]>();\n\n    // Find assets via parameters (for LegacyStackSynthesizer)\n    const paramRe = /^AssetParameters([a-zA-Z0-9]{64})(S3Bucket|S3VersionKey|ArtifactHash)([a-zA-Z0-9]{8})$/;\n    for (const paramName of Object.keys(template?.Parameters || {})) {\n      const m = paramRe.exec(paramName);\n      if (!m) { continue; }\n      if (assetsSeen.has(m[1])) { continue; }\n\n      assetsSeen.add(m[1]);\n      const ix = assetsSeen.size;\n\n      // Full parameter reference\n      stringSubstitutions.push([\n        new RegExp(`AssetParameters${m[1]}(S3Bucket|S3VersionKey|ArtifactHash)([a-zA-Z0-9]{8})`),\n        `Asset${ix}$1`,\n      ]);\n      // Substring asset hash reference\n      stringSubstitutions.push([\n        new RegExp(`${m[1]}`),\n        `Asset${ix}Hash`,\n      ]);\n    }\n\n    // find assets defined in the asset manifest\n    try {\n      assets.forEach(asset => {\n        if (!assetsSeen.has(asset)) {\n          assetsSeen.add(asset);\n          const ix = assetsSeen.size;\n          stringSubstitutions.push([\n            new RegExp(asset),\n            `Asset${ix}$1`,\n          ]);\n        }\n      });\n    } catch {\n      // if there is no asset manifest that is fine.\n    }\n\n    // Substitute them out\n    return substitute(template);\n\n    function substitute(what: any): any {\n      if (Array.isArray(what)) {\n        return what.map(substitute);\n      }\n\n      if (typeof what === 'object' && what !== null) {\n        const ret: any = {};\n        for (const [k, v] of Object.entries(what)) {\n          ret[stringSub(k)] = substitute(v);\n        }\n        return ret;\n      }\n\n      if (typeof what === 'string') {\n        return stringSub(what);\n      }\n\n      return what;\n    }\n\n    function stringSub(x: string) {\n      for (const [re, replacement] of stringSubstitutions) {\n        x = x.replace(re, replacement);\n      }\n      return x;\n    }\n  }\n}\n\nclass StringWritable extends Writable {\n  public data: string;\n  private _decoder: StringDecoder;\n  constructor(options: WritableOptions) {\n    super(options);\n    this._decoder = new StringDecoder();\n    this.data = '';\n  }\n\n  _write(chunk: any, encoding: string, callback: (error?: Error | null) => void): void {\n    if (encoding === 'buffer') {\n      chunk = this._decoder.write(chunk);\n    }\n\n    this.data += chunk;\n    callback();\n  }\n\n  _final(callback: (error?: Error | null) => void): void {\n    this.data += this._decoder.end();\n    callback();\n  }\n}\n"]}
package/lib/utils.js CHANGED
@@ -73,6 +73,7 @@ class WorkList {
73
73
  }
74
74
  done() {
75
75
  this.remaining.clear();
76
+ this.stopTimer();
76
77
  }
77
78
  stopTimer() {
78
79
  if (this.timer) {
@@ -89,4 +90,4 @@ class WorkList {
89
90
  }
90
91
  }
91
92
  exports.WorkList = WorkList;
92
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJ1dGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxnQ0FBZ0M7QUFDaEMsaURBQTBDO0FBRTFDOztHQUVHO0FBQ0gsU0FBZ0IsSUFBSSxDQUFDLFdBQXFCLEVBQUUsVUFBMEQsRUFBRztJQUN2RyxNQUFNLElBQUksR0FBRyx5QkFBUyxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUMsRUFBRSxXQUFXLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxFQUFFO1FBQzNELEtBQUssRUFBRSxDQUFDLFFBQVEsRUFBRSxNQUFNLEVBQUUsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUM7UUFDL0QsR0FBRyxFQUFFO1lBQ0gsR0FBRyxPQUFPLENBQUMsR0FBRztZQUNkLEdBQUcsT0FBTyxDQUFDLEdBQUc7U0FDZjtRQUNELEdBQUcsRUFBRSxPQUFPLENBQUMsR0FBRztLQUNqQixDQUFDLENBQUM7SUFFSCxJQUFJLElBQUksQ0FBQyxLQUFLLEVBQUU7UUFBRSxNQUFNLElBQUksQ0FBQyxLQUFLLENBQUM7S0FBRTtJQUNyQyxJQUFJLElBQUksQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFO1FBQ3JCLElBQUksT0FBTyxDQUFDLE1BQU0sRUFBRSxFQUFFLGlDQUFpQztZQUNyRCxPQUFPLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7U0FDbkM7UUFDRCxNQUFNLElBQUksS0FBSyxDQUFDLHVCQUF1QixJQUFJLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxVQUFVLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQyxDQUFDLENBQUMsVUFBVSxJQUFJLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQyxDQUFDO0tBQzNHO0lBRUQsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLENBQUMsSUFBSSxFQUFFLENBQUM7SUFFcEQsT0FBTyxNQUFNLENBQUM7QUFDaEIsQ0FBQztBQXJCRCxvQkFxQkM7QUFFRDs7R0FFRztBQUNILFNBQWdCLE9BQU8sQ0FBSSxFQUFTO0lBQ2xDLE9BQU8sS0FBSyxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLEVBQUUsRUFBRSxFQUFFLENBQUMsQ0FBQztBQUM5QyxDQUFDO0FBRkQsMEJBRUM7QUFFRDs7R0FFRztBQUNILFNBQWdCLEtBQUssQ0FBQyxRQUFrQjtJQUN0QyxPQUFPLFFBQVEsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO0FBQ2hELENBQUM7QUFGRCxzQkFFQztBQUVEOztHQUVHO0FBQ0gsU0FBZ0IsTUFBTSxDQUFDLE9BQWU7SUFDcEMsTUFBTSxNQUFNLEdBQUcsT0FBTyxDQUFDLEtBQUssQ0FBQyx1QkFBdUIsQ0FBQyxDQUFDO0lBQ3RELE9BQU8sTUFBTSxhQUFOLE1BQU0sY0FBTixNQUFNLEdBQUksRUFBRSxDQUFDO0FBQ3RCLENBQUM7QUFIRCx3QkFHQztBQUdEOzs7O0dBSUc7QUFDSCxNQUFhLFFBQVE7SUFLbkIsWUFBNkIsS0FBVSxFQUFtQixVQUE4QixFQUFFOztRQUE3RCxVQUFLLEdBQUwsS0FBSyxDQUFLO1FBQW1CLFlBQU8sR0FBUCxPQUFPLENBQXlCO1FBSnpFLGNBQVMsR0FBRyxJQUFJLEdBQUcsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFLL0MsSUFBSSxDQUFDLE9BQU8sU0FBRyxPQUFPLENBQUMsT0FBTyxtQ0FBSSxLQUFNLENBQUM7UUFDekMsSUFBSSxDQUFDLGFBQWEsRUFBRSxDQUFDO0lBQ3ZCLENBQUM7SUFFTSxRQUFRLENBQUMsSUFBTztRQUNyQixJQUFJLENBQUMsU0FBUyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUM1QixJQUFJLENBQUMsU0FBUyxFQUFFLENBQUM7UUFDakIsSUFBSSxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksR0FBRyxDQUFDLEVBQUU7WUFDM0IsSUFBSSxDQUFDLGFBQWEsRUFBRSxDQUFDO1NBQ3RCO0lBQ0gsQ0FBQztJQUVNLElBQUk7UUFDVCxJQUFJLENBQUMsU0FBUyxDQUFDLEtBQUssRUFBRSxDQUFDO0lBQ3pCLENBQUM7SUFFTyxTQUFTO1FBQ2YsSUFBSSxJQUFJLENBQUMsS0FBSyxFQUFFO1lBQ2QsWUFBWSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsQ0FBQztZQUN6QixJQUFJLENBQUMsS0FBSyxHQUFHLFNBQVMsQ0FBQztTQUN4QjtJQUNILENBQUM7SUFFTyxhQUFhO1FBQ25CLElBQUksQ0FBQyxLQUFLLEdBQUcsVUFBVSxDQUFDLEdBQUcsRUFBRSxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsRUFBRSxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDN0QsQ0FBQztJQUVPLE1BQU07O1FBQ1osTUFBQSxNQUFBLElBQUksQ0FBQyxPQUFPLEVBQUMsU0FBUyxtREFBRyxJQUFJLENBQUMsU0FBUyxFQUFFO0lBQzNDLENBQUM7Q0FDRjtBQXBDRCw0QkFvQ0MiLCJzb3VyY2VzQ29udGVudCI6WyIvLyBIZWxwZXIgZnVuY3Rpb25zIGZvciBDREsgRXhlY1xuaW1wb3J0IHsgc3Bhd25TeW5jIH0gZnJvbSAnY2hpbGRfcHJvY2Vzcyc7XG5cbi8qKlxuICogT3VyIG93biBleGVjdXRlIGZ1bmN0aW9uIHdoaWNoIGRvZXNuJ3QgdXNlIHNoZWxscyBhbmQgc3RyaW5ncy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGV4ZWMoY29tbWFuZExpbmU6IHN0cmluZ1tdLCBvcHRpb25zOiB7IGN3ZD86IHN0cmluZywgdmVyYm9zZT86IGJvb2xlYW4sIGVudj86IGFueSB9ID0geyB9KTogYW55IHtcbiAgY29uc3QgcHJvYyA9IHNwYXduU3luYyhjb21tYW5kTGluZVswXSwgY29tbWFuZExpbmUuc2xpY2UoMSksIHtcbiAgICBzdGRpbzogWydpZ25vcmUnLCAncGlwZScsIG9wdGlvbnMudmVyYm9zZSA/ICdpbmhlcml0JyA6ICdwaXBlJ10sIC8vIGluaGVyaXQgU1RERVJSIGluIHZlcmJvc2UgbW9kZVxuICAgIGVudjoge1xuICAgICAgLi4ucHJvY2Vzcy5lbnYsXG4gICAgICAuLi5vcHRpb25zLmVudixcbiAgICB9LFxuICAgIGN3ZDogb3B0aW9ucy5jd2QsXG4gIH0pO1xuXG4gIGlmIChwcm9jLmVycm9yKSB7IHRocm93IHByb2MuZXJyb3I7IH1cbiAgaWYgKHByb2Muc3RhdHVzICE9PSAwKSB7XG4gICAgaWYgKHByb2Nlc3Muc3RkZXJyKSB7IC8vIHdpbGwgYmUgJ251bGwnIGluIHZlcmJvc2UgbW9kZVxuICAgICAgcHJvY2Vzcy5zdGRlcnIud3JpdGUocHJvYy5zdGRlcnIpO1xuICAgIH1cbiAgICB0aHJvdyBuZXcgRXJyb3IoYENvbW1hbmQgZXhpdGVkIHdpdGggJHtwcm9jLnN0YXR1cyA/IGBzdGF0dXMgJHtwcm9jLnN0YXR1c31gIDogYHNpZ25hbCAke3Byb2Muc2lnbmFsfWB9YCk7XG4gIH1cblxuICBjb25zdCBvdXRwdXQgPSBwcm9jLnN0ZG91dC50b1N0cmluZygndXRmLTgnKS50cmltKCk7XG5cbiAgcmV0dXJuIG91dHB1dDtcbn1cblxuLyoqXG4gKiBGbGF0dGVuIGEgbGlzdCBvZiBsaXN0cyBpbnRvIGEgbGlzdCBvZiBlbGVtZW50c1xuICovXG5leHBvcnQgZnVuY3Rpb24gZmxhdHRlbjxUPih4czogVFtdW10pOiBUW10ge1xuICByZXR1cm4gQXJyYXkucHJvdG90eXBlLmNvbmNhdC5hcHBseShbXSwgeHMpO1xufVxuXG4vKipcbiAqIENoYWluIGNvbW1hbmRzXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjaGFpbihjb21tYW5kczogc3RyaW5nW10pOiBzdHJpbmcge1xuICByZXR1cm4gY29tbWFuZHMuZmlsdGVyKGMgPT4gISFjKS5qb2luKCcgJiYgJyk7XG59XG5cbi8qKlxuICogU3BsaXQgY29tbWFuZCB0byBjaHVua3MgYnkgc3BhY2VcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGNodW5rcyhjb21tYW5kOiBzdHJpbmcpOiBzdHJpbmdbXSB7XG4gIGNvbnN0IHJlc3VsdCA9IGNvbW1hbmQubWF0Y2goLyg/OlteXFxzXCJdK3xcIlteXCJdKlwiKSsvZyk7XG4gIHJldHVybiByZXN1bHQgPz8gW107XG59XG5cblxuLyoqXG4gKiBBIGNsYXNzIGhvbGRpbmcgYSBzZXQgb2YgaXRlbXMgd2hpY2ggYXJlIGJlaW5nIGNyb3NzZWQgb2ZmIGluIHRpbWVcbiAqXG4gKiBJZiBpdCB0YWtlcyB0b28gbG9uZyB0byBjcm9zcyBvZmYgYSBuZXcgaXRlbSwgcHJpbnQgdGhlIGxpc3QuXG4gKi9cbmV4cG9ydCBjbGFzcyBXb3JrTGlzdDxBPiB7XG4gIHByaXZhdGUgcmVhZG9ubHkgcmVtYWluaW5nID0gbmV3IFNldCh0aGlzLml0ZW1zKTtcbiAgcHJpdmF0ZSByZWFkb25seSB0aW1lb3V0OiBudW1iZXI7XG4gIHByaXZhdGUgdGltZXI/OiBOb2RlSlMuVGltZW91dDtcblxuICBjb25zdHJ1Y3Rvcihwcml2YXRlIHJlYWRvbmx5IGl0ZW1zOiBBW10sIHByaXZhdGUgcmVhZG9ubHkgb3B0aW9uczogV29ya0xpc3RPcHRpb25zPEE+ID0ge30pIHtcbiAgICB0aGlzLnRpbWVvdXQgPSBvcHRpb25zLnRpbWVvdXQgPz8gNjBfMDAwO1xuICAgIHRoaXMuc2NoZWR1bGVUaW1lcigpO1xuICB9XG5cbiAgcHVibGljIGNyb3NzT2ZmKGl0ZW06IEEpIHtcbiAgICB0aGlzLnJlbWFpbmluZy5kZWxldGUoaXRlbSk7XG4gICAgdGhpcy5zdG9wVGltZXIoKTtcbiAgICBpZiAodGhpcy5yZW1haW5pbmcuc2l6ZSA+IDApIHtcbiAgICAgIHRoaXMuc2NoZWR1bGVUaW1lcigpO1xuICAgIH1cbiAgfVxuXG4gIHB1YmxpYyBkb25lKCkge1xuICAgIHRoaXMucmVtYWluaW5nLmNsZWFyKCk7XG4gIH1cblxuICBwcml2YXRlIHN0b3BUaW1lcigpIHtcbiAgICBpZiAodGhpcy50aW1lcikge1xuICAgICAgY2xlYXJUaW1lb3V0KHRoaXMudGltZXIpO1xuICAgICAgdGhpcy50aW1lciA9IHVuZGVmaW5lZDtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIHNjaGVkdWxlVGltZXIoKSB7XG4gICAgdGhpcy50aW1lciA9IHNldFRpbWVvdXQoKCkgPT4gdGhpcy5yZXBvcnQoKSwgdGhpcy50aW1lb3V0KTtcbiAgfVxuXG4gIHByaXZhdGUgcmVwb3J0KCkge1xuICAgIHRoaXMub3B0aW9ucy5vblRpbWVvdXQ/Lih0aGlzLnJlbWFpbmluZyk7XG4gIH1cbn1cblxuZXhwb3J0IGludGVyZmFjZSBXb3JrTGlzdE9wdGlvbnM8QT4ge1xuICAvKipcbiAgICogV2hlbiB0byByZXBseSB3aXRoIHJlbWFpbmluZyBpdGVtc1xuICAgKlxuICAgKiBAZGVmYXVsdCA2MDAwMFxuICAgKi9cbiAgcmVhZG9ubHkgdGltZW91dD86IG51bWJlcjtcblxuICAvKipcbiAgICogRnVuY3Rpb24gdG8gY2FsbCB3aGVuIHRpbWVvdXQgaGl0c1xuICAgKi9cbiAgcmVhZG9ubHkgb25UaW1lb3V0PzogKHg6IFNldDxBPikgPT4gdm9pZDtcbn0iXX0=
93
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyJ1dGlscy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQSxnQ0FBZ0M7QUFDaEMsaURBQTBDO0FBRTFDOztHQUVHO0FBQ0gsU0FBZ0IsSUFBSSxDQUFDLFdBQXFCLEVBQUUsVUFBMEQsRUFBRztJQUN2RyxNQUFNLElBQUksR0FBRyx5QkFBUyxDQUFDLFdBQVcsQ0FBQyxDQUFDLENBQUMsRUFBRSxXQUFXLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxFQUFFO1FBQzNELEtBQUssRUFBRSxDQUFDLFFBQVEsRUFBRSxNQUFNLEVBQUUsT0FBTyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUM7UUFDL0QsR0FBRyxFQUFFO1lBQ0gsR0FBRyxPQUFPLENBQUMsR0FBRztZQUNkLEdBQUcsT0FBTyxDQUFDLEdBQUc7U0FDZjtRQUNELEdBQUcsRUFBRSxPQUFPLENBQUMsR0FBRztLQUNqQixDQUFDLENBQUM7SUFFSCxJQUFJLElBQUksQ0FBQyxLQUFLLEVBQUU7UUFBRSxNQUFNLElBQUksQ0FBQyxLQUFLLENBQUM7S0FBRTtJQUNyQyxJQUFJLElBQUksQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFO1FBQ3JCLElBQUksT0FBTyxDQUFDLE1BQU0sRUFBRSxFQUFFLGlDQUFpQztZQUNyRCxPQUFPLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7U0FDbkM7UUFDRCxNQUFNLElBQUksS0FBSyxDQUFDLHVCQUF1QixJQUFJLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxVQUFVLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQyxDQUFDLENBQUMsVUFBVSxJQUFJLENBQUMsTUFBTSxFQUFFLEVBQUUsQ0FBQyxDQUFDO0tBQzNHO0lBRUQsTUFBTSxNQUFNLEdBQUcsSUFBSSxDQUFDLE1BQU0sQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLENBQUMsSUFBSSxFQUFFLENBQUM7SUFFcEQsT0FBTyxNQUFNLENBQUM7QUFDaEIsQ0FBQztBQXJCRCxvQkFxQkM7QUFFRDs7R0FFRztBQUNILFNBQWdCLE9BQU8sQ0FBSSxFQUFTO0lBQ2xDLE9BQU8sS0FBSyxDQUFDLFNBQVMsQ0FBQyxNQUFNLENBQUMsS0FBSyxDQUFDLEVBQUUsRUFBRSxFQUFFLENBQUMsQ0FBQztBQUM5QyxDQUFDO0FBRkQsMEJBRUM7QUFFRDs7R0FFRztBQUNILFNBQWdCLEtBQUssQ0FBQyxRQUFrQjtJQUN0QyxPQUFPLFFBQVEsQ0FBQyxNQUFNLENBQUMsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO0FBQ2hELENBQUM7QUFGRCxzQkFFQztBQUVEOztHQUVHO0FBQ0gsU0FBZ0IsTUFBTSxDQUFDLE9BQWU7SUFDcEMsTUFBTSxNQUFNLEdBQUcsT0FBTyxDQUFDLEtBQUssQ0FBQyx1QkFBdUIsQ0FBQyxDQUFDO0lBQ3RELE9BQU8sTUFBTSxhQUFOLE1BQU0sY0FBTixNQUFNLEdBQUksRUFBRSxDQUFDO0FBQ3RCLENBQUM7QUFIRCx3QkFHQztBQUdEOzs7O0dBSUc7QUFDSCxNQUFhLFFBQVE7SUFLbkIsWUFBNkIsS0FBVSxFQUFtQixVQUE4QixFQUFFOztRQUE3RCxVQUFLLEdBQUwsS0FBSyxDQUFLO1FBQW1CLFlBQU8sR0FBUCxPQUFPLENBQXlCO1FBSnpFLGNBQVMsR0FBRyxJQUFJLEdBQUcsQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFLL0MsSUFBSSxDQUFDLE9BQU8sU0FBRyxPQUFPLENBQUMsT0FBTyxtQ0FBSSxLQUFNLENBQUM7UUFDekMsSUFBSSxDQUFDLGFBQWEsRUFBRSxDQUFDO0lBQ3ZCLENBQUM7SUFFTSxRQUFRLENBQUMsSUFBTztRQUNyQixJQUFJLENBQUMsU0FBUyxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsQ0FBQztRQUM1QixJQUFJLENBQUMsU0FBUyxFQUFFLENBQUM7UUFDakIsSUFBSSxJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksR0FBRyxDQUFDLEVBQUU7WUFDM0IsSUFBSSxDQUFDLGFBQWEsRUFBRSxDQUFDO1NBQ3RCO0lBQ0gsQ0FBQztJQUVNLElBQUk7UUFDVCxJQUFJLENBQUMsU0FBUyxDQUFDLEtBQUssRUFBRSxDQUFDO1FBQ3ZCLElBQUksQ0FBQyxTQUFTLEVBQUUsQ0FBQztJQUNuQixDQUFDO0lBRU8sU0FBUztRQUNmLElBQUksSUFBSSxDQUFDLEtBQUssRUFBRTtZQUNkLFlBQVksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7WUFDekIsSUFBSSxDQUFDLEtBQUssR0FBRyxTQUFTLENBQUM7U0FDeEI7SUFDSCxDQUFDO0lBRU8sYUFBYTtRQUNuQixJQUFJLENBQUMsS0FBSyxHQUFHLFVBQVUsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsTUFBTSxFQUFFLEVBQUUsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBQzdELENBQUM7SUFFTyxNQUFNOztRQUNaLE1BQUEsTUFBQSxJQUFJLENBQUMsT0FBTyxFQUFDLFNBQVMsbURBQUcsSUFBSSxDQUFDLFNBQVMsRUFBRTtJQUMzQyxDQUFDO0NBQ0Y7QUFyQ0QsNEJBcUNDIiwic291cmNlc0NvbnRlbnQiOlsiLy8gSGVscGVyIGZ1bmN0aW9ucyBmb3IgQ0RLIEV4ZWNcbmltcG9ydCB7IHNwYXduU3luYyB9IGZyb20gJ2NoaWxkX3Byb2Nlc3MnO1xuXG4vKipcbiAqIE91ciBvd24gZXhlY3V0ZSBmdW5jdGlvbiB3aGljaCBkb2Vzbid0IHVzZSBzaGVsbHMgYW5kIHN0cmluZ3MuXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBleGVjKGNvbW1hbmRMaW5lOiBzdHJpbmdbXSwgb3B0aW9uczogeyBjd2Q/OiBzdHJpbmcsIHZlcmJvc2U/OiBib29sZWFuLCBlbnY/OiBhbnkgfSA9IHsgfSk6IGFueSB7XG4gIGNvbnN0IHByb2MgPSBzcGF3blN5bmMoY29tbWFuZExpbmVbMF0sIGNvbW1hbmRMaW5lLnNsaWNlKDEpLCB7XG4gICAgc3RkaW86IFsnaWdub3JlJywgJ3BpcGUnLCBvcHRpb25zLnZlcmJvc2UgPyAnaW5oZXJpdCcgOiAncGlwZSddLCAvLyBpbmhlcml0IFNUREVSUiBpbiB2ZXJib3NlIG1vZGVcbiAgICBlbnY6IHtcbiAgICAgIC4uLnByb2Nlc3MuZW52LFxuICAgICAgLi4ub3B0aW9ucy5lbnYsXG4gICAgfSxcbiAgICBjd2Q6IG9wdGlvbnMuY3dkLFxuICB9KTtcblxuICBpZiAocHJvYy5lcnJvcikgeyB0aHJvdyBwcm9jLmVycm9yOyB9XG4gIGlmIChwcm9jLnN0YXR1cyAhPT0gMCkge1xuICAgIGlmIChwcm9jZXNzLnN0ZGVycikgeyAvLyB3aWxsIGJlICdudWxsJyBpbiB2ZXJib3NlIG1vZGVcbiAgICAgIHByb2Nlc3Muc3RkZXJyLndyaXRlKHByb2Muc3RkZXJyKTtcbiAgICB9XG4gICAgdGhyb3cgbmV3IEVycm9yKGBDb21tYW5kIGV4aXRlZCB3aXRoICR7cHJvYy5zdGF0dXMgPyBgc3RhdHVzICR7cHJvYy5zdGF0dXN9YCA6IGBzaWduYWwgJHtwcm9jLnNpZ25hbH1gfWApO1xuICB9XG5cbiAgY29uc3Qgb3V0cHV0ID0gcHJvYy5zdGRvdXQudG9TdHJpbmcoJ3V0Zi04JykudHJpbSgpO1xuXG4gIHJldHVybiBvdXRwdXQ7XG59XG5cbi8qKlxuICogRmxhdHRlbiBhIGxpc3Qgb2YgbGlzdHMgaW50byBhIGxpc3Qgb2YgZWxlbWVudHNcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGZsYXR0ZW48VD4oeHM6IFRbXVtdKTogVFtdIHtcbiAgcmV0dXJuIEFycmF5LnByb3RvdHlwZS5jb25jYXQuYXBwbHkoW10sIHhzKTtcbn1cblxuLyoqXG4gKiBDaGFpbiBjb21tYW5kc1xuICovXG5leHBvcnQgZnVuY3Rpb24gY2hhaW4oY29tbWFuZHM6IHN0cmluZ1tdKTogc3RyaW5nIHtcbiAgcmV0dXJuIGNvbW1hbmRzLmZpbHRlcihjID0+ICEhYykuam9pbignICYmICcpO1xufVxuXG4vKipcbiAqIFNwbGl0IGNvbW1hbmQgdG8gY2h1bmtzIGJ5IHNwYWNlXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBjaHVua3MoY29tbWFuZDogc3RyaW5nKTogc3RyaW5nW10ge1xuICBjb25zdCByZXN1bHQgPSBjb21tYW5kLm1hdGNoKC8oPzpbXlxcc1wiXSt8XCJbXlwiXSpcIikrL2cpO1xuICByZXR1cm4gcmVzdWx0ID8/IFtdO1xufVxuXG5cbi8qKlxuICogQSBjbGFzcyBob2xkaW5nIGEgc2V0IG9mIGl0ZW1zIHdoaWNoIGFyZSBiZWluZyBjcm9zc2VkIG9mZiBpbiB0aW1lXG4gKlxuICogSWYgaXQgdGFrZXMgdG9vIGxvbmcgdG8gY3Jvc3Mgb2ZmIGEgbmV3IGl0ZW0sIHByaW50IHRoZSBsaXN0LlxuICovXG5leHBvcnQgY2xhc3MgV29ya0xpc3Q8QT4ge1xuICBwcml2YXRlIHJlYWRvbmx5IHJlbWFpbmluZyA9IG5ldyBTZXQodGhpcy5pdGVtcyk7XG4gIHByaXZhdGUgcmVhZG9ubHkgdGltZW91dDogbnVtYmVyO1xuICBwcml2YXRlIHRpbWVyPzogTm9kZUpTLlRpbWVvdXQ7XG5cbiAgY29uc3RydWN0b3IocHJpdmF0ZSByZWFkb25seSBpdGVtczogQVtdLCBwcml2YXRlIHJlYWRvbmx5IG9wdGlvbnM6IFdvcmtMaXN0T3B0aW9uczxBPiA9IHt9KSB7XG4gICAgdGhpcy50aW1lb3V0ID0gb3B0aW9ucy50aW1lb3V0ID8/IDYwXzAwMDtcbiAgICB0aGlzLnNjaGVkdWxlVGltZXIoKTtcbiAgfVxuXG4gIHB1YmxpYyBjcm9zc09mZihpdGVtOiBBKSB7XG4gICAgdGhpcy5yZW1haW5pbmcuZGVsZXRlKGl0ZW0pO1xuICAgIHRoaXMuc3RvcFRpbWVyKCk7XG4gICAgaWYgKHRoaXMucmVtYWluaW5nLnNpemUgPiAwKSB7XG4gICAgICB0aGlzLnNjaGVkdWxlVGltZXIoKTtcbiAgICB9XG4gIH1cblxuICBwdWJsaWMgZG9uZSgpIHtcbiAgICB0aGlzLnJlbWFpbmluZy5jbGVhcigpO1xuICAgIHRoaXMuc3RvcFRpbWVyKCk7XG4gIH1cblxuICBwcml2YXRlIHN0b3BUaW1lcigpIHtcbiAgICBpZiAodGhpcy50aW1lcikge1xuICAgICAgY2xlYXJUaW1lb3V0KHRoaXMudGltZXIpO1xuICAgICAgdGhpcy50aW1lciA9IHVuZGVmaW5lZDtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIHNjaGVkdWxlVGltZXIoKSB7XG4gICAgdGhpcy50aW1lciA9IHNldFRpbWVvdXQoKCkgPT4gdGhpcy5yZXBvcnQoKSwgdGhpcy50aW1lb3V0KTtcbiAgfVxuXG4gIHByaXZhdGUgcmVwb3J0KCkge1xuICAgIHRoaXMub3B0aW9ucy5vblRpbWVvdXQ/Lih0aGlzLnJlbWFpbmluZyk7XG4gIH1cbn1cblxuZXhwb3J0IGludGVyZmFjZSBXb3JrTGlzdE9wdGlvbnM8QT4ge1xuICAvKipcbiAgICogV2hlbiB0byByZXBseSB3aXRoIHJlbWFpbmluZyBpdGVtc1xuICAgKlxuICAgKiBAZGVmYXVsdCA2MDAwMFxuICAgKi9cbiAgcmVhZG9ubHkgdGltZW91dD86IG51bWJlcjtcblxuICAvKipcbiAgICogRnVuY3Rpb24gdG8gY2FsbCB3aGVuIHRpbWVvdXQgaGl0c1xuICAgKi9cbiAgcmVhZG9ubHkgb25UaW1lb3V0PzogKHg6IFNldDxBPikgPT4gdm9pZDtcbn0iXX0=
@@ -191,6 +191,10 @@ export interface Diagnostic {
191
191
  * The name of the test
192
192
  */
193
193
  readonly testName: string;
194
+ /**
195
+ * The name of the stack
196
+ */
197
+ readonly stackName: string;
194
198
  /**
195
199
  * The diagnostic message
196
200
  */
@@ -207,6 +211,10 @@ export interface Diagnostic {
207
211
  * Additional messages to print
208
212
  */
209
213
  readonly additionalMessages?: string[];
214
+ /**
215
+ * Relevant config options that were used for the integ test
216
+ */
217
+ readonly config?: Record<string, any>;
210
218
  }
211
219
  export declare function printSummary(total: number, failed: number): void;
212
220
  /**
@@ -106,4 +106,4 @@ function printLaggards(testNames) {
106
106
  logger.print(chalk.grey(parts.filter(x => x).join(' ')));
107
107
  }
108
108
  exports.printLaggards = printLaggards;
109
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["common.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'util';\nimport { ResourceImpact } from '@aws-cdk/cloudformation-diff';\nimport * as chalk from 'chalk';\nimport * as logger from '../logger';\nimport { IntegTestInfo } from '../runner/integration-tests';\n\n/**\n * The aggregate results from running assertions on a test case\n */\nexport type AssertionResults = { [id: string]: AssertionResult };\n\n/**\n * The result of an individual assertion\n */\nexport interface AssertionResult {\n  /**\n   * The assertion message. If the assertion failed, this will\n   * include the reason.\n   */\n  readonly message: string;\n\n  /**\n   * Whether the assertion succeeded or failed\n   */\n  readonly status: 'success' | 'fail';\n}\n\n/**\n * Config for an integration test\n */\nexport interface IntegTestWorkerConfig extends IntegTestInfo {\n  /**\n   * A list of any destructive changes\n   *\n   * @default []\n   */\n  readonly destructiveChanges?: DestructiveChange[];\n}\n\n/**\n * Information on any destructive changes\n */\nexport interface DestructiveChange {\n  /**\n   * The logicalId of the resource with a destructive change\n   */\n  readonly logicalId: string;\n\n  /**\n   * The name of the stack that contains the destructive change\n   */\n  readonly stackName: string;\n\n  /**\n   * The impact of the destructive change\n   */\n  readonly impact: ResourceImpact;\n}\n\n\n/**\n * Represents integration tests metrics for a given worker\n */\nexport interface IntegRunnerMetrics {\n  /**\n   * The region the test was run in\n   */\n  readonly region: string;\n\n  /**\n   * The total duration of the worker.\n   * This will be the sum of all individual test durations\n   */\n  readonly duration: number;\n\n  /**\n   * Contains the duration of individual tests that the\n   * worker executed.\n   *\n   * Map of testName to duration.\n   */\n  readonly tests: { [testName: string]: number };\n\n  /**\n   * The profile that was used to run the test\n   *\n   * @default - default profile\n   */\n  readonly profile?: string;\n}\n\nexport interface SnapshotVerificationOptions {\n  /**\n   * Retain failed snapshot comparisons\n   *\n   * @default false\n   */\n  readonly retain?: boolean;\n\n  /**\n   * Verbose mode\n   *\n   * @default false\n   */\n  readonly verbose?: boolean;\n}\n\n/**\n * Integration test results\n */\nexport interface IntegBatchResponse {\n  /**\n   * List of failed tests\n   */\n  readonly failedTests: IntegTestInfo[];\n\n  /**\n   * List of Integration test metrics. Each entry in the\n   * list represents metrics from a single worker (account + region).\n   */\n  readonly metrics: IntegRunnerMetrics[];\n}\n\n/**\n * Common options for running integration tests\n */\nexport interface IntegTestOptions {\n  /**\n   * A list of integration tests to run\n   * in this batch\n   */\n  readonly tests: IntegTestWorkerConfig[];\n\n  /**\n   * Whether or not to destroy the stacks at the\n   * end of the test\n   *\n   * @default true\n   */\n  readonly clean?: boolean;\n\n  /**\n   * When this is set to `true` the snapshot will\n   * be created _without_ running the integration test\n   * The resulting snapshot SHOULD NOT be checked in\n   *\n   * @default false\n   */\n  readonly dryRun?: boolean;\n\n  /**\n   * The level of verbosity for logging.\n   * Higher number means more output.\n   *\n   * @default 0\n   */\n  readonly verbosity?: number;\n\n  /**\n   * If this is set to true then the stack update workflow will be disabled\n   *\n   * @default true\n   */\n  readonly updateWorkflow?: boolean;\n}\n\n/**\n * Represents possible reasons for a diagnostic\n */\nexport enum DiagnosticReason {\n  /**\n   * The integration test failed because there\n   * is not existing snapshot\n   */\n  NO_SNAPSHOT = 'NO_SNAPSHOT',\n\n  /**\n   * The integration test failed\n   */\n  TEST_FAILED = 'TEST_FAILED',\n\n  /**\n   * There was an error running the integration test\n   */\n  TEST_ERROR = 'TEST_ERROR',\n\n  /**\n   * The snapshot test failed because the actual\n   * snapshot was different than the expected snapshot\n   */\n  SNAPSHOT_FAILED = 'SNAPSHOT_FAILED',\n\n  /**\n   * The snapshot test failed because there was an error executing it\n   */\n  SNAPSHOT_ERROR = 'SNAPSHOT_ERROR',\n\n  /**\n   * The snapshot test succeeded\n   */\n  SNAPSHOT_SUCCESS = 'SNAPSHOT_SUCCESS',\n\n  /**\n   * The integration test succeeded\n   */\n  TEST_SUCCESS = 'TEST_SUCCESS',\n\n  /**\n   * The assertion failed\n   */\n  ASSERTION_FAILED = 'ASSERTION_FAILED',\n}\n\n/**\n * Integration test diagnostics\n * This is used to report back the status of each test\n */\nexport interface Diagnostic {\n  /**\n   * The name of the test\n   */\n  readonly testName: string;\n\n  /**\n   * The diagnostic message\n   */\n  readonly message: string;\n\n  /**\n   * The time it took to run the test\n   */\n  readonly duration?: number;\n\n  /**\n   * The reason for the diagnostic\n   */\n  readonly reason: DiagnosticReason;\n\n  /**\n   * Additional messages to print\n   */\n  readonly additionalMessages?: string[];\n}\n\nexport function printSummary(total: number, failed: number): void {\n  if (failed > 0) {\n    logger.print('%s:    %s %s, %s total', chalk.bold('Tests'), chalk.red(failed), chalk.red('failed'), total);\n  } else {\n    logger.print('%s:    %s %s, %s total', chalk.bold('Tests'), chalk.green(total), chalk.green('passed'), total);\n  }\n}\n\n/**\n * Format the assertion results so that the results can be\n * printed\n */\nexport function formatAssertionResults(results: AssertionResults): string {\n  return Object.entries(results)\n    .map(([id, result]) => format('%s%s', id, result.status === 'success' ? ` - ${result.status}` : `\\n${result.message}`))\n    .join('\\n      ');\n}\n\n/**\n * Print out the results from tests\n */\nexport function printResults(diagnostic: Diagnostic): void {\n  switch (diagnostic.reason) {\n    case DiagnosticReason.SNAPSHOT_SUCCESS:\n      logger.success('  UNCHANGED  %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));\n      break;\n    case DiagnosticReason.TEST_SUCCESS:\n      logger.success('  SUCCESS    %s %s\\n      ', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.NO_SNAPSHOT:\n      logger.error('  NEW        %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));\n      break;\n    case DiagnosticReason.SNAPSHOT_FAILED:\n      logger.error('  CHANGED    %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.SNAPSHOT_ERROR:\n    case DiagnosticReason.TEST_ERROR:\n      logger.error('  ERROR      %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.TEST_FAILED:\n      logger.error('  FAILED     %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.ASSERTION_FAILED:\n      logger.error('  ASSERT     %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n  }\n  for (const addl of diagnostic.additionalMessages ?? []) {\n    logger.print(`      ${addl}`);\n  }\n}\n\nexport function printLaggards(testNames: Set<string>) {\n  const parts = [\n    '  ',\n    `Waiting for ${testNames.size} more`,\n    testNames.size < 10 ? ['(', Array.from(testNames).join(', '), ')'].join('') : '',\n  ];\n\n  logger.print(chalk.grey(parts.filter(x => x).join(' ')));\n}\n"]}
109
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["common.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'util';\nimport { ResourceImpact } from '@aws-cdk/cloudformation-diff';\nimport * as chalk from 'chalk';\nimport * as logger from '../logger';\nimport { IntegTestInfo } from '../runner/integration-tests';\n\n/**\n * The aggregate results from running assertions on a test case\n */\nexport type AssertionResults = { [id: string]: AssertionResult };\n\n/**\n * The result of an individual assertion\n */\nexport interface AssertionResult {\n  /**\n   * The assertion message. If the assertion failed, this will\n   * include the reason.\n   */\n  readonly message: string;\n\n  /**\n   * Whether the assertion succeeded or failed\n   */\n  readonly status: 'success' | 'fail';\n}\n\n/**\n * Config for an integration test\n */\nexport interface IntegTestWorkerConfig extends IntegTestInfo {\n  /**\n   * A list of any destructive changes\n   *\n   * @default []\n   */\n  readonly destructiveChanges?: DestructiveChange[];\n}\n\n/**\n * Information on any destructive changes\n */\nexport interface DestructiveChange {\n  /**\n   * The logicalId of the resource with a destructive change\n   */\n  readonly logicalId: string;\n\n  /**\n   * The name of the stack that contains the destructive change\n   */\n  readonly stackName: string;\n\n  /**\n   * The impact of the destructive change\n   */\n  readonly impact: ResourceImpact;\n}\n\n\n/**\n * Represents integration tests metrics for a given worker\n */\nexport interface IntegRunnerMetrics {\n  /**\n   * The region the test was run in\n   */\n  readonly region: string;\n\n  /**\n   * The total duration of the worker.\n   * This will be the sum of all individual test durations\n   */\n  readonly duration: number;\n\n  /**\n   * Contains the duration of individual tests that the\n   * worker executed.\n   *\n   * Map of testName to duration.\n   */\n  readonly tests: { [testName: string]: number };\n\n  /**\n   * The profile that was used to run the test\n   *\n   * @default - default profile\n   */\n  readonly profile?: string;\n}\n\nexport interface SnapshotVerificationOptions {\n  /**\n   * Retain failed snapshot comparisons\n   *\n   * @default false\n   */\n  readonly retain?: boolean;\n\n  /**\n   * Verbose mode\n   *\n   * @default false\n   */\n  readonly verbose?: boolean;\n}\n\n/**\n * Integration test results\n */\nexport interface IntegBatchResponse {\n  /**\n   * List of failed tests\n   */\n  readonly failedTests: IntegTestInfo[];\n\n  /**\n   * List of Integration test metrics. Each entry in the\n   * list represents metrics from a single worker (account + region).\n   */\n  readonly metrics: IntegRunnerMetrics[];\n}\n\n/**\n * Common options for running integration tests\n */\nexport interface IntegTestOptions {\n  /**\n   * A list of integration tests to run\n   * in this batch\n   */\n  readonly tests: IntegTestWorkerConfig[];\n\n  /**\n   * Whether or not to destroy the stacks at the\n   * end of the test\n   *\n   * @default true\n   */\n  readonly clean?: boolean;\n\n  /**\n   * When this is set to `true` the snapshot will\n   * be created _without_ running the integration test\n   * The resulting snapshot SHOULD NOT be checked in\n   *\n   * @default false\n   */\n  readonly dryRun?: boolean;\n\n  /**\n   * The level of verbosity for logging.\n   * Higher number means more output.\n   *\n   * @default 0\n   */\n  readonly verbosity?: number;\n\n  /**\n   * If this is set to true then the stack update workflow will be disabled\n   *\n   * @default true\n   */\n  readonly updateWorkflow?: boolean;\n}\n\n/**\n * Represents possible reasons for a diagnostic\n */\nexport enum DiagnosticReason {\n  /**\n   * The integration test failed because there\n   * is not existing snapshot\n   */\n  NO_SNAPSHOT = 'NO_SNAPSHOT',\n\n  /**\n   * The integration test failed\n   */\n  TEST_FAILED = 'TEST_FAILED',\n\n  /**\n   * There was an error running the integration test\n   */\n  TEST_ERROR = 'TEST_ERROR',\n\n  /**\n   * The snapshot test failed because the actual\n   * snapshot was different than the expected snapshot\n   */\n  SNAPSHOT_FAILED = 'SNAPSHOT_FAILED',\n\n  /**\n   * The snapshot test failed because there was an error executing it\n   */\n  SNAPSHOT_ERROR = 'SNAPSHOT_ERROR',\n\n  /**\n   * The snapshot test succeeded\n   */\n  SNAPSHOT_SUCCESS = 'SNAPSHOT_SUCCESS',\n\n  /**\n   * The integration test succeeded\n   */\n  TEST_SUCCESS = 'TEST_SUCCESS',\n\n  /**\n   * The assertion failed\n   */\n  ASSERTION_FAILED = 'ASSERTION_FAILED',\n}\n\n/**\n * Integration test diagnostics\n * This is used to report back the status of each test\n */\nexport interface Diagnostic {\n  /**\n   * The name of the test\n   */\n  readonly testName: string;\n\n  /**\n   * The name of the stack\n   */\n  readonly stackName: string;\n\n  /**\n   * The diagnostic message\n   */\n  readonly message: string;\n\n  /**\n   * The time it took to run the test\n   */\n  readonly duration?: number;\n\n  /**\n   * The reason for the diagnostic\n   */\n  readonly reason: DiagnosticReason;\n\n  /**\n   * Additional messages to print\n   */\n  readonly additionalMessages?: string[];\n\n  /**\n   * Relevant config options that were used for the integ test\n   */\n  readonly config?: Record<string, any>;\n}\n\nexport function printSummary(total: number, failed: number): void {\n  if (failed > 0) {\n    logger.print('%s:    %s %s, %s total', chalk.bold('Tests'), chalk.red(failed), chalk.red('failed'), total);\n  } else {\n    logger.print('%s:    %s %s, %s total', chalk.bold('Tests'), chalk.green(total), chalk.green('passed'), total);\n  }\n}\n\n/**\n * Format the assertion results so that the results can be\n * printed\n */\nexport function formatAssertionResults(results: AssertionResults): string {\n  return Object.entries(results)\n    .map(([id, result]) => format('%s%s', id, result.status === 'success' ? ` - ${result.status}` : `\\n${result.message}`))\n    .join('\\n      ');\n}\n\n/**\n * Print out the results from tests\n */\nexport function printResults(diagnostic: Diagnostic): void {\n  switch (diagnostic.reason) {\n    case DiagnosticReason.SNAPSHOT_SUCCESS:\n      logger.success('  UNCHANGED  %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));\n      break;\n    case DiagnosticReason.TEST_SUCCESS:\n      logger.success('  SUCCESS    %s %s\\n      ', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.NO_SNAPSHOT:\n      logger.error('  NEW        %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));\n      break;\n    case DiagnosticReason.SNAPSHOT_FAILED:\n      logger.error('  CHANGED    %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.SNAPSHOT_ERROR:\n    case DiagnosticReason.TEST_ERROR:\n      logger.error('  ERROR      %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.TEST_FAILED:\n      logger.error('  FAILED     %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.ASSERTION_FAILED:\n      logger.error('  ASSERT     %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n  }\n  for (const addl of diagnostic.additionalMessages ?? []) {\n    logger.print(`      ${addl}`);\n  }\n}\n\nexport function printLaggards(testNames: Set<string>) {\n  const parts = [\n    '  ',\n    `Waiting for ${testNames.size} more`,\n    testNames.size < 10 ? ['(', Array.from(testNames).join(', '), ')'].join('') : '',\n  ];\n\n  logger.print(chalk.grey(parts.filter(x => x).join(' ')));\n}\n"]}