UNPKG

@aws-cdk/integ-runner

Version:

CDK Integration Testing Tool

204 lines 26.3 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.LegacyIntegTestSuite = exports.IntegTestSuite = void 0; const osPath = require("path"); const cloud_assembly_schema_1 = require("@aws-cdk/cloud-assembly-schema"); const fs = require("fs-extra"); const integ_manifest_1 = require("./private/integ-manifest"); const CDK_INTEG_STACK_PRAGMA = '/// !cdk-integ'; const PRAGMA_PREFIX = 'pragma:'; const SET_CONTEXT_PRAGMA_PREFIX = 'pragma:set-context:'; const VERIFY_ASSET_HASHES = 'pragma:include-assets-hashes'; const DISABLE_UPDATE_WORKFLOW = 'pragma:disable-update-workflow'; const ENABLE_LOOKUPS_PRAGMA = 'pragma:enable-lookups'; /** * Helper class for working with Integration tests * This requires an `integ.json` file in the snapshot * directory. For legacy test cases use LegacyIntegTestCases */ class IntegTestSuite { enableLookups; testSuite; synthContext; /** * Loads integ tests from a snapshot directory */ static fromPath(path) { const reader = integ_manifest_1.IntegManifestReader.fromPath(path); return new IntegTestSuite(reader.tests.enableLookups, reader.tests.testCases, reader.tests.synthContext); } type = 'test-suite'; constructor(enableLookups, testSuite, synthContext) { this.enableLookups = enableLookups; this.testSuite = testSuite; this.synthContext = synthContext; } /** * Returns a list of stacks that have stackUpdateWorkflow disabled */ getStacksWithoutUpdateWorkflow() { return Object.values(this.testSuite) .filter(testCase => !(testCase.stackUpdateWorkflow ?? true)) .flatMap((testCase) => testCase.stacks); } /** * Returns test case options for a given stack */ getOptionsForStack(stackId) { for (const testCase of Object.values(this.testSuite ?? {})) { if (testCase.stacks.includes(stackId)) { return { hooks: testCase.hooks, regions: testCase.regions, diffAssets: testCase.diffAssets ?? false, allowDestroy: testCase.allowDestroy, cdkCommandOptions: testCase.cdkCommandOptions, stackUpdateWorkflow: testCase.stackUpdateWorkflow ?? true, }; } } return undefined; } /** * Get a list of stacks in the test suite */ get stacks() { return Object.values(this.testSuite).flatMap(testCase => testCase.stacks); } } exports.IntegTestSuite = IntegTestSuite; /** * Helper class for creating an integ manifest for legacy * test cases, i.e. tests without a `integ.json`. */ class LegacyIntegTestSuite extends IntegTestSuite { enableLookups; testSuite; synthContext; /** * Returns the single test stack to use. * * If the test has a single stack, it will be chosen. Otherwise a pragma is expected within the * test file the name of the stack: * * @example * * /// !cdk-integ <stack-name> * */ static async fromLegacy(config) { const pragmas = this.pragmas(config.integSourceFilePath); const tests = { stacks: [], diffAssets: pragmas.includes(VERIFY_ASSET_HASHES), stackUpdateWorkflow: !pragmas.includes(DISABLE_UPDATE_WORKFLOW), }; const pragma = this.readStackPragma(config.integSourceFilePath); if (pragma.length > 0) { tests.stacks.push(...pragma); } else { const options = { ...config.listOptions, notices: false, }; const stacks = await config.cdk.list(options); if (stacks.length !== 1) { throw new Error('"integ-runner" can only operate on apps with a single stack.\n\n' + ' If your app has multiple stacks, specify which stack to select by adding this to your test source:\n\n' + ` ${CDK_INTEG_STACK_PRAGMA} STACK ...\n\n` + ` Available stacks: ${stacks.join(' ')} (wildcards are also supported)\n`); } if (stacks.length === 1 && stacks[0] === '') { throw new Error(`No stack found for test ${config.testName}`); } tests.stacks.push(...stacks); } return new LegacyIntegTestSuite(pragmas.includes(ENABLE_LOOKUPS_PRAGMA), { [config.testName]: tests, }, LegacyIntegTestSuite.getPragmaContext(config.integSourceFilePath)); } static getPragmaContext(integSourceFilePath) { const ctxPragmaContext = {}; // apply context from set-context pragma // usage: pragma:set-context:key=value const ctxPragmas = (this.pragmas(integSourceFilePath)).filter(p => p.startsWith(SET_CONTEXT_PRAGMA_PREFIX)); for (const p of ctxPragmas) { const instruction = p.substring(SET_CONTEXT_PRAGMA_PREFIX.length); const [key, value] = instruction.split('='); if (key == null || value == null) { throw new Error(`invalid "set-context" pragma syntax. example: "pragma:set-context:@aws-cdk/core:newStyleStackSynthesis=true" got: ${p}`); } ctxPragmaContext[key] = value; } return { ...ctxPragmaContext, }; } /** * Reads stack names from the "!cdk-integ" pragma. * * Every word that's NOT prefixed by "pragma:" is considered a stack name. * * @example * * /// !cdk-integ <stack-name> */ static readStackPragma(integSourceFilePath) { return (this.readIntegPragma(integSourceFilePath)).filter(p => !p.startsWith(PRAGMA_PREFIX)); } /** * Read arbitrary cdk-integ pragma directives * * Reads the test source file and looks for the "!cdk-integ" pragma. If it exists, returns it's * contents. This allows integ tests to supply custom command line arguments to "cdk deploy" and "cdk synth". * * @example * * /// !cdk-integ [...] */ static readIntegPragma(integSourceFilePath) { const source = fs.readFileSync(integSourceFilePath, { encoding: 'utf-8' }); const pragmaLine = source.split('\n').find(x => x.startsWith(CDK_INTEG_STACK_PRAGMA + ' ')); if (!pragmaLine) { return []; } const args = pragmaLine.substring(CDK_INTEG_STACK_PRAGMA.length).trim().split(' '); if (args.length === 0) { throw new Error(`Invalid syntax for cdk-integ pragma. Usage: "${CDK_INTEG_STACK_PRAGMA} [STACK] [pragma:PRAGMA] [...]"`); } return args; } /** * Return the non-stack pragmas * * These are all pragmas that start with "pragma:". * * For backwards compatibility reasons, all pragmas that DON'T start with this * string are considered to be stack names. */ static pragmas(integSourceFilePath) { return (this.readIntegPragma(integSourceFilePath)).filter(p => p.startsWith(PRAGMA_PREFIX)); } type = 'legacy-test-suite'; constructor(enableLookups, testSuite, synthContext) { super(enableLookups, testSuite); this.enableLookups = enableLookups; this.testSuite = testSuite; this.synthContext = synthContext; } /** * Save the integ manifest to a directory */ saveManifest(directory, context) { const manifest = { version: cloud_assembly_schema_1.Manifest.version(), testCases: this.testSuite, synthContext: context, enableLookups: this.enableLookups, }; cloud_assembly_schema_1.Manifest.saveIntegManifest(manifest, osPath.join(directory, integ_manifest_1.IntegManifestReader.DEFAULT_FILENAME)); } } exports.LegacyIntegTestSuite = LegacyIntegTestSuite; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"integ-test-suite.js","sourceRoot":"","sources":["integ-test-suite.ts"],"names":[],"mappings":"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","sourcesContent":["import * as osPath from 'path';\nimport type { TestCase, TestOptions, IntegManifest } from '@aws-cdk/cloud-assembly-schema';\nimport { Manifest } from '@aws-cdk/cloud-assembly-schema';\nimport * as fs from 'fs-extra';\nimport type { ICdk, ListOptions } from '../engines/cdk-interface';\nimport { IntegManifestReader } from './private/integ-manifest';\n\nconst CDK_INTEG_STACK_PRAGMA = '/// !cdk-integ';\nconst PRAGMA_PREFIX = 'pragma:';\nconst SET_CONTEXT_PRAGMA_PREFIX = 'pragma:set-context:';\nconst VERIFY_ASSET_HASHES = 'pragma:include-assets-hashes';\nconst DISABLE_UPDATE_WORKFLOW = 'pragma:disable-update-workflow';\nconst ENABLE_LOOKUPS_PRAGMA = 'pragma:enable-lookups';\n\n/**\n * Represents an integration test\n */\nexport type TestSuite = { [testName: string]: TestCase };\n\nexport type TestSuiteType = 'test-suite' | 'legacy-test-suite';\n\n/**\n * Helper class for working with Integration tests\n * This requires an `integ.json` file in the snapshot\n * directory. For legacy test cases use LegacyIntegTestCases\n */\nexport class IntegTestSuite {\n  /**\n   * Loads integ tests from a snapshot directory\n   */\n  public static fromPath(path: string): IntegTestSuite {\n    const reader = IntegManifestReader.fromPath(path);\n    return new IntegTestSuite(\n      reader.tests.enableLookups,\n      reader.tests.testCases,\n      reader.tests.synthContext,\n    );\n  }\n\n  public readonly type: TestSuiteType = 'test-suite';\n\n  constructor(\n    public readonly enableLookups: boolean,\n    public readonly testSuite: TestSuite,\n    public readonly synthContext?: { [name: string]: string },\n  ) {\n  }\n\n  /**\n   * Returns a list of stacks that have stackUpdateWorkflow disabled\n   */\n  public getStacksWithoutUpdateWorkflow(): string[] {\n    return Object.values(this.testSuite)\n      .filter(testCase => !(testCase.stackUpdateWorkflow ?? true))\n      .flatMap((testCase: TestCase) => testCase.stacks);\n  }\n\n  /**\n   * Returns test case options for a given stack\n   */\n  public getOptionsForStack(stackId: string): TestOptions | undefined {\n    for (const testCase of Object.values(this.testSuite ?? {})) {\n      if (testCase.stacks.includes(stackId)) {\n        return {\n          hooks: testCase.hooks,\n          regions: testCase.regions,\n          diffAssets: testCase.diffAssets ?? false,\n          allowDestroy: testCase.allowDestroy,\n          cdkCommandOptions: testCase.cdkCommandOptions,\n          stackUpdateWorkflow: testCase.stackUpdateWorkflow ?? true,\n        };\n      }\n    }\n    return undefined;\n  }\n\n  /**\n   * Get a list of stacks in the test suite\n   */\n  public get stacks(): string[] {\n    return Object.values(this.testSuite).flatMap(testCase => testCase.stacks);\n  }\n}\n\n/**\n * Options for a reading a legacy test case manifest\n */\nexport interface LegacyTestCaseConfig {\n  /**\n   * The name of the test case\n   */\n  readonly testName: string;\n\n  /**\n   * Options to use when performing `cdk list`\n   * This is used to determine the name of the stacks\n   * in the test case\n   */\n  readonly listOptions: ListOptions;\n\n  /**\n   * An instance of the CDK CLI (e.g. CdkCliWrapper)\n   */\n  readonly cdk: ICdk;\n\n  /**\n   * The path to the integration test file\n   * i.e. integ.test.js\n   */\n  readonly integSourceFilePath: string;\n}\n\n/**\n * Helper class for creating an integ manifest for legacy\n * test cases, i.e. tests without a `integ.json`.\n */\nexport class LegacyIntegTestSuite extends IntegTestSuite {\n  /**\n   * Returns the single test stack to use.\n   *\n   * If the test has a single stack, it will be chosen. Otherwise a pragma is expected within the\n   * test file the name of the stack:\n   *\n   * @example\n   *\n   *    /// !cdk-integ <stack-name>\n   *\n   */\n  public static async fromLegacy(config: LegacyTestCaseConfig): Promise<LegacyIntegTestSuite> {\n    const pragmas = this.pragmas(config.integSourceFilePath);\n    const tests: TestCase = {\n      stacks: [],\n      diffAssets: pragmas.includes(VERIFY_ASSET_HASHES),\n      stackUpdateWorkflow: !pragmas.includes(DISABLE_UPDATE_WORKFLOW),\n    };\n    const pragma = this.readStackPragma(config.integSourceFilePath);\n    if (pragma.length > 0) {\n      tests.stacks.push(...pragma);\n    } else {\n      const options: ListOptions = {\n        ...config.listOptions,\n        notices: false,\n      };\n      const stacks = await config.cdk.list(options);\n      if (stacks.length !== 1) {\n        throw new Error('\"integ-runner\" can only operate on apps with a single stack.\\n\\n' +\n          '  If your app has multiple stacks, specify which stack to select by adding this to your test source:\\n\\n' +\n          `      ${CDK_INTEG_STACK_PRAGMA} STACK ...\\n\\n` +\n          `  Available stacks: ${stacks.join(' ')} (wildcards are also supported)\\n`);\n      }\n      if (stacks.length === 1 && stacks[0] === '') {\n        throw new Error(`No stack found for test ${config.testName}`);\n      }\n      tests.stacks.push(...stacks);\n    }\n\n    return new LegacyIntegTestSuite(\n      pragmas.includes(ENABLE_LOOKUPS_PRAGMA),\n      {\n        [config.testName]: tests,\n      },\n      LegacyIntegTestSuite.getPragmaContext(config.integSourceFilePath),\n    );\n  }\n\n  public static getPragmaContext(integSourceFilePath: string): Record<string, any> {\n    const ctxPragmaContext: Record<string, any> = {};\n\n    // apply context from set-context pragma\n    // usage: pragma:set-context:key=value\n    const ctxPragmas = (this.pragmas(integSourceFilePath)).filter(p => p.startsWith(SET_CONTEXT_PRAGMA_PREFIX));\n    for (const p of ctxPragmas) {\n      const instruction = p.substring(SET_CONTEXT_PRAGMA_PREFIX.length);\n      const [key, value] = instruction.split('=');\n      if (key == null || value == null) {\n        throw new Error(`invalid \"set-context\" pragma syntax. example: \"pragma:set-context:@aws-cdk/core:newStyleStackSynthesis=true\" got: ${p}`);\n      }\n\n      ctxPragmaContext[key] = value;\n    }\n    return {\n      ...ctxPragmaContext,\n    };\n  }\n\n  /**\n   * Reads stack names from the \"!cdk-integ\" pragma.\n   *\n   * Every word that's NOT prefixed by \"pragma:\" is considered a stack name.\n   *\n   * @example\n   *\n   *    /// !cdk-integ <stack-name>\n   */\n  private static readStackPragma(integSourceFilePath: string): string[] {\n    return (this.readIntegPragma(integSourceFilePath)).filter(p => !p.startsWith(PRAGMA_PREFIX));\n  }\n\n  /**\n   * Read arbitrary cdk-integ pragma directives\n   *\n   * Reads the test source file and looks for the \"!cdk-integ\" pragma. If it exists, returns it's\n   * contents. This allows integ tests to supply custom command line arguments to \"cdk deploy\" and \"cdk synth\".\n   *\n   * @example\n   *\n   *    /// !cdk-integ [...]\n   */\n  private static readIntegPragma(integSourceFilePath: string): string[] {\n    const source = fs.readFileSync(integSourceFilePath, { encoding: 'utf-8' });\n    const pragmaLine = source.split('\\n').find(x => x.startsWith(CDK_INTEG_STACK_PRAGMA + ' '));\n    if (!pragmaLine) {\n      return [];\n    }\n\n    const args = pragmaLine.substring(CDK_INTEG_STACK_PRAGMA.length).trim().split(' ');\n    if (args.length === 0) {\n      throw new Error(`Invalid syntax for cdk-integ pragma. Usage: \"${CDK_INTEG_STACK_PRAGMA} [STACK] [pragma:PRAGMA] [...]\"`);\n    }\n    return args;\n  }\n\n  /**\n   * Return the non-stack pragmas\n   *\n   * These are all pragmas that start with \"pragma:\".\n   *\n   * For backwards compatibility reasons, all pragmas that DON'T start with this\n   * string are considered to be stack names.\n   */\n  private static pragmas(integSourceFilePath: string): string[] {\n    return (this.readIntegPragma(integSourceFilePath)).filter(p => p.startsWith(PRAGMA_PREFIX));\n  }\n\n  public readonly type: TestSuiteType = 'legacy-test-suite';\n\n  constructor(\n    public readonly enableLookups: boolean,\n    public readonly testSuite: TestSuite,\n    public readonly synthContext?: { [name: string]: string },\n  ) {\n    super(enableLookups, testSuite);\n  }\n\n  /**\n   * Save the integ manifest to a directory\n   */\n  public saveManifest(directory: string, context?: Record<string, any>): void {\n    const manifest: IntegManifest = {\n      version: Manifest.version(),\n      testCases: this.testSuite,\n      synthContext: context,\n      enableLookups: this.enableLookups,\n    };\n    Manifest.saveIntegManifest(manifest, osPath.join(directory, IntegManifestReader.DEFAULT_FILENAME));\n  }\n}\n"]}