@aws-cdk/integ-runner 2.38.1 → 2.40.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -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"]}
@@ -41,11 +41,21 @@ export declare class AssemblyManifestReader {
41
41
  * Write trace data to the assembly manifest metadata
42
42
  */
43
43
  recordTrace(trace: ManifestTrace): void;
44
+ /**
45
+ * Return a list of assets for a given stack
46
+ */
47
+ getAssetIdsForStack(stackId: string): string[];
44
48
  /**
45
49
  * For a given stackId return a list of assets that belong to the stack
46
50
  */
47
- getAssetsForStack(stackId: string): string[];
51
+ getAssetLocationsForStack(stackId: string): string[];
52
+ /**
53
+ * Get a list of assets from the assembly manifest
54
+ */
48
55
  private assetsFromAssemblyManifest;
56
+ /**
57
+ * Get a list of assets from the asset manifest
58
+ */
49
59
  private assetsFromAssetManifest;
50
60
  /**
51
61
  * Clean the manifest of any unneccesary data. Currently that includes
@@ -71,37 +71,66 @@ class AssemblyManifestReader {
71
71
  };
72
72
  cloud_assembly_schema_1.Manifest.saveAssemblyManifest(newManifest, this.manifestFileName);
73
73
  }
74
+ /**
75
+ * Return a list of assets for a given stack
76
+ */
77
+ getAssetIdsForStack(stackId) {
78
+ var _a, _b;
79
+ const assets = [];
80
+ for (const artifact of Object.values((_a = this.manifest.artifacts) !== null && _a !== void 0 ? _a : {})) {
81
+ if (artifact.type === cloud_assembly_schema_1.ArtifactType.ASSET_MANIFEST && ((_b = artifact.properties) === null || _b === void 0 ? void 0 : _b.file) === `${stackId}.assets.json`) {
82
+ assets.push(...this.assetsFromAssetManifest(artifact).map(asset => asset.id.assetId));
83
+ }
84
+ else if (artifact.type === cloud_assembly_schema_1.ArtifactType.AWS_CLOUDFORMATION_STACK) {
85
+ assets.push(...this.assetsFromAssemblyManifest(artifact).map(asset => asset.id));
86
+ }
87
+ }
88
+ return assets;
89
+ }
74
90
  /**
75
91
  * For a given stackId return a list of assets that belong to the stack
76
92
  */
77
- getAssetsForStack(stackId) {
93
+ getAssetLocationsForStack(stackId) {
78
94
  var _a, _b;
79
95
  const assets = [];
80
96
  for (const artifact of Object.values((_a = this.manifest.artifacts) !== null && _a !== void 0 ? _a : {})) {
81
97
  if (artifact.type === cloud_assembly_schema_1.ArtifactType.ASSET_MANIFEST && ((_b = artifact.properties) === null || _b === void 0 ? void 0 : _b.file) === `${stackId}.assets.json`) {
82
- assets.push(...this.assetsFromAssetManifest(artifact));
98
+ assets.push(...this.assetsFromAssetManifest(artifact).map(asset => {
99
+ if (asset.type === 'file') {
100
+ return asset.source.path;
101
+ }
102
+ else {
103
+ return asset.source.directory;
104
+ }
105
+ }));
83
106
  }
84
107
  else if (artifact.type === cloud_assembly_schema_1.ArtifactType.AWS_CLOUDFORMATION_STACK) {
85
- assets.push(...this.assetsFromAssemblyManifest(artifact));
108
+ assets.push(...this.assetsFromAssemblyManifest(artifact).map(asset => asset.path));
86
109
  }
87
110
  }
88
111
  return assets;
89
112
  }
113
+ /**
114
+ * Get a list of assets from the assembly manifest
115
+ */
90
116
  assetsFromAssemblyManifest(artifact) {
91
117
  var _a;
92
118
  const assets = [];
93
119
  for (const metadata of Object.values((_a = artifact.metadata) !== null && _a !== void 0 ? _a : {})) {
94
120
  metadata.forEach(data => {
95
121
  if (data.type === cloud_assembly_schema_1.ArtifactMetadataEntryType.ASSET) {
96
- const assetPath = data.data.path;
97
- if (assetPath.startsWith('asset.')) {
98
- assets.push(assetPath);
122
+ const asset = data.data;
123
+ if (asset.path.startsWith('asset.')) {
124
+ assets.push(asset);
99
125
  }
100
126
  }
101
127
  });
102
128
  }
103
129
  return assets;
104
130
  }
131
+ /**
132
+ * Get a list of assets from the asset manifest
133
+ */
105
134
  assetsFromAssetManifest(artifact) {
106
135
  const assets = [];
107
136
  const fileName = artifact.properties.file;
@@ -110,13 +139,13 @@ class AssemblyManifestReader {
110
139
  if (entry.type === 'file') {
111
140
  const source = entry.source;
112
141
  if (source.path && source.path.startsWith('asset.')) {
113
- assets.push(entry.source.path);
142
+ assets.push(entry);
114
143
  }
115
144
  }
116
145
  else if (entry.type === 'docker-image') {
117
146
  const source = entry.source;
118
147
  if (source.directory && source.directory.startsWith('asset.')) {
119
- assets.push(entry.source.directory);
148
+ assets.push(entry);
120
149
  }
121
150
  }
122
151
  });
@@ -188,4 +217,4 @@ class AssemblyManifestReader {
188
217
  }
189
218
  exports.AssemblyManifestReader = AssemblyManifestReader;
190
219
  AssemblyManifestReader.DEFAULT_FILENAME = 'manifest.json';
191
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-assembly.js","sourceRoot":"","sources":["cloud-assembly.ts"],"names":[],"mappings":"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","sourcesContent":["import * as path from 'path';\nimport { AssemblyManifest, Manifest, ArtifactType, AwsCloudFormationStackProperties, ArtifactManifest, MetadataEntry, AssetManifestProperties, ArtifactMetadataEntryType, ContainerImageAssetMetadataEntry, FileAssetMetadataEntry } from '@aws-cdk/cloud-assembly-schema';\nimport { AssetManifest, FileManifestEntry, DockerImageManifestEntry } from 'cdk-assets';\nimport * as fs from 'fs-extra';\n\n/**\n * Trace information for stack\n * map of resource logicalId to trace message\n */\nexport type StackTrace = Map<string, string>;\n\n/**\n * Trace information for a assembly\n *\n * map of stackId to StackTrace\n */\nexport type ManifestTrace = Map<string, StackTrace>;\n\n/**\n * Reads a Cloud Assembly manifest\n */\nexport class AssemblyManifestReader {\n  public static readonly DEFAULT_FILENAME = 'manifest.json';\n\n  /**\n   * Reads a Cloud Assembly manifest from a file\n   */\n  public static fromFile(fileName: string): AssemblyManifestReader {\n    try {\n      const obj = Manifest.loadAssemblyManifest(fileName);\n      return new AssemblyManifestReader(path.dirname(fileName), obj, fileName);\n\n    } catch (e) {\n      throw new Error(`Cannot read integ manifest '${fileName}': ${e.message}`);\n    }\n  }\n\n  /**\n   * Reads a Cloud Assembly manifest from a file or a directory\n   * If the given filePath is a directory then it will look for\n   * a file within the directory with the DEFAULT_FILENAME\n   */\n  public static fromPath(filePath: string): AssemblyManifestReader {\n    let st;\n    try {\n      st = fs.statSync(filePath);\n    } catch (e) {\n      throw new Error(`Cannot read integ manifest at '${filePath}': ${e.message}`);\n    }\n    if (st.isDirectory()) {\n      return AssemblyManifestReader.fromFile(path.join(filePath, AssemblyManifestReader.DEFAULT_FILENAME));\n    }\n    return AssemblyManifestReader.fromFile(filePath);\n  }\n\n  /**\n   * The directory where the manifest was found\n   */\n  public readonly directory: string;\n\n  constructor(directory: string, private readonly manifest: AssemblyManifest, private readonly manifestFileName: string) {\n    this.directory = directory;\n  }\n\n  /**\n   * Get the stacks from the manifest\n   * returns a map of artifactId to CloudFormation template\n   */\n  public get stacks(): Record<string, any> {\n    const stacks: Record<string, any> = {};\n    for (const [artifactId, artifact] of Object.entries(this.manifest.artifacts ?? {})) {\n      if (artifact.type !== ArtifactType.AWS_CLOUDFORMATION_STACK) { continue; }\n      const props = artifact.properties as AwsCloudFormationStackProperties;\n\n      const template = fs.readJSONSync(path.resolve(this.directory, props.templateFile));\n      stacks[artifactId] = template;\n    }\n    return stacks;\n  }\n\n  /**\n   * Write trace data to the assembly manifest metadata\n   */\n  public recordTrace(trace: ManifestTrace): void {\n    const newManifest = {\n      ...this.manifest,\n      artifacts: this.renderArtifacts(trace),\n    };\n    Manifest.saveAssemblyManifest(newManifest, this.manifestFileName);\n  }\n\n  /**\n   * For a given stackId return a list of assets that belong to the stack\n   */\n  public getAssetsForStack(stackId: string): string[] {\n    const assets: string[] = [];\n    for (const artifact of Object.values(this.manifest.artifacts ?? {})) {\n      if (artifact.type === ArtifactType.ASSET_MANIFEST && (artifact.properties as AssetManifestProperties)?.file === `${stackId}.assets.json`) {\n        assets.push(...this.assetsFromAssetManifest(artifact));\n      } else if (artifact.type === ArtifactType.AWS_CLOUDFORMATION_STACK) {\n        assets.push(...this.assetsFromAssemblyManifest(artifact));\n      }\n    }\n    return assets;\n  }\n\n  private assetsFromAssemblyManifest(artifact: ArtifactManifest): string[] {\n    const assets: string[] = [];\n    for (const metadata of Object.values(artifact.metadata ?? {})) {\n      metadata.forEach(data => {\n        if (data.type === ArtifactMetadataEntryType.ASSET) {\n          const assetPath = (data.data as ContainerImageAssetMetadataEntry | FileAssetMetadataEntry).path;\n          if (assetPath.startsWith('asset.')) {\n            assets.push(assetPath);\n          }\n        }\n      });\n    }\n    return assets;\n  }\n\n  private assetsFromAssetManifest(artifact: ArtifactManifest): string[] {\n    const assets: string[] = [];\n    const fileName = (artifact.properties as AssetManifestProperties).file;\n    const assetManifest = AssetManifest.fromFile(path.join(this.directory, fileName));\n    assetManifest.entries.forEach(entry => {\n      if (entry.type === 'file') {\n        const source = (entry as FileManifestEntry).source;\n        if (source.path && source.path.startsWith('asset.')) {\n          assets.push((entry as FileManifestEntry).source.path!);\n        }\n      } else if (entry.type === 'docker-image') {\n        const source = (entry as DockerImageManifestEntry).source;\n        if (source.directory && source.directory.startsWith('asset.')) {\n          assets.push((entry as DockerImageManifestEntry).source.directory!);\n        }\n      }\n    });\n    return assets;\n  }\n\n  /**\n   * Clean the manifest of any unneccesary data. Currently that includes\n   * the metadata trace information since this includes trace information like\n   * file system locations and file lines that will change depending on what machine the test is run on\n   */\n  public cleanManifest(): void {\n    const newManifest = {\n      ...this.manifest,\n      artifacts: this.renderArtifacts(),\n    };\n    Manifest.saveAssemblyManifest(newManifest, this.manifestFileName);\n  }\n\n  private renderArtifactMetadata(artifact: ArtifactManifest, trace?: StackTrace): { [id: string]: MetadataEntry[] } | undefined {\n    const newMetadata: { [id: string]: MetadataEntry[] } = {};\n    if (!artifact.metadata) return artifact.metadata;\n    for (const [metadataId, metadataEntry] of Object.entries(artifact.metadata ?? {})) {\n      newMetadata[metadataId] = metadataEntry.map((meta: MetadataEntry) => {\n        if (meta.type === 'aws:cdk:logicalId' && trace && meta.data) {\n          const traceData = trace.get(meta.data.toString());\n          if (traceData) {\n            trace.delete(meta.data.toString());\n            return {\n              type: meta.type,\n              data: meta.data,\n              trace: [traceData],\n            };\n          }\n        }\n        // return metadata without the trace data\n        return {\n          type: meta.type,\n          data: meta.data,\n        };\n      });\n    }\n    if (trace && trace.size > 0) {\n      for (const [id, data] of trace.entries()) {\n        newMetadata[id] = [{\n          type: 'aws:cdk:logicalId',\n          data: id,\n          trace: [data],\n        }];\n      }\n    }\n    return newMetadata;\n  }\n\n  private renderArtifacts(trace?: ManifestTrace): { [id: string]: ArtifactManifest } | undefined {\n    const newArtifacts: { [id: string]: ArtifactManifest } = {};\n    for (const [artifactId, artifact] of Object.entries(this.manifest.artifacts ?? {})) {\n      let stackTrace: StackTrace | undefined = undefined;\n      if (artifact.type === ArtifactType.AWS_CLOUDFORMATION_STACK && trace) {\n        stackTrace = trace.get(artifactId);\n      }\n      newArtifacts[artifactId] = {\n        ...artifact,\n        metadata: this.renderArtifactMetadata(artifact, stackTrace),\n      };\n    }\n    return newArtifacts;\n  }\n}\n"]}
220
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cloud-assembly.js","sourceRoot":"","sources":["cloud-assembly.ts"],"names":[],"mappings":";;;AAAA,6BAA6B;AAC7B,0EAA2Q;AAC3Q,2CAAwF;AACxF,+BAA+B;AAe/B;;GAEG;AACH,MAAa,sBAAsB;IAuCjC,YAAY,SAAiB,EAAmB,QAA0B,EAAmB,gBAAwB;QAArE,aAAQ,GAAR,QAAQ,CAAkB;QAAmB,qBAAgB,GAAhB,gBAAgB,CAAQ;QACnH,IAAI,CAAC,SAAS,GAAG,SAAS,CAAC;IAC7B,CAAC;IAtCD;;OAEG;IACI,MAAM,CAAC,QAAQ,CAAC,QAAgB;QACrC,IAAI;YACF,MAAM,GAAG,GAAG,gCAAQ,CAAC,oBAAoB,CAAC,QAAQ,CAAC,CAAC;YACpD,OAAO,IAAI,sBAAsB,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,GAAG,EAAE,QAAQ,CAAC,CAAC;SAE1E;QAAC,OAAO,CAAC,EAAE;YACV,MAAM,IAAI,KAAK,CAAC,+BAA+B,QAAQ,MAAM,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;SAC3E;IACH,CAAC;IAED;;;;OAIG;IACI,MAAM,CAAC,QAAQ,CAAC,QAAgB;QACrC,IAAI,EAAE,CAAC;QACP,IAAI;YACF,EAAE,GAAG,EAAE,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;SAC5B;QAAC,OAAO,CAAC,EAAE;YACV,MAAM,IAAI,KAAK,CAAC,kCAAkC,QAAQ,MAAM,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;SAC9E;QACD,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE;YACpB,OAAO,sBAAsB,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,sBAAsB,CAAC,gBAAgB,CAAC,CAAC,CAAC;SACtG;QACD,OAAO,sBAAsB,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;IACnD,CAAC;IAWD;;;OAGG;IACH,IAAW,MAAM;;QACf,MAAM,MAAM,GAAwB,EAAE,CAAC;QACvC,KAAK,MAAM,CAAC,UAAU,EAAE,QAAQ,CAAC,IAAI,MAAM,CAAC,OAAO,OAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,mCAAI,EAAE,CAAC,EAAE;YAClF,IAAI,QAAQ,CAAC,IAAI,KAAK,oCAAY,CAAC,wBAAwB,EAAE;gBAAE,SAAS;aAAE;YAC1E,MAAM,KAAK,GAAG,QAAQ,CAAC,UAA8C,CAAC;YAEtE,MAAM,QAAQ,GAAG,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,EAAE,KAAK,CAAC,YAAY,CAAC,CAAC,CAAC;YACnF,MAAM,CAAC,UAAU,CAAC,GAAG,QAAQ,CAAC;SAC/B;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACI,WAAW,CAAC,KAAoB;QACrC,MAAM,WAAW,GAAG;YAClB,GAAG,IAAI,CAAC,QAAQ;YAChB,SAAS,EAAE,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC;SACvC,CAAC;QACF,gCAAQ,CAAC,oBAAoB,CAAC,WAAW,EAAE,IAAI,CAAC,gBAAgB,CAAC,CAAC;IACpE,CAAC;IAED;;OAEG;IACI,mBAAmB,CAAC,OAAe;;QACxC,MAAM,MAAM,GAAa,EAAE,CAAC;QAC5B,KAAK,MAAM,QAAQ,IAAI,MAAM,CAAC,MAAM,OAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,mCAAI,EAAE,CAAC,EAAE;YACnE,IAAI,QAAQ,CAAC,IAAI,KAAK,oCAAY,CAAC,cAAc,IAAI,OAAC,QAAQ,CAAC,UAAsC,0CAAE,IAAI,MAAK,GAAG,OAAO,cAAc,EAAE;gBACxI,MAAM,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,uBAAuB,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC;aACvF;iBAAM,IAAI,QAAQ,CAAC,IAAI,KAAK,oCAAY,CAAC,wBAAwB,EAAE;gBAClE,MAAM,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,0BAA0B,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC;aAClF;SACF;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACI,yBAAyB,CAAC,OAAe;;QAC9C,MAAM,MAAM,GAAa,EAAE,CAAC;QAC5B,KAAK,MAAM,QAAQ,IAAI,MAAM,CAAC,MAAM,OAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,mCAAI,EAAE,CAAC,EAAE;YACnE,IAAI,QAAQ,CAAC,IAAI,KAAK,oCAAY,CAAC,cAAc,IAAI,OAAC,QAAQ,CAAC,UAAsC,0CAAE,IAAI,MAAK,GAAG,OAAO,cAAc,EAAE;gBACxI,MAAM,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,uBAAuB,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE;oBAChE,IAAI,KAAK,CAAC,IAAI,KAAK,MAAM,EAAE;wBACzB,OAAO,KAAK,CAAC,MAAM,CAAC,IAAK,CAAC;qBAC3B;yBAAM;wBACL,OAAO,KAAK,CAAC,MAAM,CAAC,SAAU,CAAC;qBAChC;gBACH,CAAC,CAAC,CAAC,CAAC;aACL;iBAAM,IAAI,QAAQ,CAAC,IAAI,KAAK,oCAAY,CAAC,wBAAwB,EAAE;gBAClE,MAAM,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,0BAA0B,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC;aACpF;SACF;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACK,0BAA0B,CAAC,QAA0B;;QAC3D,MAAM,MAAM,GAAkE,EAAE,CAAC;QACjF,KAAK,MAAM,QAAQ,IAAI,MAAM,CAAC,MAAM,OAAC,QAAQ,CAAC,QAAQ,mCAAI,EAAE,CAAC,EAAE;YAC7D,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;gBACtB,IAAI,IAAI,CAAC,IAAI,KAAK,iDAAyB,CAAC,KAAK,EAAE;oBACjD,MAAM,KAAK,GAAI,IAAI,CAAC,IAAkE,CAAC;oBACvF,IAAI,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE;wBACnC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;qBACpB;iBACF;YACH,CAAC,CAAC,CAAC;SACJ;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACK,uBAAuB,CAAC,QAA0B;QACxD,MAAM,MAAM,GAAqD,EAAE,CAAC;QACpE,MAAM,QAAQ,GAAI,QAAQ,CAAC,UAAsC,CAAC,IAAI,CAAC;QACvE,MAAM,aAAa,GAAG,0BAAa,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC,CAAC;QAClF,aAAa,CAAC,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;YACpC,IAAI,KAAK,CAAC,IAAI,KAAK,MAAM,EAAE;gBACzB,MAAM,MAAM,GAAI,KAA2B,CAAC,MAAM,CAAC;gBACnD,IAAI,MAAM,CAAC,IAAI,IAAI,MAAM,CAAC,IAAI,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE;oBACnD,MAAM,CAAC,IAAI,CAAC,KAA0B,CAAC,CAAC;iBACzC;aACF;iBAAM,IAAI,KAAK,CAAC,IAAI,KAAK,cAAc,EAAE;gBACxC,MAAM,MAAM,GAAI,KAAkC,CAAC,MAAM,CAAC;gBAC1D,IAAI,MAAM,CAAC,SAAS,IAAI,MAAM,CAAC,SAAS,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE;oBAC7D,MAAM,CAAC,IAAI,CAAC,KAAiC,CAAC,CAAC;iBAChD;aACF;QACH,CAAC,CAAC,CAAC;QACH,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;OAIG;IACI,aAAa;QAClB,MAAM,WAAW,GAAG;YAClB,GAAG,IAAI,CAAC,QAAQ;YAChB,SAAS,EAAE,IAAI,CAAC,eAAe,EAAE;SAClC,CAAC;QACF,gCAAQ,CAAC,oBAAoB,CAAC,WAAW,EAAE,IAAI,CAAC,gBAAgB,CAAC,CAAC;IACpE,CAAC;IAEO,sBAAsB,CAAC,QAA0B,EAAE,KAAkB;;QAC3E,MAAM,WAAW,GAAsC,EAAE,CAAC;QAC1D,IAAI,CAAC,QAAQ,CAAC,QAAQ;YAAE,OAAO,QAAQ,CAAC,QAAQ,CAAC;QACjD,KAAK,MAAM,CAAC,UAAU,EAAE,aAAa,CAAC,IAAI,MAAM,CAAC,OAAO,OAAC,QAAQ,CAAC,QAAQ,mCAAI,EAAE,CAAC,EAAE;YACjF,WAAW,CAAC,UAAU,CAAC,GAAG,aAAa,CAAC,GAAG,CAAC,CAAC,IAAmB,EAAE,EAAE;gBAClE,IAAI,IAAI,CAAC,IAAI,KAAK,mBAAmB,IAAI,KAAK,IAAI,IAAI,CAAC,IAAI,EAAE;oBAC3D,MAAM,SAAS,GAAG,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC;oBAClD,IAAI,SAAS,EAAE;wBACb,KAAK,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC;wBACnC,OAAO;4BACL,IAAI,EAAE,IAAI,CAAC,IAAI;4BACf,IAAI,EAAE,IAAI,CAAC,IAAI;4BACf,KAAK,EAAE,CAAC,SAAS,CAAC;yBACnB,CAAC;qBACH;iBACF;gBACD,yCAAyC;gBACzC,OAAO;oBACL,IAAI,EAAE,IAAI,CAAC,IAAI;oBACf,IAAI,EAAE,IAAI,CAAC,IAAI;iBAChB,CAAC;YACJ,CAAC,CAAC,CAAC;SACJ;QACD,IAAI,KAAK,IAAI,KAAK,CAAC,IAAI,GAAG,CAAC,EAAE;YAC3B,KAAK,MAAM,CAAC,EAAE,EAAE,IAAI,CAAC,IAAI,KAAK,CAAC,OAAO,EAAE,EAAE;gBACxC,WAAW,CAAC,EAAE,CAAC,GAAG,CAAC;wBACjB,IAAI,EAAE,mBAAmB;wBACzB,IAAI,EAAE,EAAE;wBACR,KAAK,EAAE,CAAC,IAAI,CAAC;qBACd,CAAC,CAAC;aACJ;SACF;QACD,OAAO,WAAW,CAAC;IACrB,CAAC;IAEO,eAAe,CAAC,KAAqB;;QAC3C,MAAM,YAAY,GAAuC,EAAE,CAAC;QAC5D,KAAK,MAAM,CAAC,UAAU,EAAE,QAAQ,CAAC,IAAI,MAAM,CAAC,OAAO,OAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,mCAAI,EAAE,CAAC,EAAE;YAClF,IAAI,UAAU,GAA2B,SAAS,CAAC;YACnD,IAAI,QAAQ,CAAC,IAAI,KAAK,oCAAY,CAAC,wBAAwB,IAAI,KAAK,EAAE;gBACpE,UAAU,GAAG,KAAK,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;aACpC;YACD,YAAY,CAAC,UAAU,CAAC,GAAG;gBACzB,GAAG,QAAQ;gBACX,QAAQ,EAAE,IAAI,CAAC,sBAAsB,CAAC,QAAQ,EAAE,UAAU,CAAC;aAC5D,CAAC;SACH;QACD,OAAO,YAAY,CAAC;IACtB,CAAC;;AAhNH,wDAiNC;AAhNwB,uCAAgB,GAAG,eAAe,CAAC","sourcesContent":["import * as path from 'path';\nimport { AssemblyManifest, Manifest, ArtifactType, AwsCloudFormationStackProperties, ArtifactManifest, MetadataEntry, AssetManifestProperties, ArtifactMetadataEntryType, ContainerImageAssetMetadataEntry, FileAssetMetadataEntry } from '@aws-cdk/cloud-assembly-schema';\nimport { AssetManifest, FileManifestEntry, DockerImageManifestEntry } from 'cdk-assets';\nimport * as fs from 'fs-extra';\n\n/**\n * Trace information for stack\n * map of resource logicalId to trace message\n */\nexport type StackTrace = Map<string, string>;\n\n/**\n * Trace information for a assembly\n *\n * map of stackId to StackTrace\n */\nexport type ManifestTrace = Map<string, StackTrace>;\n\n/**\n * Reads a Cloud Assembly manifest\n */\nexport class AssemblyManifestReader {\n  public static readonly DEFAULT_FILENAME = 'manifest.json';\n\n  /**\n   * Reads a Cloud Assembly manifest from a file\n   */\n  public static fromFile(fileName: string): AssemblyManifestReader {\n    try {\n      const obj = Manifest.loadAssemblyManifest(fileName);\n      return new AssemblyManifestReader(path.dirname(fileName), obj, fileName);\n\n    } catch (e) {\n      throw new Error(`Cannot read integ manifest '${fileName}': ${e.message}`);\n    }\n  }\n\n  /**\n   * Reads a Cloud Assembly manifest from a file or a directory\n   * If the given filePath is a directory then it will look for\n   * a file within the directory with the DEFAULT_FILENAME\n   */\n  public static fromPath(filePath: string): AssemblyManifestReader {\n    let st;\n    try {\n      st = fs.statSync(filePath);\n    } catch (e) {\n      throw new Error(`Cannot read integ manifest at '${filePath}': ${e.message}`);\n    }\n    if (st.isDirectory()) {\n      return AssemblyManifestReader.fromFile(path.join(filePath, AssemblyManifestReader.DEFAULT_FILENAME));\n    }\n    return AssemblyManifestReader.fromFile(filePath);\n  }\n\n  /**\n   * The directory where the manifest was found\n   */\n  public readonly directory: string;\n\n  constructor(directory: string, private readonly manifest: AssemblyManifest, private readonly manifestFileName: string) {\n    this.directory = directory;\n  }\n\n  /**\n   * Get the stacks from the manifest\n   * returns a map of artifactId to CloudFormation template\n   */\n  public get stacks(): Record<string, any> {\n    const stacks: Record<string, any> = {};\n    for (const [artifactId, artifact] of Object.entries(this.manifest.artifacts ?? {})) {\n      if (artifact.type !== ArtifactType.AWS_CLOUDFORMATION_STACK) { continue; }\n      const props = artifact.properties as AwsCloudFormationStackProperties;\n\n      const template = fs.readJSONSync(path.resolve(this.directory, props.templateFile));\n      stacks[artifactId] = template;\n    }\n    return stacks;\n  }\n\n  /**\n   * Write trace data to the assembly manifest metadata\n   */\n  public recordTrace(trace: ManifestTrace): void {\n    const newManifest = {\n      ...this.manifest,\n      artifacts: this.renderArtifacts(trace),\n    };\n    Manifest.saveAssemblyManifest(newManifest, this.manifestFileName);\n  }\n\n  /**\n   * Return a list of assets for a given stack\n   */\n  public getAssetIdsForStack(stackId: string): string[] {\n    const assets: string[] = [];\n    for (const artifact of Object.values(this.manifest.artifacts ?? {})) {\n      if (artifact.type === ArtifactType.ASSET_MANIFEST && (artifact.properties as AssetManifestProperties)?.file === `${stackId}.assets.json`) {\n        assets.push(...this.assetsFromAssetManifest(artifact).map(asset => asset.id.assetId));\n      } else if (artifact.type === ArtifactType.AWS_CLOUDFORMATION_STACK) {\n        assets.push(...this.assetsFromAssemblyManifest(artifact).map(asset => asset.id));\n      }\n    }\n    return assets;\n  }\n\n  /**\n   * For a given stackId return a list of assets that belong to the stack\n   */\n  public getAssetLocationsForStack(stackId: string): string[] {\n    const assets: string[] = [];\n    for (const artifact of Object.values(this.manifest.artifacts ?? {})) {\n      if (artifact.type === ArtifactType.ASSET_MANIFEST && (artifact.properties as AssetManifestProperties)?.file === `${stackId}.assets.json`) {\n        assets.push(...this.assetsFromAssetManifest(artifact).map(asset => {\n          if (asset.type === 'file') {\n            return asset.source.path!;\n          } else {\n            return asset.source.directory!;\n          }\n        }));\n      } else if (artifact.type === ArtifactType.AWS_CLOUDFORMATION_STACK) {\n        assets.push(...this.assetsFromAssemblyManifest(artifact).map(asset => asset.path));\n      }\n    }\n    return assets;\n  }\n\n  /**\n   * Get a list of assets from the assembly manifest\n   */\n  private assetsFromAssemblyManifest(artifact: ArtifactManifest): (ContainerImageAssetMetadataEntry | FileAssetMetadataEntry)[] {\n    const assets: (ContainerImageAssetMetadataEntry | FileAssetMetadataEntry)[] = [];\n    for (const metadata of Object.values(artifact.metadata ?? {})) {\n      metadata.forEach(data => {\n        if (data.type === ArtifactMetadataEntryType.ASSET) {\n          const asset = (data.data as ContainerImageAssetMetadataEntry | FileAssetMetadataEntry);\n          if (asset.path.startsWith('asset.')) {\n            assets.push(asset);\n          }\n        }\n      });\n    }\n    return assets;\n  }\n\n  /**\n   * Get a list of assets from the asset manifest\n   */\n  private assetsFromAssetManifest(artifact: ArtifactManifest): (FileManifestEntry | DockerImageManifestEntry)[] {\n    const assets: (FileManifestEntry | DockerImageManifestEntry)[] = [];\n    const fileName = (artifact.properties as AssetManifestProperties).file;\n    const assetManifest = AssetManifest.fromFile(path.join(this.directory, fileName));\n    assetManifest.entries.forEach(entry => {\n      if (entry.type === 'file') {\n        const source = (entry as FileManifestEntry).source;\n        if (source.path && source.path.startsWith('asset.')) {\n          assets.push(entry as FileManifestEntry);\n        }\n      } else if (entry.type === 'docker-image') {\n        const source = (entry as DockerImageManifestEntry).source;\n        if (source.directory && source.directory.startsWith('asset.')) {\n          assets.push(entry as DockerImageManifestEntry);\n        }\n      }\n    });\n    return assets;\n  }\n\n  /**\n   * Clean the manifest of any unneccesary data. Currently that includes\n   * the metadata trace information since this includes trace information like\n   * file system locations and file lines that will change depending on what machine the test is run on\n   */\n  public cleanManifest(): void {\n    const newManifest = {\n      ...this.manifest,\n      artifacts: this.renderArtifacts(),\n    };\n    Manifest.saveAssemblyManifest(newManifest, this.manifestFileName);\n  }\n\n  private renderArtifactMetadata(artifact: ArtifactManifest, trace?: StackTrace): { [id: string]: MetadataEntry[] } | undefined {\n    const newMetadata: { [id: string]: MetadataEntry[] } = {};\n    if (!artifact.metadata) return artifact.metadata;\n    for (const [metadataId, metadataEntry] of Object.entries(artifact.metadata ?? {})) {\n      newMetadata[metadataId] = metadataEntry.map((meta: MetadataEntry) => {\n        if (meta.type === 'aws:cdk:logicalId' && trace && meta.data) {\n          const traceData = trace.get(meta.data.toString());\n          if (traceData) {\n            trace.delete(meta.data.toString());\n            return {\n              type: meta.type,\n              data: meta.data,\n              trace: [traceData],\n            };\n          }\n        }\n        // return metadata without the trace data\n        return {\n          type: meta.type,\n          data: meta.data,\n        };\n      });\n    }\n    if (trace && trace.size > 0) {\n      for (const [id, data] of trace.entries()) {\n        newMetadata[id] = [{\n          type: 'aws:cdk:logicalId',\n          data: id,\n          trace: [data],\n        }];\n      }\n    }\n    return newMetadata;\n  }\n\n  private renderArtifacts(trace?: ManifestTrace): { [id: string]: ArtifactManifest } | undefined {\n    const newArtifacts: { [id: string]: ArtifactManifest } = {};\n    for (const [artifactId, artifact] of Object.entries(this.manifest.artifacts ?? {})) {\n      let stackTrace: StackTrace | undefined = undefined;\n      if (artifact.type === ArtifactType.AWS_CLOUDFORMATION_STACK && trace) {\n        stackTrace = trace.get(artifactId);\n      }\n      newArtifacts[artifactId] = {\n        ...artifact,\n        metadata: this.renderArtifactMetadata(artifact, stackTrace),\n      };\n    }\n    return newArtifacts;\n  }\n}\n"]}
@@ -156,7 +156,7 @@ class IntegRunner {
156
156
  const manifest = cloud_assembly_1.AssemblyManifestReader.fromPath(this.snapshotDir);
157
157
  const assets = utils_1.flatten(stacks.map(stack => {
158
158
  var _a;
159
- return (_a = manifest.getAssetsForStack(stack)) !== null && _a !== void 0 ? _a : [];
159
+ return (_a = manifest.getAssetLocationsForStack(stack)) !== null && _a !== void 0 ? _a : [];
160
160
  }));
161
161
  assets.forEach(asset => {
162
162
  const fileName = path.join(this.snapshotDir, asset);
@@ -306,4 +306,4 @@ exports.DEFAULT_SYNTH_OPTIONS = {
306
306
  CDK_INTEG_REGION: 'test-region',
307
307
  },
308
308
  };
309
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"runner-base.js","sourceRoot":"","sources":["runner-base.ts"],"names":[],"mappings":";;;AAAA,6BAA6B;AAE7B,4CAAmK;AACnK,qDAAsD;AACtD,+BAA+B;AAC/B,oCAAmC;AAEnC,yDAA0E;AAE1E,6DAAiF;AAEjF,MAAM,mBAAmB,GAAG,wBAAwB,CAAC;AAgDrD;;GAEG;AACH;;GAEG;AACH,MAAsB,WAAW;IAwE/B,YAAY,OAA2B;;QAtBvC;;WAEG;QACgB,gBAAW,GAAsB;YAClD,YAAY,EAAE,KAAK;YACnB,aAAa,EAAE,KAAK;YACpB,gBAAgB,EAAE,KAAK;SACxB,CAAA;QAgBC,IAAI,CAAC,IAAI,GAAG,OAAO,CAAC,IAAI,CAAC;QACzB,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC;QACrC,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC;QACnC,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC;QACzC,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,kBAAkB,CAAC,CAAC;QAEpE,IAAI,CAAC,GAAG,SAAG,OAAO,CAAC,GAAG,mCAAI,IAAI,+BAAa,CAAC;YAC1C,SAAS,EAAE,IAAI,CAAC,SAAS;YACzB,UAAU,EAAE,OAAO,CAAC,UAAU;YAC9B,GAAG,EAAE;gBACH,GAAG,OAAO,CAAC,GAAG;aACf;SACF,CAAC,CAAC;QACH,IAAI,CAAC,SAAS,SAAG,OAAO,CAAC,WAAW,mCAAI,IAAI,CAAC,IAAI,CAAC,kBAAkB,CAAC;QACrE,IAAI,CAAC,MAAM,GAAG,QAAQ,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,EAAE,CAAC;QAC1E,IAAI,CAAC,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC;QAC/B,IAAI,IAAI,CAAC,WAAW,EAAE,EAAE;YACtB,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,YAAY,EAAE,CAAC;SAC9C;QACD,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,sBAAsB,EAAE,CAAC;IACvD,CAAC;IAED;;OAEG;IACI,aAAa;;QAClB,aAAO,IAAI,CAAC,iBAAiB,0CAAE,SAAS,CAAC;IAC3C,CAAC;IAED;;OAEG;IACI,WAAW;QAChB,OAAO,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC;IACxC,CAAC;IAED;;;;OAIG;IACI,sBAAsB;;QAC3B,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC;YACjB,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC;YAC/B,GAAG,EAAE;gBACH,GAAG,6BAAqB,CAAC,GAAG;gBAC5B,oFAAoF;gBACpF,6EAA6E;gBAC7E,gBAAgB,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,OAAC,IAAI,CAAC,iBAAiB,0CAAE,YAAY,CAAC,CAAC;aACxF;YACD,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC;SACtD,CAAC,CAAC;QACH,OAAO,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAC3C,CAAC;IAED;;OAEG;IACI,WAAW;QAChB,OAAO,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;IACzC,CAAC;IAED;;;;;;OAMG;IACO,YAAY,CAAC,GAAY;QACjC,IAAI;YACF,MAAM,SAAS,GAAG,iCAAc,CAAC,QAAQ,CAAC,GAAG,aAAH,GAAG,cAAH,GAAG,GAAI,IAAI,CAAC,WAAW,CAAC,CAAC;YACnE,OAAO,SAAS,CAAC;SAClB;QAAC,OAAO,CAAC,EAAE;YACV,MAAM,SAAS,GAAG,uCAAoB,CAAC,UAAU,CAAC;gBAChD,GAAG,EAAE,IAAI,CAAC,GAAG;gBACb,QAAQ,EAAE,IAAI,CAAC,IAAI,CAAC,kBAAkB;gBACtC,mBAAmB,EAAE,IAAI,CAAC,IAAI,CAAC,QAAQ;gBACvC,WAAW,EAAE;oBACX,GAAG,IAAI,CAAC,WAAW;oBACnB,GAAG,EAAE,IAAI;oBACT,GAAG,EAAE,IAAI,CAAC,MAAM;oBAChB,OAAO,EAAE,IAAI,CAAC,OAAO;oBACrB,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,SAAS,CAAC;iBACtD;aACF,CAAC,CAAC;YACH,IAAI,CAAC,aAAa,GAAG,uCAAoB,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC/E,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;YACzB,OAAO,SAAS,CAAC;SAClB;IACH,CAAC;IAES,OAAO;QACf,MAAM,UAAU,GAAG,IAAI,CAAC,SAAS,CAAC;QAClC,IAAI,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE;YAC7B,EAAE,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;SAC3B;IACH,CAAC;IAED;;;OAGG;IACK,eAAe;;QACrB,MAAM,SAAS,GAAkB,IAAI,GAAG,EAAE,CAAC;QAC3C,MAAM,kBAAkB,SAAG,IAAI,CAAC,mBAAmB,mCAAI,EAAE,CAAC;QAC1D,kBAAkB,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;YAClC,MAAM,KAAK,GAAG,SAAS,CAAC,GAAG,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;YAC9C,IAAI,KAAK,EAAE;gBACT,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,SAAS,EAAE,GAAG,mBAAmB,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC,CAAC;aACxE;iBAAM;gBACL,SAAS,CAAC,GAAG,CAAC,MAAM,CAAC,SAAS,EAAE,IAAI,GAAG,CAAC;oBACtC,CAAC,MAAM,CAAC,SAAS,EAAE,GAAG,mBAAmB,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;iBAC9D,CAAC,CAAC,CAAC;aACL;QACH,CAAC,CAAC,CAAC;QACH,OAAO,SAAS,CAAC;IACnB,CAAC;IAED;;;;;;;;OAQG;IACO,wBAAwB;;QAChC,MAAM,MAAM,SAAG,IAAI,CAAC,eAAe,CAAC,8BAA8B,EAAE,mCAAI,EAAE,CAAC;QAC3E,MAAM,QAAQ,GAAG,uCAAsB,CAAC,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACnE,MAAM,MAAM,GAAG,eAAO,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE;;YACxC,aAAO,QAAQ,CAAC,iBAAiB,CAAC,KAAK,CAAC,mCAAI,EAAE,CAAC;QACjD,CAAC,CAAC,CAAC,CAAC;QAEJ,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;YACrB,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,KAAK,CAAC,CAAC;YACpD,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,EAAE;gBAC3B,IAAI,EAAE,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,WAAW,EAAE,EAAE;oBACxC,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;iBACzB;qBAAM;oBACL,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;iBACzB;aACF;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;OAIG;IACO,6BAA6B;QACrC,MAAM,KAAK,GAAG,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAC/C,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;YACnB,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,CAAC;YACnD,IAAI,EAAE,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,WAAW,EAAE,IAAI,IAAI,KAAK,QAAQ,EAAE;gBAC7D,EAAE,CAAC,YAAY,CAAC,QAAQ,CAAC,CAAC;gBAC1B,EAAE,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;aACxB;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;;;;;OASG;IACO,cAAc;QACtB,IAAI,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE;YACnC,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;SACjC;QAED,qDAAqD;QACrD,oDAAoD;QACpD,IAAI,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE;YACtC,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC;gBACjB,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC;gBAC/B,GAAG,EAAE;oBACH,GAAG,6BAAqB,CAAC,GAAG;oBAC5B,gBAAgB,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;iBACpD;gBACD,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,WAAW,CAAC;aACxD,CAAC,CAAC;SACJ;aAAM;YACL,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,SAAS,EAAE,IAAI,CAAC,WAAW,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;SACpE;QAED,IAAI,CAAC,eAAe,EAAE,CAAC;IACzB,CAAC;IAED;;;;OAIG;IACK,eAAe;QACrB,IAAI,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE;YACnC,IAAI,CAAC,wBAAwB,EAAE,CAAC;YAChC,IAAI,CAAC,6BAA6B,EAAE,CAAC;YACrC,MAAM,QAAQ,GAAG,uCAAsB,CAAC,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACnE,QAAQ,CAAC,aAAa,EAAE,CAAC;YACzB,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,eAAe,EAAE,CAAC,CAAC;SAC9C;QAED,yDAAyD;QACzD,sDAAsD;QACtD,yEAAyE;QACzE,gBAAgB;QAChB,IAAI,IAAI,CAAC,eAAe,CAAC,IAAI,KAAK,mBAAmB,EAAE;YACpD,IAAI,CAAC,eAAwC,CAAC,YAAY,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC;SACnG;IACH,CAAC;IAES,UAAU,CAAC,iBAAuC;;QAC1D,MAAM,WAAW,GAA2B,EAAE,CAAC;QAC/C,MAAM,CAAC,OAAO,CAAC,qBAAY,CAAC;aACzB,MAAM,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,6BAAoB,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC;aACrD,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE,CAAC,WAAW,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;QAE3C,MAAM,aAAa,SAAG,OAAC,IAAI,CAAC,eAAe,mCAAI,IAAI,CAAC,iBAAiB,CAAC,0CAAE,aAAa,CAAC;QACtF,OAAO;YACL,+CAA+C;YAC/C,2BAA2B;YAC3B,GAAG,aAAa,CAAC,CAAC,CAAC,6BAAqB,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE;YACrD,0DAA0D;YAC1D,sBAAsB;YACtB,CAAC,0CAAiC,CAAC,EAAE,EAAE;YACvC,GAAG,WAAW;YACd,GAAG,IAAI,CAAC,aAAa;YACrB,GAAG,iBAAiB;SACrB,CAAC;IACJ,CAAC;CACF;AAvTD,kCAuTC;AAGD,2EAA2E;AAC3E,kCAAkC;AACrB,QAAA,qBAAqB,GAAG;IACnC,OAAO,EAAE;QACP,CAAC,+CAAsC,CAAC,EAAE,CAAC,gBAAgB,EAAE,gBAAgB,EAAE,gBAAgB,CAAC;QAChG,wDAAwD,EAAE,CAAC,gBAAgB,EAAE,gBAAgB,EAAE,gBAAgB,CAAC;QAChH,oHAAoH,EAAE,UAAU;QAChI,qHAAqH,EAAE,UAAU;QACjI,+GAA+G,EAAE,0BAA0B;QAC3I,mCAAmC;QACnC,kJAAkJ,EAAE,UAAU;QAC9J,qGAAqG,EAAE;YACrG,KAAK,EAAE,cAAc;YACrB,YAAY,EAAE;gBACZ;oBACE,IAAI,EAAE,QAAQ;oBACd,IAAI,EAAE,QAAQ;oBACd,OAAO,EAAE;wBACP;4BACE,QAAQ,EAAE,iBAAiB;4BAC3B,gBAAgB,EAAE,YAAY;4BAC9B,YAAY,EAAE,cAAc;yBAC7B;wBACD;4BACE,QAAQ,EAAE,iBAAiB;4BAC3B,gBAAgB,EAAE,YAAY;4BAC9B,YAAY,EAAE,cAAc;yBAC7B;wBACD;4BACE,QAAQ,EAAE,iBAAiB;4BAC3B,gBAAgB,EAAE,YAAY;4BAC9B,YAAY,EAAE,cAAc;yBAC7B;qBACF;iBACF;aACF;SACF;QAED,qFAAqF;QACrF,uFAAuF;QACvF,0FAA0F;QAC1F,gFAAgF;QAChF,CAAC,0BAAiB,CAAC,EAAE,CAAC,KAAK,EAAE,QAAQ,CAAC;KACvC;IACD,GAAG,EAAE;QACH,iBAAiB,EAAE,UAAU;QAC7B,gBAAgB,EAAE,aAAa;KAChC;CACF,CAAC","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"]}
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.getAssetLocationsForStack(stack) ?? [];\n    }));\n\n    assets.forEach(asset => {\n      const fileName = path.join(this.snapshotDir, asset);\n      if (fs.existsSync(fileName)) {\n        if (fs.lstatSync(fileName).isDirectory()) {\n          fs.removeSync(fileName);\n        } else {\n          fs.unlinkSync(fileName);\n        }\n      }\n    });\n  }\n\n  /**\n   * Remove the asset cache (.cache/) files from the snapshot.\n   * These are a cache of the asset zips, but we are fine with\n   * re-zipping on deploy\n   */\n  protected removeAssetsCacheFromSnapshot(): void {\n    const files = fs.readdirSync(this.snapshotDir);\n    files.forEach(file => {\n      const fileName = path.join(this.snapshotDir, file);\n      if (fs.lstatSync(fileName).isDirectory() && file === '.cache') {\n        fs.emptyDirSync(fileName);\n        fs.rmdirSync(fileName);\n      }\n    });\n  }\n\n  /**\n   * Create the new snapshot.\n   *\n   * If lookups are enabled, then we need create the snapshot by 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"]}
@@ -33,4 +33,11 @@ export declare class IntegSnapshotRunner extends IntegRunner {
33
33
  */
34
34
  private diffAssembly;
35
35
  private readAssembly;
36
+ /**
37
+ * Reduce template to a normal form where asset references have been normalized
38
+ *
39
+ * This makes it possible to compare templates if all that's different between
40
+ * them is the hashes of the asset values.
41
+ */
42
+ private canonicalizeTemplate;
36
43
  }