@aws-cdk/integ-runner 2.196.0 → 2.196.1
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/lib/cli.d.ts +1 -0
- package/lib/cli.d.ts.map +1 -0
- package/lib/engines/cdk-interface.d.ts +34 -99
- package/lib/engines/cdk-interface.d.ts.map +1 -0
- package/lib/engines/cdk-interface.js +1 -34
- package/lib/engines/toolkit-lib.d.ts +4 -12
- package/lib/engines/toolkit-lib.d.ts.map +1 -0
- package/lib/engines/toolkit-lib.js +100 -64
- package/lib/index.d.ts +1 -0
- package/lib/index.d.ts.map +1 -0
- package/lib/index.js +52 -0
- package/lib/logger.d.ts +1 -0
- package/lib/logger.d.ts.map +1 -0
- package/lib/runner/engine.d.ts +1 -0
- package/lib/runner/engine.d.ts.map +1 -0
- package/lib/runner/index.d.ts +1 -0
- package/lib/runner/index.d.ts.map +1 -0
- package/lib/runner/integ-test-runner.d.ts +1 -0
- package/lib/runner/integ-test-runner.d.ts.map +1 -0
- package/lib/runner/integ-test-runner.js +16 -15
- package/lib/runner/integ-test-suite.d.ts +1 -0
- package/lib/runner/integ-test-suite.d.ts.map +1 -0
- package/lib/runner/integ-test-suite.js +9 -3
- package/lib/runner/integration-tests.d.ts +1 -0
- package/lib/runner/integration-tests.d.ts.map +1 -0
- package/lib/runner/integration-tests.js +52 -1
- package/lib/runner/private/cloud-assembly.d.ts +1 -0
- package/lib/runner/private/cloud-assembly.d.ts.map +1 -0
- package/lib/runner/private/cloud-assembly.js +8 -2
- package/lib/runner/private/integ-manifest.d.ts +1 -0
- package/lib/runner/private/integ-manifest.d.ts.map +1 -0
- package/lib/runner/private/integ-manifest.js +11 -2
- package/lib/runner/runner-base.d.ts +1 -0
- package/lib/runner/runner-base.d.ts.map +1 -0
- package/lib/runner/runner-base.js +63 -13
- package/lib/runner/snapshot-test-runner.d.ts +1 -0
- package/lib/runner/snapshot-test-runner.d.ts.map +1 -0
- package/lib/runner/snapshot-test-runner.js +7 -4
- package/lib/unstable-features.d.ts +1 -0
- package/lib/unstable-features.d.ts.map +1 -0
- package/lib/utils.d.ts +1 -0
- package/lib/utils.d.ts.map +1 -0
- package/lib/utils.js +7 -2
- package/lib/workers/common.d.ts +1 -0
- package/lib/workers/common.d.ts.map +1 -0
- package/lib/workers/extract/extract_worker.d.ts +1 -0
- package/lib/workers/extract/extract_worker.d.ts.map +1 -0
- package/lib/workers/extract/extract_worker.js +2 -2
- package/lib/workers/extract/index.d.ts +1 -0
- package/lib/workers/extract/index.d.ts.map +1 -0
- package/lib/workers/extract/index.js +1844 -1742
- package/lib/workers/index.d.ts +1 -0
- package/lib/workers/index.d.ts.map +1 -0
- package/lib/workers/integ-snapshot-worker.d.ts +1 -0
- package/lib/workers/integ-snapshot-worker.d.ts.map +1 -0
- package/lib/workers/integ-test-worker.d.ts +1 -0
- package/lib/workers/integ-test-worker.d.ts.map +1 -0
- package/lib/workers/integ-watch-worker.d.ts +1 -0
- package/lib/workers/integ-watch-worker.d.ts.map +1 -0
- package/package.json +7 -7
|
@@ -22,15 +22,65 @@ const DESTRUCTIVE_CHANGES = '!!DESTRUCTIVE_CHANGES:';
|
|
|
22
22
|
* Represents an Integration test runner
|
|
23
23
|
*/
|
|
24
24
|
class IntegRunner {
|
|
25
|
+
/**
|
|
26
|
+
* The directory where the snapshot will be stored
|
|
27
|
+
*/
|
|
28
|
+
snapshotDir;
|
|
29
|
+
/**
|
|
30
|
+
* An instance of the CDK CLI
|
|
31
|
+
*/
|
|
32
|
+
cdk;
|
|
33
|
+
/**
|
|
34
|
+
* Pretty name of the test
|
|
35
|
+
*/
|
|
36
|
+
testName;
|
|
37
|
+
/**
|
|
38
|
+
* The value used in the '--app' CLI parameter
|
|
39
|
+
*
|
|
40
|
+
* Path to the integ test source file, relative to `this.directory`.
|
|
41
|
+
*/
|
|
42
|
+
cdkApp;
|
|
43
|
+
/**
|
|
44
|
+
* The path where the `cdk.context.json` file
|
|
45
|
+
* will be created
|
|
46
|
+
*/
|
|
47
|
+
cdkContextPath;
|
|
48
|
+
/**
|
|
49
|
+
* The working directory that the integration tests will be
|
|
50
|
+
* executed from
|
|
51
|
+
*/
|
|
52
|
+
directory;
|
|
53
|
+
/**
|
|
54
|
+
* The test to run
|
|
55
|
+
*/
|
|
56
|
+
test;
|
|
57
|
+
/**
|
|
58
|
+
* Default options to pass to the CDK CLI
|
|
59
|
+
*/
|
|
60
|
+
defaultArgs = {
|
|
61
|
+
pathMetadata: false,
|
|
62
|
+
assetMetadata: false,
|
|
63
|
+
versionReporting: false,
|
|
64
|
+
};
|
|
65
|
+
/**
|
|
66
|
+
* The directory where the CDK will be synthed to
|
|
67
|
+
*
|
|
68
|
+
* Relative to cwd.
|
|
69
|
+
*/
|
|
70
|
+
cdkOutDir;
|
|
71
|
+
/**
|
|
72
|
+
* The profile to use for the CDK CLI calls
|
|
73
|
+
*/
|
|
74
|
+
profile;
|
|
75
|
+
/**
|
|
76
|
+
* Show output from the integ test run.
|
|
77
|
+
*/
|
|
78
|
+
showOutput;
|
|
79
|
+
_destructiveChanges;
|
|
80
|
+
legacyContext;
|
|
81
|
+
_expectedTestSuite;
|
|
82
|
+
_actualTestSuite;
|
|
25
83
|
constructor(options) {
|
|
26
|
-
/**
|
|
27
|
-
* Default options to pass to the CDK CLI
|
|
28
|
-
*/
|
|
29
|
-
this.defaultArgs = {
|
|
30
|
-
pathMetadata: false,
|
|
31
|
-
assetMetadata: false,
|
|
32
|
-
versionReporting: false,
|
|
33
|
-
};
|
|
34
84
|
this.test = options.test;
|
|
35
85
|
this.directory = this.test.directory;
|
|
36
86
|
this.testName = this.test.testName;
|
|
@@ -61,8 +111,8 @@ class IntegRunner {
|
|
|
61
111
|
* This will synth and then load the integration test manifest
|
|
62
112
|
*/
|
|
63
113
|
async generateActualSnapshot() {
|
|
64
|
-
await this.cdk.
|
|
65
|
-
|
|
114
|
+
await this.cdk.synth({
|
|
115
|
+
app: this.cdkApp,
|
|
66
116
|
// we don't know the "actual" context yet (this method is what generates it) so just
|
|
67
117
|
// use the "expected" context. This is only run in order to read the manifest
|
|
68
118
|
context: this.getContext((await this.expectedTestSuite())?.synthContext),
|
|
@@ -222,8 +272,8 @@ class IntegRunner {
|
|
|
222
272
|
const actualTestSuite = await this.actualTestSuite();
|
|
223
273
|
// if lookups are enabled then we need to synth again
|
|
224
274
|
// using dummy context and save that as the snapshot
|
|
225
|
-
await this.cdk.
|
|
226
|
-
|
|
275
|
+
await this.cdk.synth({
|
|
276
|
+
app: this.cdkApp,
|
|
227
277
|
context: this.getContext(actualTestSuite.enableLookups ? exports.DEFAULT_SYNTH_OPTIONS.context : {}),
|
|
228
278
|
env: exports.DEFAULT_SYNTH_OPTIONS.env,
|
|
229
279
|
output: path.relative(this.directory, this.snapshotDir),
|
|
@@ -329,4 +379,4 @@ exports.DEFAULT_SYNTH_OPTIONS = {
|
|
|
329
379
|
function currentlyRecommendedAwsCdkLibFlags() {
|
|
330
380
|
return recommendedFlagsFile;
|
|
331
381
|
}
|
|
332
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runner-base.js","sourceRoot":"","sources":["runner-base.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @cdklabs/no-literal-partition */\nimport * as path from 'path';\nimport { AVAILABILITY_ZONE_FALLBACK_CONTEXT_KEY } from '@aws-cdk/cloud-assembly-api';\nimport type { TestCase, DefaultCdkOptions } from '@aws-cdk/cloud-assembly-schema';\nimport { TARGET_PARTITIONS } from '@aws-cdk/cx-api';\nimport * as fs from 'fs-extra';\nimport { IntegTestSuite, LegacyIntegTestSuite } from './integ-test-suite';\nimport type { IntegTest } from './integration-tests';\nimport * as recommendedFlagsFile from '../recommended-feature-flags.json';\nimport { flatten } from '../utils';\nimport { makeEngine } from './engine';\nimport type { ICdk } from '../engines/cdk-interface';\nimport * as logger from '../logger';\nimport type { ManifestTrace } from './private/cloud-assembly';\nimport { AssemblyManifestReader } from './private/cloud-assembly';\nimport type { DestructiveChange } from '../workers/common';\nimport { NoManifestError } from './private/integ-manifest';\n\nconst DESTRUCTIVE_CHANGES = '!!DESTRUCTIVE_CHANGES:';\n\n/**\n * Options for creating an integration test runner\n */\nexport interface IntegRunnerOptions {\n  /**\n   * Information about the test to run\n   */\n  readonly test: IntegTest;\n\n  /**\n   * The region where the test should be deployed\n   */\n  readonly region: string;\n\n  /**\n   * The AWS profile to use when invoking the CDK CLI\n   *\n   * @default - no profile is passed, the default profile is used\n   */\n  readonly profile?: string;\n\n  /**\n   * Additional environment variables that will be available\n   * to the CDK CLI\n   *\n   * @default - no additional environment variables\n   */\n  readonly env?: { [name: string]: string };\n\n  /**\n   * tmp cdk.out directory\n   *\n   * @default - directory will be `cdk-integ.out.${testName}`\n   */\n  readonly integOutDir?: string;\n\n  /**\n   * Instance of the CDK Toolkit Engine to use\n   *\n   * @default - based on `engine` option\n   */\n  readonly cdk?: ICdk;\n\n  /**\n   * Show output from running integration tests\n   *\n   * @default false\n   */\n  readonly showOutput?: boolean;\n}\n\n/**\n * The different components of a test name\n */\n/**\n * Represents an Integration test runner\n */\nexport abstract class IntegRunner {\n  /**\n   * The directory where the snapshot will be stored\n   */\n  public readonly snapshotDir: string;\n\n  /**\n   * An instance of the CDK  CLI\n   */\n  public readonly cdk: ICdk;\n\n  /**\n   * Pretty name of the test\n   */\n  public readonly testName: string;\n\n  /**\n   * The value used in the '--app' CLI parameter\n   *\n   * Path to the integ test source file, relative to `this.directory`.\n   */\n  protected readonly cdkApp: string;\n\n  /**\n   * The path where the `cdk.context.json` file\n   * will be created\n   */\n  protected readonly cdkContextPath: string;\n\n  /**\n   * The working directory that the integration tests will be\n   * executed from\n   */\n  protected readonly directory: string;\n\n  /**\n   * The test to run\n   */\n  protected readonly test: IntegTest;\n\n  /**\n   * Default options to pass to the CDK CLI\n   */\n  protected readonly defaultArgs: DefaultCdkOptions = {\n    pathMetadata: false,\n    assetMetadata: false,\n    versionReporting: false,\n  };\n\n  /**\n   * The directory where the CDK will be synthed to\n   *\n   * Relative to cwd.\n   */\n  protected readonly cdkOutDir: string;\n\n  /**\n   * The profile to use for the CDK CLI calls\n   */\n  protected readonly profile?: string;\n\n  /**\n   * Show output from the integ test run.\n   */\n  protected readonly showOutput: boolean;\n\n  protected _destructiveChanges?: DestructiveChange[];\n  private legacyContext?: Record<string, any>;\n  private _expectedTestSuite?: IntegTestSuite | LegacyIntegTestSuite;\n  private _actualTestSuite?: IntegTestSuite | LegacyIntegTestSuite;\n\n  constructor(options: IntegRunnerOptions) {\n    this.test = options.test;\n    this.directory = this.test.directory;\n    this.testName = this.test.testName;\n    this.snapshotDir = this.test.snapshotDir;\n    this.cdkContextPath = path.join(this.directory, 'cdk.context.json');\n    this.profile = options.profile;\n    this.showOutput = options.showOutput ?? false;\n\n    this.cdk = options.cdk ?? makeEngine(options);\n    this.cdkOutDir = options.integOutDir ?? this.test.temporaryOutputDir;\n\n    const testRunCommand = this.test.appCommand;\n    this.cdkApp = testRunCommand.replace('{filePath}', path.relative(this.directory, this.test.fileName));\n  }\n\n  /**\n   * Return the list of expected (i.e. existing) test cases for this integration test\n   */\n  public async expectedTests(): Promise<{ [testName: string]: TestCase } | undefined> {\n    return (await this.expectedTestSuite())?.testSuite;\n  }\n\n  /**\n   * Return the list of actual (i.e. new) test cases for this integration test\n   */\n  public async actualTests(): Promise<{ [testName: string]: TestCase } | undefined> {\n    return (await this.actualTestSuite()).testSuite;\n  }\n\n  /**\n   * Generate a new \"actual\" snapshot which will be compared to the\n   * existing \"expected\" snapshot\n   * This will synth and then load the integration test manifest\n   */\n  public async generateActualSnapshot(): Promise<IntegTestSuite | LegacyIntegTestSuite> {\n    await this.cdk.synthFast({\n      execCmd: this.cdkApp.split(' '),\n      // we don't know the \"actual\" context yet (this method is what generates it) so just\n      // use the \"expected\" context. This is only run in order to read the manifest\n      context: this.getContext((await this.expectedTestSuite())?.synthContext),\n      env: DEFAULT_SYNTH_OPTIONS.env,\n      output: path.relative(this.directory, this.cdkOutDir),\n    });\n    const manifest = await this.loadManifest(this.cdkOutDir);\n    // after we load the manifest remove the tmp snapshot\n    // so that it doesn't mess up the real snapshot created later\n    this.cleanup();\n    return manifest;\n  }\n\n  /**\n   * Returns true if a snapshot already exists for this test\n   */\n  public hasSnapshot(): boolean {\n    return fs.existsSync(this.snapshotDir);\n  }\n\n  /**\n   * The test suite from the existing snapshot\n   */\n  protected async expectedTestSuite(): Promise<IntegTestSuite | LegacyIntegTestSuite | undefined> {\n    if (!this._expectedTestSuite && this.hasSnapshot()) {\n      this._expectedTestSuite = await this.loadManifest();\n    }\n    return this._expectedTestSuite;\n  }\n\n  /**\n   * The test suite from the new \"actual\" snapshot\n   */\n  protected async actualTestSuite(): Promise<IntegTestSuite | LegacyIntegTestSuite> {\n    if (!this._actualTestSuite) {\n      this._actualTestSuite = await this.generateActualSnapshot();\n    }\n    return this._actualTestSuite;\n  }\n\n  /**\n   * Load the integ manifest which contains information\n   * on how to execute the tests\n   * First we try and load the manifest from the integ manifest (i.e. integ.json)\n   * from the cloud assembly. If it doesn't exist, then we fallback to the\n   * \"legacy mode\" and create a manifest from pragma\n   */\n  protected async loadManifest(dir?: string): Promise<IntegTestSuite | LegacyIntegTestSuite> {\n    const manifest = dir ?? this.snapshotDir;\n    try {\n      const testSuite = IntegTestSuite.fromPath(manifest);\n      return testSuite;\n    } catch (modernError: any) {\n      // Only attempt legacy test case if the integ test manifest was not found\n      // For any other errors, e.g. when parsing the manifest fails, we abort.\n      if (!(modernError instanceof NoManifestError)) {\n        throw modernError;\n      }\n\n      if (this.showOutput) {\n        logger.trace(\n          \"Failed to load integ test manifest for '%s'. Attempting as deprecated legacy test instead. Error was: %s\",\n          manifest,\n          modernError.message ?? String(modernError),\n        );\n      }\n\n      const testCases = await LegacyIntegTestSuite.fromLegacy({\n        cdk: this.cdk,\n        testName: this.test.normalizedTestName,\n        integSourceFilePath: this.test.fileName,\n        listOptions: {\n          ...this.defaultArgs,\n          all: true,\n          app: this.cdkApp,\n          profile: this.profile,\n          output: path.relative(this.directory, this.cdkOutDir),\n        },\n      });\n      this.legacyContext = LegacyIntegTestSuite.getPragmaContext(this.test.fileName);\n      return testCases;\n    }\n  }\n\n  protected cleanup(): void {\n    const cdkOutPath = this.cdkOutDir;\n    if (fs.existsSync(cdkOutPath)) {\n      fs.removeSync(cdkOutPath);\n    }\n  }\n\n  /**\n   * If there are any destructive changes to a stack then this will record\n   * those in the manifest.json file\n   */\n  private renderTraceData(): ManifestTrace {\n    const traceData: ManifestTrace = new Map();\n    const destructiveChanges = this._destructiveChanges ?? [];\n    destructiveChanges.forEach(change => {\n      const trace = traceData.get(change.stackName);\n      if (trace) {\n        trace.set(change.logicalId, `${DESTRUCTIVE_CHANGES} ${change.impact}`);\n      } else {\n        traceData.set(change.stackName, new Map([\n          [change.logicalId, `${DESTRUCTIVE_CHANGES} ${change.impact}`],\n        ]));\n      }\n    });\n    return traceData;\n  }\n\n  /**\n   * In cases where we do not want to retain the assets,\n   * for example, if the assets are very large.\n   *\n   * Since it is possible to disable the update workflow for individual test\n   * cases, this needs to first get a list of stacks that have the update workflow\n   * disabled and then delete assets that relate to that stack. It does that\n   * by reading the asset manifest for the stack and deleting the asset source\n   */\n  protected async removeAssetsFromSnapshot(): Promise<void> {\n    const stacks = (await this.actualTestSuite()).getStacksWithoutUpdateWorkflow() ?? [];\n    const manifest = AssemblyManifestReader.fromPath(this.snapshotDir);\n    const assets = flatten(stacks.map(stack => {\n      return manifest.getAssetLocationsForStack(stack) ?? [];\n    }));\n\n    assets.forEach(asset => {\n      const fileName = path.join(this.snapshotDir, asset);\n      if (fs.existsSync(fileName)) {\n        if (fs.lstatSync(fileName).isDirectory()) {\n          fs.removeSync(fileName);\n        } else {\n          fs.unlinkSync(fileName);\n        }\n      }\n    });\n  }\n\n  /**\n   * Remove the asset cache (.cache/) files from the snapshot.\n   * These are a cache of the asset zips, but we are fine with\n   * re-zipping on deploy\n   */\n  protected removeAssetsCacheFromSnapshot(): void {\n    const files = fs.readdirSync(this.snapshotDir);\n    files.forEach(file => {\n      const fileName = path.join(this.snapshotDir, file);\n      if (fs.lstatSync(fileName).isDirectory() && file === '.cache') {\n        fs.emptyDirSync(fileName);\n        fs.rmdirSync(fileName);\n      }\n    });\n  }\n\n  /**\n   * Create the new snapshot.\n   *\n   * If lookups are enabled, then we need create the snapshot by synth'ing again\n   * with the dummy context so that each time the test is run on different machines\n   * (and with different context/env) the diff will not change.\n   *\n   * If lookups are disabled (which means the stack is env agnostic) then just copy\n   * the assembly that was output by the deployment\n   */\n  protected async createSnapshot(): Promise<void> {\n    if (fs.existsSync(this.snapshotDir)) {\n      fs.removeSync(this.snapshotDir);\n    }\n\n    const actualTestSuite = await this.actualTestSuite();\n\n    // if lookups are enabled then we need to synth again\n    // using dummy context and save that as the snapshot\n    await this.cdk.synthFast({\n      execCmd: this.cdkApp.split(' '),\n      context: this.getContext(actualTestSuite.enableLookups ? DEFAULT_SYNTH_OPTIONS.context : {}),\n      env: DEFAULT_SYNTH_OPTIONS.env,\n      output: path.relative(this.directory, this.snapshotDir),\n    });\n\n    await this.cleanupSnapshot();\n  }\n\n  /**\n   * Perform some cleanup steps after the snapshot is created\n   * Anytime the snapshot needs to be modified after creation\n   * the logic should live here.\n   */\n  private async cleanupSnapshot(): Promise<void> {\n    if (fs.existsSync(this.snapshotDir)) {\n      await this.removeAssetsFromSnapshot();\n      this.removeAssetsCacheFromSnapshot();\n      const assembly = AssemblyManifestReader.fromPath(this.snapshotDir);\n      assembly.cleanManifest();\n      assembly.recordTrace(this.renderTraceData());\n    }\n\n    // if this is a legacy test then create an integ manifest\n    // in the snapshot directory which can be used for the\n    // update workflow. Save any legacyContext as well so that it can be read\n    // the next time\n    const actualTestSuite = await this.actualTestSuite();\n    if (actualTestSuite.type === 'legacy-test-suite') {\n      (actualTestSuite as LegacyIntegTestSuite).saveManifest(this.snapshotDir, this.legacyContext);\n    }\n  }\n\n  protected getContext(additionalContext?: Record<string, any>): Record<string, any> {\n    return {\n      ...currentlyRecommendedAwsCdkLibFlags(),\n      ...this.legacyContext,\n      ...additionalContext,\n\n      // We originally had PLANNED to set this to ['aws', 'aws-cn'], but due to a programming mistake\n      // it was set to everything. In this PR, set it to everything to not mess up all the snapshots.\n      [TARGET_PARTITIONS]: undefined,\n\n      /* ---------------- THE FUTURE LIVES BELOW----------------------------\n      // Restricting to these target partitions makes most service principals synthesize to\n      // `service.${URL_SUFFIX}`, which is technically *incorrect* (it's only `amazonaws.com`\n      // or `amazonaws.com.cn`, never UrlSuffix for any of the restricted regions) but it's what\n      // most existing integ tests contain, and we want to disturb as few as possible.\n      // [TARGET_PARTITIONS]: ['aws', 'aws-cn'],\n      /* ---------------- END OF THE FUTURE ------------------------------- */\n    };\n  }\n}\n\n// Default context we run all integ tests with, so they don't depend on the\n// account of the exercising user.\nexport const DEFAULT_SYNTH_OPTIONS = {\n  context: {\n    [AVAILABILITY_ZONE_FALLBACK_CONTEXT_KEY]: ['test-region-1a', 'test-region-1b', 'test-region-1c'],\n    'availability-zones:account=12345678:region=test-region': ['test-region-1a', 'test-region-1b', 'test-region-1c'],\n    'ssm:account=12345678:parameterName=/aws/service/ami-amazon-linux-latest/amzn-ami-hvm-x86_64-gp2:region=test-region': 'ami-1234',\n    'ssm:account=12345678:parameterName=/aws/service/ami-amazon-linux-latest/amzn2-ami-hvm-x86_64-gp2:region=test-region': 'ami-1234',\n    'ssm:account=12345678:parameterName=/aws/service/ecs/optimized-ami/amazon-linux/recommended:region=test-region': '{\"image_id\": \"ami-1234\"}',\n    // eslint-disable-next-line @stylistic/max-len\n    'ami:account=12345678:filters.image-type.0=machine:filters.name.0=amzn-ami-vpc-nat-*:filters.state.0=available:owners.0=amazon:region=test-region': 'ami-1234',\n    'vpc-provider:account=12345678:filter.isDefault=true:region=test-region:returnAsymmetricSubnets=true': {\n      vpcId: 'vpc-60900905',\n      subnetGroups: [\n        {\n          type: 'Public',\n          name: 'Public',\n          subnets: [\n            {\n              subnetId: 'subnet-e19455ca',\n              availabilityZone: 'us-east-1a',\n              routeTableId: 'rtb-e19455ca',\n            },\n            {\n              subnetId: 'subnet-e0c24797',\n              availabilityZone: 'us-east-1b',\n              routeTableId: 'rtb-e0c24797',\n            },\n            {\n              subnetId: 'subnet-ccd77395',\n              availabilityZone: 'us-east-1c',\n              routeTableId: 'rtb-ccd77395',\n            },\n          ],\n        },\n      ],\n    },\n  },\n  env: {\n    CDK_INTEG_ACCOUNT: '12345678',\n    CDK_INTEG_REGION: 'test-region',\n    CDK_INTEG_HOSTED_ZONE_ID: 'Z23ABC4XYZL05B',\n    CDK_INTEG_HOSTED_ZONE_NAME: 'example.com',\n    CDK_INTEG_DOMAIN_NAME: '*.example.com',\n    CDK_INTEG_CERT_ARN: 'arn:aws:acm:test-region:12345678:certificate/86468209-a272-595d-b831-0efb6421265z',\n    CDK_INTEG_SUBNET_ID: 'subnet-0dff1a399d8f6f92c',\n  },\n};\n\n/**\n * Return the currently recommended flags for `aws-cdk-lib`.\n *\n * These have been built into the CLI at build time. If this ever gets changed\n * back to a dynamic load, remember that this source file may be bundled into\n * a JavaScript bundle, and `__dirname` might not point where you think it does.\n */\nexport function currentlyRecommendedAwsCdkLibFlags() {\n  return recommendedFlagsFile;\n}\n"]}
|
|
382
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runner-base.js","sourceRoot":"","sources":["runner-base.ts"],"names":[],"mappings":";;;AAudA,gFAEC;AAzdD,kDAAkD;AAClD,6BAA6B;AAC7B,oEAAqF;AAErF,4CAAoD;AACpD,+BAA+B;AAC/B,yDAA0E;AAE1E,0EAA0E;AAC1E,oCAAmC;AACnC,qCAAsC;AAEtC,oCAAoC;AAEpC,6DAAkE;AAElE,6DAA2D;AAE3D,MAAM,mBAAmB,GAAG,wBAAwB,CAAC;AAqDrD;;GAEG;AACH;;GAEG;AACH,MAAsB,WAAW;IAC/B;;OAEG;IACa,WAAW,CAAS;IAEpC;;OAEG;IACa,GAAG,CAAO;IAE1B;;OAEG;IACa,QAAQ,CAAS;IAEjC;;;;OAIG;IACgB,MAAM,CAAS;IAElC;;;OAGG;IACgB,cAAc,CAAS;IAE1C;;;OAGG;IACgB,SAAS,CAAS;IAErC;;OAEG;IACgB,IAAI,CAAY;IAEnC;;OAEG;IACgB,WAAW,GAAsB;QAClD,YAAY,EAAE,KAAK;QACnB,aAAa,EAAE,KAAK;QACpB,gBAAgB,EAAE,KAAK;KACxB,CAAC;IAEF;;;;OAIG;IACgB,SAAS,CAAS;IAErC;;OAEG;IACgB,OAAO,CAAU;IAEpC;;OAEG;IACgB,UAAU,CAAU;IAE7B,mBAAmB,CAAuB;IAC5C,aAAa,CAAuB;IACpC,kBAAkB,CAAyC;IAC3D,gBAAgB,CAAyC;IAEjE,YAAY,OAA2B;QACrC,IAAI,CAAC,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;QACzB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC;QACrC,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC;QACnC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC;QACzC,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,kBAAkB,CAAC,CAAC;QACpE,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC;QAC/B,IAAI,CAAC,UAAU,GAAG,OAAO,CAAC,UAAU,IAAI,KAAK,CAAC;QAE9C,IAAI,CAAC,GAAG,GAAG,OAAO,CAAC,GAAG,IAAI,IAAA,mBAAU,EAAC,OAAO,CAAC,CAAC;QAC9C,IAAI,CAAC,SAAS,GAAG,OAAO,CAAC,WAAW,IAAI,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC;QAErE,MAAM,cAAc,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC;QAC5C,IAAI,CAAC,MAAM,GAAG,cAAc,CAAC,OAAO,CAAC,YAAY,EAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;IACxG,CAAC;IAED;;OAEG;IACI,KAAK,CAAC,aAAa;QACxB,OAAO,CAAC,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC,EAAE,SAAS,CAAC;IACrD,CAAC;IAED;;OAEG;IACI,KAAK,CAAC,WAAW;QACtB,OAAO,CAAC,MAAM,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC,SAAS,CAAC;IAClD,CAAC;IAED;;;;OAIG;IACI,KAAK,CAAC,sBAAsB;QACjC,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC;YACnB,GAAG,EAAE,IAAI,CAAC,MAAM;YAChB,oFAAoF;YACpF,6EAA6E;YAC7E,OAAO,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC,EAAE,YAAY,CAAC;YACxE,GAAG,EAAE,6BAAqB,CAAC,GAAG;YAC9B,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC;SACtD,CAAC,CAAC;QACH,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACzD,qDAAqD;QACrD,6DAA6D;QAC7D,IAAI,CAAC,OAAO,EAAE,CAAC;QACf,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED;;OAEG;IACI,WAAW;QAChB,OAAO,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IACzC,CAAC;IAED;;OAEG;IACO,KAAK,CAAC,iBAAiB;QAC/B,IAAI,CAAC,IAAI,CAAC,kBAAkB,IAAI,IAAI,CAAC,WAAW,EAAE,EAAE,CAAC;YACnD,IAAI,CAAC,kBAAkB,GAAG,MAAM,IAAI,CAAC,YAAY,EAAE,CAAC;QACtD,CAAC;QACD,OAAO,IAAI,CAAC,kBAAkB,CAAC;IACjC,CAAC;IAED;;OAEG;IACO,KAAK,CAAC,eAAe;QAC7B,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC3B,IAAI,CAAC,gBAAgB,GAAG,MAAM,IAAI,CAAC,sBAAsB,EAAE,CAAC;QAC9D,CAAC;QACD,OAAO,IAAI,CAAC,gBAAgB,CAAC;IAC/B,CAAC;IAED;;;;;;OAMG;IACO,KAAK,CAAC,YAAY,CAAC,GAAY;QACvC,MAAM,QAAQ,GAAG,GAAG,IAAI,IAAI,CAAC,WAAW,CAAC;QACzC,IAAI,CAAC;YACH,MAAM,SAAS,GAAG,iCAAc,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;YACpD,OAAO,SAAS,CAAC;QACnB,CAAC;QAAC,OAAO,WAAgB,EAAE,CAAC;YAC1B,yEAAyE;YACzE,wEAAwE;YACxE,IAAI,CAAC,CAAC,WAAW,YAAY,gCAAe,CAAC,EAAE,CAAC;gBAC9C,MAAM,WAAW,CAAC;YACpB,CAAC;YAED,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;gBACpB,MAAM,CAAC,KAAK,CACV,0GAA0G,EAC1G,QAAQ,EACR,WAAW,CAAC,OAAO,IAAI,MAAM,CAAC,WAAW,CAAC,CAC3C,CAAC;YACJ,CAAC;YAED,MAAM,SAAS,GAAG,MAAM,uCAAoB,CAAC,UAAU,CAAC;gBACtD,GAAG,EAAE,IAAI,CAAC,GAAG;gBACb,QAAQ,EAAE,IAAI,CAAC,IAAI,CAAC,kBAAkB;gBACtC,mBAAmB,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ;gBACvC,WAAW,EAAE;oBACX,GAAG,IAAI,CAAC,WAAW;oBACnB,GAAG,EAAE,IAAI;oBACT,GAAG,EAAE,IAAI,CAAC,MAAM;oBAChB,OAAO,EAAE,IAAI,CAAC,OAAO;oBACrB,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC;iBACtD;aACF,CAAC,CAAC;YACH,IAAI,CAAC,aAAa,GAAG,uCAAoB,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC/E,OAAO,SAAS,CAAC;QACnB,CAAC;IACH,CAAC;IAES,OAAO;QACf,MAAM,UAAU,GAAG,IAAI,CAAC,SAAS,CAAC;QAClC,IAAI,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;YAC9B,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;QAC5B,CAAC;IACH,CAAC;IAED;;;OAGG;IACK,eAAe;QACrB,MAAM,SAAS,GAAkB,IAAI,GAAG,EAAE,CAAC;QAC3C,MAAM,kBAAkB,GAAG,IAAI,CAAC,mBAAmB,IAAI,EAAE,CAAC;QAC1D,kBAAkB,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;YAClC,MAAM,KAAK,GAAG,SAAS,CAAC,GAAG,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;YAC9C,IAAI,KAAK,EAAE,CAAC;gBACV,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,SAAS,EAAE,GAAG,mBAAmB,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC,CAAC;YACzE,CAAC;iBAAM,CAAC;gBACN,SAAS,CAAC,GAAG,CAAC,MAAM,CAAC,SAAS,EAAE,IAAI,GAAG,CAAC;oBACtC,CAAC,MAAM,CAAC,SAAS,EAAE,GAAG,mBAAmB,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;iBAC9D,CAAC,CAAC,CAAC;YACN,CAAC;QACH,CAAC,CAAC,CAAC;QACH,OAAO,SAAS,CAAC;IACnB,CAAC;IAED;;;;;;;;OAQG;IACO,KAAK,CAAC,wBAAwB;QACtC,MAAM,MAAM,GAAG,CAAC,MAAM,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC,8BAA8B,EAAE,IAAI,EAAE,CAAC;QACrF,MAAM,QAAQ,GAAG,uCAAsB,CAAC,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACnE,MAAM,MAAM,GAAG,IAAA,eAAO,EAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE;YACxC,OAAO,QAAQ,CAAC,yBAAyB,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;QACzD,CAAC,CAAC,CAAC,CAAC;QAEJ,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;YACrB,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,KAAK,CAAC,CAAC;YACpD,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE,CAAC;gBAC5B,IAAI,EAAE,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,WAAW,EAAE,EAAE,CAAC;oBACzC,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;gBAC1B,CAAC;qBAAM,CAAC;oBACN,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;gBAC1B,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACO,6BAA6B;QACrC,MAAM,KAAK,GAAG,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAC/C,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;YACnB,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC;YACnD,IAAI,EAAE,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,WAAW,EAAE,IAAI,IAAI,KAAK,QAAQ,EAAE,CAAC;gBAC9D,EAAE,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC;gBAC1B,EAAE,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;YACzB,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;;;;;OASG;IACO,KAAK,CAAC,cAAc;QAC5B,IAAI,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC;YACpC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAClC,CAAC;QAED,MAAM,eAAe,GAAG,MAAM,IAAI,CAAC,eAAe,EAAE,CAAC;QAErD,qDAAqD;QACrD,oDAAoD;QACpD,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC;YACnB,GAAG,EAAE,IAAI,CAAC,MAAM;YAChB,OAAO,EAAE,IAAI,CAAC,UAAU,CAAC,eAAe,CAAC,aAAa,CAAC,CAAC,CAAC,6BAAqB,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC;YAC5F,GAAG,EAAE,6BAAqB,CAAC,GAAG;YAC9B,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,WAAW,CAAC;SACxD,CAAC,CAAC;QAEH,MAAM,IAAI,CAAC,eAAe,EAAE,CAAC;IAC/B,CAAC;IAED;;;;OAIG;IACK,KAAK,CAAC,eAAe;QAC3B,IAAI,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC;YACpC,MAAM,IAAI,CAAC,wBAAwB,EAAE,CAAC;YACtC,IAAI,CAAC,6BAA6B,EAAE,CAAC;YACrC,MAAM,QAAQ,GAAG,uCAAsB,CAAC,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACnE,QAAQ,CAAC,aAAa,EAAE,CAAC;YACzB,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC;QAC/C,CAAC;QAED,yDAAyD;QACzD,sDAAsD;QACtD,yEAAyE;QACzE,gBAAgB;QAChB,MAAM,eAAe,GAAG,MAAM,IAAI,CAAC,eAAe,EAAE,CAAC;QACrD,IAAI,eAAe,CAAC,IAAI,KAAK,mBAAmB,EAAE,CAAC;YAChD,eAAwC,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC;QAC/F,CAAC;IACH,CAAC;IAES,UAAU,CAAC,iBAAuC;QAC1D,OAAO;YACL,GAAG,kCAAkC,EAAE;YACvC,GAAG,IAAI,CAAC,aAAa;YACrB,GAAG,iBAAiB;YAEpB,+FAA+F;YAC/F,+FAA+F;YAC/F,CAAC,0BAAiB,CAAC,EAAE,SAAS;YAE9B;;;;;;oFAMwE;SACzE,CAAC;IACJ,CAAC;CACF;AAhVD,kCAgVC;AAED,2EAA2E;AAC3E,kCAAkC;AACrB,QAAA,qBAAqB,GAAG;IACnC,OAAO,EAAE;QACP,CAAC,2DAAsC,CAAC,EAAE,CAAC,gBAAgB,EAAE,gBAAgB,EAAE,gBAAgB,CAAC;QAChG,wDAAwD,EAAE,CAAC,gBAAgB,EAAE,gBAAgB,EAAE,gBAAgB,CAAC;QAChH,oHAAoH,EAAE,UAAU;QAChI,qHAAqH,EAAE,UAAU;QACjI,+GAA+G,EAAE,0BAA0B;QAC3I,8CAA8C;QAC9C,kJAAkJ,EAAE,UAAU;QAC9J,qGAAqG,EAAE;YACrG,KAAK,EAAE,cAAc;YACrB,YAAY,EAAE;gBACZ;oBACE,IAAI,EAAE,QAAQ;oBACd,IAAI,EAAE,QAAQ;oBACd,OAAO,EAAE;wBACP;4BACE,QAAQ,EAAE,iBAAiB;4BAC3B,gBAAgB,EAAE,YAAY;4BAC9B,YAAY,EAAE,cAAc;yBAC7B;wBACD;4BACE,QAAQ,EAAE,iBAAiB;4BAC3B,gBAAgB,EAAE,YAAY;4BAC9B,YAAY,EAAE,cAAc;yBAC7B;wBACD;4BACE,QAAQ,EAAE,iBAAiB;4BAC3B,gBAAgB,EAAE,YAAY;4BAC9B,YAAY,EAAE,cAAc;yBAC7B;qBACF;iBACF;aACF;SACF;KACF;IACD,GAAG,EAAE;QACH,iBAAiB,EAAE,UAAU;QAC7B,gBAAgB,EAAE,aAAa;QAC/B,wBAAwB,EAAE,gBAAgB;QAC1C,0BAA0B,EAAE,aAAa;QACzC,qBAAqB,EAAE,eAAe;QACtC,kBAAkB,EAAE,mFAAmF;QACvG,mBAAmB,EAAE,0BAA0B;KAChD;CACF,CAAC;AAEF;;;;;;GAMG;AACH,SAAgB,kCAAkC;IAChD,OAAO,oBAAoB,CAAC;AAC9B,CAAC","sourcesContent":["/* eslint-disable @cdklabs/no-literal-partition */\nimport * as path from 'path';\nimport { AVAILABILITY_ZONE_FALLBACK_CONTEXT_KEY } from '@aws-cdk/cloud-assembly-api';\nimport type { TestCase, DefaultCdkOptions } from '@aws-cdk/cloud-assembly-schema';\nimport { TARGET_PARTITIONS } from '@aws-cdk/cx-api';\nimport * as fs from 'fs-extra';\nimport { IntegTestSuite, LegacyIntegTestSuite } from './integ-test-suite';\nimport type { IntegTest } from './integration-tests';\nimport * as recommendedFlagsFile from '../recommended-feature-flags.json';\nimport { flatten } from '../utils';\nimport { makeEngine } from './engine';\nimport type { ICdk } from '../engines/cdk-interface';\nimport * as logger from '../logger';\nimport type { ManifestTrace } from './private/cloud-assembly';\nimport { AssemblyManifestReader } from './private/cloud-assembly';\nimport type { DestructiveChange } from '../workers/common';\nimport { NoManifestError } from './private/integ-manifest';\n\nconst DESTRUCTIVE_CHANGES = '!!DESTRUCTIVE_CHANGES:';\n\n/**\n * Options for creating an integration test runner\n */\nexport interface IntegRunnerOptions {\n  /**\n   * Information about the test to run\n   */\n  readonly test: IntegTest;\n\n  /**\n   * The region where the test should be deployed\n   */\n  readonly region: string;\n\n  /**\n   * The AWS profile to use when invoking the CDK CLI\n   *\n   * @default - no profile is passed, the default profile is used\n   */\n  readonly profile?: string;\n\n  /**\n   * Additional environment variables that will be available\n   * to the CDK CLI\n   *\n   * @default - no additional environment variables\n   */\n  readonly env?: { [name: string]: string };\n\n  /**\n   * tmp cdk.out directory\n   *\n   * @default - directory will be `cdk-integ.out.${testName}`\n   */\n  readonly integOutDir?: string;\n\n  /**\n   * Instance of the CDK Toolkit Engine to use\n   *\n   * @default - based on `engine` option\n   */\n  readonly cdk?: ICdk;\n\n  /**\n   * Show output from running integration tests\n   *\n   * @default false\n   */\n  readonly showOutput?: boolean;\n}\n\n/**\n * The different components of a test name\n */\n/**\n * Represents an Integration test runner\n */\nexport abstract class IntegRunner {\n  /**\n   * The directory where the snapshot will be stored\n   */\n  public readonly snapshotDir: string;\n\n  /**\n   * An instance of the CDK  CLI\n   */\n  public readonly cdk: ICdk;\n\n  /**\n   * Pretty name of the test\n   */\n  public readonly testName: string;\n\n  /**\n   * The value used in the '--app' CLI parameter\n   *\n   * Path to the integ test source file, relative to `this.directory`.\n   */\n  protected readonly cdkApp: string;\n\n  /**\n   * The path where the `cdk.context.json` file\n   * will be created\n   */\n  protected readonly cdkContextPath: string;\n\n  /**\n   * The working directory that the integration tests will be\n   * executed from\n   */\n  protected readonly directory: string;\n\n  /**\n   * The test to run\n   */\n  protected readonly test: IntegTest;\n\n  /**\n   * Default options to pass to the CDK CLI\n   */\n  protected readonly defaultArgs: DefaultCdkOptions = {\n    pathMetadata: false,\n    assetMetadata: false,\n    versionReporting: false,\n  };\n\n  /**\n   * The directory where the CDK will be synthed to\n   *\n   * Relative to cwd.\n   */\n  protected readonly cdkOutDir: string;\n\n  /**\n   * The profile to use for the CDK CLI calls\n   */\n  protected readonly profile?: string;\n\n  /**\n   * Show output from the integ test run.\n   */\n  protected readonly showOutput: boolean;\n\n  protected _destructiveChanges?: DestructiveChange[];\n  private legacyContext?: Record<string, any>;\n  private _expectedTestSuite?: IntegTestSuite | LegacyIntegTestSuite;\n  private _actualTestSuite?: IntegTestSuite | LegacyIntegTestSuite;\n\n  constructor(options: IntegRunnerOptions) {\n    this.test = options.test;\n    this.directory = this.test.directory;\n    this.testName = this.test.testName;\n    this.snapshotDir = this.test.snapshotDir;\n    this.cdkContextPath = path.join(this.directory, 'cdk.context.json');\n    this.profile = options.profile;\n    this.showOutput = options.showOutput ?? false;\n\n    this.cdk = options.cdk ?? makeEngine(options);\n    this.cdkOutDir = options.integOutDir ?? this.test.temporaryOutputDir;\n\n    const testRunCommand = this.test.appCommand;\n    this.cdkApp = testRunCommand.replace('{filePath}', path.relative(this.directory, this.test.fileName));\n  }\n\n  /**\n   * Return the list of expected (i.e. existing) test cases for this integration test\n   */\n  public async expectedTests(): Promise<{ [testName: string]: TestCase } | undefined> {\n    return (await this.expectedTestSuite())?.testSuite;\n  }\n\n  /**\n   * Return the list of actual (i.e. new) test cases for this integration test\n   */\n  public async actualTests(): Promise<{ [testName: string]: TestCase } | undefined> {\n    return (await this.actualTestSuite()).testSuite;\n  }\n\n  /**\n   * Generate a new \"actual\" snapshot which will be compared to the\n   * existing \"expected\" snapshot\n   * This will synth and then load the integration test manifest\n   */\n  public async generateActualSnapshot(): Promise<IntegTestSuite | LegacyIntegTestSuite> {\n    await this.cdk.synth({\n      app: this.cdkApp,\n      // we don't know the \"actual\" context yet (this method is what generates it) so just\n      // use the \"expected\" context. This is only run in order to read the manifest\n      context: this.getContext((await this.expectedTestSuite())?.synthContext),\n      env: DEFAULT_SYNTH_OPTIONS.env,\n      output: path.relative(this.directory, this.cdkOutDir),\n    });\n    const manifest = await this.loadManifest(this.cdkOutDir);\n    // after we load the manifest remove the tmp snapshot\n    // so that it doesn't mess up the real snapshot created later\n    this.cleanup();\n    return manifest;\n  }\n\n  /**\n   * Returns true if a snapshot already exists for this test\n   */\n  public hasSnapshot(): boolean {\n    return fs.existsSync(this.snapshotDir);\n  }\n\n  /**\n   * The test suite from the existing snapshot\n   */\n  protected async expectedTestSuite(): Promise<IntegTestSuite | LegacyIntegTestSuite | undefined> {\n    if (!this._expectedTestSuite && this.hasSnapshot()) {\n      this._expectedTestSuite = await this.loadManifest();\n    }\n    return this._expectedTestSuite;\n  }\n\n  /**\n   * The test suite from the new \"actual\" snapshot\n   */\n  protected async actualTestSuite(): Promise<IntegTestSuite | LegacyIntegTestSuite> {\n    if (!this._actualTestSuite) {\n      this._actualTestSuite = await this.generateActualSnapshot();\n    }\n    return this._actualTestSuite;\n  }\n\n  /**\n   * Load the integ manifest which contains information\n   * on how to execute the tests\n   * First we try and load the manifest from the integ manifest (i.e. integ.json)\n   * from the cloud assembly. If it doesn't exist, then we fallback to the\n   * \"legacy mode\" and create a manifest from pragma\n   */\n  protected async loadManifest(dir?: string): Promise<IntegTestSuite | LegacyIntegTestSuite> {\n    const manifest = dir ?? this.snapshotDir;\n    try {\n      const testSuite = IntegTestSuite.fromPath(manifest);\n      return testSuite;\n    } catch (modernError: any) {\n      // Only attempt legacy test case if the integ test manifest was not found\n      // For any other errors, e.g. when parsing the manifest fails, we abort.\n      if (!(modernError instanceof NoManifestError)) {\n        throw modernError;\n      }\n\n      if (this.showOutput) {\n        logger.trace(\n          \"Failed to load integ test manifest for '%s'. Attempting as deprecated legacy test instead. Error was: %s\",\n          manifest,\n          modernError.message ?? String(modernError),\n        );\n      }\n\n      const testCases = await LegacyIntegTestSuite.fromLegacy({\n        cdk: this.cdk,\n        testName: this.test.normalizedTestName,\n        integSourceFilePath: this.test.fileName,\n        listOptions: {\n          ...this.defaultArgs,\n          all: true,\n          app: this.cdkApp,\n          profile: this.profile,\n          output: path.relative(this.directory, this.cdkOutDir),\n        },\n      });\n      this.legacyContext = LegacyIntegTestSuite.getPragmaContext(this.test.fileName);\n      return testCases;\n    }\n  }\n\n  protected cleanup(): void {\n    const cdkOutPath = this.cdkOutDir;\n    if (fs.existsSync(cdkOutPath)) {\n      fs.removeSync(cdkOutPath);\n    }\n  }\n\n  /**\n   * If there are any destructive changes to a stack then this will record\n   * those in the manifest.json file\n   */\n  private renderTraceData(): ManifestTrace {\n    const traceData: ManifestTrace = new Map();\n    const destructiveChanges = this._destructiveChanges ?? [];\n    destructiveChanges.forEach(change => {\n      const trace = traceData.get(change.stackName);\n      if (trace) {\n        trace.set(change.logicalId, `${DESTRUCTIVE_CHANGES} ${change.impact}`);\n      } else {\n        traceData.set(change.stackName, new Map([\n          [change.logicalId, `${DESTRUCTIVE_CHANGES} ${change.impact}`],\n        ]));\n      }\n    });\n    return traceData;\n  }\n\n  /**\n   * In cases where we do not want to retain the assets,\n   * for example, if the assets are very large.\n   *\n   * Since it is possible to disable the update workflow for individual test\n   * cases, this needs to first get a list of stacks that have the update workflow\n   * disabled and then delete assets that relate to that stack. It does that\n   * by reading the asset manifest for the stack and deleting the asset source\n   */\n  protected async removeAssetsFromSnapshot(): Promise<void> {\n    const stacks = (await this.actualTestSuite()).getStacksWithoutUpdateWorkflow() ?? [];\n    const manifest = AssemblyManifestReader.fromPath(this.snapshotDir);\n    const assets = flatten(stacks.map(stack => {\n      return manifest.getAssetLocationsForStack(stack) ?? [];\n    }));\n\n    assets.forEach(asset => {\n      const fileName = path.join(this.snapshotDir, asset);\n      if (fs.existsSync(fileName)) {\n        if (fs.lstatSync(fileName).isDirectory()) {\n          fs.removeSync(fileName);\n        } else {\n          fs.unlinkSync(fileName);\n        }\n      }\n    });\n  }\n\n  /**\n   * Remove the asset cache (.cache/) files from the snapshot.\n   * These are a cache of the asset zips, but we are fine with\n   * re-zipping on deploy\n   */\n  protected removeAssetsCacheFromSnapshot(): void {\n    const files = fs.readdirSync(this.snapshotDir);\n    files.forEach(file => {\n      const fileName = path.join(this.snapshotDir, file);\n      if (fs.lstatSync(fileName).isDirectory() && file === '.cache') {\n        fs.emptyDirSync(fileName);\n        fs.rmdirSync(fileName);\n      }\n    });\n  }\n\n  /**\n   * Create the new snapshot.\n   *\n   * If lookups are enabled, then we need create the snapshot by synth'ing again\n   * with the dummy context so that each time the test is run on different machines\n   * (and with different context/env) the diff will not change.\n   *\n   * If lookups are disabled (which means the stack is env agnostic) then just copy\n   * the assembly that was output by the deployment\n   */\n  protected async createSnapshot(): Promise<void> {\n    if (fs.existsSync(this.snapshotDir)) {\n      fs.removeSync(this.snapshotDir);\n    }\n\n    const actualTestSuite = await this.actualTestSuite();\n\n    // if lookups are enabled then we need to synth again\n    // using dummy context and save that as the snapshot\n    await this.cdk.synth({\n      app: this.cdkApp,\n      context: this.getContext(actualTestSuite.enableLookups ? DEFAULT_SYNTH_OPTIONS.context : {}),\n      env: DEFAULT_SYNTH_OPTIONS.env,\n      output: path.relative(this.directory, this.snapshotDir),\n    });\n\n    await this.cleanupSnapshot();\n  }\n\n  /**\n   * Perform some cleanup steps after the snapshot is created\n   * Anytime the snapshot needs to be modified after creation\n   * the logic should live here.\n   */\n  private async cleanupSnapshot(): Promise<void> {\n    if (fs.existsSync(this.snapshotDir)) {\n      await this.removeAssetsFromSnapshot();\n      this.removeAssetsCacheFromSnapshot();\n      const assembly = AssemblyManifestReader.fromPath(this.snapshotDir);\n      assembly.cleanManifest();\n      assembly.recordTrace(this.renderTraceData());\n    }\n\n    // if this is a legacy test then create an integ manifest\n    // in the snapshot directory which can be used for the\n    // update workflow. Save any legacyContext as well so that it can be read\n    // the next time\n    const actualTestSuite = await this.actualTestSuite();\n    if (actualTestSuite.type === 'legacy-test-suite') {\n      (actualTestSuite as LegacyIntegTestSuite).saveManifest(this.snapshotDir, this.legacyContext);\n    }\n  }\n\n  protected getContext(additionalContext?: Record<string, any>): Record<string, any> {\n    return {\n      ...currentlyRecommendedAwsCdkLibFlags(),\n      ...this.legacyContext,\n      ...additionalContext,\n\n      // We originally had PLANNED to set this to ['aws', 'aws-cn'], but due to a programming mistake\n      // it was set to everything. In this PR, set it to everything to not mess up all the snapshots.\n      [TARGET_PARTITIONS]: undefined,\n\n      /* ---------------- THE FUTURE LIVES BELOW----------------------------\n      // Restricting to these target partitions makes most service principals synthesize to\n      // `service.${URL_SUFFIX}`, which is technically *incorrect* (it's only `amazonaws.com`\n      // or `amazonaws.com.cn`, never UrlSuffix for any of the restricted regions) but it's what\n      // most existing integ tests contain, and we want to disturb as few as possible.\n      // [TARGET_PARTITIONS]: ['aws', 'aws-cn'],\n      /* ---------------- END OF THE FUTURE ------------------------------- */\n    };\n  }\n}\n\n// Default context we run all integ tests with, so they don't depend on the\n// account of the exercising user.\nexport const DEFAULT_SYNTH_OPTIONS = {\n  context: {\n    [AVAILABILITY_ZONE_FALLBACK_CONTEXT_KEY]: ['test-region-1a', 'test-region-1b', 'test-region-1c'],\n    'availability-zones:account=12345678:region=test-region': ['test-region-1a', 'test-region-1b', 'test-region-1c'],\n    'ssm:account=12345678:parameterName=/aws/service/ami-amazon-linux-latest/amzn-ami-hvm-x86_64-gp2:region=test-region': 'ami-1234',\n    'ssm:account=12345678:parameterName=/aws/service/ami-amazon-linux-latest/amzn2-ami-hvm-x86_64-gp2:region=test-region': 'ami-1234',\n    'ssm:account=12345678:parameterName=/aws/service/ecs/optimized-ami/amazon-linux/recommended:region=test-region': '{\"image_id\": \"ami-1234\"}',\n    // eslint-disable-next-line @stylistic/max-len\n    'ami:account=12345678:filters.image-type.0=machine:filters.name.0=amzn-ami-vpc-nat-*:filters.state.0=available:owners.0=amazon:region=test-region': 'ami-1234',\n    'vpc-provider:account=12345678:filter.isDefault=true:region=test-region:returnAsymmetricSubnets=true': {\n      vpcId: 'vpc-60900905',\n      subnetGroups: [\n        {\n          type: 'Public',\n          name: 'Public',\n          subnets: [\n            {\n              subnetId: 'subnet-e19455ca',\n              availabilityZone: 'us-east-1a',\n              routeTableId: 'rtb-e19455ca',\n            },\n            {\n              subnetId: 'subnet-e0c24797',\n              availabilityZone: 'us-east-1b',\n              routeTableId: 'rtb-e0c24797',\n            },\n            {\n              subnetId: 'subnet-ccd77395',\n              availabilityZone: 'us-east-1c',\n              routeTableId: 'rtb-ccd77395',\n            },\n          ],\n        },\n      ],\n    },\n  },\n  env: {\n    CDK_INTEG_ACCOUNT: '12345678',\n    CDK_INTEG_REGION: 'test-region',\n    CDK_INTEG_HOSTED_ZONE_ID: 'Z23ABC4XYZL05B',\n    CDK_INTEG_HOSTED_ZONE_NAME: 'example.com',\n    CDK_INTEG_DOMAIN_NAME: '*.example.com',\n    CDK_INTEG_CERT_ARN: 'arn:aws:acm:test-region:12345678:certificate/86468209-a272-595d-b831-0efb6421265z',\n    CDK_INTEG_SUBNET_ID: 'subnet-0dff1a399d8f6f92c',\n  },\n};\n\n/**\n * Return the currently recommended flags for `aws-cdk-lib`.\n *\n * These have been built into the CLI at build time. If this ever gets changed\n * back to a dynamic load, remember that this source file may be bundled into\n * a JavaScript bundle, and `__dirname` might not point where you think it does.\n */\nexport function currentlyRecommendedAwsCdkLibFlags() {\n  return recommendedFlagsFile;\n}\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"snapshot-test-runner.d.ts","sourceRoot":"","sources":["snapshot-test-runner.ts"],"names":[],"mappings":"AAQA,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,eAAe,CAAC;AACxD,OAAO,EAAE,WAAW,EAAyB,MAAM,eAAe,CAAC;AACnE,OAAO,KAAK,EAAE,UAAU,EAAE,iBAAiB,EAAE,2BAA2B,EAAE,MAAM,mBAAmB,CAAC;AAsBpG;;;GAGG;AACH,qBAAa,mBAAoB,SAAQ,WAAW;gBACtC,OAAO,EAAE,IAAI,CAAC,kBAAkB,EAAE,QAAQ,CAAC;IAOvD;;;;;OAKG;IACU,YAAY,CAAC,OAAO,GAAE,2BAAgC,GAAG,OAAO,CAAC;QAC5E,WAAW,EAAE,UAAU,EAAE,CAAC;QAC1B,kBAAkB,EAAE,iBAAiB,EAAE,CAAC;KACzC,CAAC;IAkEF;;;;;;;OAOG;IACH,OAAO,CAAC,mBAAmB;IAsB3B;;;;;;OAMG;YACW,8BAA8B;IAS5C;;;;;;OAMG;YACW,YAAY;IAuG1B,OAAO,CAAC,YAAY;IAIpB;;;;;OAKG;IACH,OAAO,CAAC,oBAAoB;CA4E7B"}
|
|
@@ -4,6 +4,7 @@ exports.IntegSnapshotRunner = void 0;
|
|
|
4
4
|
const path = require("path");
|
|
5
5
|
const stream_1 = require("stream");
|
|
6
6
|
const string_decoder_1 = require("string_decoder");
|
|
7
|
+
const cloud_assembly_api_1 = require("@aws-cdk/cloud-assembly-api");
|
|
7
8
|
const cloudformation_diff_1 = require("@aws-cdk/cloudformation-diff");
|
|
8
9
|
const cloud_assembly_1 = require("./private/cloud-assembly");
|
|
9
10
|
const runner_base_1 = require("./runner-base");
|
|
@@ -16,7 +17,7 @@ class IntegSnapshotRunner extends runner_base_1.IntegRunner {
|
|
|
16
17
|
constructor(options) {
|
|
17
18
|
super({
|
|
18
19
|
...options,
|
|
19
|
-
region:
|
|
20
|
+
region: cloud_assembly_api_1.UNKNOWN_REGION,
|
|
20
21
|
});
|
|
21
22
|
}
|
|
22
23
|
/**
|
|
@@ -37,8 +38,8 @@ class IntegSnapshotRunner extends runner_base_1.IntegRunner {
|
|
|
37
38
|
// generates an incorrect snapshot and I have no idea why so synth again here
|
|
38
39
|
// to produce the "correct" snapshot
|
|
39
40
|
const env = runner_base_1.DEFAULT_SYNTH_OPTIONS.env;
|
|
40
|
-
await this.cdk.
|
|
41
|
-
|
|
41
|
+
await this.cdk.synth({
|
|
42
|
+
app: this.cdkApp,
|
|
42
43
|
context: this.getContext(actualTestSuite.enableLookups ? runner_base_1.DEFAULT_SYNTH_OPTIONS.context : {}),
|
|
43
44
|
env,
|
|
44
45
|
output: path.relative(this.directory, this.cdkOutDir),
|
|
@@ -300,6 +301,8 @@ class IntegSnapshotRunner extends runner_base_1.IntegRunner {
|
|
|
300
301
|
}
|
|
301
302
|
exports.IntegSnapshotRunner = IntegSnapshotRunner;
|
|
302
303
|
class StringWritable extends stream_1.Writable {
|
|
304
|
+
data;
|
|
305
|
+
_decoder;
|
|
303
306
|
constructor(options) {
|
|
304
307
|
super(options);
|
|
305
308
|
this._decoder = new string_decoder_1.StringDecoder();
|
|
@@ -317,4 +320,4 @@ class StringWritable extends stream_1.Writable {
|
|
|
317
320
|
callback();
|
|
318
321
|
}
|
|
319
322
|
}
|
|
320
|
-
//# 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 type { WritableOptions } from 'stream';\nimport { Writable } from 'stream';\nimport { StringDecoder } from 'string_decoder';\nimport type { ResourceDifference } from '@aws-cdk/cloudformation-diff';\nimport { fullDiff, formatDifferences, ResourceImpact } from '@aws-cdk/cloudformation-diff';\nimport { AssemblyManifestReader } from './private/cloud-assembly';\nimport type { IntegRunnerOptions } from './runner-base';\nimport { IntegRunner, DEFAULT_SYNTH_OPTIONS } from './runner-base';\nimport type { Diagnostic, DestructiveChange, SnapshotVerificationOptions } from '../workers/common';\nimport { DiagnosticReason } from '../workers/common';\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: Omit<IntegRunnerOptions, 'region'>) {\n    super({\n      ...options,\n      region: 'unused',\n    });\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 async testSnapshot(options: SnapshotVerificationOptions = {}): Promise<{\n    diagnostics: Diagnostic[];\n    destructiveChanges: DestructiveChange[];\n  }> {\n    let doClean = true;\n    try {\n      const expectedTestSuite = await this.expectedTestSuite();\n      const actualTestSuite = await this.actualTestSuite();\n      const expectedSnapshotAssembly = this.getSnapshotAssembly(this.snapshotDir, 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 = DEFAULT_SYNTH_OPTIONS.env;\n      await this.cdk.synthFast({\n        execCmd: this.cdkApp.split(' '),\n        context: this.getContext(actualTestSuite.enableLookups ? DEFAULT_SYNTH_OPTIONS.context : {}),\n        env,\n        output: path.relative(this.directory, this.cdkOutDir),\n      });\n\n      // read the \"actual\" snapshot\n      const actualSnapshotAssembly = this.getSnapshotAssembly(this.cdkOutDir, actualTestSuite.stacks);\n\n      // diff the existing snapshot (expected) with the integration test (actual)\n      const diagnostics = await 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).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 async getAllowedDestroyTypesForStack(stackId: string): Promise<string[] | undefined> {\n    for (const testCase of Object.values((await 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 async diffAssembly(\n    expected: SnapshotAssembly,\n    actual: SnapshotAssembly,\n  ): Promise<{ 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: (await 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 = fullDiff(expectedTemplate, actualTemplate);\n          if (!templateDiff.isEmpty) {\n            const allowedDestroyTypes = (await 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) {\n        continue;\n      }\n      if (assetsSeen.has(m[1])) {\n        continue;\n      }\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"]}
|
|
323
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snapshot-test-runner.js","sourceRoot":"","sources":["snapshot-test-runner.ts"],"names":[],"mappings":";;;AAAA,6BAA6B;AAE7B,mCAAkC;AAClC,mDAA+C;AAC/C,oEAA6D;AAE7D,sEAA2F;AAC3F,6DAAkE;AAElE,+CAAmE;AAEnE,8CAAqD;AAqBrD;;;GAGG;AACH,MAAa,mBAAoB,SAAQ,yBAAW;IAClD,YAAY,OAA2C;QACrD,KAAK,CAAC;YACJ,GAAG,OAAO;YACV,MAAM,EAAE,mCAAc;SACvB,CAAC,CAAC;IACL,CAAC;IAED;;;;;OAKG;IACI,KAAK,CAAC,YAAY,CAAC,UAAuC,EAAE;QAIjE,IAAI,OAAO,GAAG,IAAI,CAAC;QACnB,IAAI,CAAC;YACH,MAAM,iBAAiB,GAAG,MAAM,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzD,MAAM,eAAe,GAAG,MAAM,IAAI,CAAC,eAAe,EAAE,CAAC;YACrD,MAAM,wBAAwB,GAAG,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,WAAW,EAAE,iBAAiB,EAAE,MAAM,CAAC,CAAC;YAEvG,6BAA6B;YAC7B,yDAAyD;YACzD,2EAA2E;YAC3E,6EAA6E;YAC7E,oCAAoC;YACpC,MAAM,GAAG,GAAG,mCAAqB,CAAC,GAAG,CAAC;YACtC,MAAM,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC;gBACnB,GAAG,EAAE,IAAI,CAAC,MAAM;gBAChB,OAAO,EAAE,IAAI,CAAC,UAAU,CAAC,eAAe,CAAC,aAAa,CAAC,CAAC,CAAC,mCAAqB,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC;gBAC5F,GAAG;gBACH,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC;aACtD,CAAC,CAAC;YAEH,6BAA6B;YAC7B,MAAM,sBAAsB,GAAG,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,SAAS,EAAE,eAAe,CAAC,MAAM,CAAC,CAAC;YAEhG,2EAA2E;YAC3E,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,YAAY,CAAC,wBAAwB,EAAE,sBAAsB,CAAC,CAAC;YAE9F,IAAI,WAAW,CAAC,WAAW,CAAC,MAAM,EAAE,CAAC;gBACnC,qDAAqD;gBACrD,MAAM,kBAAkB,GAAa,EAAE,CAAC;gBAExC,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC;oBACnB,kBAAkB,CAAC,IAAI,CACrB,gCAAgC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,WAAW,CAAC,EAAE,EAChF,gCAAgC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,CAC/E;wBACD,OAAO,GAAG,KAAK,CAAC;gBAClB,CAAC;gBAED,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;oBACpB,2CAA2C;oBAC3C,MAAM,MAAM,GAAG,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;oBAClE,MAAM,MAAM,GAAG,MAAM,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;oBAE3D,kBAAkB,CAAC,IAAI,CACrB,QAAQ,EACR,KAAK,CAAC,GAAG,MAAM,EAAE,WAAW,EAAE,OAAO,IAAI,CAAC,MAAM,GAAG,EAAE,OAAO,IAAI,CAAC,SAAS,GAAG,EAAE,GAAG,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,KAAK,QAAQ,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAEtM,CAAC;gBACJ,CAAC;gBAED,WAAW,CAAC,WAAW,CAAC,CAAC,CAAC,GAAG;oBAC3B,GAAG,WAAW,CAAC,WAAW,CAAC,CAAC,CAAC;oBAC7B,kBAAkB;iBACnB,CAAC;YACJ,CAAC;YAED,OAAO,WAAW,CAAC;QACrB,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,MAAM,CAAC,CAAC;QACV,CAAC;gBAAS,CAAC;YACT,IAAI,OAAO,EAAE,CAAC;gBACZ,IAAI,CAAC,OAAO,EAAE,CAAC;YACjB,CAAC;QACH,CAAC;IACH,CAAC;IAED;;;;;;;OAOG;IACK,mBAAmB,CAAC,gBAAwB,EAAE,aAAuB,EAAE;QAC7E,MAAM,QAAQ,GAAG,IAAI,CAAC,YAAY,CAAC,gBAAgB,CAAC,CAAC;QACrD,MAAM,MAAM,GAAG,QAAQ,CAAC,MAAM,CAAC;QAC/B,MAAM,SAAS,GAAqB,EAAE,CAAC;QACvC,KAAK,MAAM,CAAC,SAAS,EAAE,aAAa,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;YAChE,IAAI,UAAU,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;gBACnC,MAAM,QAAQ,GAAG,uCAAsB,CAAC,QAAQ,CAAC,gBAAgB,CAAC,CAAC;gBACnE,MAAM,MAAM,GAAG,QAAQ,CAAC,mBAAmB,CAAC,SAAS,CAAC,CAAC;gBAEvD,SAAS,CAAC,SAAS,CAAC,GAAG;oBACrB,SAAS,EAAE;wBACT,CAAC,SAAS,CAAC,EAAE,aAAa;wBAC1B,GAAG,QAAQ,CAAC,uBAAuB,CAAC,SAAS,CAAC;qBAC/C;oBACD,MAAM;iBACP,CAAC;YACJ,CAAC;QACH,CAAC;QAED,OAAO,SAAS,CAAC;IACnB,CAAC;IAED;;;;;;OAMG;IACK,KAAK,CAAC,8BAA8B,CAAC,OAAe;QAC1D,KAAK,MAAM,QAAQ,IAAI,MAAM,CAAC,MAAM,CAAC,CAAC,MAAM,IAAI,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC;YACvE,IAAI,QAAQ,CAAC,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;gBACtC,OAAO,QAAQ,CAAC,YAAY,CAAC;YAC/B,CAAC;QACH,CAAC;QACD,OAAO,SAAS,CAAC;IACnB,CAAC;IAED;;;;;;OAMG;IACK,KAAK,CAAC,YAAY,CACxB,QAA0B,EAC1B,MAAwB;QAExB,MAAM,QAAQ,GAAiB,EAAE,CAAC;QAClC,MAAM,kBAAkB,GAAwB,EAAE,CAAC;QAEnD,2DAA2D;QAC3D,+CAA+C;QAC/C,KAAK,MAAM,CAAC,OAAO,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC;YACxD,KAAK,MAAM,UAAU,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,EAAE,CAAC;gBACtD,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,SAAS,CAAC,UAAU,CAAC,EAAE,CAAC;oBAC5C,QAAQ,CAAC,IAAI,CAAC;wBACZ,QAAQ,EAAE,IAAI,CAAC,QAAQ;wBACvB,SAAS,EAAE,UAAU;wBACrB,MAAM,EAAE,yBAAgB,CAAC,eAAe;wBACxC,OAAO,EAAE,GAAG,UAAU,wCAAwC;qBAC/D,CAAC,CAAC;gBACL,CAAC;YACH,CAAC;QACH,CAAC;QAED,KAAK,MAAM,CAAC,OAAO,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;YACtD,KAAK,MAAM,UAAU,IAAI,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC,EAAE,CAAC;gBACxD,4DAA4D;gBAC5D,8CAA8C;gBAC5C,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,SAAS,CAAC,UAAU,CAAC,EAAE,CAAC;oBAC9C,QAAQ,CAAC,IAAI,CAAC;wBACZ,QAAQ,EAAE,IAAI,CAAC,QAAQ;wBACvB,SAAS,EAAE,UAAU;wBACrB,MAAM,EAAE,yBAAgB,CAAC,eAAe;wBACxC,OAAO,EAAE,GAAG,UAAU,iDAAiD;qBACxE,CAAC,CAAC;oBACH,SAAS;gBACX,CAAC;qBAAM,CAAC;oBACN,MAAM,MAAM,GAAG;wBACb,UAAU,EAAE,CAAC,MAAM,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC,kBAAkB,CAAC,OAAO,CAAC,EAAE,UAAU;qBACnF,CAAC;oBACF,IAAI,cAAc,GAAG,MAAM,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;oBAC3D,IAAI,gBAAgB,GAAG,QAAQ,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;oBAE/D,gEAAgE;oBAChE,mEAAmE;oBACnE,aAAa;oBACb,IAAI,CAAC,MAAM,CAAC,UAAU,EAAE,CAAC;wBACvB,cAAc,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,EAAE,MAAM,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,CAAC;wBACnF,gBAAgB,GAAG,IAAI,CAAC,oBAAoB,CAAC,gBAAgB,EAAE,QAAQ,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,CAAC;oBAC3F,CAAC;oBACD,MAAM,YAAY,GAAG,IAAA,8BAAQ,EAAC,gBAAgB,EAAE,cAAc,CAAC,CAAC;oBAChE,IAAI,CAAC,YAAY,CAAC,OAAO,EAAE,CAAC;wBAC1B,MAAM,mBAAmB,GAAG,CAAC,MAAM,IAAI,CAAC,8BAA8B,CAAC,OAAO,CAAC,CAAC,IAAI,EAAE,CAAC;wBAEvF,4DAA4D;wBAC5D,wBAAwB;wBACxB,YAAY,CAAC,SAAS,CAAC,iBAAiB,CAAC,CAAC,SAAiB,EAAE,MAA0B,EAAE,EAAE;4BAC3F,qEAAqE;4BACrE,uEAAuE;4BACvE,qCAAqC;4BACnC,MAAM,YAAY,GAAG,MAAM,CAAC,QAAQ,EAAE,IAAI,IAAI,MAAM,CAAC,QAAQ,EAAE,IAAI,CAAC;4BACpE,IAAI,YAAY,IAAI,mBAAmB,CAAC,QAAQ,CAAC,YAAY,CAAC,EAAE,CAAC;gCAC/D,OAAO;4BACT,CAAC;4BACD,IAAI,MAAM,CAAC,SAAS,EAAE,CAAC;gCACrB,kBAAkB,CAAC,IAAI,CAAC;oCACtB,MAAM,EAAE,oCAAc,CAAC,YAAY;oCACnC,SAAS;oCACT,SAAS,EAAE,UAAU;iCACtB,CAAC,CAAC;4BACL,CAAC;iCAAM,CAAC;gCACN,QAAQ,MAAM,CAAC,YAAY,EAAE,CAAC;oCAC5B,KAAK,oCAAc,CAAC,WAAW,CAAC;oCAChC,KAAK,oCAAc,CAAC,WAAW,CAAC;oCAChC,KAAK,oCAAc,CAAC,YAAY,CAAC;oCACjC,KAAK,oCAAc,CAAC,YAAY;wCAC9B,kBAAkB,CAAC,IAAI,CAAC;4CACtB,MAAM,EAAE,MAAM,CAAC,YAAY;4CAC3B,SAAS;4CACT,SAAS,EAAE,UAAU;yCACtB,CAAC,CAAC;wCACH,MAAM;gCACV,CAAC;4BACH,CAAC;wBACH,CAAC,CAAC,CAAC;wBACH,MAAM,QAAQ,GAAG,IAAI,cAAc,CAAC,EAAE,CAAC,CAAC;wBACxC,IAAA,uCAAiB,EAAC,QAAQ,EAAE,YAAY,CAAC,CAAC;wBAC1C,QAAQ,CAAC,IAAI,CAAC;4BACZ,MAAM,EAAE,yBAAgB,CAAC,eAAe;4BACxC,OAAO,EAAE,QAAQ,CAAC,IAAI;4BACtB,SAAS,EAAE,UAAU;4BACrB,QAAQ,EAAE,IAAI,CAAC,QAAQ;4BACvB,MAAM;yBACP,CAAC,CAAC;oBACL,CAAC;gBACH,CAAC;YACH,CAAC;QACH,CAAC;QAED,OAAO;YACL,WAAW,EAAE,QAAQ;YACrB,kBAAkB;SACnB,CAAC;IACJ,CAAC;IAEO,YAAY,CAAC,GAAW;QAC9B,OAAO,uCAAsB,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC;IAC9C,CAAC;IAED;;;;;OAKG;IACK,oBAAoB,CAAC,QAAa,EAAE,MAAgB;QAC1D,MAAM,UAAU,GAAG,IAAI,GAAG,EAAU,CAAC;QACrC,MAAM,mBAAmB,GAAG,IAAI,KAAK,EAAoB,CAAC;QAE1D,0DAA0D;QAC1D,MAAM,OAAO,GAAG,wFAAwF,CAAC;QACzG,KAAK,MAAM,SAAS,IAAI,MAAM,CAAC,IAAI,CAAC,QAAQ,EAAE,UAAU,IAAI,EAAE,CAAC,EAAE,CAAC;YAChE,MAAM,CAAC,GAAG,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YAClC,IAAI,CAAC,CAAC,EAAE,CAAC;gBACP,SAAS;YACX,CAAC;YACD,IAAI,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;gBACzB,SAAS;YACX,CAAC;YAED,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YACrB,MAAM,EAAE,GAAG,UAAU,CAAC,IAAI,CAAC;YAE3B,2BAA2B;YAC3B,mBAAmB,CAAC,IAAI,CAAC;gBACvB,IAAI,MAAM,CAAC,kBAAkB,CAAC,CAAC,CAAC,CAAC,sDAAsD,CAAC;gBACxF,QAAQ,EAAE,IAAI;aACf,CAAC,CAAC;YACH,iCAAiC;YACjC,mBAAmB,CAAC,IAAI,CAAC;gBACvB,IAAI,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;gBACrB,QAAQ,EAAE,MAAM;aACjB,CAAC,CAAC;QACL,CAAC;QAED,4CAA4C;QAC5C,IAAI,CAAC;YACH,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;gBACrB,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC;oBAC3B,UAAU,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;oBACtB,MAAM,EAAE,GAAG,UAAU,CAAC,IAAI,CAAC;oBAC3B,mBAAmB,CAAC,IAAI,CAAC;wBACvB,IAAI,MAAM,CAAC,KAAK,CAAC;wBACjB,QAAQ,EAAE,IAAI;qBACf,CAAC,CAAC;gBACL,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC;QAAC,MAAM,CAAC;YACP,8CAA8C;QAChD,CAAC;QAED,sBAAsB;QACtB,OAAO,UAAU,CAAC,QAAQ,CAAC,CAAC;QAE5B,SAAS,UAAU,CAAC,IAAS;YAC3B,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC;gBACxB,OAAO,IAAI,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;YAC9B,CAAC;YAED,IAAI,OAAO,IAAI,KAAK,QAAQ,IAAI,IAAI,KAAK,IAAI,EAAE,CAAC;gBAC9C,MAAM,GAAG,GAAQ,EAAE,CAAC;gBACpB,KAAK,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC;oBAC1C,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC;gBACpC,CAAC;gBACD,OAAO,GAAG,CAAC;YACb,CAAC;YAED,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;gBAC7B,OAAO,SAAS,CAAC,IAAI,CAAC,CAAC;YACzB,CAAC;YAED,OAAO,IAAI,CAAC;QACd,CAAC;QAED,SAAS,SAAS,CAAC,CAAS;YAC1B,KAAK,MAAM,CAAC,EAAE,EAAE,WAAW,CAAC,IAAI,mBAAmB,EAAE,CAAC;gBACpD,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,EAAE,EAAE,WAAW,CAAC,CAAC;YACjC,CAAC;YACD,OAAO,CAAC,CAAC;QACX,CAAC;IACH,CAAC;CACF;AArUD,kDAqUC;AAED,MAAM,cAAe,SAAQ,iBAAQ;IAC5B,IAAI,CAAS;IACZ,QAAQ,CAAgB;IAChC,YAAY,OAAwB;QAClC,KAAK,CAAC,OAAO,CAAC,CAAC;QACf,IAAI,CAAC,QAAQ,GAAG,IAAI,8BAAa,EAAE,CAAC;QACpC,IAAI,CAAC,IAAI,GAAG,EAAE,CAAC;IACjB,CAAC;IAED,MAAM,CAAC,KAAU,EAAE,QAAgB,EAAE,QAAwC;QAC3E,IAAI,QAAQ,KAAK,QAAQ,EAAE,CAAC;YAC1B,KAAK,GAAG,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;QACrC,CAAC;QAED,IAAI,CAAC,IAAI,IAAI,KAAK,CAAC;QACnB,QAAQ,EAAE,CAAC;IACb,CAAC;IAED,MAAM,CAAC,QAAwC;QAC7C,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,QAAQ,CAAC,GAAG,EAAE,CAAC;QACjC,QAAQ,EAAE,CAAC;IACb,CAAC;CACF","sourcesContent":["import * as path from 'path';\nimport type { WritableOptions } from 'stream';\nimport { Writable } from 'stream';\nimport { StringDecoder } from 'string_decoder';\nimport { UNKNOWN_REGION } from '@aws-cdk/cloud-assembly-api';\nimport type { ResourceDifference } from '@aws-cdk/cloudformation-diff';\nimport { fullDiff, formatDifferences, ResourceImpact } from '@aws-cdk/cloudformation-diff';\nimport { AssemblyManifestReader } from './private/cloud-assembly';\nimport type { IntegRunnerOptions } from './runner-base';\nimport { IntegRunner, DEFAULT_SYNTH_OPTIONS } from './runner-base';\nimport type { Diagnostic, DestructiveChange, SnapshotVerificationOptions } from '../workers/common';\nimport { DiagnosticReason } from '../workers/common';\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: Omit<IntegRunnerOptions, 'region'>) {\n    super({\n      ...options,\n      region: UNKNOWN_REGION,\n    });\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 async testSnapshot(options: SnapshotVerificationOptions = {}): Promise<{\n    diagnostics: Diagnostic[];\n    destructiveChanges: DestructiveChange[];\n  }> {\n    let doClean = true;\n    try {\n      const expectedTestSuite = await this.expectedTestSuite();\n      const actualTestSuite = await this.actualTestSuite();\n      const expectedSnapshotAssembly = this.getSnapshotAssembly(this.snapshotDir, 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 = DEFAULT_SYNTH_OPTIONS.env;\n      await this.cdk.synth({\n        app: this.cdkApp,\n        context: this.getContext(actualTestSuite.enableLookups ? DEFAULT_SYNTH_OPTIONS.context : {}),\n        env,\n        output: path.relative(this.directory, this.cdkOutDir),\n      });\n\n      // read the \"actual\" snapshot\n      const actualSnapshotAssembly = this.getSnapshotAssembly(this.cdkOutDir, actualTestSuite.stacks);\n\n      // diff the existing snapshot (expected) with the integration test (actual)\n      const diagnostics = await 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).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 async getAllowedDestroyTypesForStack(stackId: string): Promise<string[] | undefined> {\n    for (const testCase of Object.values((await 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 async diffAssembly(\n    expected: SnapshotAssembly,\n    actual: SnapshotAssembly,\n  ): Promise<{ 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: (await 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 = fullDiff(expectedTemplate, actualTemplate);\n          if (!templateDiff.isEmpty) {\n            const allowedDestroyTypes = (await 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) {\n        continue;\n      }\n      if (assetsSeen.has(m[1])) {\n        continue;\n      }\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"]}
|
|
@@ -37,3 +37,4 @@ export declare function processUnstableFeatures(unstableFeatures?: string[]): st
|
|
|
37
37
|
* @returns A string describing available features or indicating none are available
|
|
38
38
|
*/
|
|
39
39
|
export declare function availableFeaturesDescription(): string;
|
|
40
|
+
//# sourceMappingURL=unstable-features.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"unstable-features.d.ts","sourceRoot":"","sources":["unstable-features.ts"],"names":[],"mappings":"AAEA;;GAEG;AACH,oBAAY,aAAa;IACvB,oDAAoD;IACpD,SAAS,cAAc;IACvB,8DAA8D;IAC9D,UAAU,eAAe;IACzB,wDAAwD;IACxD,OAAO,YAAY;CACpB;AAED;;GAEG;AACH,MAAM,WAAW,eAAe;IAC9B,oCAAoC;IACpC,QAAQ,CAAC,MAAM,EAAE,aAAa,CAAC;IAC/B,sEAAsE;IACtE,QAAQ,CAAC,kBAAkB,EAAE,MAAM,CAAC;IACpC,iEAAiE;IACjE,QAAQ,CAAC,cAAc,EAAE,MAAM,CAAC;CACjC;AAED;;GAEG;AACH,eAAO,MAAM,iBAAiB,EAAE,MAAM,CAAC,MAAM,EAAE,eAAe,CAW7D,CAAC;AAEF;;;;;GAKG;AACH,wBAAgB,uBAAuB,CAAC,gBAAgB,GAAE,MAAM,EAAO,GAAG,MAAM,EAAE,CAmCjF;AAED;;;;GAIG;AACH,wBAAgB,4BAA4B,IAAI,MAAM,CAUrD"}
|
package/lib/utils.d.ts
CHANGED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["utils.ts"],"names":[],"mappings":"AAGA;;GAEG;AACH,wBAAgB,IAAI,CAAC,WAAW,EAAE,MAAM,EAAE,EAAE,OAAO,GAAE;IAAE,GAAG,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,OAAO,CAAC;IAAC,GAAG,CAAC,EAAE,GAAG,CAAA;CAAQ,GAAG,GAAG,CAuB9G;AAED,KAAK,OAAO,GAAG,KAAK,CAAC,MAAM,GAAG,OAAO,CAAC,CAAC;AAEvC;;;;;;;;;;;;;;;GAeG;AACH,wBAAgB,gBAAgB,CAAC,OAAO,EAAE,OAAO,EAAE,OAAO,GAAE;IAAE,GAAG,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,OAAO,CAAC;IAAC,GAAG,CAAC,EAAE,GAAG,CAAA;CAAQ,GAAG,GAAG,CASrH;AAED;;GAEG;AACH,wBAAgB,aAAa,CAAC,OAAO,EAAE,OAAO,GAAG,MAAM,CAOtD;AAED;;GAEG;AACH,wBAAgB,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,EAAE,EAAE,GAAG,CAAC,EAAE,CAEzC;AAED;;GAEG;AACH,wBAAgB,KAAK,CAAC,QAAQ,EAAE,MAAM,EAAE,GAAG,MAAM,CAEhD;AAED;;GAEG;AACH,wBAAgB,MAAM,CAAC,OAAO,EAAE,MAAM,GAAG,MAAM,EAAE,CAGhD;AAED;;;;GAIG;AACH,qBAAa,QAAQ,CAAC,CAAC;IAKT,OAAO,CAAC,QAAQ,CAAC,KAAK;IAAO,OAAO,CAAC,QAAQ,CAAC,OAAO;IAJjE,OAAO,CAAC,QAAQ,CAAC,SAAS,CAAS;IACnC,OAAO,CAAC,QAAQ,CAAC,OAAO,CAAS;IACjC,OAAO,CAAC,KAAK,CAAC,CAAiB;gBAEF,KAAK,EAAE,CAAC,EAAE,EAAmB,OAAO,GAAE,eAAe,CAAC,CAAC,CAAM;IAMnF,QAAQ,CAAC,IAAI,EAAE,CAAC;IAQhB,IAAI;IAKX,OAAO,CAAC,SAAS;IAOjB,OAAO,CAAC,aAAa;IAIrB,OAAO,CAAC,MAAM;CAGf;AAED,MAAM,WAAW,eAAe,CAAC,CAAC;IAChC;;;;OAIG;IACH,QAAQ,CAAC,OAAO,CAAC,EAAE,MAAM,CAAC;IAE1B;;OAEG;IACH,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC,CAAC,KAAK,IAAI,CAAC;CAC1C;AAED;;;;GAIG;AACH,wBAAgB,oBAAoB,CAAC,CAAC,KAAK,mBAAmB,CAAC,CAAC,CAAC,CAOhE;AAED,UAAU,mBAAmB,CAAC,CAAC;IAC7B,OAAO,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC;IACpB,OAAO,EAAE,CAAC,KAAK,EAAE,CAAC,KAAK,IAAI,CAAC;IAC5B,MAAM,EAAE,CAAC,MAAM,EAAE,GAAG,KAAK,IAAI,CAAC;CAC/B"}
|
package/lib/utils.js
CHANGED
|
@@ -95,11 +95,16 @@ function chunks(command) {
|
|
|
95
95
|
* If it takes too long to cross off a new item, print the list.
|
|
96
96
|
*/
|
|
97
97
|
class WorkList {
|
|
98
|
+
items;
|
|
99
|
+
options;
|
|
100
|
+
remaining;
|
|
101
|
+
timeout;
|
|
102
|
+
timer;
|
|
98
103
|
constructor(items, options = {}) {
|
|
99
104
|
this.items = items;
|
|
100
105
|
this.options = options;
|
|
101
106
|
this.remaining = new Set(this.items);
|
|
102
|
-
this.timeout = options.timeout ??
|
|
107
|
+
this.timeout = options.timeout ?? 60_000;
|
|
103
108
|
this.scheduleTimer();
|
|
104
109
|
}
|
|
105
110
|
crossOff(item) {
|
|
@@ -140,4 +145,4 @@ function promiseWithResolvers() {
|
|
|
140
145
|
});
|
|
141
146
|
return { promise, resolve: resolve, reject: reject };
|
|
142
147
|
}
|
|
143
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["utils.ts"],"names":[],"mappings":"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","sourcesContent":["// Helper functions for CDK Exec\nimport { spawnSync } from 'child_process';\n\n/**\n * Our own execute function which doesn't use shells and strings.\n */\nexport function exec(commandLine: string[], options: { cwd?: string; verbose?: boolean; env?: any } = { }): any {\n  const proc = spawnSync(commandLine[0], commandLine.slice(1), {\n    stdio: ['ignore', 'pipe', options.verbose ? 'inherit' : 'pipe'], // inherit STDERR in verbose mode\n    env: {\n      ...process.env,\n      ...options.env,\n    },\n    cwd: options.cwd,\n  });\n\n  if (proc.error) {\n    throw proc.error;\n  }\n  if (proc.status !== 0) {\n    if (process.stderr) { // will be 'null' in verbose mode\n      process.stderr.write(proc.stderr);\n    }\n    throw new Error(`Command exited with ${proc.status ? `status ${proc.status}` : `signal ${proc.signal}`}`);\n  }\n\n  const output = proc.stdout.toString('utf-8').trim();\n\n  return output;\n}\n\ntype Command = Array<string | Command>;\n\n/**\n * Like exec, but any arrays encountered inside the command array are pull out and executed first, than their value is inserted again.\n * This mimics execution a command with sub shell behavior.\n *\n * For example this input:\n * ```\n * [\"git\", \"checkout\", [\"git\", \"merge-base\", \"HEAD\"], \"--,\" \"path/to/file\"]\n * ```\n * will run something like this:\n * ```\n * git checkout $(git merge-base HEAD) -- path/to/file\n * ```\n *\n * Note that the algorithm will detect sub shells first, exec them and then\n * substitute the return values in.\n */\nexport function execWithSubShell(command: Command, options: { cwd?: string; verbose?: boolean; env?: any } = { }): any {\n  const resolvedCommand: string[] = command.map((cmd) => {\n    if (Array.isArray(cmd)) {\n      return execWithSubShell(cmd, options);\n    }\n    return cmd;\n  });\n\n  return exec(resolvedCommand, options);\n}\n\n/**\n * Takes the same input as `execWithSubShell` and returns a string with sub shells.\n */\nexport function renderCommand(command: Command): string {\n  return command.map((cmd) => {\n    if (Array.isArray(cmd)) {\n      return `$(${renderCommand(cmd)})`;\n    }\n    return cmd;\n  }).join(' ');\n}\n\n/**\n * Flatten a list of lists into a list of elements\n */\nexport function flatten<T>(xs: T[][]): T[] {\n  return Array.prototype.concat.apply([], xs);\n}\n\n/**\n * Chain commands\n */\nexport function chain(commands: string[]): string {\n  return commands.filter(c => !!c).join(' && ');\n}\n\n/**\n * Split command to chunks by space\n */\nexport function chunks(command: string): string[] {\n  const result = command.match(/(?:[^\\s\"]+|\"[^\"]*\")+/g);\n  return result ?? [];\n}\n\n/**\n * A class holding a set of items which are being crossed off in time\n *\n * If it takes too long to cross off a new item, print the list.\n */\nexport class WorkList<A> {\n  private readonly remaining = new Set(this.items);\n  private readonly timeout: number;\n  private timer?: NodeJS.Timeout;\n\n  constructor(private readonly items: A[], private readonly options: WorkListOptions<A> = {}) {\n    this.timeout = options.timeout ?? 60_000;\n    this.scheduleTimer();\n  }\n\n  public crossOff(item: A) {\n    this.remaining.delete(item);\n    this.stopTimer();\n    if (this.remaining.size > 0) {\n      this.scheduleTimer();\n    }\n  }\n\n  public done() {\n    this.remaining.clear();\n    this.stopTimer();\n  }\n\n  private stopTimer() {\n    if (this.timer) {\n      clearTimeout(this.timer);\n      this.timer = undefined;\n    }\n  }\n\n  private scheduleTimer() {\n    this.timer = setTimeout(() => this.report(), this.timeout);\n  }\n\n  private report() {\n    this.options.onTimeout?.(this.remaining);\n  }\n}\n\nexport interface WorkListOptions<A> {\n  /**\n   * When to reply with remaining items\n   *\n   * @default 60000\n   */\n  readonly timeout?: number;\n\n  /**\n   * Function to call when timeout hits\n   */\n  readonly onTimeout?: (x: Set<A>) => void;\n}\n\n/**\n * A backport of Promiser.withResolvers\n *\n * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise/withResolvers\n */\nexport function promiseWithResolvers<A>(): PromiseAndResolvers<A> {\n  let resolve: PromiseAndResolvers<A>['resolve'], reject: PromiseAndResolvers<A>['reject'];\n  const promise = new Promise<A>((res, rej) => {\n    resolve = res;\n    reject = rej;\n  });\n  return { promise, resolve: resolve!, reject: reject! };\n}\n\ninterface PromiseAndResolvers<A> {\n  promise: Promise<A>;\n  resolve: (value: A) => void;\n  reject: (reason: any) => void;\n}\n"]}
|
|
148
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["utils.ts"],"names":[],"mappings":"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","sourcesContent":["// Helper functions for CDK Exec\nimport { spawnSync } from 'child_process';\n\n/**\n * Our own execute function which doesn't use shells and strings.\n */\nexport function exec(commandLine: string[], options: { cwd?: string; verbose?: boolean; env?: any } = { }): any {\n  const proc = spawnSync(commandLine[0], commandLine.slice(1), {\n    stdio: ['ignore', 'pipe', options.verbose ? 'inherit' : 'pipe'], // inherit STDERR in verbose mode\n    env: {\n      ...process.env,\n      ...options.env,\n    },\n    cwd: options.cwd,\n  });\n\n  if (proc.error) {\n    throw proc.error;\n  }\n  if (proc.status !== 0) {\n    if (process.stderr) { // will be 'null' in verbose mode\n      process.stderr.write(proc.stderr);\n    }\n    throw new Error(`Command exited with ${proc.status ? `status ${proc.status}` : `signal ${proc.signal}`}`);\n  }\n\n  const output = proc.stdout.toString('utf-8').trim();\n\n  return output;\n}\n\ntype Command = Array<string | Command>;\n\n/**\n * Like exec, but any arrays encountered inside the command array are pull out and executed first, than their value is inserted again.\n * This mimics execution a command with sub shell behavior.\n *\n * For example this input:\n * ```\n * [\"git\", \"checkout\", [\"git\", \"merge-base\", \"HEAD\"], \"--,\" \"path/to/file\"]\n * ```\n * will run something like this:\n * ```\n * git checkout $(git merge-base HEAD) -- path/to/file\n * ```\n *\n * Note that the algorithm will detect sub shells first, exec them and then\n * substitute the return values in.\n */\nexport function execWithSubShell(command: Command, options: { cwd?: string; verbose?: boolean; env?: any } = { }): any {\n  const resolvedCommand: string[] = command.map((cmd) => {\n    if (Array.isArray(cmd)) {\n      return execWithSubShell(cmd, options);\n    }\n    return cmd;\n  });\n\n  return exec(resolvedCommand, options);\n}\n\n/**\n * Takes the same input as `execWithSubShell` and returns a string with sub shells.\n */\nexport function renderCommand(command: Command): string {\n  return command.map((cmd) => {\n    if (Array.isArray(cmd)) {\n      return `$(${renderCommand(cmd)})`;\n    }\n    return cmd;\n  }).join(' ');\n}\n\n/**\n * Flatten a list of lists into a list of elements\n */\nexport function flatten<T>(xs: T[][]): T[] {\n  return Array.prototype.concat.apply([], xs);\n}\n\n/**\n * Chain commands\n */\nexport function chain(commands: string[]): string {\n  return commands.filter(c => !!c).join(' && ');\n}\n\n/**\n * Split command to chunks by space\n */\nexport function chunks(command: string): string[] {\n  const result = command.match(/(?:[^\\s\"]+|\"[^\"]*\")+/g);\n  return result ?? [];\n}\n\n/**\n * A class holding a set of items which are being crossed off in time\n *\n * If it takes too long to cross off a new item, print the list.\n */\nexport class WorkList<A> {\n  private readonly remaining: Set<A>;\n  private readonly timeout: number;\n  private timer?: NodeJS.Timeout;\n\n  constructor(private readonly items: A[], private readonly options: WorkListOptions<A> = {}) {\n    this.remaining = new Set(this.items);\n    this.timeout = options.timeout ?? 60_000;\n    this.scheduleTimer();\n  }\n\n  public crossOff(item: A) {\n    this.remaining.delete(item);\n    this.stopTimer();\n    if (this.remaining.size > 0) {\n      this.scheduleTimer();\n    }\n  }\n\n  public done() {\n    this.remaining.clear();\n    this.stopTimer();\n  }\n\n  private stopTimer() {\n    if (this.timer) {\n      clearTimeout(this.timer);\n      this.timer = undefined;\n    }\n  }\n\n  private scheduleTimer() {\n    this.timer = setTimeout(() => this.report(), this.timeout);\n  }\n\n  private report() {\n    this.options.onTimeout?.(this.remaining);\n  }\n}\n\nexport interface WorkListOptions<A> {\n  /**\n   * When to reply with remaining items\n   *\n   * @default 60000\n   */\n  readonly timeout?: number;\n\n  /**\n   * Function to call when timeout hits\n   */\n  readonly onTimeout?: (x: Set<A>) => void;\n}\n\n/**\n * A backport of Promiser.withResolvers\n *\n * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise/withResolvers\n */\nexport function promiseWithResolvers<A>(): PromiseAndResolvers<A> {\n  let resolve: PromiseAndResolvers<A>['resolve'], reject: PromiseAndResolvers<A>['reject'];\n  const promise = new Promise<A>((res, rej) => {\n    resolve = res;\n    reject = rej;\n  });\n  return { promise, resolve: resolve!, reject: reject! };\n}\n\ninterface PromiseAndResolvers<A> {\n  promise: Promise<A>;\n  resolve: (value: A) => void;\n  reject: (reason: any) => void;\n}\n"]}
|
package/lib/workers/common.d.ts
CHANGED
|
@@ -238,3 +238,4 @@ export declare function formatAssertionResults(results: AssertionResults): strin
|
|
|
238
238
|
export declare function printResults(diagnostic: Diagnostic): void;
|
|
239
239
|
export declare function printLaggards(testNames: Set<string>): void;
|
|
240
240
|
export declare function formatError(error: any): string;
|
|
241
|
+
//# sourceMappingURL=common.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"common.d.ts","sourceRoot":"","sources":["common.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,8BAA8B,CAAC;AAGnE,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,6BAA6B,CAAC;AAEjE;;GAEG;AACH,MAAM,MAAM,gBAAgB,GAAG;IAAE,CAAC,EAAE,EAAE,MAAM,GAAG,eAAe,CAAA;CAAE,CAAC;AAEjE;;GAEG;AACH,MAAM,WAAW,eAAe;IAC9B;;;OAGG;IACH,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC;IAEzB;;OAEG;IACH,QAAQ,CAAC,MAAM,EAAE,SAAS,GAAG,MAAM,CAAC;CACrC;AAED;;GAEG;AACH,MAAM,WAAW,qBAAsB,SAAQ,aAAa;IAC1D;;;;OAIG;IACH,QAAQ,CAAC,kBAAkB,CAAC,EAAE,iBAAiB,EAAE,CAAC;CACnD;AAED;;GAEG;AACH,MAAM,WAAW,iBAAiB;IAChC;;OAEG;IACH,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC;IAE3B;;OAEG;IACH,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC;IAE3B;;OAEG;IACH,QAAQ,CAAC,MAAM,EAAE,cAAc,CAAC;CACjC;AAED;;GAEG;AACH,MAAM,WAAW,kBAAkB;IACjC;;OAEG;IACH,QAAQ,CAAC,MAAM,EAAE,MAAM,CAAC;IAExB;;;OAGG;IACH,QAAQ,CAAC,QAAQ,EAAE,MAAM,CAAC;IAE1B;;;;;OAKG;IACH,QAAQ,CAAC,KAAK,EAAE;QAAE,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM,CAAA;KAAE,CAAC;IAE/C;;;;OAIG;IACH,QAAQ,CAAC,OAAO,CAAC,EAAE,MAAM,CAAC;CAC3B;AAED,MAAM,WAAW,2BAA2B;IAC1C;;;;OAIG;IACH,QAAQ,CAAC,MAAM,CAAC,EAAE,OAAO,CAAC;IAE1B;;;;OAIG;IACH,QAAQ,CAAC,OAAO,CAAC,EAAE,OAAO,CAAC;CAC5B;AAED;;GAEG;AACH,MAAM,WAAW,kBAAkB;IACjC;;OAEG;IACH,QAAQ,CAAC,WAAW,EAAE,aAAa,EAAE,CAAC;IAEtC;;;OAGG;IACH,QAAQ,CAAC,OAAO,EAAE,kBAAkB,EAAE,CAAC;CACxC;AAED;;GAEG;AACH,MAAM,WAAW,gBAAgB;IAC/B;;;OAGG;IACH,QAAQ,CAAC,KAAK,EAAE,qBAAqB,EAAE,CAAC;IAExC;;;;;OAKG;IACH,QAAQ,CAAC,KAAK,CAAC,EAAE,OAAO,CAAC;IAEzB;;;;;;OAMG;IACH,QAAQ,CAAC,MAAM,CAAC,EAAE,OAAO,CAAC;IAE1B;;;;;OAKG;IACH,QAAQ,CAAC,SAAS,CAAC,EAAE,MAAM,CAAC;IAE5B;;;;OAIG;IACH,QAAQ,CAAC,cAAc,CAAC,EAAE,OAAO,CAAC;IAElC;;;;OAIG;IACH,QAAQ,CAAC,KAAK,CAAC,EAAE,OAAO,CAAC;CAC1B;AAED;;GAEG;AACH,oBAAY,gBAAgB;IAC1B;;;OAGG;IACH,WAAW,gBAAgB;IAE3B;;OAEG;IACH,WAAW,gBAAgB;IAE3B;;OAEG;IACH,UAAU,eAAe;IAEzB;;OAEG;IACH,YAAY,iBAAiB;IAE7B;;;OAGG;IACH,eAAe,oBAAoB;IAEnC;;OAEG;IACH,cAAc,mBAAmB;IAEjC;;OAEG;IACH,gBAAgB,qBAAqB;IAErC;;OAEG;IACH,YAAY,iBAAiB;IAE7B;;OAEG;IACH,gBAAgB,qBAAqB;CACtC;AAED;;;GAGG;AACH,MAAM,WAAW,UAAU;IACzB;;OAEG;IACH,QAAQ,CAAC,QAAQ,EAAE,MAAM,CAAC;IAE1B;;OAEG;IACH,QAAQ,CAAC,SAAS,EAAE,MAAM,CAAC;IAE3B;;OAEG;IACH,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC;IAEzB;;OAEG;IACH,QAAQ,CAAC,QAAQ,CAAC,EAAE,MAAM,CAAC;IAE3B;;OAEG;IACH,QAAQ,CAAC,MAAM,EAAE,gBAAgB,CAAC;IAElC;;OAEG;IACH,QAAQ,CAAC,kBAAkB,CAAC,EAAE,MAAM,EAAE,CAAC;IAEvC;;OAEG;IACH,QAAQ,CAAC,MAAM,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;CACvC;AAED,wBAAgB,YAAY,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,IAAI,CAMhE;AAED;;;GAGG;AACH,wBAAgB,sBAAsB,CAAC,OAAO,EAAE,gBAAgB,GAAG,MAAM,CAIxE;AAED;;GAEG;AACH,wBAAgB,YAAY,CAAC,UAAU,EAAE,UAAU,GAAG,IAAI,CA+BzD;AASD,wBAAgB,aAAa,CAAC,SAAS,EAAE,GAAG,CAAC,MAAM,CAAC,QAQnD;AAED,wBAAgB,WAAW,CAAC,KAAK,EAAE,GAAG,GAAG,MAAM,CAS9C"}
|
|
@@ -19,3 +19,4 @@ export declare function watchTestWorker(options: IntegWatchOptions): Promise<voi
|
|
|
19
19
|
* check if there are any changes
|
|
20
20
|
*/
|
|
21
21
|
export declare function snapshotTestWorker(testInfo: IntegTestInfo, options?: SnapshotVerificationOptions): Promise<IntegTestWorkerConfig[]>;
|
|
22
|
+
//# sourceMappingURL=extract_worker.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"extract_worker.d.ts","sourceRoot":"","sources":["extract_worker.ts"],"names":[],"mappings":"AAEA,OAAO,KAAK,EAAE,aAAa,EAAE,MAAM,gCAAgC,CAAC;AAEpE,OAAO,KAAK,EAAE,qBAAqB,EAAE,2BAA2B,EAAc,MAAM,WAAW,CAAC;AAEhG,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,sBAAsB,CAAC;AAClE,OAAO,KAAK,EAAE,iBAAiB,EAAE,MAAM,uBAAuB,CAAC;AAE/D;;;;;;;GAOG;AACH,wBAAsB,eAAe,CAAC,OAAO,EAAE,qBAAqB,GAAG,OAAO,CAAC,qBAAqB,EAAE,CAAC,CA0EtG;AAED,wBAAsB,eAAe,CAAC,OAAO,EAAE,iBAAiB,GAAG,OAAO,CAAC,IAAI,CAAC,CAwB/E;AAED;;;;;GAKG;AACH,wBAAsB,kBAAkB,CAAC,QAAQ,EAAE,aAAa,EAAE,OAAO,GAAE,2BAAgC,GAAG,OAAO,CAAC,qBAAqB,EAAE,CAAC,CA4D7I"}
|