@synap-ac/node-dot-extra-reporter
Version:
Custom dot reporter for the node test runner
209 lines • 28 kB
JavaScript
import { Transform } from "node:stream";
import chalk from "chalk";
import test from "node:test";
var EventType;
(function (EventType) {
EventType["ENQUEUE"] = "test:enqueue";
EventType["DEQUEUE"] = "test:dequeue";
EventType["START"] = "test:start";
EventType["PASS"] = "test:pass";
EventType["FAIL"] = "test:fail";
EventType["PLAN"] = "test:plan";
EventType["WATCH_DRAINED"] = "test:watch:drained";
EventType["STDOUT"] = "test:stdout";
EventType["STDERR"] = "test:stderr";
})(EventType || (EventType = {}));
class DotExtraReporter extends Transform {
passedTests;
skippedTests;
todoTests;
totalDuration;
failedTests;
testSuite;
currentSuite;
currentSuiteLineage = [];
nesting = 0;
constructor(options = {}) {
super({ ...options, writableObjectMode: true });
this.passedTests = 0;
this.skippedTests = [];
this.todoTests = [];
this.totalDuration = 0;
this.failedTests = [];
this.testSuite = {};
}
handleTestStart(test) {
// Add new child
const newItem = {
file: test.file,
name: test.name,
nesting: test.nesting,
children: {},
};
// If nesting > current nesting then
// We've moved into a nested suite so:
// - Add the current suite to the lineage
// - Add new item as a child of the current suite
if (test.nesting === 0) {
// First entry
this.currentSuite = newItem;
this.testSuite[`${test.file}:${test.name}`] = newItem;
}
else if (test.nesting > this.nesting) {
this.currentSuiteLineage.push(this.currentSuite);
this.currentSuite.children[test.name] = newItem;
this.currentSuite = newItem;
}
else {
// Not deeper so add to current parent, and set new item to current
const parent = this.currentSuiteLineage.pop();
parent.children[test.name] = newItem;
this.currentSuiteLineage.push(parent);
this.currentSuite = newItem;
}
this.nesting = test.nesting;
}
handleTestResult(test, eventType) {
// Update test result details
this.currentSuite.passed = eventType === EventType.PASS;
if (test.details) {
this.currentSuite.details = test.details;
}
if (test.skip) {
process.stdout.write(chalk.yellow("*"));
this.currentSuite.skip = test.skip;
}
else if (test.todo) {
process.stdout.write(chalk.blue("-"));
this.currentSuite.todo = test.todo;
}
else if (this.currentSuite.passed) {
process.stdout.write(chalk.green("."));
}
else {
process.stdout.write(chalk.red("F"));
}
this.nesting = test.nesting;
}
handleTestEnd(test) {
// Suite finished, step up to parent
this.currentSuite = this.currentSuiteLineage.pop();
this.nesting = test.nesting;
}
_transform(event, _encoding, callback) {
try {
const test = event.data;
switch (event.type) {
case EventType.START:
this.handleTestStart(test);
break;
case EventType.PASS:
case EventType.FAIL:
if (test.name !== test.file) {
this.handleTestResult(test, event.type);
}
break;
case EventType.PLAN:
this.handleTestEnd(test);
break;
case EventType.STDOUT:
process.stdout.write(event.data.message ?? "");
break;
case EventType.STDERR:
process.stderr.write(event.data.message ?? "");
break;
default:
break;
}
callback();
}
catch (error) {
console.log(error);
callback(error);
}
}
processOutcomes() {
Object.entries(this.testSuite).forEach(([file, suite]) => {
this.buildStats(suite);
// File suite doesn't have a duration so collate from all top level suites.
Object.values(suite.children ?? {}).forEach((child) => {
this.totalDuration += child.details?.duration_ms || 0;
});
});
}
buildStats(suite, prefix) {
const isSuite = (obj) => obj.details?.type === "suite";
if (isSuite(suite)) {
// Iterate over each child
Object.values(suite.children).forEach((child) => this.buildStats(child, prefix ? `${prefix} > ${suite.name}` : suite.name));
}
if (prefix) {
suite.name = `${prefix} > ${suite.name}`;
}
if (suite.skip) {
this.skippedTests.push(suite);
}
else if (suite.todo) {
this.todoTests.push(suite);
}
else if (!isSuite(suite)) {
if (suite.passed === false) {
this.failedTests.push(suite);
}
else {
this.passedTests++;
}
}
}
_flush() {
this.processOutcomes();
if (this.skippedTests.length > 0) {
console.log("\n\nSkipped:");
this.skippedTests.forEach((test, i) => {
console.log(`\n${i + 1}) ${test.name}`);
if (typeof test.skip === "string") {
console.log(chalk.yellow(`\tSkipped: ${test.skip}`));
}
else {
console.log(chalk.yellow(`\t${"Skipped: No reason"}`));
}
});
}
if (this.todoTests.length > 0) {
console.log("\n\nTodo:");
this.todoTests.forEach((test, i) => {
console.log(`\n${i + 1}) ${test.name}`);
if (typeof test.todo === "string") {
console.log(chalk.blue(`\tTODO: ${test.todo}`));
}
else {
console.log(chalk.blue(`\t${"TODO: No reason"}`));
}
});
}
if (this.failedTests.length > 0) {
console.log(chalk.red("\n✖") + " Failing tests:");
this.failedTests.forEach((test, i) => {
console.log(chalk.red(`\n${i + 1}) ${test.name} (${test.details.duration_ms}ms)`));
console.log(test.details.error?.cause);
});
}
console.log(`\n\nℹ Tests ${this.passedTests +
this.failedTests.length +
this.skippedTests.length +
this.todoTests.length}`);
console.log(`ℹ Passed ${chalk.green(this.passedTests)}`);
console.log(`ℹ Failed ${chalk.red(this.failedTests.length)}`);
console.log(`ℹ Skipped ${chalk.yellow(this.skippedTests.length)}`);
console.log(`ℹ Todo ${chalk.blue(this.todoTests.length)}`);
console.log(`ℹ Duration ${this.totalDuration.toFixed(2)}ms`);
this.passedTests = 0;
this.skippedTests = [];
this.todoTests = [];
this.totalDuration = 0;
this.failedTests = [];
this.testSuite = {};
}
}
export default DotExtraReporter;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dot-extra-reporter.js","sourceRoot":"","sources":["../src/dot-extra-reporter.ts"],"names":[],"mappings":"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","sourcesContent":["import { Transform, type TransformCallback } from \"node:stream\";\nimport chalk from \"chalk\";\nimport test from \"node:test\";\n\n// Useful documentation here: https://nodejs.org/api/test.html#class-testsstream\n\n// Map of file name to\ntype TestSuite = Record<string, Suite>;\n\ntype Suite = {\n  file: string;\n  name: string;\n  nesting: number;\n  children?: Record<string, Test | Suite>;\n  passed?: boolean;\n  details?: { duration_ms: number; type?: \"suite\"; error?: { cause?: Error } };\n  skip?: boolean | string;\n  todo?: boolean | string;\n};\n\ntype Test = {\n  file: string;\n  name: string;\n  nesting: number;\n  message?: string;\n  details: {\n    duration_ms: number;\n    type?: \"suite\";\n    error?: {\n      cause: Error;\n    };\n  };\n  skip?: boolean | string;\n  todo?: boolean | string;\n  passed?: boolean;\n};\n\nenum EventType {\n  ENQUEUE = \"test:enqueue\",\n  DEQUEUE = \"test:dequeue\",\n  START = \"test:start\",\n  PASS = \"test:pass\",\n  FAIL = \"test:fail\",\n  PLAN = \"test:plan\",\n  WATCH_DRAINED = \"test:watch:drained\",\n  STDOUT = \"test:stdout\",\n  STDERR = \"test:stderr\",\n}\n\ntype Event = {\n  type: EventType;\n  data: Test;\n};\n\nclass DotExtraReporter extends Transform {\n  private passedTests: number;\n\n  private skippedTests: Test[];\n\n  private todoTests: Test[];\n\n  private totalDuration: number;\n\n  private failedTests: Test[];\n\n  private testSuite: TestSuite;\n\n  private currentSuite: Suite | undefined;\n\n  private currentSuiteLineage: Suite[] = [];\n\n  private nesting = 0;\n\n  constructor(options = {}) {\n    super({ ...options, writableObjectMode: true });\n    this.passedTests = 0;\n    this.skippedTests = [];\n    this.todoTests = [];\n    this.totalDuration = 0;\n    this.failedTests = [];\n    this.testSuite = {};\n  }\n\n  private handleTestStart(test: Test) {\n    // Add new child\n    const newItem = {\n      file: test.file,\n      name: test.name,\n      nesting: test.nesting,\n      children: {},\n    };\n\n    // If nesting > current nesting then\n    // We've moved into a nested suite so:\n    // - Add the current suite to the lineage\n    // - Add new item as a child of the current suite\n    if (test.nesting === 0) {\n      // First entry\n      this.currentSuite = newItem;\n      this.testSuite[`${test.file}:${test.name}`] = newItem;\n    } else if (test.nesting > this.nesting) {\n      this.currentSuiteLineage.push(this.currentSuite!);\n      this.currentSuite!.children![test.name] = newItem;\n\n      this.currentSuite = newItem;\n    } else {\n      // Not deeper so add to current parent, and set new item to current\n      const parent = this.currentSuiteLineage.pop();\n      parent!.children![test.name] = newItem;\n      this.currentSuiteLineage.push(parent!);\n      this.currentSuite = newItem;\n    }\n    this.nesting = test.nesting;\n  }\n\n  private handleTestResult(test: Test, eventType: EventType) {\n    // Update test result details\n    this.currentSuite!.passed = eventType === EventType.PASS;\n\n    if (test.details) {\n      this.currentSuite!.details = test.details;\n    }\n    if (test.skip) {\n      process.stdout.write(chalk.yellow(\"*\"));\n      this.currentSuite!.skip = test.skip;\n    } else if (test.todo) {\n      process.stdout.write(chalk.blue(\"-\"));\n      this.currentSuite!.todo = test.todo;\n    } else if (this.currentSuite!.passed) {\n      process.stdout.write(chalk.green(\".\"));\n    } else {\n      process.stdout.write(chalk.red(\"F\"));\n    }\n    this.nesting = test.nesting;\n  }\n\n  private handleTestEnd(test: Test) {\n    // Suite finished, step up to parent\n    this.currentSuite = this.currentSuiteLineage.pop();\n    this.nesting = test.nesting;\n  }\n\n  public override _transform(\n    event: Event,\n    _encoding: BufferEncoding,\n    callback: TransformCallback\n  ): void {\n    try {\n      const test = event.data as Test;\n\n      switch (event.type) {\n        case EventType.START:\n          this.handleTestStart(test);\n          break;\n        case EventType.PASS:\n        case EventType.FAIL:\n          if (test.name !== test.file) {\n            this.handleTestResult(test, event.type);\n          }\n          break;\n        case EventType.PLAN:\n          this.handleTestEnd(test);\n          break;\n        case EventType.STDOUT:\n          process.stdout.write(event.data.message ?? \"\");\n          break;\n        case EventType.STDERR:\n          process.stderr.write(event.data.message ?? \"\");\n          break;\n        default:\n          break;\n      }\n\n      callback();\n    } catch (error) {\n      console.log(error);\n      callback(error as Error);\n    }\n  }\n\n  public processOutcomes() {\n    Object.entries(this.testSuite).forEach(([file, suite]) => {\n      this.buildStats(suite);\n      // File suite doesn't have a duration so collate from all top level suites.\n      Object.values(suite.children ?? {}).forEach((child) => {\n        this.totalDuration += child.details?.duration_ms || 0;\n      });\n    });\n  }\n\n  private buildStats(suite: Suite | Test, prefix?: string) {\n    const isSuite = (obj: Suite | Test): obj is Suite =>\n      obj.details?.type === \"suite\";\n\n    if (isSuite(suite)) {\n      // Iterate over each child\n      Object.values(suite.children!).forEach((child) =>\n        this.buildStats(\n          child,\n          prefix ? `${prefix} > ${suite.name}` : suite.name\n        )\n      );\n    }\n\n    if (prefix) {\n      suite.name = `${prefix} > ${suite.name}`;\n    }\n    if (suite.skip) {\n      this.skippedTests.push(suite as Test);\n    } else if (suite.todo) {\n      this.todoTests.push(suite as Test);\n    } else if (!isSuite(suite)) {\n      if (suite.passed === false) {\n        this.failedTests.push(suite as Test);\n      } else {\n        this.passedTests++;\n      }\n    }\n  }\n\n  public override _flush() {\n    this.processOutcomes();\n\n    if (this.skippedTests.length > 0) {\n      console.log(\"\\n\\nSkipped:\");\n      this.skippedTests.forEach((test, i) => {\n        console.log(`\\n${i + 1}) ${test.name}`);\n        if (typeof test.skip === \"string\") {\n          console.log(chalk.yellow(`\\tSkipped: ${test.skip}`));\n        } else {\n          console.log(chalk.yellow(`\\t${\"Skipped: No reason\"}`));\n        }\n      });\n    }\n\n    if (this.todoTests.length > 0) {\n      console.log(\"\\n\\nTodo:\");\n      this.todoTests.forEach((test, i) => {\n        console.log(`\\n${i + 1}) ${test.name}`);\n        if (typeof test.todo === \"string\") {\n          console.log(chalk.blue(`\\tTODO: ${test.todo}`));\n        } else {\n          console.log(chalk.blue(`\\t${\"TODO: No reason\"}`));\n        }\n      });\n    }\n\n    if (this.failedTests.length > 0) {\n      console.log(chalk.red(\"\\n✖\") + \" Failing tests:\");\n      this.failedTests.forEach((test, i) => {\n        console.log(\n          chalk.red(`\\n${i + 1}) ${test.name} (${test.details.duration_ms}ms)`)\n        );\n        console.log(test.details.error?.cause);\n      });\n    }\n\n    console.log(\n      `\\n\\nℹ Tests    ${\n        this.passedTests +\n        this.failedTests.length +\n        this.skippedTests.length +\n        this.todoTests.length\n      }`\n    );\n    console.log(`ℹ Passed   ${chalk.green(this.passedTests)}`);\n    console.log(`ℹ Failed   ${chalk.red(this.failedTests.length)}`);\n    console.log(`ℹ Skipped  ${chalk.yellow(this.skippedTests.length)}`);\n    console.log(`ℹ Todo     ${chalk.blue(this.todoTests.length)}`);\n    console.log(`ℹ Duration ${this.totalDuration.toFixed(2)}ms`);\n\n    this.passedTests = 0;\n    this.skippedTests = [];\n    this.todoTests = [];\n    this.totalDuration = 0;\n    this.failedTests = [];\n    this.testSuite = {};\n  }\n}\n\nexport default DotExtraReporter;\n"]}