@aws-cdk/integ-runner 1.166.1 → 2.38.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -38,6 +38,12 @@ export interface RunOptions {
38
38
  * @default true
39
39
  */
40
40
  readonly updateWorkflow?: boolean;
41
+ /**
42
+ * The level of verbosity for logging.
43
+ *
44
+ * @default 0
45
+ */
46
+ readonly verbosity?: number;
41
47
  }
42
48
  /**
43
49
  * An integration test runner that orchestrates executing
@@ -15,6 +15,14 @@ class IntegTestRunner extends runner_base_1.IntegRunner {
15
15
  constructor(options, destructiveChanges) {
16
16
  super(options);
17
17
  this._destructiveChanges = destructiveChanges;
18
+ // We don't want new tests written in the legacy mode.
19
+ // If there is no existing snapshot _and_ this is a legacy
20
+ // test then point the user to the new `IntegTest` construct
21
+ if (!this.hasSnapshot() && this.isLegacyTest) {
22
+ throw new Error(`${this.testName} is a new test. Please use the IntegTest construct ` +
23
+ 'to configure the test\n' +
24
+ 'https://github.com/aws/aws-cdk/tree/main/packages/%40aws-cdk/integ-tests');
25
+ }
18
26
  }
19
27
  /**
20
28
  * When running integration tests with the update path workflow
@@ -87,12 +95,19 @@ class IntegTestRunner extends runner_base_1.IntegRunner {
87
95
  const clean = (_a = options.clean) !== null && _a !== void 0 ? _a : true;
88
96
  const updateWorkflowEnabled = ((_b = options.updateWorkflow) !== null && _b !== void 0 ? _b : true)
89
97
  && ((_c = actualTestCase.stackUpdateWorkflow) !== null && _c !== void 0 ? _c : true);
98
+ const enableForVerbosityLevel = (needed = 1) => {
99
+ var _a;
100
+ const verbosity = (_a = options.verbosity) !== null && _a !== void 0 ? _a : 0;
101
+ return (verbosity >= needed) ? true : undefined;
102
+ };
90
103
  try {
91
104
  if (!options.dryRun && ((_f = (_e = (_d = actualTestCase.cdkCommandOptions) === null || _d === void 0 ? void 0 : _d.deploy) === null || _e === void 0 ? void 0 : _e.enabled) !== null && _f !== void 0 ? _f : true)) {
92
105
  assertionResults = this.deploy({
93
106
  ...this.defaultArgs,
94
107
  profile: this.profile,
95
108
  requireApproval: cloud_assembly_schema_1.RequireApproval.NEVER,
109
+ verbose: enableForVerbosityLevel(3),
110
+ debug: enableForVerbosityLevel(4),
96
111
  }, updateWorkflowEnabled, options.testCaseName);
97
112
  }
98
113
  else {
@@ -127,6 +142,8 @@ class IntegTestRunner extends runner_base_1.IntegRunner {
127
142
  output: path.relative(this.directory, this.cdkOutDir),
128
143
  ...(_l = (_k = actualTestCase.cdkCommandOptions) === null || _k === void 0 ? void 0 : _k.destroy) === null || _l === void 0 ? void 0 : _l.args,
129
144
  context: this.getContext((_p = (_o = (_m = actualTestCase.cdkCommandOptions) === null || _m === void 0 ? void 0 : _m.destroy) === null || _o === void 0 ? void 0 : _o.args) === null || _p === void 0 ? void 0 : _p.context),
145
+ verbose: enableForVerbosityLevel(3),
146
+ debug: enableForVerbosityLevel(4),
130
147
  });
131
148
  }
132
149
  }
@@ -277,4 +294,4 @@ class IntegTestRunner extends runner_base_1.IntegRunner {
277
294
  }
278
295
  }
279
296
  exports.IntegTestRunner = IntegTestRunner;
280
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"integ-test-runner.js","sourceRoot":"","sources":["integ-test-runner.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { RequireApproval } from '@aws-cdk/cloud-assembly-schema';\nimport { DeployOptions, DestroyOptions } from 'cdk-cli-wrapper';\nimport * as fs from 'fs-extra';\nimport * as logger from '../logger';\nimport { chain, exec } from '../utils';\nimport { DestructiveChange, AssertionResults, AssertionResult } from '../workers/common';\nimport { IntegRunnerOptions, IntegRunner, DEFAULT_SYNTH_OPTIONS } from './runner-base';\n\n/**\n * Options for the integration test runner\n */\nexport interface RunOptions {\n  /**\n   * The name of the test case\n   */\n  readonly testCaseName: string;\n\n  /**\n   * Whether or not to run `cdk destroy` and cleanup the\n   * integration test stacks.\n   *\n   * Set this to false if you need to perform any validation\n   * or troubleshooting after deployment.\n   *\n   * @default true\n   */\n  readonly clean?: boolean;\n\n  /**\n   * If set to true, the integration test will not deploy\n   * anything and will simply update the snapshot.\n   *\n   * You should NOT use this method since you are essentially\n   * bypassing the integration test.\n   *\n   * @default false\n   */\n  readonly dryRun?: boolean;\n\n  /**\n   * If this is set to false then the stack update workflow will\n   * not be run\n   *\n   * The update workflow exists to check for cases where a change would cause\n   * a failure to an existing stack, but not for a newly created stack.\n   *\n   * @default true\n   */\n  readonly updateWorkflow?: boolean;\n}\n\n/**\n * An integration test runner that orchestrates executing\n * integration tests\n */\nexport class IntegTestRunner extends IntegRunner {\n  constructor(options: IntegRunnerOptions, destructiveChanges?: DestructiveChange[]) {\n    super(options);\n    this._destructiveChanges = destructiveChanges;\n  }\n\n  /**\n   * When running integration tests with the update path workflow\n   * it is important that the snapshot that is deployed is the current snapshot\n   * from the upstream branch. In order to guarantee that, first checkout the latest\n   * (to the user) snapshot from upstream\n   *\n   * It is not straightforward to figure out what branch the current\n   * working branch was created from. This is a best effort attempt to do so.\n   * This assumes that there is an 'origin'. `git remote show origin` returns a list of\n   * all branches and we then search for one that starts with `HEAD branch: `\n   */\n  private checkoutSnapshot(): void {\n    const cwd = this.directory;\n\n    // https://git-scm.com/docs/git-merge-base\n    let baseBranch: string | undefined = undefined;\n    // try to find the base branch that the working branch was created from\n    try {\n      const origin: string = exec(['git', 'remote', 'show', 'origin'], {\n        cwd,\n      });\n      const originLines = origin.split('\\n');\n      for (const line of originLines) {\n        if (line.trim().startsWith('HEAD branch: ')) {\n          baseBranch = line.trim().split('HEAD branch: ')[1];\n        }\n      }\n    } catch (e) {\n      logger.warning('%s\\n%s',\n        'Could not determine git origin branch.',\n        `You need to manually checkout the snapshot directory ${this.snapshotDir}` +\n        'from the merge-base (https://git-scm.com/docs/git-merge-base)',\n      );\n      logger.warning('error: %s', e);\n    }\n\n    // if we found the base branch then get the merge-base (most recent common commit)\n    // and checkout the snapshot using that commit\n    if (baseBranch) {\n      const relativeSnapshotDir = path.relative(this.directory, this.snapshotDir);\n\n      try {\n        const base = exec(['git', 'merge-base', 'HEAD', baseBranch], {\n          cwd,\n        });\n        exec(['git', 'checkout', base, '--', relativeSnapshotDir], {\n          cwd,\n        });\n      } catch (e) {\n        logger.warning('%s\\n%s',\n          `Could not checkout snapshot directory ${this.snapshotDir} using these commands: `,\n          `git merge-base HEAD ${baseBranch} && git checkout {merge-base} -- ${relativeSnapshotDir}`,\n        );\n        logger.warning('error: %s', e);\n      }\n    }\n  }\n\n  /**\n   * Orchestrates running integration tests. Currently this includes\n   *\n   * 1. (if update workflow is enabled) Deploying the snapshot test stacks\n   * 2. Deploying the integration test stacks\n   * 2. Saving the snapshot (if successful)\n   * 3. Destroying the integration test stacks (if clean=false)\n   *\n   * The update workflow exists to check for cases where a change would cause\n   * a failure to an existing stack, but not for a newly created stack.\n   */\n  public runIntegTestCase(options: RunOptions): AssertionResults | undefined {\n    let assertionResults: AssertionResults | undefined;\n    const actualTestCase = this.actualTestSuite.testSuite[options.testCaseName];\n    if (!actualTestCase) {\n      throw new Error(`Did not find test case name '${options.testCaseName}' in '${Object.keys(this.actualTestSuite.testSuite)}'`);\n    }\n    const clean = options.clean ?? true;\n    const updateWorkflowEnabled = (options.updateWorkflow ?? true)\n      && (actualTestCase.stackUpdateWorkflow ?? true);\n    try {\n      if (!options.dryRun && (actualTestCase.cdkCommandOptions?.deploy?.enabled ?? true)) {\n        assertionResults = this.deploy(\n          {\n            ...this.defaultArgs,\n            profile: this.profile,\n            requireApproval: RequireApproval.NEVER,\n          },\n          updateWorkflowEnabled,\n          options.testCaseName,\n        );\n      } else {\n        const env: Record<string, any> = {\n          ...DEFAULT_SYNTH_OPTIONS.env,\n          CDK_CONTEXT_JSON: JSON.stringify(this.getContext()),\n        };\n        this.cdk.synthFast({\n          execCmd: this.cdkApp.split(' '),\n          env,\n          output: path.relative(this.directory, this.cdkOutDir),\n        });\n      }\n      // only create the snapshot if there are no assertion assertion results\n      // (i.e. no failures)\n      if (!assertionResults) {\n        this.createSnapshot();\n      }\n    } catch (e) {\n      throw e;\n    } finally {\n      if (!options.dryRun) {\n        if (clean && (actualTestCase.cdkCommandOptions?.destroy?.enabled ?? true)) {\n          this.destroy(options.testCaseName, {\n            ...this.defaultArgs,\n            profile: this.profile,\n            all: true,\n            force: true,\n            app: this.cdkApp,\n            output: path.relative(this.directory, this.cdkOutDir),\n            ...actualTestCase.cdkCommandOptions?.destroy?.args,\n            context: this.getContext(actualTestCase.cdkCommandOptions?.destroy?.args?.context),\n          });\n        }\n      }\n      this.cleanup();\n    }\n    return assertionResults;\n  }\n\n  /**\n   * Perform a integ test case stack destruction\n   */\n  private destroy(testCaseName: string, destroyArgs: DestroyOptions) {\n    const actualTestCase = this.actualTestSuite.testSuite[testCaseName];\n    try {\n      if (actualTestCase.hooks?.preDestroy) {\n        exec([chain(actualTestCase.hooks.preDestroy)], {\n          cwd: path.dirname(this.snapshotDir),\n        });\n      }\n      this.cdk.destroy({\n        ...destroyArgs,\n      });\n\n      if (actualTestCase.hooks?.postDestroy) {\n        exec([chain(actualTestCase.hooks.postDestroy)], {\n          cwd: path.dirname(this.snapshotDir),\n        });\n      }\n    } catch (e) {\n      this.parseError(e,\n        actualTestCase.cdkCommandOptions?.destroy?.expectError ?? false,\n        actualTestCase.cdkCommandOptions?.destroy?.expectedMessage,\n      );\n    }\n  }\n\n  /**\n   * Perform a integ test case deployment, including\n   * peforming the update workflow\n   */\n  private deploy(\n    deployArgs: DeployOptions,\n    updateWorkflowEnabled: boolean,\n    testCaseName: string,\n  ): AssertionResults | undefined {\n    const actualTestCase = this.actualTestSuite.testSuite[testCaseName];\n    try {\n      if (actualTestCase.hooks?.preDeploy) {\n        exec([chain(actualTestCase.hooks?.preDeploy)], {\n          cwd: path.dirname(this.snapshotDir),\n        });\n      }\n      // if the update workflow is not disabled, first\n      // perform a deployment with the exising snapshot\n      // then perform a deployment (which will be a stack update)\n      // with the current integration test\n      // We also only want to run the update workflow if there is an existing\n      // snapshot (otherwise there is nothing to update)\n      if (updateWorkflowEnabled && this.hasSnapshot() &&\n        (this.expectedTestSuite && testCaseName in this.expectedTestSuite?.testSuite)) {\n        // make sure the snapshot is the latest from 'origin'\n        this.checkoutSnapshot();\n        const expectedTestCase = this.expectedTestSuite.testSuite[testCaseName];\n        this.cdk.deploy({\n          ...deployArgs,\n          stacks: expectedTestCase.stacks,\n          ...expectedTestCase?.cdkCommandOptions?.deploy?.args,\n          context: this.getContext(expectedTestCase?.cdkCommandOptions?.deploy?.args?.context),\n          app: path.relative(this.directory, this.snapshotDir),\n          lookups: this.expectedTestSuite?.enableLookups,\n        });\n      }\n      // now deploy the \"actual\" test. If there are any assertions\n      // deploy the assertion stack as well\n      this.cdk.deploy({\n        ...deployArgs,\n        lookups: this.actualTestSuite.enableLookups,\n        stacks: [\n          ...actualTestCase.stacks,\n          ...actualTestCase.assertionStack ? [actualTestCase.assertionStack] : [],\n        ],\n        rollback: false,\n        output: path.relative(this.directory, this.cdkOutDir),\n        ...actualTestCase?.cdkCommandOptions?.deploy?.args,\n        ...actualTestCase.assertionStack ? { outputsFile: path.relative(this.directory, path.join(this.cdkOutDir, 'assertion-results.json')) } : undefined,\n        context: this.getContext(actualTestCase?.cdkCommandOptions?.deploy?.args?.context),\n        app: this.cdkApp,\n      });\n\n      if (actualTestCase.hooks?.postDeploy) {\n        exec([chain(actualTestCase.hooks?.postDeploy)], {\n          cwd: path.dirname(this.snapshotDir),\n        });\n      }\n\n      if (actualTestCase.assertionStack) {\n        return this.processAssertionResults(\n          path.join(this.cdkOutDir, 'assertion-results.json'),\n          actualTestCase.assertionStack,\n        );\n      }\n    } catch (e) {\n      this.parseError(e,\n        actualTestCase.cdkCommandOptions?.deploy?.expectError ?? false,\n        actualTestCase.cdkCommandOptions?.deploy?.expectedMessage,\n      );\n    }\n    return;\n  }\n\n  /**\n   * Process the outputsFile which contains the assertions results as stack\n   * outputs\n   */\n  private processAssertionResults(file: string, assertionStackId: string): AssertionResults | undefined {\n    const results: AssertionResults = {};\n    if (fs.existsSync(file)) {\n      try {\n        const outputs: { [key: string]: { [key: string]: string } } = fs.readJSONSync(file);\n\n        if (assertionStackId in outputs) {\n          for (const [assertionId, result] of Object.entries(outputs[assertionStackId])) {\n            if (assertionId.startsWith('AssertionResults')) {\n              const assertionResult: AssertionResult = JSON.parse(result.replace(/\\n/g, '\\\\n'));\n              if (assertionResult.status === 'fail') {\n                results[assertionId] = assertionResult;\n              }\n            }\n          }\n        }\n      } catch (e) {\n        // if there are outputs, but they cannot be processed, then throw an error\n        // so that the test fails\n        results[assertionStackId] = {\n          status: 'fail',\n          message: `error processing assertion results: ${e}`,\n        };\n      } finally {\n        // remove the outputs file so it is not part of the snapshot\n        // it will contain env specific information from values\n        // resolved at deploy time\n        fs.unlinkSync(file);\n      }\n    }\n    return Object.keys(results).length > 0 ? results : undefined;\n  }\n\n  /**\n   * Parses an error message returned from a CDK command\n   */\n  private parseError(e: unknown, expectError: boolean, expectedMessage?: string) {\n    if (expectError) {\n      if (expectedMessage) {\n        const message = (e as Error).message;\n        if (!message.match(expectedMessage)) {\n          throw (e);\n        }\n      }\n    } else {\n      throw e;\n    }\n  }\n}\n"]}
297
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"integ-test-runner.js","sourceRoot":"","sources":["integ-test-runner.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { RequireApproval } from '@aws-cdk/cloud-assembly-schema';\nimport { DeployOptions, DestroyOptions } from 'cdk-cli-wrapper';\nimport * as fs from 'fs-extra';\nimport * as logger from '../logger';\nimport { chain, exec } from '../utils';\nimport { DestructiveChange, AssertionResults, AssertionResult } from '../workers/common';\nimport { IntegRunnerOptions, IntegRunner, DEFAULT_SYNTH_OPTIONS } from './runner-base';\n\n/**\n * Options for the integration test runner\n */\nexport interface RunOptions {\n  /**\n   * The name of the test case\n   */\n  readonly testCaseName: string;\n\n  /**\n   * Whether or not to run `cdk destroy` and cleanup the\n   * integration test stacks.\n   *\n   * Set this to false if you need to perform any validation\n   * or troubleshooting after deployment.\n   *\n   * @default true\n   */\n  readonly clean?: boolean;\n\n  /**\n   * If set to true, the integration test will not deploy\n   * anything and will simply update the snapshot.\n   *\n   * You should NOT use this method since you are essentially\n   * bypassing the integration test.\n   *\n   * @default false\n   */\n  readonly dryRun?: boolean;\n\n  /**\n   * If this is set to false then the stack update workflow will\n   * not be run\n   *\n   * The update workflow exists to check for cases where a change would cause\n   * a failure to an existing stack, but not for a newly created stack.\n   *\n   * @default true\n   */\n  readonly updateWorkflow?: boolean;\n\n  /**\n   * The level of verbosity for logging.\n   *\n   * @default 0\n   */\n  readonly verbosity?: number;\n}\n\n/**\n * An integration test runner that orchestrates executing\n * integration tests\n */\nexport class IntegTestRunner extends IntegRunner {\n  constructor(options: IntegRunnerOptions, destructiveChanges?: DestructiveChange[]) {\n    super(options);\n    this._destructiveChanges = destructiveChanges;\n\n    // We don't want new tests written in the legacy mode.\n    // If there is no existing snapshot _and_ this is a legacy\n    // test then point the user to the new `IntegTest` construct\n    if (!this.hasSnapshot() && this.isLegacyTest) {\n      throw new Error(`${this.testName} is a new test. Please use the IntegTest construct ` +\n       'to configure the test\\n' +\n        'https://github.com/aws/aws-cdk/tree/main/packages/%40aws-cdk/integ-tests',\n      );\n    }\n  }\n\n  /**\n   * When running integration tests with the update path workflow\n   * it is important that the snapshot that is deployed is the current snapshot\n   * from the upstream branch. In order to guarantee that, first checkout the latest\n   * (to the user) snapshot from upstream\n   *\n   * It is not straightforward to figure out what branch the current\n   * working branch was created from. This is a best effort attempt to do so.\n   * This assumes that there is an 'origin'. `git remote show origin` returns a list of\n   * all branches and we then search for one that starts with `HEAD branch: `\n   */\n  private checkoutSnapshot(): void {\n    const cwd = this.directory;\n\n    // https://git-scm.com/docs/git-merge-base\n    let baseBranch: string | undefined = undefined;\n    // try to find the base branch that the working branch was created from\n    try {\n      const origin: string = exec(['git', 'remote', 'show', 'origin'], {\n        cwd,\n      });\n      const originLines = origin.split('\\n');\n      for (const line of originLines) {\n        if (line.trim().startsWith('HEAD branch: ')) {\n          baseBranch = line.trim().split('HEAD branch: ')[1];\n        }\n      }\n    } catch (e) {\n      logger.warning('%s\\n%s',\n        'Could not determine git origin branch.',\n        `You need to manually checkout the snapshot directory ${this.snapshotDir}` +\n        'from the merge-base (https://git-scm.com/docs/git-merge-base)',\n      );\n      logger.warning('error: %s', e);\n    }\n\n    // if we found the base branch then get the merge-base (most recent common commit)\n    // and checkout the snapshot using that commit\n    if (baseBranch) {\n      const relativeSnapshotDir = path.relative(this.directory, this.snapshotDir);\n\n      try {\n        const base = exec(['git', 'merge-base', 'HEAD', baseBranch], {\n          cwd,\n        });\n        exec(['git', 'checkout', base, '--', relativeSnapshotDir], {\n          cwd,\n        });\n      } catch (e) {\n        logger.warning('%s\\n%s',\n          `Could not checkout snapshot directory ${this.snapshotDir} using these commands: `,\n          `git merge-base HEAD ${baseBranch} && git checkout {merge-base} -- ${relativeSnapshotDir}`,\n        );\n        logger.warning('error: %s', e);\n      }\n    }\n  }\n\n  /**\n   * Orchestrates running integration tests. Currently this includes\n   *\n   * 1. (if update workflow is enabled) Deploying the snapshot test stacks\n   * 2. Deploying the integration test stacks\n   * 2. Saving the snapshot (if successful)\n   * 3. Destroying the integration test stacks (if clean=false)\n   *\n   * The update workflow exists to check for cases where a change would cause\n   * a failure to an existing stack, but not for a newly created stack.\n   */\n  public runIntegTestCase(options: RunOptions): AssertionResults | undefined {\n    let assertionResults: AssertionResults | undefined;\n    const actualTestCase = this.actualTestSuite.testSuite[options.testCaseName];\n    if (!actualTestCase) {\n      throw new Error(`Did not find test case name '${options.testCaseName}' in '${Object.keys(this.actualTestSuite.testSuite)}'`);\n    }\n    const clean = options.clean ?? true;\n    const updateWorkflowEnabled = (options.updateWorkflow ?? true)\n      && (actualTestCase.stackUpdateWorkflow ?? true);\n    const enableForVerbosityLevel = (needed = 1) => {\n      const verbosity = options.verbosity ?? 0;\n      return (verbosity >= needed) ? true : undefined;\n    };\n\n    try {\n      if (!options.dryRun && (actualTestCase.cdkCommandOptions?.deploy?.enabled ?? true)) {\n        assertionResults = this.deploy(\n          {\n            ...this.defaultArgs,\n            profile: this.profile,\n            requireApproval: RequireApproval.NEVER,\n            verbose: enableForVerbosityLevel(3),\n            debug: enableForVerbosityLevel(4),\n          },\n          updateWorkflowEnabled,\n          options.testCaseName,\n        );\n      } else {\n        const env: Record<string, any> = {\n          ...DEFAULT_SYNTH_OPTIONS.env,\n          CDK_CONTEXT_JSON: JSON.stringify(this.getContext()),\n        };\n        this.cdk.synthFast({\n          execCmd: this.cdkApp.split(' '),\n          env,\n          output: path.relative(this.directory, this.cdkOutDir),\n        });\n      }\n      // only create the snapshot if there are no assertion assertion results\n      // (i.e. no failures)\n      if (!assertionResults) {\n        this.createSnapshot();\n      }\n    } catch (e) {\n      throw e;\n    } finally {\n      if (!options.dryRun) {\n        if (clean && (actualTestCase.cdkCommandOptions?.destroy?.enabled ?? true)) {\n          this.destroy(options.testCaseName, {\n            ...this.defaultArgs,\n            profile: this.profile,\n            all: true,\n            force: true,\n            app: this.cdkApp,\n            output: path.relative(this.directory, this.cdkOutDir),\n            ...actualTestCase.cdkCommandOptions?.destroy?.args,\n            context: this.getContext(actualTestCase.cdkCommandOptions?.destroy?.args?.context),\n            verbose: enableForVerbosityLevel(3),\n            debug: enableForVerbosityLevel(4),\n          });\n        }\n      }\n      this.cleanup();\n    }\n    return assertionResults;\n  }\n\n  /**\n   * Perform a integ test case stack destruction\n   */\n  private destroy(testCaseName: string, destroyArgs: DestroyOptions) {\n    const actualTestCase = this.actualTestSuite.testSuite[testCaseName];\n    try {\n      if (actualTestCase.hooks?.preDestroy) {\n        exec([chain(actualTestCase.hooks.preDestroy)], {\n          cwd: path.dirname(this.snapshotDir),\n        });\n      }\n      this.cdk.destroy({\n        ...destroyArgs,\n      });\n\n      if (actualTestCase.hooks?.postDestroy) {\n        exec([chain(actualTestCase.hooks.postDestroy)], {\n          cwd: path.dirname(this.snapshotDir),\n        });\n      }\n    } catch (e) {\n      this.parseError(e,\n        actualTestCase.cdkCommandOptions?.destroy?.expectError ?? false,\n        actualTestCase.cdkCommandOptions?.destroy?.expectedMessage,\n      );\n    }\n  }\n\n  /**\n   * Perform a integ test case deployment, including\n   * peforming the update workflow\n   */\n  private deploy(\n    deployArgs: DeployOptions,\n    updateWorkflowEnabled: boolean,\n    testCaseName: string,\n  ): AssertionResults | undefined {\n    const actualTestCase = this.actualTestSuite.testSuite[testCaseName];\n    try {\n      if (actualTestCase.hooks?.preDeploy) {\n        exec([chain(actualTestCase.hooks?.preDeploy)], {\n          cwd: path.dirname(this.snapshotDir),\n        });\n      }\n      // if the update workflow is not disabled, first\n      // perform a deployment with the exising snapshot\n      // then perform a deployment (which will be a stack update)\n      // with the current integration test\n      // We also only want to run the update workflow if there is an existing\n      // snapshot (otherwise there is nothing to update)\n      if (updateWorkflowEnabled && this.hasSnapshot() &&\n        (this.expectedTestSuite && testCaseName in this.expectedTestSuite?.testSuite)) {\n        // make sure the snapshot is the latest from 'origin'\n        this.checkoutSnapshot();\n        const expectedTestCase = this.expectedTestSuite.testSuite[testCaseName];\n        this.cdk.deploy({\n          ...deployArgs,\n          stacks: expectedTestCase.stacks,\n          ...expectedTestCase?.cdkCommandOptions?.deploy?.args,\n          context: this.getContext(expectedTestCase?.cdkCommandOptions?.deploy?.args?.context),\n          app: path.relative(this.directory, this.snapshotDir),\n          lookups: this.expectedTestSuite?.enableLookups,\n        });\n      }\n      // now deploy the \"actual\" test. If there are any assertions\n      // deploy the assertion stack as well\n      this.cdk.deploy({\n        ...deployArgs,\n        lookups: this.actualTestSuite.enableLookups,\n        stacks: [\n          ...actualTestCase.stacks,\n          ...actualTestCase.assertionStack ? [actualTestCase.assertionStack] : [],\n        ],\n        rollback: false,\n        output: path.relative(this.directory, this.cdkOutDir),\n        ...actualTestCase?.cdkCommandOptions?.deploy?.args,\n        ...actualTestCase.assertionStack ? { outputsFile: path.relative(this.directory, path.join(this.cdkOutDir, 'assertion-results.json')) } : undefined,\n        context: this.getContext(actualTestCase?.cdkCommandOptions?.deploy?.args?.context),\n        app: this.cdkApp,\n      });\n\n      if (actualTestCase.hooks?.postDeploy) {\n        exec([chain(actualTestCase.hooks?.postDeploy)], {\n          cwd: path.dirname(this.snapshotDir),\n        });\n      }\n\n      if (actualTestCase.assertionStack) {\n        return this.processAssertionResults(\n          path.join(this.cdkOutDir, 'assertion-results.json'),\n          actualTestCase.assertionStack,\n        );\n      }\n    } catch (e) {\n      this.parseError(e,\n        actualTestCase.cdkCommandOptions?.deploy?.expectError ?? false,\n        actualTestCase.cdkCommandOptions?.deploy?.expectedMessage,\n      );\n    }\n    return;\n  }\n\n  /**\n   * Process the outputsFile which contains the assertions results as stack\n   * outputs\n   */\n  private processAssertionResults(file: string, assertionStackId: string): AssertionResults | undefined {\n    const results: AssertionResults = {};\n    if (fs.existsSync(file)) {\n      try {\n        const outputs: { [key: string]: { [key: string]: string } } = fs.readJSONSync(file);\n\n        if (assertionStackId in outputs) {\n          for (const [assertionId, result] of Object.entries(outputs[assertionStackId])) {\n            if (assertionId.startsWith('AssertionResults')) {\n              const assertionResult: AssertionResult = JSON.parse(result.replace(/\\n/g, '\\\\n'));\n              if (assertionResult.status === 'fail') {\n                results[assertionId] = assertionResult;\n              }\n            }\n          }\n        }\n      } catch (e) {\n        // if there are outputs, but they cannot be processed, then throw an error\n        // so that the test fails\n        results[assertionStackId] = {\n          status: 'fail',\n          message: `error processing assertion results: ${e}`,\n        };\n      } finally {\n        // remove the outputs file so it is not part of the snapshot\n        // it will contain env specific information from values\n        // resolved at deploy time\n        fs.unlinkSync(file);\n      }\n    }\n    return Object.keys(results).length > 0 ? results : undefined;\n  }\n\n  /**\n   * Parses an error message returned from a CDK command\n   */\n  private parseError(e: unknown, expectError: boolean, expectedMessage?: string) {\n    if (expectError) {\n      if (expectedMessage) {\n        const message = (e as Error).message;\n        if (!message.match(expectedMessage)) {\n          throw (e);\n        }\n      }\n    } else {\n      throw e;\n    }\n  }\n}\n"]}
@@ -39,6 +39,12 @@ export declare class IntegTest {
39
39
  * The absolute path to the file
40
40
  */
41
41
  readonly absoluteFileName: string;
42
+ /**
43
+ * The normalized name of the test. This name
44
+ * will be the same regardless of what directory the tool
45
+ * is run from.
46
+ */
47
+ readonly normalizedTestName: string;
42
48
  /**
43
49
  * Directory the test is in
44
50
  */
@@ -25,6 +25,7 @@ class IntegTest {
25
25
  ? parsed.name
26
26
  : path.join(path.relative(this.info.discoveryRoot, parsed.dir), parsed.name);
27
27
  const nakedTestName = parsed.name.slice(6); // Leave name without 'integ.' and '.ts'
28
+ this.normalizedTestName = parsed.name;
28
29
  this.snapshotDir = path.join(this.directory, `${nakedTestName}.integ.snapshot`);
29
30
  this.temporaryOutputDir = path.join(this.directory, `${CDK_OUTDIR_PREFIX}.${nakedTestName}`);
30
31
  }
@@ -133,4 +134,4 @@ class IntegrationTests {
133
134
  }
134
135
  }
135
136
  exports.IntegrationTests = IntegrationTests;
136
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"integration-tests.js","sourceRoot":"","sources":["integration-tests.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport * as fs from 'fs-extra';\n\nconst CDK_OUTDIR_PREFIX = 'cdk-integ.out';\n\n/**\n * Represents a single integration test\n *\n * This type is a data-only structure, so it can trivially be passed to workers.\n * Derived attributes are calculated using the `IntegTest` class.\n */\nexport interface IntegTestInfo {\n  /**\n   * Path to the file to run\n   *\n   * Path is relative to the current working directory.\n   */\n  readonly fileName: string;\n\n  /**\n   * The root directory we discovered this test from\n   *\n   * Path is relative to the current working directory.\n   */\n  readonly discoveryRoot: string;\n}\n\n/**\n * Derived information for IntegTests\n */\nexport class IntegTest {\n  /**\n   * The name of the file to run\n   *\n   * Path is relative to the current working directory.\n   */\n  public readonly fileName: string;\n\n  /**\n   * Relative path to the file to run\n   *\n   * Relative from the \"discovery root\".\n   */\n  public readonly discoveryRelativeFileName: string;\n\n  /**\n   * The absolute path to the file\n   */\n  public readonly absoluteFileName: string;\n\n  /**\n   * Directory the test is in\n   */\n  public readonly directory: string;\n\n  /**\n   * Display name for the test\n   *\n   * Depends on the discovery directory.\n   *\n   * Looks like `integ.mytest` or `package/test/integ.mytest`.\n   */\n  public readonly testName: string;\n\n  /**\n   * Path of the snapshot directory for this test\n   */\n  public readonly snapshotDir: string;\n\n  /**\n   * Path to the temporary output directory for this test\n   */\n  public readonly temporaryOutputDir: string;\n\n  constructor(public readonly info: IntegTestInfo) {\n    this.absoluteFileName = path.resolve(info.fileName);\n    this.fileName = path.relative(process.cwd(), info.fileName);\n\n    const parsed = path.parse(this.fileName);\n    this.discoveryRelativeFileName = path.relative(info.discoveryRoot, info.fileName);\n    this.directory = parsed.dir;\n\n    // if we are running in a package directory then just use the fileName\n    // as the testname, but if we are running in a parent directory with\n    // multiple packages then use the directory/filename as the testname\n    //\n    // Looks either like `integ.mytest` or `package/test/integ.mytest`.\n    const relDiscoveryRoot = path.relative(process.cwd(), info.discoveryRoot);\n    this.testName = this.directory === path.join(relDiscoveryRoot, 'test') || this.directory === path.join(relDiscoveryRoot)\n      ? parsed.name\n      : path.join(path.relative(this.info.discoveryRoot, parsed.dir), parsed.name);\n\n    const nakedTestName = parsed.name.slice(6); // Leave name without 'integ.' and '.ts'\n    this.snapshotDir = path.join(this.directory, `${nakedTestName}.integ.snapshot`);\n    this.temporaryOutputDir = path.join(this.directory, `${CDK_OUTDIR_PREFIX}.${nakedTestName}`);\n  }\n\n  /**\n   * Whether this test matches the user-given name\n   *\n   * We are very lenient here. A name matches if it matches:\n   *\n   * - The CWD-relative filename\n   * - The discovery root-relative filename\n   * - The suite name\n   * - The absolute filename\n   */\n  public matches(name: string) {\n    return [\n      this.fileName,\n      this.discoveryRelativeFileName,\n      this.testName,\n      this.absoluteFileName,\n    ].includes(name);\n  }\n}\n\n/**\n * The list of tests to run can be provided in a file\n * instead of as command line arguments.\n */\nexport interface IntegrationTestFileConfig {\n  /**\n   * If this is set to true then the list of tests\n   * provided will be excluded\n   *\n   * @default false\n   */\n  readonly exclude?: boolean;\n\n  /**\n   * List of tests to include (or exclude if `exclude=true`)\n   */\n  readonly tests: string[];\n}\n\n/**\n * Discover integration tests\n */\nexport class IntegrationTests {\n  constructor(private readonly directory: string) {\n  }\n\n  /**\n   * Takes a file name of a file that contains a list of test\n   * to either run or exclude and returns a list of Integration Tests to run\n   */\n  public async fromFile(fileName: string): Promise<IntegTest[]> {\n    const file: IntegrationTestFileConfig = JSON.parse(fs.readFileSync(fileName, { encoding: 'utf-8' }));\n    const foundTests = await this.discover();\n\n    const allTests = this.filterTests(foundTests, file.tests, file.exclude);\n\n    return allTests;\n  }\n\n  /**\n   * If the user provides a list of tests, these can either be a list of tests to include or a list of tests to exclude.\n   *\n   * - If it is a list of tests to include then we discover all available tests and check whether they have provided valid tests.\n   *   If they have provided a test name that we don't find, then we write out that error message.\n   * - If it is a list of tests to exclude, then we discover all available tests and filter out the tests that were provided by the user.\n   */\n  private filterTests(discoveredTests: IntegTest[], requestedTests?: string[], exclude?: boolean): IntegTest[] {\n    if (!requestedTests) {\n      return discoveredTests;\n    }\n\n\n    const allTests = discoveredTests.filter(t => {\n      const matches = requestedTests.some(pattern => t.matches(pattern));\n      return matches !== !!exclude; // Looks weird but is equal to (matches && !exclude) || (!matches && exclude)\n    });\n\n    // If not excluding, all patterns must have matched at least one test\n    if (!exclude) {\n      const unmatchedPatterns = requestedTests.filter(pattern => !discoveredTests.some(t => t.matches(pattern)));\n      for (const unmatched of unmatchedPatterns) {\n        process.stderr.write(`No such integ test: ${unmatched}\\n`);\n      }\n      if (unmatchedPatterns.length > 0) {\n        process.stderr.write(`Available tests: ${discoveredTests.map(t => t.discoveryRelativeFileName).join(' ')}\\n`);\n        return [];\n      }\n    }\n\n    return allTests;\n  }\n\n  /**\n   * Takes an optional list of tests to look for, otherwise\n   * it will look for all tests from the directory\n   *\n   * @param tests Tests to include or exclude, undefined means include all tests.\n   * @param exclude Whether the 'tests' list is inclusive or exclusive (inclusive by default).\n   */\n  public async fromCliArgs(tests?: string[], exclude?: boolean): Promise<IntegTest[]> {\n    const discoveredTests = await this.discover();\n\n    const allTests = this.filterTests(discoveredTests, tests, exclude);\n\n    return allTests;\n  }\n\n  private async discover(): Promise<IntegTest[]> {\n    const files = await this.readTree();\n    const integs = files.filter(fileName => path.basename(fileName).startsWith('integ.') && path.basename(fileName).endsWith('.js'));\n    return this.request(integs);\n  }\n\n  private request(files: string[]): IntegTest[] {\n    return files.map(fileName => new IntegTest({ discoveryRoot: this.directory, fileName }));\n  }\n\n  private async readTree(): Promise<string[]> {\n    const ret = new Array<string>();\n\n    async function recurse(dir: string) {\n      const files = await fs.readdir(dir);\n      for (const file of files) {\n        const fullPath = path.join(dir, file);\n        const statf = await fs.stat(fullPath);\n        if (statf.isFile()) { ret.push(fullPath); }\n        if (statf.isDirectory()) { await recurse(path.join(fullPath)); }\n      }\n    }\n\n    await recurse(this.directory);\n    return ret;\n  }\n}\n"]}
137
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"integration-tests.js","sourceRoot":"","sources":["integration-tests.ts"],"names":[],"mappings":";;;AAAA,6BAA6B;AAC7B,+BAA+B;AAE/B,MAAM,iBAAiB,GAAG,eAAe,CAAC;AAwB1C;;GAEG;AACH,MAAa,SAAS;IAmDpB,YAA4B,IAAmB;QAAnB,SAAI,GAAJ,IAAI,CAAe;QAC7C,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QACpD,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;QAE5D,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QACzC,IAAI,CAAC,yBAAyB,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,aAAa,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;QAClF,IAAI,CAAC,SAAS,GAAG,MAAM,CAAC,GAAG,CAAC;QAE5B,sEAAsE;QACtE,oEAAoE;QACpE,oEAAoE;QACpE,EAAE;QACF,mEAAmE;QACnE,MAAM,gBAAgB,GAAG,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,GAAG,EAAE,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC;QAC1E,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,SAAS,KAAK,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,MAAM,CAAC,IAAI,IAAI,CAAC,SAAS,KAAK,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC;YACtH,CAAC,CAAC,MAAM,CAAC,IAAI;YACb,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,MAAM,CAAC,IAAI,CAAC,CAAC;QAE/E,MAAM,aAAa,GAAG,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,wCAAwC;QACpF,IAAI,CAAC,kBAAkB,GAAG,MAAM,CAAC,IAAI,CAAC;QACtC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,GAAG,aAAa,iBAAiB,CAAC,CAAC;QAChF,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,GAAG,iBAAiB,IAAI,aAAa,EAAE,CAAC,CAAC;IAC/F,CAAC;IAED;;;;;;;;;OASG;IACI,OAAO,CAAC,IAAY;QACzB,OAAO;YACL,IAAI,CAAC,QAAQ;YACb,IAAI,CAAC,yBAAyB;YAC9B,IAAI,CAAC,QAAQ;YACb,IAAI,CAAC,gBAAgB;SACtB,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;IACnB,CAAC;CACF;AA7FD,8BA6FC;AAqBD;;GAEG;AACH,MAAa,gBAAgB;IAC3B,YAA6B,SAAiB;QAAjB,cAAS,GAAT,SAAS,CAAQ;IAC9C,CAAC;IAED;;;OAGG;IACI,KAAK,CAAC,QAAQ,CAAC,QAAgB;QACpC,MAAM,IAAI,GAA8B,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,EAAE,QAAQ,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC;QACrG,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,QAAQ,EAAE,CAAC;QAEzC,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,UAAU,EAAE,IAAI,CAAC,KAAK,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;QAExE,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED;;;;;;OAMG;IACK,WAAW,CAAC,eAA4B,EAAE,cAAyB,EAAE,OAAiB;QAC5F,IAAI,CAAC,cAAc,EAAE;YACnB,OAAO,eAAe,CAAC;SACxB;QAGD,MAAM,QAAQ,GAAG,eAAe,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE;YAC1C,MAAM,OAAO,GAAG,cAAc,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC;YACnE,OAAO,OAAO,KAAK,CAAC,CAAC,OAAO,CAAC,CAAC,6EAA6E;QAC7G,CAAC,CAAC,CAAC;QAEH,qEAAqE;QACrE,IAAI,CAAC,OAAO,EAAE;YACZ,MAAM,iBAAiB,GAAG,cAAc,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;YAC3G,KAAK,MAAM,SAAS,IAAI,iBAAiB,EAAE;gBACzC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,uBAAuB,SAAS,IAAI,CAAC,CAAC;aAC5D;YACD,IAAI,iBAAiB,CAAC,MAAM,GAAG,CAAC,EAAE;gBAChC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,oBAAoB,eAAe,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,yBAAyB,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;gBAC9G,OAAO,EAAE,CAAC;aACX;SACF;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED;;;;;;OAMG;IACI,KAAK,CAAC,WAAW,CAAC,KAAgB,EAAE,OAAiB;QAC1D,MAAM,eAAe,GAAG,MAAM,IAAI,CAAC,QAAQ,EAAE,CAAC;QAE9C,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,eAAe,EAAE,KAAK,EAAE,OAAO,CAAC,CAAC;QAEnE,OAAO,QAAQ,CAAC;IAClB,CAAC;IAEO,KAAK,CAAC,QAAQ;QACpB,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,QAAQ,EAAE,CAAC;QACpC,MAAM,MAAM,GAAG,KAAK,CAAC,MAAM,CAAC,QAAQ,CAAC,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,UAAU,CAAC,QAAQ,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;QACjI,OAAO,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;IAC9B,CAAC;IAEO,OAAO,CAAC,KAAe;QAC7B,OAAO,KAAK,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC,IAAI,SAAS,CAAC,EAAE,aAAa,EAAE,IAAI,CAAC,SAAS,EAAE,QAAQ,EAAE,CAAC,CAAC,CAAC;IAC3F,CAAC;IAEO,KAAK,CAAC,QAAQ;QACpB,MAAM,GAAG,GAAG,IAAI,KAAK,EAAU,CAAC;QAEhC,KAAK,UAAU,OAAO,CAAC,GAAW;YAChC,MAAM,KAAK,GAAG,MAAM,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;YACpC,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE;gBACxB,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;gBACtC,MAAM,KAAK,GAAG,MAAM,EAAE,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;gBACtC,IAAI,KAAK,CAAC,MAAM,EAAE,EAAE;oBAAE,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;iBAAE;gBAC3C,IAAI,KAAK,CAAC,WAAW,EAAE,EAAE;oBAAE,MAAM,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;iBAAE;aACjE;QACH,CAAC;QAED,MAAM,OAAO,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAC9B,OAAO,GAAG,CAAC;IACb,CAAC;CACF;AA3FD,4CA2FC","sourcesContent":["import * as path from 'path';\nimport * as fs from 'fs-extra';\n\nconst CDK_OUTDIR_PREFIX = 'cdk-integ.out';\n\n/**\n * Represents a single integration test\n *\n * This type is a data-only structure, so it can trivially be passed to workers.\n * Derived attributes are calculated using the `IntegTest` class.\n */\nexport interface IntegTestInfo {\n  /**\n   * Path to the file to run\n   *\n   * Path is relative to the current working directory.\n   */\n  readonly fileName: string;\n\n  /**\n   * The root directory we discovered this test from\n   *\n   * Path is relative to the current working directory.\n   */\n  readonly discoveryRoot: string;\n}\n\n/**\n * Derived information for IntegTests\n */\nexport class IntegTest {\n  /**\n   * The name of the file to run\n   *\n   * Path is relative to the current working directory.\n   */\n  public readonly fileName: string;\n\n  /**\n   * Relative path to the file to run\n   *\n   * Relative from the \"discovery root\".\n   */\n  public readonly discoveryRelativeFileName: string;\n\n  /**\n   * The absolute path to the file\n   */\n  public readonly absoluteFileName: string;\n\n  /**\n   * The normalized name of the test. This name\n   * will be the same regardless of what directory the tool\n   * is run from.\n   */\n  public readonly normalizedTestName: string;\n\n  /**\n   * Directory the test is in\n   */\n  public readonly directory: string;\n\n  /**\n   * Display name for the test\n   *\n   * Depends on the discovery directory.\n   *\n   * Looks like `integ.mytest` or `package/test/integ.mytest`.\n   */\n  public readonly testName: string;\n\n  /**\n   * Path of the snapshot directory for this test\n   */\n  public readonly snapshotDir: string;\n\n  /**\n   * Path to the temporary output directory for this test\n   */\n  public readonly temporaryOutputDir: string;\n\n  constructor(public readonly info: IntegTestInfo) {\n    this.absoluteFileName = path.resolve(info.fileName);\n    this.fileName = path.relative(process.cwd(), info.fileName);\n\n    const parsed = path.parse(this.fileName);\n    this.discoveryRelativeFileName = path.relative(info.discoveryRoot, info.fileName);\n    this.directory = parsed.dir;\n\n    // if we are running in a package directory then just use the fileName\n    // as the testname, but if we are running in a parent directory with\n    // multiple packages then use the directory/filename as the testname\n    //\n    // Looks either like `integ.mytest` or `package/test/integ.mytest`.\n    const relDiscoveryRoot = path.relative(process.cwd(), info.discoveryRoot);\n    this.testName = this.directory === path.join(relDiscoveryRoot, 'test') || this.directory === path.join(relDiscoveryRoot)\n      ? parsed.name\n      : path.join(path.relative(this.info.discoveryRoot, parsed.dir), parsed.name);\n\n    const nakedTestName = parsed.name.slice(6); // Leave name without 'integ.' and '.ts'\n    this.normalizedTestName = parsed.name;\n    this.snapshotDir = path.join(this.directory, `${nakedTestName}.integ.snapshot`);\n    this.temporaryOutputDir = path.join(this.directory, `${CDK_OUTDIR_PREFIX}.${nakedTestName}`);\n  }\n\n  /**\n   * Whether this test matches the user-given name\n   *\n   * We are very lenient here. A name matches if it matches:\n   *\n   * - The CWD-relative filename\n   * - The discovery root-relative filename\n   * - The suite name\n   * - The absolute filename\n   */\n  public matches(name: string) {\n    return [\n      this.fileName,\n      this.discoveryRelativeFileName,\n      this.testName,\n      this.absoluteFileName,\n    ].includes(name);\n  }\n}\n\n/**\n * The list of tests to run can be provided in a file\n * instead of as command line arguments.\n */\nexport interface IntegrationTestFileConfig {\n  /**\n   * If this is set to true then the list of tests\n   * provided will be excluded\n   *\n   * @default false\n   */\n  readonly exclude?: boolean;\n\n  /**\n   * List of tests to include (or exclude if `exclude=true`)\n   */\n  readonly tests: string[];\n}\n\n/**\n * Discover integration tests\n */\nexport class IntegrationTests {\n  constructor(private readonly directory: string) {\n  }\n\n  /**\n   * Takes a file name of a file that contains a list of test\n   * to either run or exclude and returns a list of Integration Tests to run\n   */\n  public async fromFile(fileName: string): Promise<IntegTest[]> {\n    const file: IntegrationTestFileConfig = JSON.parse(fs.readFileSync(fileName, { encoding: 'utf-8' }));\n    const foundTests = await this.discover();\n\n    const allTests = this.filterTests(foundTests, file.tests, file.exclude);\n\n    return allTests;\n  }\n\n  /**\n   * If the user provides a list of tests, these can either be a list of tests to include or a list of tests to exclude.\n   *\n   * - If it is a list of tests to include then we discover all available tests and check whether they have provided valid tests.\n   *   If they have provided a test name that we don't find, then we write out that error message.\n   * - If it is a list of tests to exclude, then we discover all available tests and filter out the tests that were provided by the user.\n   */\n  private filterTests(discoveredTests: IntegTest[], requestedTests?: string[], exclude?: boolean): IntegTest[] {\n    if (!requestedTests) {\n      return discoveredTests;\n    }\n\n\n    const allTests = discoveredTests.filter(t => {\n      const matches = requestedTests.some(pattern => t.matches(pattern));\n      return matches !== !!exclude; // Looks weird but is equal to (matches && !exclude) || (!matches && exclude)\n    });\n\n    // If not excluding, all patterns must have matched at least one test\n    if (!exclude) {\n      const unmatchedPatterns = requestedTests.filter(pattern => !discoveredTests.some(t => t.matches(pattern)));\n      for (const unmatched of unmatchedPatterns) {\n        process.stderr.write(`No such integ test: ${unmatched}\\n`);\n      }\n      if (unmatchedPatterns.length > 0) {\n        process.stderr.write(`Available tests: ${discoveredTests.map(t => t.discoveryRelativeFileName).join(' ')}\\n`);\n        return [];\n      }\n    }\n\n    return allTests;\n  }\n\n  /**\n   * Takes an optional list of tests to look for, otherwise\n   * it will look for all tests from the directory\n   *\n   * @param tests Tests to include or exclude, undefined means include all tests.\n   * @param exclude Whether the 'tests' list is inclusive or exclusive (inclusive by default).\n   */\n  public async fromCliArgs(tests?: string[], exclude?: boolean): Promise<IntegTest[]> {\n    const discoveredTests = await this.discover();\n\n    const allTests = this.filterTests(discoveredTests, tests, exclude);\n\n    return allTests;\n  }\n\n  private async discover(): Promise<IntegTest[]> {\n    const files = await this.readTree();\n    const integs = files.filter(fileName => path.basename(fileName).startsWith('integ.') && path.basename(fileName).endsWith('.js'));\n    return this.request(integs);\n  }\n\n  private request(files: string[]): IntegTest[] {\n    return files.map(fileName => new IntegTest({ discoveryRoot: this.directory, fileName }));\n  }\n\n  private async readTree(): Promise<string[]> {\n    const ret = new Array<string>();\n\n    async function recurse(dir: string) {\n      const files = await fs.readdir(dir);\n      for (const file of files) {\n        const fullPath = path.join(dir, file);\n        const statf = await fs.stat(fullPath);\n        if (statf.isFile()) { ret.push(fullPath); }\n        if (statf.isDirectory()) { await recurse(path.join(fullPath)); }\n      }\n    }\n\n    await recurse(this.directory);\n    return ret;\n  }\n}\n"]}
@@ -38,6 +38,12 @@ export interface IntegRunnerOptions {
38
38
  * @default - CdkCliWrapper
39
39
  */
40
40
  readonly cdk?: ICdk;
41
+ /**
42
+ * Show output from running integration tests
43
+ *
44
+ * @default false
45
+ */
46
+ readonly showOutput?: boolean;
41
47
  }
42
48
  /**
43
49
  * The different components of a test name
@@ -99,6 +105,7 @@ export declare abstract class IntegRunner {
99
105
  protected readonly profile?: string;
100
106
  protected _destructiveChanges?: DestructiveChange[];
101
107
  private legacyContext?;
108
+ protected isLegacyTest?: boolean;
102
109
  constructor(options: IntegRunnerOptions);
103
110
  /**
104
111
  * Return the list of expected (i.e. existing) test cases for this integration test
@@ -33,6 +33,7 @@ class IntegRunner {
33
33
  this.cdkContextPath = path.join(this.directory, 'cdk.context.json');
34
34
  this.cdk = (_a = options.cdk) !== null && _a !== void 0 ? _a : new cdk_cli_wrapper_1.CdkCliWrapper({
35
35
  directory: this.directory,
36
+ showOutput: options.showOutput,
36
37
  env: {
37
38
  ...options.env,
38
39
  },
@@ -98,7 +99,7 @@ class IntegRunner {
98
99
  catch (e) {
99
100
  const testCases = integ_test_suite_1.LegacyIntegTestSuite.fromLegacy({
100
101
  cdk: this.cdk,
101
- testName: this.testName,
102
+ testName: this.test.normalizedTestName,
102
103
  integSourceFilePath: this.test.fileName,
103
104
  listOptions: {
104
105
  ...this.defaultArgs,
@@ -109,6 +110,7 @@ class IntegRunner {
109
110
  },
110
111
  });
111
112
  this.legacyContext = integ_test_suite_1.LegacyIntegTestSuite.getPragmaContext(this.test.fileName);
113
+ this.isLegacyTest = true;
112
114
  return testCases;
113
115
  }
114
116
  }
@@ -304,4 +306,4 @@ exports.DEFAULT_SYNTH_OPTIONS = {
304
306
  CDK_INTEG_REGION: 'test-region',
305
307
  },
306
308
  };
307
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runner-base.js","sourceRoot":"","sources":["runner-base.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { TestCase, DefaultCdkOptions } from '@aws-cdk/cloud-assembly-schema';\nimport { AVAILABILITY_ZONE_FALLBACK_CONTEXT_KEY, FUTURE_FLAGS, TARGET_PARTITIONS, FUTURE_FLAGS_EXPIRED, NEW_STYLE_STACK_SYNTHESIS_CONTEXT } from '@aws-cdk/cx-api';\nimport { CdkCliWrapper, ICdk } from 'cdk-cli-wrapper';\nimport * as fs from 'fs-extra';\nimport { flatten } from '../utils';\nimport { DestructiveChange } from '../workers/common';\nimport { IntegTestSuite, LegacyIntegTestSuite } from './integ-test-suite';\nimport { IntegTest } from './integration-tests';\nimport { AssemblyManifestReader, ManifestTrace } from './private/cloud-assembly';\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 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 CLI to use\n   *\n   * @default - CdkCliWrapper\n   */\n  readonly cdk?: ICdk;\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 test suite from the existing snapshot\n   */\n  protected readonly expectedTestSuite?: IntegTestSuite | LegacyIntegTestSuite;\n\n  /**\n   * The test suite from the new \"actual\" snapshot\n   */\n  protected readonly actualTestSuite: IntegTestSuite | LegacyIntegTestSuite;\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  protected readonly profile?: string;\n\n  protected _destructiveChanges?: DestructiveChange[];\n  private legacyContext?: Record<string, any>;\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\n    this.cdk = options.cdk ?? new CdkCliWrapper({\n      directory: this.directory,\n      env: {\n        ...options.env,\n      },\n    });\n    this.cdkOutDir = options.integOutDir ?? this.test.temporaryOutputDir;\n    this.cdkApp = `node ${path.relative(this.directory, this.test.fileName)}`;\n    this.profile = options.profile;\n    if (this.hasSnapshot()) {\n      this.expectedTestSuite = this.loadManifest();\n    }\n    this.actualTestSuite = this.generateActualSnapshot();\n  }\n\n  /**\n   * Return the list of expected (i.e. existing) test cases for this integration test\n   */\n  public expectedTests(): { [testName: string]: TestCase } | undefined {\n    return this.expectedTestSuite?.testSuite;\n  }\n\n  /**\n   * Return the list of actual (i.e. new) test cases for this integration test\n   */\n  public actualTests(): { [testName: string]: TestCase } | undefined {\n    return 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 generateActualSnapshot(): IntegTestSuite | LegacyIntegTestSuite {\n    this.cdk.synthFast({\n      execCmd: this.cdkApp.split(' '),\n      env: {\n        ...DEFAULT_SYNTH_OPTIONS.env,\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        CDK_CONTEXT_JSON: JSON.stringify(this.getContext(this.expectedTestSuite?.synthContext)),\n      },\n      output: path.relative(this.directory, this.cdkOutDir),\n    });\n    return this.loadManifest(this.cdkOutDir);\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   * 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 loadManifest(dir?: string): IntegTestSuite | LegacyIntegTestSuite {\n    try {\n      const testSuite = IntegTestSuite.fromPath(dir ?? this.snapshotDir);\n      return testSuite;\n    } catch (e) {\n      const testCases = LegacyIntegTestSuite.fromLegacy({\n        cdk: this.cdk,\n        testName: this.testName,\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 removeAssetsFromSnapshot(): void {\n    const stacks = this.actualTestSuite.getStacksWithoutUpdateWorkflow() ?? [];\n    const manifest = AssemblyManifestReader.fromPath(this.snapshotDir);\n    const assets = flatten(stacks.map(stack => {\n      return manifest.getAssetsForStack(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 synthing 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 createSnapshot(): void {\n    if (fs.existsSync(this.snapshotDir)) {\n      fs.removeSync(this.snapshotDir);\n    }\n\n    // if lookups are enabled then we need to synth again\n    // using dummy context and save that as the snapshot\n    if (this.actualTestSuite.enableLookups) {\n      this.cdk.synthFast({\n        execCmd: this.cdkApp.split(' '),\n        env: {\n          ...DEFAULT_SYNTH_OPTIONS.env,\n          CDK_CONTEXT_JSON: JSON.stringify(this.getContext()),\n        },\n        output: path.relative(this.directory, this.snapshotDir),\n      });\n    } else {\n      fs.moveSync(this.cdkOutDir, this.snapshotDir, { overwrite: true });\n    }\n\n    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 cleanupSnapshot(): void {\n    if (fs.existsSync(this.snapshotDir)) {\n      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    if (this.actualTestSuite.type === 'legacy-test-suite') {\n      (this.actualTestSuite as LegacyIntegTestSuite).saveManifest(this.snapshotDir, this.legacyContext);\n    }\n  }\n\n  protected getContext(additionalContext?: Record<string, any>): Record<string, any> {\n    const futureFlags: { [key: string]: any } = {};\n    Object.entries(FUTURE_FLAGS)\n      .filter(([k, _]) => !FUTURE_FLAGS_EXPIRED.includes(k))\n      .forEach(([k, v]) => futureFlags[k] = v);\n\n    const enableLookups = (this.actualTestSuite ?? this.expectedTestSuite)?.enableLookups;\n    return {\n      // if lookups are enabled then we need to synth\n      // with the \"dummy\" context\n      ...enableLookups ? DEFAULT_SYNTH_OPTIONS.context : {},\n      // This is needed so that there are no differences between\n      // running on v1 vs v2\n      [NEW_STYLE_STACK_SYNTHESIS_CONTEXT]: '',\n      ...futureFlags,\n      ...this.legacyContext,\n      ...additionalContext,\n    };\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 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    // 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  },\n  env: {\n    CDK_INTEG_ACCOUNT: '12345678',\n    CDK_INTEG_REGION: 'test-region',\n  },\n};\n"]}
309
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runner-base.js","sourceRoot":"","sources":["runner-base.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { TestCase, DefaultCdkOptions } from '@aws-cdk/cloud-assembly-schema';\nimport { AVAILABILITY_ZONE_FALLBACK_CONTEXT_KEY, FUTURE_FLAGS, TARGET_PARTITIONS, FUTURE_FLAGS_EXPIRED, NEW_STYLE_STACK_SYNTHESIS_CONTEXT } from '@aws-cdk/cx-api';\nimport { CdkCliWrapper, ICdk } from 'cdk-cli-wrapper';\nimport * as fs from 'fs-extra';\nimport { flatten } from '../utils';\nimport { DestructiveChange } from '../workers/common';\nimport { IntegTestSuite, LegacyIntegTestSuite } from './integ-test-suite';\nimport { IntegTest } from './integration-tests';\nimport { AssemblyManifestReader, ManifestTrace } from './private/cloud-assembly';\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 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 CLI to use\n   *\n   * @default - CdkCliWrapper\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 test suite from the existing snapshot\n   */\n  protected readonly expectedTestSuite?: IntegTestSuite | LegacyIntegTestSuite;\n\n  /**\n   * The test suite from the new \"actual\" snapshot\n   */\n  protected readonly actualTestSuite: IntegTestSuite | LegacyIntegTestSuite;\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  protected readonly profile?: string;\n\n  protected _destructiveChanges?: DestructiveChange[];\n  private legacyContext?: Record<string, any>;\n  protected isLegacyTest?: boolean;\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\n    this.cdk = options.cdk ?? new CdkCliWrapper({\n      directory: this.directory,\n      showOutput: options.showOutput,\n      env: {\n        ...options.env,\n      },\n    });\n    this.cdkOutDir = options.integOutDir ?? this.test.temporaryOutputDir;\n    this.cdkApp = `node ${path.relative(this.directory, this.test.fileName)}`;\n    this.profile = options.profile;\n    if (this.hasSnapshot()) {\n      this.expectedTestSuite = this.loadManifest();\n    }\n    this.actualTestSuite = this.generateActualSnapshot();\n  }\n\n  /**\n   * Return the list of expected (i.e. existing) test cases for this integration test\n   */\n  public expectedTests(): { [testName: string]: TestCase } | undefined {\n    return this.expectedTestSuite?.testSuite;\n  }\n\n  /**\n   * Return the list of actual (i.e. new) test cases for this integration test\n   */\n  public actualTests(): { [testName: string]: TestCase } | undefined {\n    return 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 generateActualSnapshot(): IntegTestSuite | LegacyIntegTestSuite {\n    this.cdk.synthFast({\n      execCmd: this.cdkApp.split(' '),\n      env: {\n        ...DEFAULT_SYNTH_OPTIONS.env,\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        CDK_CONTEXT_JSON: JSON.stringify(this.getContext(this.expectedTestSuite?.synthContext)),\n      },\n      output: path.relative(this.directory, this.cdkOutDir),\n    });\n    return this.loadManifest(this.cdkOutDir);\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   * 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 loadManifest(dir?: string): IntegTestSuite | LegacyIntegTestSuite {\n    try {\n      const testSuite = IntegTestSuite.fromPath(dir ?? this.snapshotDir);\n      return testSuite;\n    } catch (e) {\n      const testCases = 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      this.isLegacyTest = true;\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 removeAssetsFromSnapshot(): void {\n    const stacks = this.actualTestSuite.getStacksWithoutUpdateWorkflow() ?? [];\n    const manifest = AssemblyManifestReader.fromPath(this.snapshotDir);\n    const assets = flatten(stacks.map(stack => {\n      return manifest.getAssetsForStack(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 synthing 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 createSnapshot(): void {\n    if (fs.existsSync(this.snapshotDir)) {\n      fs.removeSync(this.snapshotDir);\n    }\n\n    // if lookups are enabled then we need to synth again\n    // using dummy context and save that as the snapshot\n    if (this.actualTestSuite.enableLookups) {\n      this.cdk.synthFast({\n        execCmd: this.cdkApp.split(' '),\n        env: {\n          ...DEFAULT_SYNTH_OPTIONS.env,\n          CDK_CONTEXT_JSON: JSON.stringify(this.getContext()),\n        },\n        output: path.relative(this.directory, this.snapshotDir),\n      });\n    } else {\n      fs.moveSync(this.cdkOutDir, this.snapshotDir, { overwrite: true });\n    }\n\n    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 cleanupSnapshot(): void {\n    if (fs.existsSync(this.snapshotDir)) {\n      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    if (this.actualTestSuite.type === 'legacy-test-suite') {\n      (this.actualTestSuite as LegacyIntegTestSuite).saveManifest(this.snapshotDir, this.legacyContext);\n    }\n  }\n\n  protected getContext(additionalContext?: Record<string, any>): Record<string, any> {\n    const futureFlags: { [key: string]: any } = {};\n    Object.entries(FUTURE_FLAGS)\n      .filter(([k, _]) => !FUTURE_FLAGS_EXPIRED.includes(k))\n      .forEach(([k, v]) => futureFlags[k] = v);\n\n    const enableLookups = (this.actualTestSuite ?? this.expectedTestSuite)?.enableLookups;\n    return {\n      // if lookups are enabled then we need to synth\n      // with the \"dummy\" context\n      ...enableLookups ? DEFAULT_SYNTH_OPTIONS.context : {},\n      // This is needed so that there are no differences between\n      // running on v1 vs v2\n      [NEW_STYLE_STACK_SYNTHESIS_CONTEXT]: '',\n      ...futureFlags,\n      ...this.legacyContext,\n      ...additionalContext,\n    };\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 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    // 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  },\n  env: {\n    CDK_INTEG_ACCOUNT: '12345678',\n    CDK_INTEG_REGION: 'test-region',\n  },\n};\n"]}
@@ -228,4 +228,4 @@ class StringWritable extends stream_1.Writable {
228
228
  callback();
229
229
  }
230
230
  }
231
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snapshot-test-runner.js","sourceRoot":"","sources":["snapshot-test-runner.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { Writable, WritableOptions } from 'stream';\nimport { StringDecoder, NodeStringDecoder } from 'string_decoder';\nimport { diffTemplate, formatDifferences, ResourceDifference, ResourceImpact } from '@aws-cdk/cloudformation-diff';\nimport { Diagnostic, DiagnosticReason, DestructiveChange, SnapshotVerificationOptions } from '../workers/common';\nimport { canonicalizeTemplate } from './private/canonicalize-assets';\nimport { AssemblyManifestReader } from './private/cloud-assembly';\nimport { IntegRunnerOptions, IntegRunner, DEFAULT_SYNTH_OPTIONS } from './runner-base';\n\n/**\n * Runner for snapshot tests. This handles orchestrating\n * the validation of the integration test snapshots\n */\nexport class IntegSnapshotRunner extends IntegRunner {\n  constructor(options: IntegRunnerOptions) {\n    super(options);\n  }\n\n  /**\n   * Synth the integration tests and compare the templates\n   * to the existing snapshot.\n   *\n   * @returns any diagnostics and any destructive changes\n   */\n  public testSnapshot(options: SnapshotVerificationOptions = {}): { diagnostics: Diagnostic[], destructiveChanges: DestructiveChange[] } {\n    let doClean = true;\n    try {\n      // read the existing snapshot\n      const expectedStacks = this.readAssembly(this.snapshotDir);\n      // only diff stacks that are part of the test case\n      const expectedStacksToDiff: Record<string, any> = {};\n      for (const [stackName, template] of Object.entries(expectedStacks)) {\n        if (this.expectedTestSuite?.stacks.includes(stackName)) {\n          expectedStacksToDiff[stackName] = template;\n        }\n      }\n\n      // synth the integration test\n      // FIXME: ideally we should not need to run this again if\n      // the cdkOutDir exists already, but for some reason generateActualSnapshot\n      // generates an incorrect snapshot and I have no idea why so synth again here\n      // to produce the \"correct\" snapshot\n      const env = {\n        ...DEFAULT_SYNTH_OPTIONS.env,\n        CDK_CONTEXT_JSON: JSON.stringify(this.getContext()),\n      };\n      this.cdk.synthFast({\n        execCmd: this.cdkApp.split(' '),\n        env,\n        output: path.relative(this.directory, this.cdkOutDir),\n      });\n\n      // read the \"actual\" snapshot\n      const actualDir = this.cdkOutDir;\n      const actualStacks = this.readAssembly(actualDir);\n      // only diff stacks that are part of the test case\n      const actualStacksToDiff: Record<string, any> = {};\n      for (const [stackName, template] of Object.entries(actualStacks)) {\n        if (this.actualTestSuite.stacks.includes(stackName)) {\n          actualStacksToDiff[stackName] = template;\n        }\n      }\n\n      // diff the existing snapshot (expected) with the integration test (actual)\n      const diagnostics = this.diffAssembly(expectedStacksToDiff, actualStacksToDiff);\n\n      if (diagnostics.diagnostics.length) {\n        // Attach additional messages to the first diagnostic\n        const additionalMessages: string[] = [];\n\n        if (options.retain) {\n          additionalMessages.push(\n            `(Failure retained) Expected: ${path.relative(process.cwd(), this.snapshotDir)}`,\n            `                   Actual:   ${path.relative(process.cwd(), actualDir)}`,\n          ),\n          doClean = false;\n        }\n\n        if (options.verbose) {\n          // Show the command necessary to repro this\n          const envSet = Object.entries(env)\n            .filter(([k, _]) => k !== 'CDK_CONTEXT_JSON')\n            .map(([k, v]) => `${k}='${v}'`);\n          const envCmd = envSet.length > 0 ? ['env', ...envSet] : [];\n\n          additionalMessages.push(\n            'Repro:',\n            `  ${[...envCmd, 'cdk synth', `-a '${this.cdkApp}'`, `-o '${this.cdkOutDir}'`, ...Object.entries(this.getContext()).flatMap(([k, v]) => typeof v !== 'object' ? [`-c '${k}=${v}'`] : [])].join(' ')}`,\n          );\n        }\n\n        diagnostics.diagnostics[0] = {\n          ...diagnostics.diagnostics[0],\n          additionalMessages,\n        };\n      }\n\n      return diagnostics;\n    } catch (e) {\n      throw e;\n    } finally {\n      if (doClean) {\n        this.cleanup();\n      }\n    }\n  }\n\n  /**\n   * For a given stack return all resource types that are allowed to be destroyed\n   * as part of a stack update\n   *\n   * @param stackId the stack id\n   * @returns a list of resource types or undefined if none are found\n   */\n  private getAllowedDestroyTypesForStack(stackId: string): string[] | undefined {\n    for (const testCase of Object.values(this.actualTests() ?? {})) {\n      if (testCase.stacks.includes(stackId)) {\n        return testCase.allowDestroy;\n      }\n    }\n    return undefined;\n  }\n\n  /**\n   * Find any differences between the existing and expected snapshots\n   *\n   * @param existing - the existing (expected) snapshot\n   * @param actual - the new (actual) snapshot\n   * @returns any diagnostics and any destructive changes\n   */\n  private diffAssembly(\n    expected: Record<string, any>,\n    actual: Record<string, any>,\n  ): { diagnostics: Diagnostic[], destructiveChanges: DestructiveChange[] } {\n    const failures: Diagnostic[] = [];\n    const destructiveChanges: DestructiveChange[] = [];\n\n    // check if there is a CFN template in the current snapshot\n    // that does not exist in the \"actual\" snapshot\n    for (const templateId of Object.keys(expected)) {\n      if (!actual.hasOwnProperty(templateId)) {\n        failures.push({\n          testName: this.testName,\n          reason: DiagnosticReason.SNAPSHOT_FAILED,\n          message: `${templateId} exists in snapshot, but not in actual`,\n        });\n      }\n    }\n\n    for (const templateId of Object.keys(actual)) {\n      // check if there is a CFN template in the \"actual\" snapshot\n      // that does not exist in the current snapshot\n      if (!expected.hasOwnProperty(templateId)) {\n        failures.push({\n          testName: this.testName,\n          reason: DiagnosticReason.SNAPSHOT_FAILED,\n          message: `${templateId} does not exist in snapshot, but does in actual`,\n        });\n        continue;\n      } else {\n        let actualTemplate = actual[templateId];\n        let expectedTemplate = expected[templateId];\n        const allowedDestroyTypes = this.getAllowedDestroyTypesForStack(templateId) ?? [];\n\n        // if we are not verifying asset hashes then remove the specific\n        // asset hashes from the templates so they are not part of the diff\n        // comparison\n        if (!this.actualTestSuite.getOptionsForStack(templateId)?.diffAssets) {\n          actualTemplate = canonicalizeTemplate(actualTemplate);\n          expectedTemplate = canonicalizeTemplate(expectedTemplate);\n        }\n        const templateDiff = diffTemplate(expectedTemplate, actualTemplate);\n        if (!templateDiff.isEmpty) {\n          // go through all the resource differences and check for any\n          // \"destructive\" changes\n          templateDiff.resources.forEachDifference((logicalId: string, change: ResourceDifference) => {\n            // if the change is a removal it will not show up as a 'changeImpact'\n            // so need to check for it separately, unless it is a resourceType that\n            // has been \"allowed\" to be destroyed\n            const resourceType = change.oldValue?.Type ?? change.newValue?.Type;\n            if (resourceType && allowedDestroyTypes.includes(resourceType)) {\n              return;\n            }\n            if (change.isRemoval) {\n              destructiveChanges.push({\n                impact: ResourceImpact.WILL_DESTROY,\n                logicalId,\n                stackName: templateId,\n              });\n            } else {\n              switch (change.changeImpact) {\n                case ResourceImpact.MAY_REPLACE:\n                case ResourceImpact.WILL_ORPHAN:\n                case ResourceImpact.WILL_DESTROY:\n                case ResourceImpact.WILL_REPLACE:\n                  destructiveChanges.push({\n                    impact: change.changeImpact,\n                    logicalId,\n                    stackName: templateId,\n                  });\n                  break;\n              }\n            }\n          });\n          const writable = new StringWritable({});\n          formatDifferences(writable, templateDiff);\n          failures.push({\n            reason: DiagnosticReason.SNAPSHOT_FAILED,\n            message: writable.data,\n            testName: this.testName,\n          });\n        }\n      }\n    }\n\n    return {\n      diagnostics: failures,\n      destructiveChanges,\n    };\n  }\n\n  private readAssembly(dir: string): Record<string, any> {\n    const assembly = AssemblyManifestReader.fromPath(dir);\n    const stacks = assembly.stacks;\n\n    return stacks;\n  }\n}\n\nclass StringWritable extends Writable {\n  public data: string;\n  private _decoder: NodeStringDecoder;\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"]}
231
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"snapshot-test-runner.js","sourceRoot":"","sources":["snapshot-test-runner.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { Writable, WritableOptions } from 'stream';\nimport { StringDecoder } from 'string_decoder';\nimport { diffTemplate, formatDifferences, ResourceDifference, ResourceImpact } from '@aws-cdk/cloudformation-diff';\nimport { Diagnostic, DiagnosticReason, DestructiveChange, SnapshotVerificationOptions } from '../workers/common';\nimport { canonicalizeTemplate } from './private/canonicalize-assets';\nimport { AssemblyManifestReader } from './private/cloud-assembly';\nimport { IntegRunnerOptions, IntegRunner, DEFAULT_SYNTH_OPTIONS } from './runner-base';\n\n/**\n * Runner for snapshot tests. This handles orchestrating\n * the validation of the integration test snapshots\n */\nexport class IntegSnapshotRunner extends IntegRunner {\n  constructor(options: IntegRunnerOptions) {\n    super(options);\n  }\n\n  /**\n   * Synth the integration tests and compare the templates\n   * to the existing snapshot.\n   *\n   * @returns any diagnostics and any destructive changes\n   */\n  public testSnapshot(options: SnapshotVerificationOptions = {}): { diagnostics: Diagnostic[], destructiveChanges: DestructiveChange[] } {\n    let doClean = true;\n    try {\n      // read the existing snapshot\n      const expectedStacks = this.readAssembly(this.snapshotDir);\n      // only diff stacks that are part of the test case\n      const expectedStacksToDiff: Record<string, any> = {};\n      for (const [stackName, template] of Object.entries(expectedStacks)) {\n        if (this.expectedTestSuite?.stacks.includes(stackName)) {\n          expectedStacksToDiff[stackName] = template;\n        }\n      }\n\n      // synth the integration test\n      // FIXME: ideally we should not need to run this again if\n      // the cdkOutDir exists already, but for some reason generateActualSnapshot\n      // generates an incorrect snapshot and I have no idea why so synth again here\n      // to produce the \"correct\" snapshot\n      const env = {\n        ...DEFAULT_SYNTH_OPTIONS.env,\n        CDK_CONTEXT_JSON: JSON.stringify(this.getContext()),\n      };\n      this.cdk.synthFast({\n        execCmd: this.cdkApp.split(' '),\n        env,\n        output: path.relative(this.directory, this.cdkOutDir),\n      });\n\n      // read the \"actual\" snapshot\n      const actualDir = this.cdkOutDir;\n      const actualStacks = this.readAssembly(actualDir);\n      // only diff stacks that are part of the test case\n      const actualStacksToDiff: Record<string, any> = {};\n      for (const [stackName, template] of Object.entries(actualStacks)) {\n        if (this.actualTestSuite.stacks.includes(stackName)) {\n          actualStacksToDiff[stackName] = template;\n        }\n      }\n\n      // diff the existing snapshot (expected) with the integration test (actual)\n      const diagnostics = this.diffAssembly(expectedStacksToDiff, actualStacksToDiff);\n\n      if (diagnostics.diagnostics.length) {\n        // Attach additional messages to the first diagnostic\n        const additionalMessages: string[] = [];\n\n        if (options.retain) {\n          additionalMessages.push(\n            `(Failure retained) Expected: ${path.relative(process.cwd(), this.snapshotDir)}`,\n            `                   Actual:   ${path.relative(process.cwd(), actualDir)}`,\n          ),\n          doClean = false;\n        }\n\n        if (options.verbose) {\n          // Show the command necessary to repro this\n          const envSet = Object.entries(env)\n            .filter(([k, _]) => k !== 'CDK_CONTEXT_JSON')\n            .map(([k, v]) => `${k}='${v}'`);\n          const envCmd = envSet.length > 0 ? ['env', ...envSet] : [];\n\n          additionalMessages.push(\n            'Repro:',\n            `  ${[...envCmd, 'cdk synth', `-a '${this.cdkApp}'`, `-o '${this.cdkOutDir}'`, ...Object.entries(this.getContext()).flatMap(([k, v]) => typeof v !== 'object' ? [`-c '${k}=${v}'`] : [])].join(' ')}`,\n          );\n        }\n\n        diagnostics.diagnostics[0] = {\n          ...diagnostics.diagnostics[0],\n          additionalMessages,\n        };\n      }\n\n      return diagnostics;\n    } catch (e) {\n      throw e;\n    } finally {\n      if (doClean) {\n        this.cleanup();\n      }\n    }\n  }\n\n  /**\n   * For a given stack return all resource types that are allowed to be destroyed\n   * as part of a stack update\n   *\n   * @param stackId the stack id\n   * @returns a list of resource types or undefined if none are found\n   */\n  private getAllowedDestroyTypesForStack(stackId: string): string[] | undefined {\n    for (const testCase of Object.values(this.actualTests() ?? {})) {\n      if (testCase.stacks.includes(stackId)) {\n        return testCase.allowDestroy;\n      }\n    }\n    return undefined;\n  }\n\n  /**\n   * Find any differences between the existing and expected snapshots\n   *\n   * @param existing - the existing (expected) snapshot\n   * @param actual - the new (actual) snapshot\n   * @returns any diagnostics and any destructive changes\n   */\n  private diffAssembly(\n    expected: Record<string, any>,\n    actual: Record<string, any>,\n  ): { diagnostics: Diagnostic[], destructiveChanges: DestructiveChange[] } {\n    const failures: Diagnostic[] = [];\n    const destructiveChanges: DestructiveChange[] = [];\n\n    // check if there is a CFN template in the current snapshot\n    // that does not exist in the \"actual\" snapshot\n    for (const templateId of Object.keys(expected)) {\n      if (!actual.hasOwnProperty(templateId)) {\n        failures.push({\n          testName: this.testName,\n          reason: DiagnosticReason.SNAPSHOT_FAILED,\n          message: `${templateId} exists in snapshot, but not in actual`,\n        });\n      }\n    }\n\n    for (const templateId of Object.keys(actual)) {\n      // check if there is a CFN template in the \"actual\" snapshot\n      // that does not exist in the current snapshot\n      if (!expected.hasOwnProperty(templateId)) {\n        failures.push({\n          testName: this.testName,\n          reason: DiagnosticReason.SNAPSHOT_FAILED,\n          message: `${templateId} does not exist in snapshot, but does in actual`,\n        });\n        continue;\n      } else {\n        let actualTemplate = actual[templateId];\n        let expectedTemplate = expected[templateId];\n        const allowedDestroyTypes = this.getAllowedDestroyTypesForStack(templateId) ?? [];\n\n        // if we are not verifying asset hashes then remove the specific\n        // asset hashes from the templates so they are not part of the diff\n        // comparison\n        if (!this.actualTestSuite.getOptionsForStack(templateId)?.diffAssets) {\n          actualTemplate = canonicalizeTemplate(actualTemplate);\n          expectedTemplate = canonicalizeTemplate(expectedTemplate);\n        }\n        const templateDiff = diffTemplate(expectedTemplate, actualTemplate);\n        if (!templateDiff.isEmpty) {\n          // go through all the resource differences and check for any\n          // \"destructive\" changes\n          templateDiff.resources.forEachDifference((logicalId: string, change: ResourceDifference) => {\n            // if the change is a removal it will not show up as a 'changeImpact'\n            // so need to check for it separately, unless it is a resourceType that\n            // has been \"allowed\" to be destroyed\n            const resourceType = change.oldValue?.Type ?? change.newValue?.Type;\n            if (resourceType && allowedDestroyTypes.includes(resourceType)) {\n              return;\n            }\n            if (change.isRemoval) {\n              destructiveChanges.push({\n                impact: ResourceImpact.WILL_DESTROY,\n                logicalId,\n                stackName: templateId,\n              });\n            } else {\n              switch (change.changeImpact) {\n                case ResourceImpact.MAY_REPLACE:\n                case ResourceImpact.WILL_ORPHAN:\n                case ResourceImpact.WILL_DESTROY:\n                case ResourceImpact.WILL_REPLACE:\n                  destructiveChanges.push({\n                    impact: change.changeImpact,\n                    logicalId,\n                    stackName: templateId,\n                  });\n                  break;\n              }\n            }\n          });\n          const writable = new StringWritable({});\n          formatDifferences(writable, templateDiff);\n          failures.push({\n            reason: DiagnosticReason.SNAPSHOT_FAILED,\n            message: writable.data,\n            testName: this.testName,\n          });\n        }\n      }\n    }\n\n    return {\n      diagnostics: failures,\n      destructiveChanges,\n    };\n  }\n\n  private readAssembly(dir: string): Record<string, any> {\n    const assembly = AssemblyManifestReader.fromPath(dir);\n    const stacks = assembly.stacks;\n\n    return stacks;\n  }\n}\n\nclass StringWritable extends Writable {\n  public data: string;\n  private _decoder: StringDecoder;\n  constructor(options: WritableOptions) {\n    super(options);\n    this._decoder = new StringDecoder();\n    this.data = '';\n  }\n\n  _write(chunk: any, encoding: string, callback: (error?: Error | null) => void): void {\n    if (encoding === 'buffer') {\n      chunk = this._decoder.write(chunk);\n    }\n\n    this.data += chunk;\n    callback();\n  }\n\n  _final(callback: (error?: Error | null) => void): void {\n    this.data += this._decoder.end();\n    callback();\n  }\n}\n"]}
@@ -130,11 +130,12 @@ export interface IntegTestOptions {
130
130
  */
131
131
  readonly dryRun?: boolean;
132
132
  /**
133
- * Whether to enable verbose logging
133
+ * The level of verbosity for logging.
134
+ * Higher number means more output.
134
135
  *
135
- * @default false
136
+ * @default 0
136
137
  */
137
- readonly verbose?: boolean;
138
+ readonly verbosity?: number;
138
139
  /**
139
140
  * If this is set to true then the stack update workflow will be disabled
140
141
  *