Skip to content

Commit 77c872d

Browse files
feat: enable eraseable syntax (#31)
* Enable eraseable types * Update the tests * Fix the benchmark * Fix the names for enum based on discussion ChainSafe/lodestar#7574 * Fix naming convention * Fix naming convention * Apply suggestions from code review Co-authored-by: Matthew Keil <[email protected]> * Rename all enums with Enum suffix * Rename all enums with Enum suffix * Rename all enums with Enum suffix * Update the name for HistoryProvier --------- Co-authored-by: Matthew Keil <[email protected]>
1 parent bce5c46 commit 77c872d

18 files changed

+114
-83
lines changed

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,7 @@
5050
"prettier": "^3.4.0",
5151
"rimraf": "^5.0.10",
5252
"ts-node": "^10.9.2",
53-
"typescript": "^5.6.3",
53+
"typescript": "^5.8.2",
5454
"typescript-eslint": "^8.16.0",
5555
"vitest": "^3.0.6",
5656
"vitest-in-process-pool": "^2.0.0"

src/benchmark/convergence/coefficientOfVariance.ts

+9-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,13 @@
11
import Debug from "debug";
22
import {BenchmarkOpts, ConvergenceCheckFn} from "../../types.js";
3-
import {calcMean, calcMedian, calcVariance, filterOutliers, OutlierSensitivity, sortData} from "../../utils/math.js";
3+
import {
4+
calcMean,
5+
calcMedian,
6+
calcVariance,
7+
filterOutliers,
8+
OutlierSensitivityEnum,
9+
sortData,
10+
} from "../../utils/math.js";
411

512
const debug = Debug("@chainsafe/benchmark/convergence");
613

@@ -43,7 +50,7 @@ export function createCVConvergenceCriteria(
4350
if (timeSinceLastCheck < sampleEveryMs) return false;
4451
lastConvergenceSample = currentMs;
4552
// For all statistical calculations we don't want to loose the precision so have to convert to numbers first
46-
const samples = filterOutliers(sortData(runsNs.map((n) => Number(n))), true, OutlierSensitivity.Mild);
53+
const samples = filterOutliers(sortData(runsNs.map((n) => Number(n))), true, OutlierSensitivityEnum.Mild);
4754

4855
// If CV does not stabilize we fallback to the median approach
4956
if (runsNs.length > maxSamplesForCV) {

src/benchmark/options.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import {AverageCalculation, BenchmarkOpts, Convergence} from "../types.js";
1+
import {AverageCalculationEnum, BenchmarkOpts, ConvergenceEnum} from "../types.js";
22

33
export const defaultBenchmarkOptions: Required<BenchmarkOpts> = {
44
minRuns: 1,
@@ -17,8 +17,8 @@ export const defaultBenchmarkOptions: Required<BenchmarkOpts> = {
1717
skip: false,
1818
only: false,
1919
threshold: 2,
20-
convergence: Convergence.Linear,
21-
averageCalculation: AverageCalculation.Simple,
20+
convergence: ConvergenceEnum.Linear,
21+
averageCalculation: AverageCalculationEnum.Simple,
2222
};
2323

2424
export function getBenchmarkOptionsWithDefaults(opts: BenchmarkOpts): Required<BenchmarkOpts> {

src/benchmark/runBenchmarkFn.ts

+7-7
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import Debug from "debug";
2-
import {BenchmarkResult, BenchmarkOpts, Convergence, ConvergenceCheckFn} from "../types.js";
3-
import {calcSum, filterOutliers, OutlierSensitivity} from "../utils/math.js";
2+
import {BenchmarkResult, BenchmarkOpts, Convergence, ConvergenceCheckFn, ConvergenceEnum} from "../types.js";
3+
import {calcSum, filterOutliers, OutlierSensitivityEnum} from "../utils/math.js";
44
import {getBenchmarkOptionsWithDefaults} from "./options.js";
55
import {createLinearConvergenceCriteria} from "./convergence/linearAverage.js";
66
import {createCVConvergenceCriteria} from "./convergence/coefficientOfVariance.js";
@@ -9,8 +9,8 @@ const debug = Debug("@chainsafe/benchmark/run");
99

1010
const convergenceCriteria: Record<Convergence, (startMs: number, opts: Required<BenchmarkOpts>) => ConvergenceCheckFn> =
1111
{
12-
[Convergence.Linear]: createLinearConvergenceCriteria,
13-
[Convergence.CV]: createCVConvergenceCriteria,
12+
[ConvergenceEnum.Linear]: createLinearConvergenceCriteria,
13+
[ConvergenceEnum.CV]: createCVConvergenceCriteria,
1414
};
1515

1616
export type BenchmarkRunOpts = BenchmarkOpts & {
@@ -45,8 +45,8 @@ export async function runBenchFn<T, T2>(
4545
throw new Error(`Average calculation logic is not defined. ${averageCalculation}`);
4646
}
4747

48-
if (!Object.values(Convergence).includes(convergence)) {
49-
throw new Error(`Unknown convergence value ${convergence}. Valid values are ${Object.values(Convergence)}`);
48+
if (!Object.values(ConvergenceEnum).includes(convergence)) {
49+
throw new Error(`Unknown convergence value ${convergence}. Valid values are ${Object.values(ConvergenceEnum)}`);
5050
}
5151

5252
// Ratio of maxMs that the warmup is allow to take from elapsedMs
@@ -138,7 +138,7 @@ either the before(), beforeEach() or fn() functions are too slow.
138138
}
139139

140140
if (averageCalculation === "clean-outliers") {
141-
const cleanData = filterOutliers(runsNs, false, OutlierSensitivity.Mild);
141+
const cleanData = filterOutliers(runsNs, false, OutlierSensitivityEnum.Mild);
142142
averageNs = Number(calcSum(cleanData) / BigInt(cleanData.length)) / runsFactor;
143143
}
144144

src/cli/compare.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ import {renderBenchmarkComparisonTable} from "../utils/render.js";
77
import {isGaRun} from "../github/context.js";
88
import {postGaComment} from "../github/comments/index.js";
99
import {benchmarkComparisonComment} from "../github/comments/comparisonReportComment.js";
10-
import {GithubCommentTag} from "../github/octokit.js";
10+
import {GithubCommentTagEnum} from "../github/octokit.js";
1111

1212
export async function compare({dirs}: {dirs: string[]}): Promise<void> {
1313
consoleLog("Comparing benchmarks:");
@@ -41,7 +41,7 @@ export async function compare({dirs}: {dirs: string[]}): Promise<void> {
4141
if (isGaRun()) {
4242
await postGaComment({
4343
commentBody: benchmarkComparisonComment(resultsComp),
44-
tag: GithubCommentTag.ComparisonReport,
44+
tag: GithubCommentTagEnum.ComparisonReport,
4545
commentOnPush: true,
4646
});
4747
}

src/cli/options.ts

+9-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,11 @@
11
import {Options} from "yargs";
2-
import {StorageOptions, BenchmarkOpts, FileCollectionOptions, Convergence, AverageCalculation} from "../types.js";
2+
import {
3+
StorageOptions,
4+
BenchmarkOpts,
5+
FileCollectionOptions,
6+
AverageCalculationEnum,
7+
ConvergenceEnum,
8+
} from "../types.js";
39
import {defaultBenchmarkOptions} from "../benchmark/options.js";
410

511
export const optionsDefault = {
@@ -209,14 +215,14 @@ export const benchmarkOptions: ICliCommandOptions<CLIBenchmarkOptions> = {
209215
type: "string",
210216
description: "The algorithm used to detect the convergence to stop benchmark runs",
211217
default: defaultBenchmarkOptions.convergence,
212-
choices: Object.values(Convergence),
218+
choices: Object.values(ConvergenceEnum),
213219
group: benchmarkGroup,
214220
},
215221
averageCalculation: {
216222
type: "string",
217223
description: "Use simple average of all runs or clean the outliers before calculating average",
218224
default: defaultBenchmarkOptions.averageCalculation,
219-
choices: Object.values(AverageCalculation),
225+
choices: Object.values(AverageCalculationEnum),
220226
group: benchmarkGroup,
221227
},
222228
};

src/cli/run.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -19,9 +19,9 @@ import {isGaRun} from "../github/context.js";
1919
import {BenchmarkRunner} from "../benchmark/runner.js";
2020
import {optionsDefault} from "./options.js";
2121
import {consoleLog} from "../utils/output.js";
22-
import {HistoryProviderType} from "../history/provider.js";
22+
import {HistoryProviderEnum} from "../history/provider.js";
2323
import {performanceReportComment} from "../github/comments/performanceReportComment.js";
24-
import {GithubCommentTag} from "../github/octokit.js";
24+
import {GithubCommentTagEnum} from "../github/octokit.js";
2525
import {defaultBenchmarkOptions} from "../benchmark/options.js";
2626

2727
const debug = Debug("@chainsafe/benchmark/cli");
@@ -81,7 +81,7 @@ export async function run(opts_: FileCollectionOptions & StorageOptions & Benchm
8181
currentCommit.branch ??
8282
parseBranchFromRef(
8383
github.context.ref ?? (await shell("git symbolic-ref HEAD")),
84-
historyProvider.type === HistoryProviderType.Local
84+
historyProvider.type === HistoryProviderEnum.Local
8585
);
8686
consoleLog(`Persisting new benchmark data for branch '${branch}' commit '${currBench.commitSha}'`);
8787
// TODO: prune and limit total entries
@@ -100,7 +100,7 @@ export async function run(opts_: FileCollectionOptions & StorageOptions & Benchm
100100
if (!opts.skipPostComment && isGaRun()) {
101101
await postGaComment({
102102
commentBody: performanceReportComment(resultsComp),
103-
tag: GithubCommentTag.PerformanceReport,
103+
tag: GithubCommentTagEnum.PerformanceReport,
104104
commentOnPush: resultsComp.someFailed,
105105
});
106106
}

src/compare/index.ts

+15-15
Original file line numberDiff line numberDiff line change
@@ -5,14 +5,14 @@ import {isGaRun} from "../github/context.js";
55
import {IHistoryProvider} from "../history/provider.js";
66
import {validateBenchmark} from "../history/schema.js";
77

8-
enum CompareWithType {
9-
latestCommitInBranch = "latestCommitInBranch",
10-
exactCommit = "exactCommit",
11-
}
8+
const CompareWithTypeEnum = {
9+
latestCommitInBranch: "latestCommitInBranch",
10+
exactCommit: "exactCommit",
11+
} as const;
1212

1313
export type CompareWith =
14-
| {type: CompareWithType.latestCommitInBranch; branch: string; before?: string}
15-
| {type: CompareWithType.exactCommit; commitSha: string};
14+
| {type: typeof CompareWithTypeEnum.latestCommitInBranch; branch: string; before?: string}
15+
| {type: typeof CompareWithTypeEnum.exactCommit; commitSha: string};
1616

1717
export async function resolveCompare(provider: IHistoryProvider, opts: StorageOptions): Promise<Benchmark | null> {
1818
const compareWith = await resolveCompareWith(opts);
@@ -29,10 +29,10 @@ export async function resolvePrevBenchmark(
2929
provider: IHistoryProvider
3030
): Promise<Benchmark | null> {
3131
switch (compareWith.type) {
32-
case CompareWithType.exactCommit:
32+
case CompareWithTypeEnum.exactCommit:
3333
return await provider.readHistoryCommit(compareWith.commitSha);
3434

35-
case CompareWithType.latestCommitInBranch: {
35+
case CompareWithTypeEnum.latestCommitInBranch: {
3636
// Try first latest commit in branch
3737
return await provider.readLatestInBranch(compareWith.branch);
3838
}
@@ -41,10 +41,10 @@ export async function resolvePrevBenchmark(
4141

4242
export function renderCompareWith(compareWith: CompareWith): string {
4343
switch (compareWith.type) {
44-
case CompareWithType.exactCommit:
44+
case CompareWithTypeEnum.exactCommit:
4545
return `exactCommit ${compareWith.commitSha}`;
4646

47-
case CompareWithType.latestCommitInBranch: {
47+
case CompareWithTypeEnum.latestCommitInBranch: {
4848
if (compareWith.before) {
4949
return `latestCommitInBranch '${compareWith.branch}' before commit ${compareWith.before}`;
5050
} else {
@@ -57,11 +57,11 @@ export function renderCompareWith(compareWith: CompareWith): string {
5757
export async function resolveCompareWith(opts: StorageOptions): Promise<CompareWith> {
5858
// compare may be a branch or commit
5959
if (opts.compareBranch) {
60-
return {type: CompareWithType.latestCommitInBranch, branch: opts.compareBranch};
60+
return {type: CompareWithTypeEnum.latestCommitInBranch, branch: opts.compareBranch};
6161
}
6262

6363
if (opts.compareCommit) {
64-
return {type: CompareWithType.exactCommit, commitSha: opts.compareCommit};
64+
return {type: CompareWithTypeEnum.exactCommit, commitSha: opts.compareCommit};
6565
}
6666

6767
// In GA CI figure out what to compare against with github actions events
@@ -70,13 +70,13 @@ export async function resolveCompareWith(opts: StorageOptions): Promise<CompareW
7070
case "pull_request": {
7171
const eventData = getGithubEventData<GithubActionsEventData["pull_request"]>();
7272
const baseBranch = eventData.pull_request.base.ref; // base.ref is already parsed
73-
return {type: CompareWithType.latestCommitInBranch, branch: baseBranch};
73+
return {type: CompareWithTypeEnum.latestCommitInBranch, branch: baseBranch};
7474
}
7575

7676
case "push": {
7777
const eventData = getGithubEventData<GithubActionsEventData["push"]>();
7878
const branch = parseBranchFromRef(github.context.ref);
79-
return {type: CompareWithType.latestCommitInBranch, branch: branch, before: eventData.before};
79+
return {type: CompareWithTypeEnum.latestCommitInBranch, branch: branch, before: eventData.before};
8080
}
8181

8282
default:
@@ -86,5 +86,5 @@ export async function resolveCompareWith(opts: StorageOptions): Promise<CompareW
8686

8787
// Otherwise compare against the default branch
8888
const defaultBranch = await getDefaultBranch(opts);
89-
return {type: CompareWithType.latestCommitInBranch, branch: defaultBranch};
89+
return {type: CompareWithTypeEnum.latestCommitInBranch, branch: defaultBranch};
9090
}

src/github/octokit.ts

+6-4
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,11 @@
1+
import {EnumLike} from "../types.js";
12
import {getContext} from "./context.js";
23

3-
export enum GithubCommentTag {
4-
PerformanceReport = "benchmarkbot/action",
5-
ComparisonReport = "benchmarkbot/compare",
6-
}
4+
export const GithubCommentTagEnum = {
5+
PerformanceReport: "benchmarkbot/action",
6+
ComparisonReport: "benchmarkbot/compare",
7+
} as const;
8+
export type GithubCommentTag = EnumLike<typeof GithubCommentTagEnum>;
79

810
export async function commentToPrUpdatable(prNumber: number, body: string, tag: GithubCommentTag): Promise<void> {
911
const {repo, octokit} = getContext();

src/history/gaCache.ts

+7-6
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ import fs from "node:fs";
22
import * as cache from "@actions/cache";
33
import {Benchmark} from "../types.js";
44
import {LocalHistoryProvider} from "./local.js";
5-
import {HistoryProviderType, IHistoryProvider} from "./provider.js";
5+
import {HistoryProviderEnum, HistoryProvider, IHistoryProvider} from "./provider.js";
66

77
/**
88
* Persist results in CSV, one benchmark result per file
@@ -23,14 +23,15 @@ export function getGaCacheHistoryProvider(cacheKey: string): IHistoryProvider {
2323
}
2424

2525
class GaCacheHistoryProvider extends LocalHistoryProvider implements IHistoryProvider {
26-
readonly type = HistoryProviderType.GaCache;
26+
readonly type: HistoryProvider = HistoryProviderEnum.GaCache;
2727
private initializePromise: Promise<unknown> | null = null;
28+
private readonly tmpDir: string;
29+
private readonly cacheKey: string;
2830

29-
constructor(
30-
private readonly tmpDir: string,
31-
private readonly cacheKey: string
32-
) {
31+
constructor(tmpDir: string, cacheKey: string) {
3332
super(tmpDir);
33+
this.tmpDir = tmpDir;
34+
this.cacheKey = cacheKey;
3435
}
3536

3637
providerInfo(): string {

src/history/local.ts

+7-3
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 {HistoryProviderType, IHistoryProvider} from "./provider.js";
3+
import {HistoryProviderEnum, HistoryProvider, IHistoryProvider} from "./provider.js";
44
import {Benchmark, BenchmarkResults} from "../types.js";
55
import {fromCsv, toCsv} from "../utils/file.js";
66
import {FsError} from "../utils/index.js";
@@ -33,8 +33,12 @@ interface CsvMeta {
3333
* ```
3434
*/
3535
export class LocalHistoryProvider implements IHistoryProvider {
36-
readonly type: HistoryProviderType = HistoryProviderType.Local;
37-
constructor(private readonly dirpath: string) {}
36+
readonly type: HistoryProvider = HistoryProviderEnum.Local;
37+
private dirpath: string;
38+
39+
constructor(dirpath: string) {
40+
this.dirpath = dirpath;
41+
}
3842

3943
providerInfo(): string {
4044
return `LocalHistoryProvider, dirpath ${this.dirpath}`;

src/history/provider.ts

+8-7
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,11 @@
1-
import {Benchmark} from "../types.js";
1+
import {Benchmark, EnumLike} from "../types.js";
22

3-
export enum HistoryProviderType {
4-
Local = "Local",
5-
GaCache = "GaCache",
6-
S3 = "S3",
7-
}
3+
export const HistoryProviderEnum = {
4+
Local: "Local",
5+
GaCache: "GaCache",
6+
S3: "S3",
7+
} as const;
8+
export type HistoryProvider = EnumLike<typeof HistoryProviderEnum>;
89

910
/**
1011
* How to organize data?
@@ -29,7 +30,7 @@ export enum HistoryProviderType {
2930
* In history you can ONLY track a single branch, which should be the main branch.
3031
*/
3132
export interface IHistoryProvider {
32-
readonly type: HistoryProviderType;
33+
readonly type: HistoryProvider;
3334
providerInfo(): string;
3435
readLatestInBranch(branch: string): Promise<Benchmark | null>;
3536
writeLatestInBranch(branch: string, benchmark: Benchmark): Promise<void>;

src/history/s3.ts

+5-3
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ import path from "node:path";
22
import S3 from "aws-sdk/clients/s3.js";
33
import {Benchmark, BenchmarkResults} from "../types.js";
44
import {fromCsv, toCsv, extendError, AwsError} from "../utils/index.js";
5-
import {HistoryProviderType, IHistoryProvider} from "./provider.js";
5+
import {HistoryProvider, HistoryProviderEnum, IHistoryProvider} from "./provider.js";
66

77
export type S3Config = Pick<S3.Types.ClientConfiguration, "accessKeyId" | "secretAccessKey" | "region" | "endpoint"> & {
88
Bucket: string;
@@ -19,10 +19,12 @@ interface CsvMeta {
1919
}
2020

2121
export class S3HistoryProvider implements IHistoryProvider {
22-
readonly type = HistoryProviderType.S3;
22+
readonly type: HistoryProvider = HistoryProviderEnum.S3;
2323
private s3: S3;
24+
private config: S3Config;
2425

25-
constructor(private readonly config: S3Config) {
26+
constructor(config: S3Config) {
27+
this.config = config;
2628
this.s3 = new S3(config);
2729
}
2830

0 commit comments

Comments
 (0)