@aws-cdk/integ-runner 2.38.1 → 2.40.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.
- package/THIRD_PARTY_LICENSES +1 -1
- package/lib/cli.js +3 -3
- package/lib/index.js +5 -5
- package/lib/runner/integ-test-runner.js +9 -9
- package/lib/runner/private/cloud-assembly.d.ts +11 -1
- package/lib/runner/private/cloud-assembly.js +38 -9
- package/lib/runner/runner-base.js +2 -2
- package/lib/runner/snapshot-test-runner.d.ts +7 -0
- package/lib/runner/snapshot-test-runner.js +78 -4
- package/lib/workers/common.js +4 -4
- package/lib/workers/extract/extract_worker.js +3 -3
- package/lib/workers/extract/index.js +8748 -2656
- package/package.json +11 -11
- package/lib/runner/private/canonicalize-assets.d.ts +0 -11
- package/lib/runner/private/canonicalize-assets.js +0 -70
|
@@ -6,7 +6,6 @@ const stream_1 = require("stream");
|
|
|
6
6
|
const string_decoder_1 = require("string_decoder");
|
|
7
7
|
const cloudformation_diff_1 = require("@aws-cdk/cloudformation-diff");
|
|
8
8
|
const common_1 = require("../workers/common");
|
|
9
|
-
const canonicalize_assets_1 = require("./private/canonicalize-assets");
|
|
10
9
|
const cloud_assembly_1 = require("./private/cloud-assembly");
|
|
11
10
|
const runner_base_1 = require("./runner-base");
|
|
12
11
|
/**
|
|
@@ -150,8 +149,8 @@ class IntegSnapshotRunner extends runner_base_1.IntegRunner {
|
|
|
150
149
|
// asset hashes from the templates so they are not part of the diff
|
|
151
150
|
// comparison
|
|
152
151
|
if (!((_b = this.actualTestSuite.getOptionsForStack(templateId)) === null || _b === void 0 ? void 0 : _b.diffAssets)) {
|
|
153
|
-
actualTemplate =
|
|
154
|
-
expectedTemplate =
|
|
152
|
+
actualTemplate = this.canonicalizeTemplate(actualTemplate, templateId, this.cdkOutDir);
|
|
153
|
+
expectedTemplate = this.canonicalizeTemplate(expectedTemplate, templateId, this.snapshotDir);
|
|
155
154
|
}
|
|
156
155
|
const templateDiff = cloudformation_diff_1.diffTemplate(expectedTemplate, actualTemplate);
|
|
157
156
|
if (!templateDiff.isEmpty) {
|
|
@@ -208,6 +207,81 @@ class IntegSnapshotRunner extends runner_base_1.IntegRunner {
|
|
|
208
207
|
const stacks = assembly.stacks;
|
|
209
208
|
return stacks;
|
|
210
209
|
}
|
|
210
|
+
/**
|
|
211
|
+
* Reduce template to a normal form where asset references have been normalized
|
|
212
|
+
*
|
|
213
|
+
* This makes it possible to compare templates if all that's different between
|
|
214
|
+
* them is the hashes of the asset values.
|
|
215
|
+
*/
|
|
216
|
+
canonicalizeTemplate(template, stackName, manifestDir) {
|
|
217
|
+
const assetsSeen = new Set();
|
|
218
|
+
const stringSubstitutions = new Array();
|
|
219
|
+
// Find assets via parameters (for LegacyStackSynthesizer)
|
|
220
|
+
const paramRe = /^AssetParameters([a-zA-Z0-9]{64})(S3Bucket|S3VersionKey|ArtifactHash)([a-zA-Z0-9]{8})$/;
|
|
221
|
+
for (const paramName of Object.keys((template === null || template === void 0 ? void 0 : template.Parameters) || {})) {
|
|
222
|
+
const m = paramRe.exec(paramName);
|
|
223
|
+
if (!m) {
|
|
224
|
+
continue;
|
|
225
|
+
}
|
|
226
|
+
if (assetsSeen.has(m[1])) {
|
|
227
|
+
continue;
|
|
228
|
+
}
|
|
229
|
+
assetsSeen.add(m[1]);
|
|
230
|
+
const ix = assetsSeen.size;
|
|
231
|
+
// Full parameter reference
|
|
232
|
+
stringSubstitutions.push([
|
|
233
|
+
new RegExp(`AssetParameters${m[1]}(S3Bucket|S3VersionKey|ArtifactHash)([a-zA-Z0-9]{8})`),
|
|
234
|
+
`Asset${ix}$1`,
|
|
235
|
+
]);
|
|
236
|
+
// Substring asset hash reference
|
|
237
|
+
stringSubstitutions.push([
|
|
238
|
+
new RegExp(`${m[1]}`),
|
|
239
|
+
`Asset${ix}Hash`,
|
|
240
|
+
]);
|
|
241
|
+
}
|
|
242
|
+
// find assets defined in the asset manifest
|
|
243
|
+
try {
|
|
244
|
+
const manifest = cloud_assembly_1.AssemblyManifestReader.fromPath(manifestDir);
|
|
245
|
+
const assets = manifest.getAssetIdsForStack(stackName);
|
|
246
|
+
assets.forEach(asset => {
|
|
247
|
+
if (!assetsSeen.has(asset)) {
|
|
248
|
+
assetsSeen.add(asset);
|
|
249
|
+
const ix = assetsSeen.size;
|
|
250
|
+
stringSubstitutions.push([
|
|
251
|
+
new RegExp(asset),
|
|
252
|
+
`Asset${ix}$1`,
|
|
253
|
+
]);
|
|
254
|
+
}
|
|
255
|
+
});
|
|
256
|
+
}
|
|
257
|
+
catch {
|
|
258
|
+
// if there is no asset manifest that is fine.
|
|
259
|
+
}
|
|
260
|
+
// Substitute them out
|
|
261
|
+
return substitute(template);
|
|
262
|
+
function substitute(what) {
|
|
263
|
+
if (Array.isArray(what)) {
|
|
264
|
+
return what.map(substitute);
|
|
265
|
+
}
|
|
266
|
+
if (typeof what === 'object' && what !== null) {
|
|
267
|
+
const ret = {};
|
|
268
|
+
for (const [k, v] of Object.entries(what)) {
|
|
269
|
+
ret[stringSub(k)] = substitute(v);
|
|
270
|
+
}
|
|
271
|
+
return ret;
|
|
272
|
+
}
|
|
273
|
+
if (typeof what === 'string') {
|
|
274
|
+
return stringSub(what);
|
|
275
|
+
}
|
|
276
|
+
return what;
|
|
277
|
+
}
|
|
278
|
+
function stringSub(x) {
|
|
279
|
+
for (const [re, replacement] of stringSubstitutions) {
|
|
280
|
+
x = x.replace(re, replacement);
|
|
281
|
+
}
|
|
282
|
+
return x;
|
|
283
|
+
}
|
|
284
|
+
}
|
|
211
285
|
}
|
|
212
286
|
exports.IntegSnapshotRunner = IntegSnapshotRunner;
|
|
213
287
|
class StringWritable extends stream_1.Writable {
|
|
@@ -228,4 +302,4 @@ class StringWritable extends stream_1.Writable {
|
|
|
228
302
|
callback();
|
|
229
303
|
}
|
|
230
304
|
}
|
|
231
|
-
//# 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 { canonicalizeTemplate } from './private/canonicalize-assets';\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      };\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        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 = canonicalizeTemplate(actualTemplate);\n          expectedTemplate = canonicalizeTemplate(expectedTemplate);\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\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"]}
|
|
305
|
+
//# 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      };\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        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"]}
|
package/lib/workers/common.js
CHANGED
|
@@ -59,8 +59,8 @@ exports.printSummary = printSummary;
|
|
|
59
59
|
*/
|
|
60
60
|
function formatAssertionResults(results) {
|
|
61
61
|
return Object.entries(results)
|
|
62
|
-
.map(([id, result]) => util_1.format('%s
|
|
63
|
-
.join('\n');
|
|
62
|
+
.map(([id, result]) => util_1.format('%s%s', id, result.status === 'success' ? ` - ${result.status}` : `\n${result.message}`))
|
|
63
|
+
.join('\n ');
|
|
64
64
|
}
|
|
65
65
|
exports.formatAssertionResults = formatAssertionResults;
|
|
66
66
|
/**
|
|
@@ -73,7 +73,7 @@ function printResults(diagnostic) {
|
|
|
73
73
|
logger.success(' UNCHANGED %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));
|
|
74
74
|
break;
|
|
75
75
|
case DiagnosticReason.TEST_SUCCESS:
|
|
76
|
-
logger.success(' SUCCESS %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));
|
|
76
|
+
logger.success(' SUCCESS %s %s\n ', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);
|
|
77
77
|
break;
|
|
78
78
|
case DiagnosticReason.NO_SNAPSHOT:
|
|
79
79
|
logger.error(' NEW %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));
|
|
@@ -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\\n%s', id, 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', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));\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}"]}
|
|
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"]}
|
|
@@ -42,7 +42,7 @@ function integTestWorker(request) {
|
|
|
42
42
|
updateWorkflow: request.updateWorkflow,
|
|
43
43
|
verbosity,
|
|
44
44
|
});
|
|
45
|
-
if (results) {
|
|
45
|
+
if (results && Object.values(results).some(result => result.status === 'fail')) {
|
|
46
46
|
failures.push(testInfo);
|
|
47
47
|
workerpool.workerEmit({
|
|
48
48
|
reason: common_1.DiagnosticReason.ASSERTION_FAILED,
|
|
@@ -55,7 +55,7 @@ function integTestWorker(request) {
|
|
|
55
55
|
workerpool.workerEmit({
|
|
56
56
|
reason: common_1.DiagnosticReason.TEST_SUCCESS,
|
|
57
57
|
testName: `${runner.testName}-${testCaseName}`,
|
|
58
|
-
message: '
|
|
58
|
+
message: results ? common_1.formatAssertionResults(results) : 'NO ASSERTIONS',
|
|
59
59
|
duration: (Date.now() - start) / 1000,
|
|
60
60
|
});
|
|
61
61
|
}
|
|
@@ -154,4 +154,4 @@ workerpool.worker({
|
|
|
154
154
|
snapshotTestWorker,
|
|
155
155
|
integTestWorker,
|
|
156
156
|
});
|
|
157
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"extract_worker.js","sourceRoot":"","sources":["extract_worker.ts"],"names":[],"mappings":"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","sourcesContent":["import * as workerpool from 'workerpool';\nimport { IntegSnapshotRunner, IntegTestRunner } from '../../runner';\nimport { IntegTest, IntegTestInfo } from '../../runner/integration-tests';\nimport { DiagnosticReason, IntegTestWorkerConfig, SnapshotVerificationOptions, Diagnostic, formatAssertionResults } from '../common';\nimport { IntegTestBatchRequest } from '../integ-test-worker';\n\n/**\n * Runs a single integration test batch request.\n * If the test does not have an existing snapshot,\n * this will first generate a snapshot and then execute\n * the integration tests.\n *\n * If the tests succeed it will then save the snapshot\n */\nexport function integTestWorker(request: IntegTestBatchRequest): IntegTestWorkerConfig[] {\n  const failures: IntegTestInfo[] = [];\n  const verbosity = request.verbosity ?? 0;\n\n  for (const testInfo of request.tests) {\n    const test = new IntegTest(testInfo); // Hydrate from data\n    const start = Date.now();\n\n    try {\n      const runner = new IntegTestRunner({\n        test,\n        profile: request.profile,\n        env: {\n          AWS_REGION: request.region,\n        },\n        showOutput: verbosity >= 2,\n      }, testInfo.destructiveChanges);\n\n      const tests = runner.actualTests();\n\n      if (!tests || Object.keys(tests).length === 0) {\n        throw new Error(`No tests defined for ${runner.testName}`);\n      }\n      for (const testCaseName of Object.keys(tests)) {\n        try {\n          const results = runner.runIntegTestCase({\n            testCaseName,\n            clean: request.clean,\n            dryRun: request.dryRun,\n            updateWorkflow: request.updateWorkflow,\n            verbosity,\n          });\n          if (results) {\n            failures.push(testInfo);\n            workerpool.workerEmit({\n              reason: DiagnosticReason.ASSERTION_FAILED,\n              testName: `${runner.testName}-${testCaseName} (${request.profile}/${request.region})`,\n              message: formatAssertionResults(results),\n              duration: (Date.now() - start) / 1000,\n            });\n          } else {\n            workerpool.workerEmit({\n              reason: DiagnosticReason.TEST_SUCCESS,\n              testName: `${runner.testName}-${testCaseName}`,\n              message: 'Success',\n              duration: (Date.now() - start) / 1000,\n            });\n          }\n        } catch (e) {\n          failures.push(testInfo);\n          workerpool.workerEmit({\n            reason: DiagnosticReason.TEST_FAILED,\n            testName: `${runner.testName}-${testCaseName} (${request.profile}/${request.region})`,\n            message: `Integration test failed: ${e}`,\n            duration: (Date.now() - start) / 1000,\n          });\n        }\n      }\n    } catch (e) {\n      failures.push(testInfo);\n      workerpool.workerEmit({\n        reason: DiagnosticReason.TEST_ERROR,\n        testName: `${testInfo.fileName} (${request.profile}/${request.region})`,\n        message: `Error during integration test: ${e}`,\n        duration: (Date.now() - start) / 1000,\n      });\n    }\n  }\n\n  return failures;\n}\n\n/**\n * Runs a single snapshot test batch request.\n * For each integration test this will check to see\n * if there is an existing snapshot, and if there is will\n * check if there are any changes\n */\nexport function snapshotTestWorker(testInfo: IntegTestInfo, options: SnapshotVerificationOptions = {}): IntegTestWorkerConfig[] {\n  const failedTests = new Array<IntegTestWorkerConfig>();\n  const start = Date.now();\n  const test = new IntegTest(testInfo); // Hydrate the data record again\n\n  const timer = setTimeout(() => {\n    workerpool.workerEmit({\n      reason: DiagnosticReason.SNAPSHOT_ERROR,\n      testName: test.testName,\n      message: 'Test is taking a very long time',\n      duration: (Date.now() - start) / 1000,\n    });\n  }, 60_000);\n\n  try {\n    const runner = new IntegSnapshotRunner({ test });\n    if (!runner.hasSnapshot()) {\n      workerpool.workerEmit({\n        reason: DiagnosticReason.NO_SNAPSHOT,\n        testName: test.testName,\n        message: 'No Snapshot',\n        duration: (Date.now() - start) / 1000,\n      });\n      failedTests.push(test.info);\n    } else {\n      const { diagnostics, destructiveChanges } = runner.testSnapshot(options);\n      if (diagnostics.length > 0) {\n        diagnostics.forEach(diagnostic => workerpool.workerEmit({\n          ...diagnostic,\n          duration: (Date.now() - start) / 1000,\n        } as Diagnostic));\n        failedTests.push({\n          ...test.info,\n          destructiveChanges,\n        });\n      } else {\n        workerpool.workerEmit({\n          reason: DiagnosticReason.SNAPSHOT_SUCCESS,\n          testName: test.testName,\n          message: 'Success',\n          duration: (Date.now() - start) / 1000,\n        } as Diagnostic);\n      }\n    }\n  } catch (e) {\n    failedTests.push(test.info);\n    workerpool.workerEmit({\n      message: e.message,\n      testName: test.testName,\n      reason: DiagnosticReason.SNAPSHOT_ERROR,\n      duration: (Date.now() - start) / 1000,\n    } as Diagnostic);\n  } finally {\n    clearTimeout(timer);\n  }\n\n  return failedTests;\n}\n\nworkerpool.worker({\n  snapshotTestWorker,\n  integTestWorker,\n});\n"]}
|
|
157
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"extract_worker.js","sourceRoot":"","sources":["extract_worker.ts"],"names":[],"mappings":"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","sourcesContent":["import * as workerpool from 'workerpool';\nimport { IntegSnapshotRunner, IntegTestRunner } from '../../runner';\nimport { IntegTest, IntegTestInfo } from '../../runner/integration-tests';\nimport { DiagnosticReason, IntegTestWorkerConfig, SnapshotVerificationOptions, Diagnostic, formatAssertionResults } from '../common';\nimport { IntegTestBatchRequest } from '../integ-test-worker';\n\n/**\n * Runs a single integration test batch request.\n * If the test does not have an existing snapshot,\n * this will first generate a snapshot and then execute\n * the integration tests.\n *\n * If the tests succeed it will then save the snapshot\n */\nexport function integTestWorker(request: IntegTestBatchRequest): IntegTestWorkerConfig[] {\n  const failures: IntegTestInfo[] = [];\n  const verbosity = request.verbosity ?? 0;\n\n  for (const testInfo of request.tests) {\n    const test = new IntegTest(testInfo); // Hydrate from data\n    const start = Date.now();\n\n    try {\n      const runner = new IntegTestRunner({\n        test,\n        profile: request.profile,\n        env: {\n          AWS_REGION: request.region,\n        },\n        showOutput: verbosity >= 2,\n      }, testInfo.destructiveChanges);\n\n      const tests = runner.actualTests();\n\n      if (!tests || Object.keys(tests).length === 0) {\n        throw new Error(`No tests defined for ${runner.testName}`);\n      }\n      for (const testCaseName of Object.keys(tests)) {\n        try {\n          const results = runner.runIntegTestCase({\n            testCaseName,\n            clean: request.clean,\n            dryRun: request.dryRun,\n            updateWorkflow: request.updateWorkflow,\n            verbosity,\n          });\n          if (results && Object.values(results).some(result => result.status === 'fail')) {\n            failures.push(testInfo);\n            workerpool.workerEmit({\n              reason: DiagnosticReason.ASSERTION_FAILED,\n              testName: `${runner.testName}-${testCaseName} (${request.profile}/${request.region})`,\n              message: formatAssertionResults(results),\n              duration: (Date.now() - start) / 1000,\n            });\n          } else {\n            workerpool.workerEmit({\n              reason: DiagnosticReason.TEST_SUCCESS,\n              testName: `${runner.testName}-${testCaseName}`,\n              message: results ? formatAssertionResults(results) : 'NO ASSERTIONS',\n              duration: (Date.now() - start) / 1000,\n            });\n          }\n        } catch (e) {\n          failures.push(testInfo);\n          workerpool.workerEmit({\n            reason: DiagnosticReason.TEST_FAILED,\n            testName: `${runner.testName}-${testCaseName} (${request.profile}/${request.region})`,\n            message: `Integration test failed: ${e}`,\n            duration: (Date.now() - start) / 1000,\n          });\n        }\n      }\n    } catch (e) {\n      failures.push(testInfo);\n      workerpool.workerEmit({\n        reason: DiagnosticReason.TEST_ERROR,\n        testName: `${testInfo.fileName} (${request.profile}/${request.region})`,\n        message: `Error during integration test: ${e}`,\n        duration: (Date.now() - start) / 1000,\n      });\n    }\n  }\n\n  return failures;\n}\n\n/**\n * Runs a single snapshot test batch request.\n * For each integration test this will check to see\n * if there is an existing snapshot, and if there is will\n * check if there are any changes\n */\nexport function snapshotTestWorker(testInfo: IntegTestInfo, options: SnapshotVerificationOptions = {}): IntegTestWorkerConfig[] {\n  const failedTests = new Array<IntegTestWorkerConfig>();\n  const start = Date.now();\n  const test = new IntegTest(testInfo); // Hydrate the data record again\n\n  const timer = setTimeout(() => {\n    workerpool.workerEmit({\n      reason: DiagnosticReason.SNAPSHOT_ERROR,\n      testName: test.testName,\n      message: 'Test is taking a very long time',\n      duration: (Date.now() - start) / 1000,\n    });\n  }, 60_000);\n\n  try {\n    const runner = new IntegSnapshotRunner({ test });\n    if (!runner.hasSnapshot()) {\n      workerpool.workerEmit({\n        reason: DiagnosticReason.NO_SNAPSHOT,\n        testName: test.testName,\n        message: 'No Snapshot',\n        duration: (Date.now() - start) / 1000,\n      });\n      failedTests.push(test.info);\n    } else {\n      const { diagnostics, destructiveChanges } = runner.testSnapshot(options);\n      if (diagnostics.length > 0) {\n        diagnostics.forEach(diagnostic => workerpool.workerEmit({\n          ...diagnostic,\n          duration: (Date.now() - start) / 1000,\n        } as Diagnostic));\n        failedTests.push({\n          ...test.info,\n          destructiveChanges,\n        });\n      } else {\n        workerpool.workerEmit({\n          reason: DiagnosticReason.SNAPSHOT_SUCCESS,\n          testName: test.testName,\n          message: 'Success',\n          duration: (Date.now() - start) / 1000,\n        } as Diagnostic);\n      }\n    }\n  } catch (e) {\n    failedTests.push(test.info);\n    workerpool.workerEmit({\n      message: e.message,\n      testName: test.testName,\n      reason: DiagnosticReason.SNAPSHOT_ERROR,\n      duration: (Date.now() - start) / 1000,\n    } as Diagnostic);\n  } finally {\n    clearTimeout(timer);\n  }\n\n  return failedTests;\n}\n\nworkerpool.worker({\n  snapshotTestWorker,\n  integTestWorker,\n});\n"]}
|