@aws-cdk/integ-runner
Version:
CDK Integration Testing Tool
204 lines • 26.3 kB
JavaScript
;
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"]}