Skip to content

Commit 80e3bad

Browse files
authored
fix: cleanup handler for task after completion (#21)
* Assign and cleanup the task fn * Bump rc version
1 parent 9dfdf16 commit 80e3bad

File tree

3 files changed

+46
-31
lines changed

3 files changed

+46
-31
lines changed

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "@chainsafe/benchmark",
3-
"version": "1.1.0-rc.5",
3+
"version": "1.1.0-rc.6",
44
"repository": "[email protected]:chainsafe/benchmark.git",
55
"author": "ChainSafe Systems",
66
"license": "MIT",

src/benchmark/benchmarkFn.ts

+28-26
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import fs from "node:fs";
22
import path from "node:path";
3-
import {getCurrentSuite} from "@vitest/runner";
3+
import {getCurrentSuite, setFn} from "@vitest/runner";
44
import {createChainable} from "@vitest/runner/utils";
55
import {store} from "./globalState.js";
66
import {BenchApi, BenchmarkOpts, BenchmarkRunOptsWithFn, PartialBy} from "../types.js";
@@ -29,6 +29,31 @@ export const bench: BenchApi = createBenchmarkFunction(function <T, T2>(
2929
timeout = minMs * 1.5;
3030
}
3131

32+
async function handler(): Promise<void> {
33+
// Ensure bench id is unique
34+
if (store.getResult(opts.id) && !opts.skip) {
35+
throw Error(`test titles must be unique, duplicated: '${opts.id}'`);
36+
}
37+
38+
// Persist full results if requested. dir is created in `beforeAll`
39+
const benchmarkResultsCsvDir = process.env.BENCHMARK_RESULTS_CSV_DIR;
40+
const persistRunsNs = Boolean(benchmarkResultsCsvDir);
41+
42+
const {result, runsNs} = await runBenchFn({...options, fn: benchTask}, persistRunsNs);
43+
44+
// Store result for:
45+
// - to persist benchmark data latter
46+
// - to render with the custom reporter
47+
store.setResult(opts.id, result);
48+
49+
if (benchmarkResultsCsvDir) {
50+
fs.mkdirSync(benchmarkResultsCsvDir, {recursive: true});
51+
const filename = `${result.id}.csv`;
52+
const filepath = path.join(benchmarkResultsCsvDir, filename);
53+
fs.writeFileSync(filepath, runsNs.join("\n"));
54+
}
55+
}
56+
3257
const task = currentSuite.task(opts.id, {
3358
skip: opts.skip ?? this.skip,
3459
only: opts.only ?? this.only,
@@ -38,32 +63,9 @@ export const bench: BenchApi = createBenchmarkFunction(function <T, T2>(
3863
meta: {
3964
"chainsafe/benchmark": true,
4065
},
41-
async handler() {
42-
// Ensure bench id is unique
43-
if (store.getResult(opts.id) && !opts.skip) {
44-
throw Error(`test titles must be unique, duplicated: '${opts.id}'`);
45-
}
46-
47-
// Persist full results if requested. dir is created in `beforeAll`
48-
const benchmarkResultsCsvDir = process.env.BENCHMARK_RESULTS_CSV_DIR;
49-
const persistRunsNs = Boolean(benchmarkResultsCsvDir);
50-
51-
const {result, runsNs} = await runBenchFn({...options, fn: benchTask}, persistRunsNs);
52-
53-
// Store result for:
54-
// - to persist benchmark data latter
55-
// - to render with the custom reporter
56-
store.setResult(opts.id, result);
57-
58-
if (benchmarkResultsCsvDir) {
59-
fs.mkdirSync(benchmarkResultsCsvDir, {recursive: true});
60-
const filename = `${result.id}.csv`;
61-
const filepath = path.join(benchmarkResultsCsvDir, filename);
62-
fs.writeFileSync(filepath, runsNs.join("\n"));
63-
}
64-
},
6566
});
6667

68+
setFn(task, handler);
6769
store.setOptions(task, opts);
6870
});
6971

@@ -85,7 +87,7 @@ function coerceToOptsObj<T, T2>(
8587

8688
if (typeof idOrOpts === "string") {
8789
if (!fn) throw Error("fn arg must be set");
88-
opts = {id: idOrOpts, fn, threshold: optionsDefault.threshold};
90+
opts = {id: idOrOpts, fn};
8991
} else {
9092
if (fn) {
9193
opts = {...idOrOpts, fn};

src/benchmark/runner.ts

+17-4
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ import {
66
VitestRunner,
77
VitestRunnerConfig,
88
VitestRunnerImportSource,
9+
setFn,
910
} from "@vitest/runner";
1011
import path from "node:path";
1112
import {Benchmark, BenchmarkOpts, BenchmarkResults} from "../types.js";
@@ -49,25 +50,37 @@ export class BenchmarkRunner implements VitestRunner {
4950
this.reporter.onTestStarted(task);
5051
}
5152

52-
onAfterRunTask(task: Task): void {
53+
async onAfterRunTask(task: Task): Promise<void> {
5354
this.reporter.onTestFinished(task);
5455
store.removeOptions(task);
5556

57+
if (task.type === "test" || task.type === "custom") {
58+
// Clear up the assigned handler to clean the memory
59+
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
60+
// @ts-expect-error
61+
setFn(task, null);
62+
}
63+
5664
// To help maintain consistent memory usage patterns
5765
// we trigger garbage collection manually
5866
if (this.triggerGC && global.gc) {
5967
global.gc();
68+
// Make sure the syn operation is off the event loop
69+
await new Promise((resolve) => setTimeout(resolve, 0));
6070
}
6171
}
6272

6373
onAfterRunFiles(files: File[]): void {
6474
this.reporter.onComplete(files);
6575
}
6676

67-
// eslint-disable-next-line @typescript-eslint/no-unused-vars
68-
async importFile(filepath: string, _source: VitestRunnerImportSource): Promise<void> {
77+
async importFile(filepath: string, source: VitestRunnerImportSource): Promise<void> {
78+
let url = filepath;
79+
if (source === "setup") {
80+
url = `${url}?key=${Date.now()}`;
81+
}
6982
// TODO: Implement file caching mechanism later
70-
await import(filepath);
83+
await import(url);
7184
}
7285

7386
async process(files: string[]): Promise<BenchmarkResults> {

0 commit comments

Comments
 (0)