@aws-cdk/integ-runner
Version:
CDK Integration Testing Tool
130 lines • 22.6 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", { value: true });
exports.DiagnosticReason = void 0;
exports.printSummary = printSummary;
exports.formatAssertionResults = formatAssertionResults;
exports.printResults = printResults;
exports.printLaggards = printLaggards;
exports.formatError = formatError;
const util_1 = require("util");
const chalk = require("chalk");
const logger = require("../logger");
/**
* Represents possible reasons for a diagnostic
*/
var DiagnosticReason;
(function (DiagnosticReason) {
/**
* The integration test failed because there
* is not existing snapshot
*/
DiagnosticReason["NO_SNAPSHOT"] = "NO_SNAPSHOT";
/**
* The integration test failed
*/
DiagnosticReason["TEST_FAILED"] = "TEST_FAILED";
/**
* There was an error running the integration test
*/
DiagnosticReason["TEST_ERROR"] = "TEST_ERROR";
/**
* A non-failing warning from the integration test run
*/
DiagnosticReason["TEST_WARNING"] = "TEST_WARNING";
/**
* The snapshot test failed because the actual
* snapshot was different than the expected snapshot
*/
DiagnosticReason["SNAPSHOT_FAILED"] = "SNAPSHOT_FAILED";
/**
* The snapshot test failed because there was an error executing it
*/
DiagnosticReason["SNAPSHOT_ERROR"] = "SNAPSHOT_ERROR";
/**
* The snapshot test succeeded
*/
DiagnosticReason["SNAPSHOT_SUCCESS"] = "SNAPSHOT_SUCCESS";
/**
* The integration test succeeded
*/
DiagnosticReason["TEST_SUCCESS"] = "TEST_SUCCESS";
/**
* The assertion failed
*/
DiagnosticReason["ASSERTION_FAILED"] = "ASSERTION_FAILED";
})(DiagnosticReason || (exports.DiagnosticReason = DiagnosticReason = {}));
function printSummary(total, failed) {
if (failed > 0) {
logger.print('%s: %s %s, %s total', chalk.bold('Tests'), chalk.red(failed), chalk.red('failed'), total);
}
else {
logger.print('%s: %s %s, %s total', chalk.bold('Tests'), chalk.green(total), chalk.green('passed'), total);
}
}
/**
* Format the assertion results so that the results can be
* printed
*/
function formatAssertionResults(results) {
return Object.entries(results)
.map(([id, result]) => (0, util_1.format)('%s%s', id, result.status === 'success' ? ` - ${result.status}` : `\n${result.message}`))
.join('\n ');
}
/**
* Print out the results from tests
*/
function printResults(diagnostic) {
switch (diagnostic.reason) {
case DiagnosticReason.SNAPSHOT_SUCCESS:
logger.success(' UNCHANGED %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));
break;
case DiagnosticReason.TEST_SUCCESS:
logger.success(' SUCCESS %s %s\n ', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));
break;
case DiagnosticReason.NO_SNAPSHOT:
logger.error(' NEW %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));
break;
case DiagnosticReason.SNAPSHOT_FAILED:
logger.error(' CHANGED %s %s\n%s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), indentLines(diagnostic.message, 6));
break;
case DiagnosticReason.TEST_WARNING:
logger.warning(' WARN %s %s\n%s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), indentLines(diagnostic.message, 6));
break;
case DiagnosticReason.SNAPSHOT_ERROR:
case DiagnosticReason.TEST_ERROR:
logger.error(' ERROR %s %s\n%s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), indentLines(diagnostic.message, 6));
break;
case DiagnosticReason.TEST_FAILED:
logger.error(' FAILED %s %s\n%s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), indentLines(diagnostic.message, 6));
break;
case DiagnosticReason.ASSERTION_FAILED:
logger.error(' ASSERT %s %s\n%s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), indentLines(diagnostic.message, 6));
break;
}
for (const addl of diagnostic.additionalMessages ?? []) {
logger.print(` ${addl}`);
}
}
/**
* Takes a multiline string and indents every line with the same number of spaces.
*/
function indentLines(message, count = 2) {
return message.split('\n').map(line => ' '.repeat(count) + line).join('\n');
}
function printLaggards(testNames) {
const parts = [
' ',
`Waiting for ${testNames.size} more`,
testNames.size < 10 ? ['(', Array.from(testNames).join(', '), ')'].join('') : '',
];
logger.print(chalk.grey(parts.filter(x => x).join(' ')));
}
function formatError(error) {
const name = error.name || 'Error';
const message = error.message || String(error);
if (error.cause) {
return `${name}: ${message}\n${chalk.gray('Cause: ' + formatError(error.cause))}`;
}
return `${name}: ${message}`;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"common.js","sourceRoot":"","sources":["common.ts"],"names":[],"mappings":"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","sourcesContent":["import { format } from 'util';\nimport type { ResourceImpact } from '@aws-cdk/cloudformation-diff';\nimport * as chalk from 'chalk';\nimport * as logger from '../logger';\nimport type { IntegTestInfo } from '../runner/integration-tests';\n\n/**\n * The aggregate results from running assertions on a test case\n */\nexport type AssertionResults = { [id: string]: AssertionResult };\n\n/**\n * The result of an individual assertion\n */\nexport interface AssertionResult {\n  /**\n   * The assertion message. If the assertion failed, this will\n   * include the reason.\n   */\n  readonly message: string;\n\n  /**\n   * Whether the assertion succeeded or failed\n   */\n  readonly status: 'success' | 'fail';\n}\n\n/**\n * Config for an integration test\n */\nexport interface IntegTestWorkerConfig extends IntegTestInfo {\n  /**\n   * A list of any destructive changes\n   *\n   * @default []\n   */\n  readonly destructiveChanges?: DestructiveChange[];\n}\n\n/**\n * Information on any destructive changes\n */\nexport interface DestructiveChange {\n  /**\n   * The logicalId of the resource with a destructive change\n   */\n  readonly logicalId: string;\n\n  /**\n   * The name of the stack that contains the destructive change\n   */\n  readonly stackName: string;\n\n  /**\n   * The impact of the destructive change\n   */\n  readonly impact: ResourceImpact;\n}\n\n/**\n * Represents integration tests metrics for a given worker\n */\nexport interface IntegRunnerMetrics {\n  /**\n   * The region the test was run in\n   */\n  readonly region: string;\n\n  /**\n   * The total duration of the worker.\n   * This will be the sum of all individual test durations\n   */\n  readonly duration: number;\n\n  /**\n   * Contains the duration of individual tests that the\n   * worker executed.\n   *\n   * Map of testName to duration.\n   */\n  readonly tests: { [testName: string]: number };\n\n  /**\n   * The profile that was used to run the test\n   *\n   * @default - default profile\n   */\n  readonly profile?: string;\n}\n\nexport interface SnapshotVerificationOptions {\n  /**\n   * Retain failed snapshot comparisons\n   *\n   * @default false\n   */\n  readonly retain?: boolean;\n\n  /**\n   * Verbose mode\n   *\n   * @default false\n   */\n  readonly verbose?: boolean;\n}\n\n/**\n * Integration test results\n */\nexport interface IntegBatchResponse {\n  /**\n   * List of failed tests\n   */\n  readonly failedTests: IntegTestInfo[];\n\n  /**\n   * List of Integration test metrics. Each entry in the\n   * list represents metrics from a single worker (account + region).\n   */\n  readonly metrics: IntegRunnerMetrics[];\n}\n\n/**\n * Common options for running integration tests\n */\nexport interface IntegTestOptions {\n  /**\n   * A list of integration tests to run\n   * in this batch\n   */\n  readonly tests: IntegTestWorkerConfig[];\n\n  /**\n   * Whether or not to destroy the stacks at the\n   * end of the test\n   *\n   * @default true\n   */\n  readonly clean?: boolean;\n\n  /**\n   * When this is set to `true` the snapshot will\n   * be created _without_ running the integration test\n   * The resulting snapshot SHOULD NOT be checked in\n   *\n   * @default false\n   */\n  readonly dryRun?: boolean;\n\n  /**\n   * The level of verbosity for logging.\n   * Higher number means more output.\n   *\n   * @default 0\n   */\n  readonly verbosity?: number;\n\n  /**\n   * If this is set to true then the stack update workflow will be disabled\n   *\n   * @default true\n   */\n  readonly updateWorkflow?: boolean;\n\n  /**\n   * true if running in watch mode\n   *\n   * @default false\n   */\n  readonly watch?: boolean;\n\n  /**\n   * Use the indicated proxy\n   *\n   * @default - no proxy\n   */\n  readonly proxy?: string;\n\n  /**\n   * Path to CA certificate to use when validating HTTPS requests\n   *\n   * @default - no additional CA bundle\n   */\n  readonly caBundlePath?: string;\n}\n\n/**\n * Represents possible reasons for a diagnostic\n */\nexport enum DiagnosticReason {\n  /**\n   * The integration test failed because there\n   * is not existing snapshot\n   */\n  NO_SNAPSHOT = 'NO_SNAPSHOT',\n\n  /**\n   * The integration test failed\n   */\n  TEST_FAILED = 'TEST_FAILED',\n\n  /**\n   * There was an error running the integration test\n   */\n  TEST_ERROR = 'TEST_ERROR',\n\n  /**\n   * A non-failing warning from the integration test run\n   */\n  TEST_WARNING = 'TEST_WARNING',\n\n  /**\n   * The snapshot test failed because the actual\n   * snapshot was different than the expected snapshot\n   */\n  SNAPSHOT_FAILED = 'SNAPSHOT_FAILED',\n\n  /**\n   * The snapshot test failed because there was an error executing it\n   */\n  SNAPSHOT_ERROR = 'SNAPSHOT_ERROR',\n\n  /**\n   * The snapshot test succeeded\n   */\n  SNAPSHOT_SUCCESS = 'SNAPSHOT_SUCCESS',\n\n  /**\n   * The integration test succeeded\n   */\n  TEST_SUCCESS = 'TEST_SUCCESS',\n\n  /**\n   * The assertion failed\n   */\n  ASSERTION_FAILED = 'ASSERTION_FAILED',\n}\n\n/**\n * Integration test diagnostics\n * This is used to report back the status of each test\n */\nexport interface Diagnostic {\n  /**\n   * The name of the test\n   */\n  readonly testName: string;\n\n  /**\n   * The name of the stack\n   */\n  readonly stackName: string;\n\n  /**\n   * The diagnostic message\n   */\n  readonly message: string;\n\n  /**\n   * The time it took to run the test\n   */\n  readonly duration?: number;\n\n  /**\n   * The reason for the diagnostic\n   */\n  readonly reason: DiagnosticReason;\n\n  /**\n   * Additional messages to print\n   */\n  readonly additionalMessages?: string[];\n\n  /**\n   * Relevant config options that were used for the integ test\n   */\n  readonly config?: Record<string, any>;\n}\n\nexport function printSummary(total: number, failed: number): void {\n  if (failed > 0) {\n    logger.print('%s:    %s %s, %s total', chalk.bold('Tests'), chalk.red(failed), chalk.red('failed'), total);\n  } else {\n    logger.print('%s:    %s %s, %s total', chalk.bold('Tests'), chalk.green(total), chalk.green('passed'), total);\n  }\n}\n\n/**\n * Format the assertion results so that the results can be\n * printed\n */\nexport function formatAssertionResults(results: AssertionResults): string {\n  return Object.entries(results)\n    .map(([id, result]) => format('%s%s', id, result.status === 'success' ? ` - ${result.status}` : `\\n${result.message}`))\n    .join('\\n      ');\n}\n\n/**\n * Print out the results from tests\n */\nexport function printResults(diagnostic: Diagnostic): void {\n  switch (diagnostic.reason) {\n    case DiagnosticReason.SNAPSHOT_SUCCESS:\n      logger.success('  UNCHANGED  %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));\n      break;\n    case DiagnosticReason.TEST_SUCCESS:\n      logger.success('  SUCCESS    %s %s\\n      ', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));\n      break;\n    case DiagnosticReason.NO_SNAPSHOT:\n      logger.error('  NEW        %s %s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`));\n      break;\n    case DiagnosticReason.SNAPSHOT_FAILED:\n      logger.error('  CHANGED    %s %s\\n%s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), indentLines(diagnostic.message, 6));\n      break;\n    case DiagnosticReason.TEST_WARNING:\n      logger.warning('  WARN       %s %s\\n%s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), indentLines(diagnostic.message, 6));\n      break;\n    case DiagnosticReason.SNAPSHOT_ERROR:\n    case DiagnosticReason.TEST_ERROR:\n      logger.error('  ERROR      %s %s\\n%s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), indentLines(diagnostic.message, 6));\n      break;\n    case DiagnosticReason.TEST_FAILED:\n      logger.error('  FAILED     %s %s\\n%s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), indentLines(diagnostic.message, 6));\n      break;\n    case DiagnosticReason.ASSERTION_FAILED:\n      logger.error('  ASSERT     %s %s\\n%s', diagnostic.testName, chalk.gray(`${diagnostic.duration}s`), indentLines(diagnostic.message, 6));\n      break;\n  }\n  for (const addl of diagnostic.additionalMessages ?? []) {\n    logger.print(`      ${addl}`);\n  }\n}\n\n/**\n * Takes a multiline string and indents every line with the same number of spaces.\n */\nfunction indentLines(message: string, count = 2): string {\n  return message.split('\\n').map(line => ' '.repeat(count) + line).join('\\n');\n}\n\nexport function printLaggards(testNames: Set<string>) {\n  const parts = [\n    '  ',\n    `Waiting for ${testNames.size} more`,\n    testNames.size < 10 ? ['(', Array.from(testNames).join(', '), ')'].join('') : '',\n  ];\n\n  logger.print(chalk.grey(parts.filter(x => x).join(' ')));\n}\n\nexport function formatError(error: any): string {\n  const name = error.name || 'Error';\n  const message = error.message || String(error);\n\n  if (error.cause) {\n    return `${name}: ${message}\\n${chalk.gray('Cause: ' + formatError(error.cause))}`;\n  }\n\n  return `${name}: ${message}`;\n}\n"]}