@aws-cdk/integ-runner 2.39.1 → 2.40.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -121,9 +121,9 @@ class IntegTestRunner extends runner_base_1.IntegRunner {
121
121
  output: path.relative(this.directory, this.cdkOutDir),
122
122
  });
123
123
  }
124
- // only create the snapshot if there are no assertion assertion results
124
+ // only create the snapshot if there are no failed assertion results
125
125
  // (i.e. no failures)
126
- if (!assertionResults) {
126
+ if (!assertionResults || !Object.values(assertionResults).some(result => result.status === 'fail')) {
127
127
  this.createSnapshot();
128
128
  }
129
129
  }
@@ -230,8 +230,8 @@ class IntegTestRunner extends runner_base_1.IntegRunner {
230
230
  cwd: path.dirname(this.snapshotDir),
231
231
  });
232
232
  }
233
- if (actualTestCase.assertionStack) {
234
- return this.processAssertionResults(path.join(this.cdkOutDir, 'assertion-results.json'), actualTestCase.assertionStack);
233
+ if (actualTestCase.assertionStack && actualTestCase.assertionStackName) {
234
+ return this.processAssertionResults(path.join(this.cdkOutDir, 'assertion-results.json'), actualTestCase.assertionStackName, actualTestCase.assertionStack);
235
235
  }
236
236
  }
237
237
  catch (e) {
@@ -243,16 +243,16 @@ class IntegTestRunner extends runner_base_1.IntegRunner {
243
243
  * Process the outputsFile which contains the assertions results as stack
244
244
  * outputs
245
245
  */
246
- processAssertionResults(file, assertionStackId) {
246
+ processAssertionResults(file, assertionStackName, assertionStackId) {
247
247
  const results = {};
248
248
  if (fs.existsSync(file)) {
249
249
  try {
250
250
  const outputs = fs.readJSONSync(file);
251
- if (assertionStackId in outputs) {
252
- for (const [assertionId, result] of Object.entries(outputs[assertionStackId])) {
251
+ if (assertionStackName in outputs) {
252
+ for (const [assertionId, result] of Object.entries(outputs[assertionStackName])) {
253
253
  if (assertionId.startsWith('AssertionResults')) {
254
254
  const assertionResult = JSON.parse(result.replace(/\n/g, '\\n'));
255
- if (assertionResult.status === 'fail') {
255
+ if (assertionResult.status === 'fail' || assertionResult.status === 'success') {
256
256
  results[assertionId] = assertionResult;
257
257
  }
258
258
  }
@@ -294,4 +294,4 @@ class IntegTestRunner extends runner_base_1.IntegRunner {
294
294
  }
295
295
  }
296
296
  exports.IntegTestRunner = IntegTestRunner;
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"]}
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 failed assertion results\n      // (i.e. no failures)\n      if (!assertionResults || !Object.values(assertionResults).some(result => result.status === 'fail')) {\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 && actualTestCase.assertionStackName) {\n        return this.processAssertionResults(\n          path.join(this.cdkOutDir, 'assertion-results.json'),\n          actualTestCase.assertionStackName,\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, assertionStackName: 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 (assertionStackName in outputs) {\n          for (const [assertionId, result] of Object.entries(outputs[assertionStackName])) {\n            if (assertionId.startsWith('AssertionResults')) {\n              const assertionResult: AssertionResult = JSON.parse(result.replace(/\\n/g, '\\\\n'));\n              if (assertionResult.status === 'fail' || assertionResult.status === 'success') {\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"]}
@@ -59,8 +59,8 @@ exports.printSummary = printSummary;
59
59
  */
60
60
  function formatAssertionResults(results) {
61
61
  return Object.entries(results)
62
- .map(([id, result]) => util_1.format('%s\n%s', id, result.message))
63
- .join('\n');
62
+ .map(([id, result]) => util_1.format('%s%s', id, result.status === 'success' ? ` - ${result.status}` : `\n${result.message}`))
63
+ .join('\n ');
64
64
  }
65
65
  exports.formatAssertionResults = formatAssertionResults;
66
66
  /**
@@ -73,7 +73,7 @@ function printResults(diagnostic) {
73
73
  logger.success(' UNCHANGED %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));
74
74
  break;
75
75
  case DiagnosticReason.TEST_SUCCESS:
76
- logger.success(' SUCCESS %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));
76
+ logger.success(' SUCCESS %s %s\n ', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);
77
77
  break;
78
78
  case DiagnosticReason.NO_SNAPSHOT:
79
79
  logger.error(' NEW %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));
@@ -106,4 +106,4 @@ function printLaggards(testNames) {
106
106
  logger.print(chalk.grey(parts.filter(x => x).join(' ')));
107
107
  }
108
108
  exports.printLaggards = printLaggards;
109
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["common.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'util';\nimport { ResourceImpact } from '@aws-cdk/cloudformation-diff';\nimport * as chalk from 'chalk';\nimport * as logger from '../logger';\nimport { IntegTestInfo } from '../runner/integration-tests';\n\n/**\n * The aggregate results from running assertions on a test case\n */\nexport type AssertionResults = { [id: string]: AssertionResult };\n\n/**\n * The result of an individual assertion\n */\nexport interface AssertionResult {\n  /**\n   * The assertion message. If the assertion failed, this will\n   * include the reason.\n   */\n  readonly message: string;\n\n  /**\n   * Whether the assertion succeeded or failed\n   */\n  readonly status: 'success' | 'fail';\n}\n\n/**\n * Config for an integration test\n */\nexport interface IntegTestWorkerConfig extends IntegTestInfo {\n  /**\n   * A list of any destructive changes\n   *\n   * @default []\n   */\n  readonly destructiveChanges?: DestructiveChange[];\n}\n\n/**\n * Information on any destructive changes\n */\nexport interface DestructiveChange {\n  /**\n   * The logicalId of the resource with a destructive change\n   */\n  readonly logicalId: string;\n\n  /**\n   * The name of the stack that contains the destructive change\n   */\n  readonly stackName: string;\n\n  /**\n   * The impact of the destructive change\n   */\n  readonly impact: ResourceImpact;\n}\n\n\n/**\n * Represents integration tests metrics for a given worker\n */\nexport interface IntegRunnerMetrics {\n  /**\n   * The region the test was run in\n   */\n  readonly region: string;\n\n  /**\n   * The total duration of the worker.\n   * This will be the sum of all individual test durations\n   */\n  readonly duration: number;\n\n  /**\n   * Contains the duration of individual tests that the\n   * worker executed.\n   *\n   * Map of testName to duration.\n   */\n  readonly tests: { [testName: string]: number };\n\n  /**\n   * The profile that was used to run the test\n   *\n   * @default - default profile\n   */\n  readonly profile?: string;\n}\n\nexport interface SnapshotVerificationOptions {\n  /**\n   * Retain failed snapshot comparisons\n   *\n   * @default false\n   */\n  readonly retain?: boolean;\n\n  /**\n   * Verbose mode\n   *\n   * @default false\n   */\n  readonly verbose?: boolean;\n}\n\n/**\n * Integration test results\n */\nexport interface IntegBatchResponse {\n  /**\n   * List of failed tests\n   */\n  readonly failedTests: IntegTestInfo[];\n\n  /**\n   * List of Integration test metrics. Each entry in the\n   * list represents metrics from a single worker (account + region).\n   */\n  readonly metrics: IntegRunnerMetrics[];\n}\n\n/**\n * Common options for running integration tests\n */\nexport interface IntegTestOptions {\n  /**\n   * A list of integration tests to run\n   * in this batch\n   */\n  readonly tests: IntegTestWorkerConfig[];\n\n  /**\n   * Whether or not to destroy the stacks at the\n   * end of the test\n   *\n   * @default true\n   */\n  readonly clean?: boolean;\n\n  /**\n   * When this is set to `true` the snapshot will\n   * be created _without_ running the integration test\n   * The resulting snapshot SHOULD NOT be checked in\n   *\n   * @default false\n   */\n  readonly dryRun?: boolean;\n\n  /**\n   * The level of verbosity for logging.\n   * Higher number means more output.\n   *\n   * @default 0\n   */\n  readonly verbosity?: number;\n\n  /**\n   * If this is set to true then the stack update workflow will be disabled\n   *\n   * @default true\n   */\n  readonly updateWorkflow?: boolean;\n}\n\n/**\n * Represents possible reasons for a diagnostic\n */\nexport enum DiagnosticReason {\n  /**\n   * The integration test failed because there\n   * is not existing snapshot\n   */\n  NO_SNAPSHOT = 'NO_SNAPSHOT',\n\n  /**\n   * The integration test failed\n   */\n  TEST_FAILED = 'TEST_FAILED',\n\n  /**\n   * There was an error running the integration test\n   */\n  TEST_ERROR = 'TEST_ERROR',\n\n  /**\n   * The snapshot test failed because the actual\n   * snapshot was different than the expected snapshot\n   */\n  SNAPSHOT_FAILED = 'SNAPSHOT_FAILED',\n\n  /**\n   * The snapshot test failed because there was an error executing it\n   */\n  SNAPSHOT_ERROR = 'SNAPSHOT_ERROR',\n\n  /**\n   * The snapshot test succeeded\n   */\n  SNAPSHOT_SUCCESS = 'SNAPSHOT_SUCCESS',\n\n  /**\n   * The integration test succeeded\n   */\n  TEST_SUCCESS = 'TEST_SUCCESS',\n\n  /**\n   * The assertion failed\n   */\n  ASSERTION_FAILED = 'ASSERTION_FAILED',\n}\n\n/**\n * Integration test diagnostics\n * This is used to report back the status of each test\n */\nexport interface Diagnostic {\n  /**\n   * The name of the test\n   */\n  readonly testName: string;\n\n  /**\n   * The diagnostic message\n   */\n  readonly message: string;\n\n  /**\n   * The time it took to run the test\n   */\n  readonly duration?: number;\n\n  /**\n   * The reason for the diagnostic\n   */\n  readonly reason: DiagnosticReason;\n\n  /**\n   * Additional messages to print\n   */\n  readonly additionalMessages?: string[];\n}\n\nexport function printSummary(total: number, failed: number): void {\n  if (failed > 0) {\n    logger.print('%s:    %s %s, %s total', chalk.bold('Tests'), chalk.red(failed), chalk.red('failed'), total);\n  } else {\n    logger.print('%s:    %s %s, %s total', chalk.bold('Tests'), chalk.green(total), chalk.green('passed'), total);\n  }\n}\n\n/**\n * Format the assertion results so that the results can be\n * printed\n */\nexport function formatAssertionResults(results: AssertionResults): string {\n  return Object.entries(results)\n    .map(([id, result]) => format('%s\\n%s', id, result.message))\n    .join('\\n');\n}\n\n/**\n * Print out the results from tests\n */\nexport function printResults(diagnostic: Diagnostic): void {\n  switch (diagnostic.reason) {\n    case DiagnosticReason.SNAPSHOT_SUCCESS:\n      logger.success('  UNCHANGED  %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));\n      break;\n    case DiagnosticReason.TEST_SUCCESS:\n      logger.success('  SUCCESS    %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));\n      break;\n    case DiagnosticReason.NO_SNAPSHOT:\n      logger.error('  NEW        %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));\n      break;\n    case DiagnosticReason.SNAPSHOT_FAILED:\n      logger.error('  CHANGED    %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.SNAPSHOT_ERROR:\n    case DiagnosticReason.TEST_ERROR:\n      logger.error('  ERROR      %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.TEST_FAILED:\n      logger.error('  FAILED     %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.ASSERTION_FAILED:\n      logger.error('  ASSERT     %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n  }\n  for (const addl of diagnostic.additionalMessages ?? []) {\n    logger.print(`      ${addl}`);\n  }\n}\n\nexport function printLaggards(testNames: Set<string>) {\n  const parts = [\n    '  ',\n    `Waiting for ${testNames.size} more`,\n    testNames.size < 10 ? ['(', Array.from(testNames).join(', '), ')'].join('') : '',\n  ];\n\n  logger.print(chalk.grey(parts.filter(x => x).join(' ')));\n}"]}
109
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["common.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'util';\nimport { ResourceImpact } from '@aws-cdk/cloudformation-diff';\nimport * as chalk from 'chalk';\nimport * as logger from '../logger';\nimport { IntegTestInfo } from '../runner/integration-tests';\n\n/**\n * The aggregate results from running assertions on a test case\n */\nexport type AssertionResults = { [id: string]: AssertionResult };\n\n/**\n * The result of an individual assertion\n */\nexport interface AssertionResult {\n  /**\n   * The assertion message. If the assertion failed, this will\n   * include the reason.\n   */\n  readonly message: string;\n\n  /**\n   * Whether the assertion succeeded or failed\n   */\n  readonly status: 'success' | 'fail';\n}\n\n/**\n * Config for an integration test\n */\nexport interface IntegTestWorkerConfig extends IntegTestInfo {\n  /**\n   * A list of any destructive changes\n   *\n   * @default []\n   */\n  readonly destructiveChanges?: DestructiveChange[];\n}\n\n/**\n * Information on any destructive changes\n */\nexport interface DestructiveChange {\n  /**\n   * The logicalId of the resource with a destructive change\n   */\n  readonly logicalId: string;\n\n  /**\n   * The name of the stack that contains the destructive change\n   */\n  readonly stackName: string;\n\n  /**\n   * The impact of the destructive change\n   */\n  readonly impact: ResourceImpact;\n}\n\n\n/**\n * Represents integration tests metrics for a given worker\n */\nexport interface IntegRunnerMetrics {\n  /**\n   * The region the test was run in\n   */\n  readonly region: string;\n\n  /**\n   * The total duration of the worker.\n   * This will be the sum of all individual test durations\n   */\n  readonly duration: number;\n\n  /**\n   * Contains the duration of individual tests that the\n   * worker executed.\n   *\n   * Map of testName to duration.\n   */\n  readonly tests: { [testName: string]: number };\n\n  /**\n   * The profile that was used to run the test\n   *\n   * @default - default profile\n   */\n  readonly profile?: string;\n}\n\nexport interface SnapshotVerificationOptions {\n  /**\n   * Retain failed snapshot comparisons\n   *\n   * @default false\n   */\n  readonly retain?: boolean;\n\n  /**\n   * Verbose mode\n   *\n   * @default false\n   */\n  readonly verbose?: boolean;\n}\n\n/**\n * Integration test results\n */\nexport interface IntegBatchResponse {\n  /**\n   * List of failed tests\n   */\n  readonly failedTests: IntegTestInfo[];\n\n  /**\n   * List of Integration test metrics. Each entry in the\n   * list represents metrics from a single worker (account + region).\n   */\n  readonly metrics: IntegRunnerMetrics[];\n}\n\n/**\n * Common options for running integration tests\n */\nexport interface IntegTestOptions {\n  /**\n   * A list of integration tests to run\n   * in this batch\n   */\n  readonly tests: IntegTestWorkerConfig[];\n\n  /**\n   * Whether or not to destroy the stacks at the\n   * end of the test\n   *\n   * @default true\n   */\n  readonly clean?: boolean;\n\n  /**\n   * When this is set to `true` the snapshot will\n   * be created _without_ running the integration test\n   * The resulting snapshot SHOULD NOT be checked in\n   *\n   * @default false\n   */\n  readonly dryRun?: boolean;\n\n  /**\n   * The level of verbosity for logging.\n   * Higher number means more output.\n   *\n   * @default 0\n   */\n  readonly verbosity?: number;\n\n  /**\n   * If this is set to true then the stack update workflow will be disabled\n   *\n   * @default true\n   */\n  readonly updateWorkflow?: boolean;\n}\n\n/**\n * Represents possible reasons for a diagnostic\n */\nexport enum DiagnosticReason {\n  /**\n   * The integration test failed because there\n   * is not existing snapshot\n   */\n  NO_SNAPSHOT = 'NO_SNAPSHOT',\n\n  /**\n   * The integration test failed\n   */\n  TEST_FAILED = 'TEST_FAILED',\n\n  /**\n   * There was an error running the integration test\n   */\n  TEST_ERROR = 'TEST_ERROR',\n\n  /**\n   * The snapshot test failed because the actual\n   * snapshot was different than the expected snapshot\n   */\n  SNAPSHOT_FAILED = 'SNAPSHOT_FAILED',\n\n  /**\n   * The snapshot test failed because there was an error executing it\n   */\n  SNAPSHOT_ERROR = 'SNAPSHOT_ERROR',\n\n  /**\n   * The snapshot test succeeded\n   */\n  SNAPSHOT_SUCCESS = 'SNAPSHOT_SUCCESS',\n\n  /**\n   * The integration test succeeded\n   */\n  TEST_SUCCESS = 'TEST_SUCCESS',\n\n  /**\n   * The assertion failed\n   */\n  ASSERTION_FAILED = 'ASSERTION_FAILED',\n}\n\n/**\n * Integration test diagnostics\n * This is used to report back the status of each test\n */\nexport interface Diagnostic {\n  /**\n   * The name of the test\n   */\n  readonly testName: string;\n\n  /**\n   * The diagnostic message\n   */\n  readonly message: string;\n\n  /**\n   * The time it took to run the test\n   */\n  readonly duration?: number;\n\n  /**\n   * The reason for the diagnostic\n   */\n  readonly reason: DiagnosticReason;\n\n  /**\n   * Additional messages to print\n   */\n  readonly additionalMessages?: string[];\n}\n\nexport function printSummary(total: number, failed: number): void {\n  if (failed > 0) {\n    logger.print('%s:    %s %s, %s total', chalk.bold('Tests'), chalk.red(failed), chalk.red('failed'), total);\n  } else {\n    logger.print('%s:    %s %s, %s total', chalk.bold('Tests'), chalk.green(total), chalk.green('passed'), total);\n  }\n}\n\n/**\n * Format the assertion results so that the results can be\n * printed\n */\nexport function formatAssertionResults(results: AssertionResults): string {\n  return Object.entries(results)\n    .map(([id, result]) => format('%s%s', id, result.status === 'success' ? ` - ${result.status}` : `\\n${result.message}`))\n    .join('\\n      ');\n}\n\n/**\n * Print out the results from tests\n */\nexport function printResults(diagnostic: Diagnostic): void {\n  switch (diagnostic.reason) {\n    case DiagnosticReason.SNAPSHOT_SUCCESS:\n      logger.success('  UNCHANGED  %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));\n      break;\n    case DiagnosticReason.TEST_SUCCESS:\n      logger.success('  SUCCESS    %s %s\\n      ', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.NO_SNAPSHOT:\n      logger.error('  NEW        %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));\n      break;\n    case DiagnosticReason.SNAPSHOT_FAILED:\n      logger.error('  CHANGED    %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.SNAPSHOT_ERROR:\n    case DiagnosticReason.TEST_ERROR:\n      logger.error('  ERROR      %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.TEST_FAILED:\n      logger.error('  FAILED     %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n    case DiagnosticReason.ASSERTION_FAILED:\n      logger.error('  ASSERT     %s %s\\n      %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), diagnostic.message);\n      break;\n  }\n  for (const addl of diagnostic.additionalMessages ?? []) {\n    logger.print(`      ${addl}`);\n  }\n}\n\nexport function printLaggards(testNames: Set<string>) {\n  const parts = [\n    '  ',\n    `Waiting for ${testNames.size} more`,\n    testNames.size < 10 ? ['(', Array.from(testNames).join(', '), ')'].join('') : '',\n  ];\n\n  logger.print(chalk.grey(parts.filter(x => x).join(' ')));\n}\n"]}
@@ -42,7 +42,7 @@ function integTestWorker(request) {
42
42
  updateWorkflow: request.updateWorkflow,
43
43
  verbosity,
44
44
  });
45
- if (results) {
45
+ if (results && Object.values(results).some(result => result.status === 'fail')) {
46
46
  failures.push(testInfo);
47
47
  workerpool.workerEmit({
48
48
  reason: common_1.DiagnosticReason.ASSERTION_FAILED,
@@ -55,7 +55,7 @@ function integTestWorker(request) {
55
55
  workerpool.workerEmit({
56
56
  reason: common_1.DiagnosticReason.TEST_SUCCESS,
57
57
  testName: `${runner.testName}-${testCaseName}`,
58
- message: 'Success',
58
+ message: results ? common_1.formatAssertionResults(results) : 'NO ASSERTIONS',
59
59
  duration: (Date.now() - start) / 1000,
60
60
  });
61
61
  }
@@ -154,4 +154,4 @@ workerpool.worker({
154
154
  snapshotTestWorker,
155
155
  integTestWorker,
156
156
  });
157
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"extract_worker.js","sourceRoot":"","sources":["extract_worker.ts"],"names":[],"mappings":"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","sourcesContent":["import * as workerpool from 'workerpool';\nimport { IntegSnapshotRunner, IntegTestRunner } from '../../runner';\nimport { IntegTest, IntegTestInfo } from '../../runner/integration-tests';\nimport { DiagnosticReason, IntegTestWorkerConfig, SnapshotVerificationOptions, Diagnostic, formatAssertionResults } from '../common';\nimport { IntegTestBatchRequest } from '../integ-test-worker';\n\n/**\n * Runs a single integration test batch request.\n * If the test does not have an existing snapshot,\n * this will first generate a snapshot and then execute\n * the integration tests.\n *\n * If the tests succeed it will then save the snapshot\n */\nexport function integTestWorker(request: IntegTestBatchRequest): IntegTestWorkerConfig[] {\n  const failures: IntegTestInfo[] = [];\n  const verbosity = request.verbosity ?? 0;\n\n  for (const testInfo of request.tests) {\n    const test = new IntegTest(testInfo); // Hydrate from data\n    const start = Date.now();\n\n    try {\n      const runner = new IntegTestRunner({\n        test,\n        profile: request.profile,\n        env: {\n          AWS_REGION: request.region,\n        },\n        showOutput: verbosity >= 2,\n      }, testInfo.destructiveChanges);\n\n      const tests = runner.actualTests();\n\n      if (!tests || Object.keys(tests).length === 0) {\n        throw new Error(`No tests defined for ${runner.testName}`);\n      }\n      for (const testCaseName of Object.keys(tests)) {\n        try {\n          const results = runner.runIntegTestCase({\n            testCaseName,\n            clean: request.clean,\n            dryRun: request.dryRun,\n            updateWorkflow: request.updateWorkflow,\n            verbosity,\n          });\n          if (results) {\n            failures.push(testInfo);\n            workerpool.workerEmit({\n              reason: DiagnosticReason.ASSERTION_FAILED,\n              testName: `${runner.testName}-${testCaseName} (${request.profile}/${request.region})`,\n              message: formatAssertionResults(results),\n              duration: (Date.now() - start) / 1000,\n            });\n          } else {\n            workerpool.workerEmit({\n              reason: DiagnosticReason.TEST_SUCCESS,\n              testName: `${runner.testName}-${testCaseName}`,\n              message: 'Success',\n              duration: (Date.now() - start) / 1000,\n            });\n          }\n        } catch (e) {\n          failures.push(testInfo);\n          workerpool.workerEmit({\n            reason: DiagnosticReason.TEST_FAILED,\n            testName: `${runner.testName}-${testCaseName} (${request.profile}/${request.region})`,\n            message: `Integration test failed: ${e}`,\n            duration: (Date.now() - start) / 1000,\n          });\n        }\n      }\n    } catch (e) {\n      failures.push(testInfo);\n      workerpool.workerEmit({\n        reason: DiagnosticReason.TEST_ERROR,\n        testName: `${testInfo.fileName} (${request.profile}/${request.region})`,\n        message: `Error during integration test: ${e}`,\n        duration: (Date.now() - start) / 1000,\n      });\n    }\n  }\n\n  return failures;\n}\n\n/**\n * Runs a single snapshot test batch request.\n * For each integration test this will check to see\n * if there is an existing snapshot, and if there is will\n * check if there are any changes\n */\nexport function snapshotTestWorker(testInfo: IntegTestInfo, options: SnapshotVerificationOptions = {}): IntegTestWorkerConfig[] {\n  const failedTests = new Array<IntegTestWorkerConfig>();\n  const start = Date.now();\n  const test = new IntegTest(testInfo); // Hydrate the data record again\n\n  const timer = setTimeout(() => {\n    workerpool.workerEmit({\n      reason: DiagnosticReason.SNAPSHOT_ERROR,\n      testName: test.testName,\n      message: 'Test is taking a very long time',\n      duration: (Date.now() - start) / 1000,\n    });\n  }, 60_000);\n\n  try {\n    const runner = new IntegSnapshotRunner({ test });\n    if (!runner.hasSnapshot()) {\n      workerpool.workerEmit({\n        reason: DiagnosticReason.NO_SNAPSHOT,\n        testName: test.testName,\n        message: 'No Snapshot',\n        duration: (Date.now() - start) / 1000,\n      });\n      failedTests.push(test.info);\n    } else {\n      const { diagnostics, destructiveChanges } = runner.testSnapshot(options);\n      if (diagnostics.length > 0) {\n        diagnostics.forEach(diagnostic => workerpool.workerEmit({\n          ...diagnostic,\n          duration: (Date.now() - start) / 1000,\n        } as Diagnostic));\n        failedTests.push({\n          ...test.info,\n          destructiveChanges,\n        });\n      } else {\n        workerpool.workerEmit({\n          reason: DiagnosticReason.SNAPSHOT_SUCCESS,\n          testName: test.testName,\n          message: 'Success',\n          duration: (Date.now() - start) / 1000,\n        } as Diagnostic);\n      }\n    }\n  } catch (e) {\n    failedTests.push(test.info);\n    workerpool.workerEmit({\n      message: e.message,\n      testName: test.testName,\n      reason: DiagnosticReason.SNAPSHOT_ERROR,\n      duration: (Date.now() - start) / 1000,\n    } as Diagnostic);\n  } finally {\n    clearTimeout(timer);\n  }\n\n  return failedTests;\n}\n\nworkerpool.worker({\n  snapshotTestWorker,\n  integTestWorker,\n});\n"]}
157
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"extract_worker.js","sourceRoot":"","sources":["extract_worker.ts"],"names":[],"mappings":"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","sourcesContent":["import * as workerpool from 'workerpool';\nimport { IntegSnapshotRunner, IntegTestRunner } from '../../runner';\nimport { IntegTest, IntegTestInfo } from '../../runner/integration-tests';\nimport { DiagnosticReason, IntegTestWorkerConfig, SnapshotVerificationOptions, Diagnostic, formatAssertionResults } from '../common';\nimport { IntegTestBatchRequest } from '../integ-test-worker';\n\n/**\n * Runs a single integration test batch request.\n * If the test does not have an existing snapshot,\n * this will first generate a snapshot and then execute\n * the integration tests.\n *\n * If the tests succeed it will then save the snapshot\n */\nexport function integTestWorker(request: IntegTestBatchRequest): IntegTestWorkerConfig[] {\n  const failures: IntegTestInfo[] = [];\n  const verbosity = request.verbosity ?? 0;\n\n  for (const testInfo of request.tests) {\n    const test = new IntegTest(testInfo); // Hydrate from data\n    const start = Date.now();\n\n    try {\n      const runner = new IntegTestRunner({\n        test,\n        profile: request.profile,\n        env: {\n          AWS_REGION: request.region,\n        },\n        showOutput: verbosity >= 2,\n      }, testInfo.destructiveChanges);\n\n      const tests = runner.actualTests();\n\n      if (!tests || Object.keys(tests).length === 0) {\n        throw new Error(`No tests defined for ${runner.testName}`);\n      }\n      for (const testCaseName of Object.keys(tests)) {\n        try {\n          const results = runner.runIntegTestCase({\n            testCaseName,\n            clean: request.clean,\n            dryRun: request.dryRun,\n            updateWorkflow: request.updateWorkflow,\n            verbosity,\n          });\n          if (results && Object.values(results).some(result => result.status === 'fail')) {\n            failures.push(testInfo);\n            workerpool.workerEmit({\n              reason: DiagnosticReason.ASSERTION_FAILED,\n              testName: `${runner.testName}-${testCaseName} (${request.profile}/${request.region})`,\n              message: formatAssertionResults(results),\n              duration: (Date.now() - start) / 1000,\n            });\n          } else {\n            workerpool.workerEmit({\n              reason: DiagnosticReason.TEST_SUCCESS,\n              testName: `${runner.testName}-${testCaseName}`,\n              message: results ? formatAssertionResults(results) : 'NO ASSERTIONS',\n              duration: (Date.now() - start) / 1000,\n            });\n          }\n        } catch (e) {\n          failures.push(testInfo);\n          workerpool.workerEmit({\n            reason: DiagnosticReason.TEST_FAILED,\n            testName: `${runner.testName}-${testCaseName} (${request.profile}/${request.region})`,\n            message: `Integration test failed: ${e}`,\n            duration: (Date.now() - start) / 1000,\n          });\n        }\n      }\n    } catch (e) {\n      failures.push(testInfo);\n      workerpool.workerEmit({\n        reason: DiagnosticReason.TEST_ERROR,\n        testName: `${testInfo.fileName} (${request.profile}/${request.region})`,\n        message: `Error during integration test: ${e}`,\n        duration: (Date.now() - start) / 1000,\n      });\n    }\n  }\n\n  return failures;\n}\n\n/**\n * Runs a single snapshot test batch request.\n * For each integration test this will check to see\n * if there is an existing snapshot, and if there is will\n * check if there are any changes\n */\nexport function snapshotTestWorker(testInfo: IntegTestInfo, options: SnapshotVerificationOptions = {}): IntegTestWorkerConfig[] {\n  const failedTests = new Array<IntegTestWorkerConfig>();\n  const start = Date.now();\n  const test = new IntegTest(testInfo); // Hydrate the data record again\n\n  const timer = setTimeout(() => {\n    workerpool.workerEmit({\n      reason: DiagnosticReason.SNAPSHOT_ERROR,\n      testName: test.testName,\n      message: 'Test is taking a very long time',\n      duration: (Date.now() - start) / 1000,\n    });\n  }, 60_000);\n\n  try {\n    const runner = new IntegSnapshotRunner({ test });\n    if (!runner.hasSnapshot()) {\n      workerpool.workerEmit({\n        reason: DiagnosticReason.NO_SNAPSHOT,\n        testName: test.testName,\n        message: 'No Snapshot',\n        duration: (Date.now() - start) / 1000,\n      });\n      failedTests.push(test.info);\n    } else {\n      const { diagnostics, destructiveChanges } = runner.testSnapshot(options);\n      if (diagnostics.length > 0) {\n        diagnostics.forEach(diagnostic => workerpool.workerEmit({\n          ...diagnostic,\n          duration: (Date.now() - start) / 1000,\n        } as Diagnostic));\n        failedTests.push({\n          ...test.info,\n          destructiveChanges,\n        });\n      } else {\n        workerpool.workerEmit({\n          reason: DiagnosticReason.SNAPSHOT_SUCCESS,\n          testName: test.testName,\n          message: 'Success',\n          duration: (Date.now() - start) / 1000,\n        } as Diagnostic);\n      }\n    }\n  } catch (e) {\n    failedTests.push(test.info);\n    workerpool.workerEmit({\n      message: e.message,\n      testName: test.testName,\n      reason: DiagnosticReason.SNAPSHOT_ERROR,\n      duration: (Date.now() - start) / 1000,\n    } as Diagnostic);\n  } finally {\n    clearTimeout(timer);\n  }\n\n  return failedTests;\n}\n\nworkerpool.worker({\n  snapshotTestWorker,\n  integTestWorker,\n});\n"]}