@rudderstack/integrations-lib 0.2.34 → 0.2.36
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/build/sdks/sfmc/index.d.ts +26 -3
- package/build/sdks/sfmc/index.d.ts.map +1 -1
- package/build/sdks/sfmc/index.js +108 -16
- package/build/sdks/sfmc/type.d.ts +44 -24
- package/build/sdks/sfmc/type.d.ts.map +1 -1
- package/build/sdks/sfmc/type.js +1 -1
- package/build/utils/benchmark.d.ts +180 -0
- package/build/utils/benchmark.d.ts.map +1 -0
- package/build/utils/benchmark.js +288 -0
- package/build/utils/misc.d.ts +24 -7
- package/build/utils/misc.d.ts.map +1 -1
- package/build/utils/misc.js +48 -11
- package/package.json +1 -1
|
@@ -0,0 +1,180 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Benchmark utilities for performance testing
|
|
3
|
+
*
|
|
4
|
+
* Provides easy-to-use tools for measuring function performance,
|
|
5
|
+
* memory usage, and creating performance comparisons.
|
|
6
|
+
*
|
|
7
|
+
* @example
|
|
8
|
+
* ```typescript
|
|
9
|
+
* import { benchmark, createBenchmarkSuite } from './benchmark';
|
|
10
|
+
*
|
|
11
|
+
* // Simple benchmark
|
|
12
|
+
* const result = await benchmark('myFunction', () => myFunction(data));
|
|
13
|
+
* console.log(`Duration: ${result.duration}ms`);
|
|
14
|
+
*
|
|
15
|
+
* // Benchmark suite
|
|
16
|
+
* const suite = createBenchmarkSuite('Array Operations');
|
|
17
|
+
* suite.add('native map', () => data.map(x => x * 2));
|
|
18
|
+
* suite.add('custom map', () => customMap(data, x => x * 2));
|
|
19
|
+
* await suite.run();
|
|
20
|
+
* ```
|
|
21
|
+
*/
|
|
22
|
+
/**
|
|
23
|
+
* Result of a benchmark measurement
|
|
24
|
+
*/
|
|
25
|
+
export type BenchmarkResult = {
|
|
26
|
+
/** Name of the benchmark */
|
|
27
|
+
name: string;
|
|
28
|
+
/** Total duration in milliseconds */
|
|
29
|
+
duration: number;
|
|
30
|
+
/** Operations per second */
|
|
31
|
+
opsPerSecond: number;
|
|
32
|
+
/** Memory used in bytes (heap difference) */
|
|
33
|
+
memoryUsed: number;
|
|
34
|
+
/** Number of iterations run */
|
|
35
|
+
iterations: number;
|
|
36
|
+
/** Average duration per iteration in milliseconds */
|
|
37
|
+
avgDuration: number;
|
|
38
|
+
/** Standard deviation of durations */
|
|
39
|
+
stdDev: number;
|
|
40
|
+
/** Minimum duration observed */
|
|
41
|
+
minDuration: number;
|
|
42
|
+
/** Maximum duration observed */
|
|
43
|
+
maxDuration: number;
|
|
44
|
+
};
|
|
45
|
+
/**
|
|
46
|
+
* Configuration options for benchmarks
|
|
47
|
+
*/
|
|
48
|
+
export type BenchmarkOptions = {
|
|
49
|
+
/** Number of iterations to run (default: 1) */
|
|
50
|
+
iterations?: number;
|
|
51
|
+
/** Number of warmup runs before measuring (default: 1) */
|
|
52
|
+
warmupRuns?: number;
|
|
53
|
+
/** Whether to force garbage collection between runs (default: false) */
|
|
54
|
+
forceGC?: boolean;
|
|
55
|
+
/** Whether to measure memory usage (default: true) */
|
|
56
|
+
measureMemory?: boolean;
|
|
57
|
+
/** Timeout in milliseconds for the entire benchmark (default: 30000) */
|
|
58
|
+
timeout?: number;
|
|
59
|
+
};
|
|
60
|
+
/**
|
|
61
|
+
* Benchmark function type
|
|
62
|
+
*/
|
|
63
|
+
export type BenchmarkFunction<T = unknown> = () => T | Promise<T>;
|
|
64
|
+
/**
|
|
65
|
+
* Run a benchmark for a single function
|
|
66
|
+
*
|
|
67
|
+
* @param name - Name of the benchmark
|
|
68
|
+
* @param fn - Function to benchmark
|
|
69
|
+
* @param options - Benchmark configuration options
|
|
70
|
+
* @returns Promise resolving to benchmark results
|
|
71
|
+
*
|
|
72
|
+
* @example
|
|
73
|
+
* ```typescript
|
|
74
|
+
* const result = await benchmark('Array.map', () => {
|
|
75
|
+
* return largeArray.map(x => x * 2);
|
|
76
|
+
* }, { iterations: 10, warmupRuns: 3 });
|
|
77
|
+
*
|
|
78
|
+
* console.log(`${result.name}: ${result.avgDuration.toFixed(2)}ms avg`);
|
|
79
|
+
* ```
|
|
80
|
+
*/
|
|
81
|
+
export declare function benchmark(name: string, fn: BenchmarkFunction, options?: BenchmarkOptions): Promise<BenchmarkResult>;
|
|
82
|
+
/**
|
|
83
|
+
* Compare multiple functions with the same benchmark configuration
|
|
84
|
+
*
|
|
85
|
+
* @param benchmarks - Object with benchmark names and functions
|
|
86
|
+
* @param options - Benchmark configuration options
|
|
87
|
+
* @returns Promise resolving to array of benchmark results
|
|
88
|
+
*
|
|
89
|
+
* @example
|
|
90
|
+
* ```typescript
|
|
91
|
+
* const results = await compareBenchmarks({
|
|
92
|
+
* 'native map': () => data.map(x => x * 2),
|
|
93
|
+
* 'for loop': () => {
|
|
94
|
+
* const result = [];
|
|
95
|
+
* for (let i = 0; i < data.length; i++) {
|
|
96
|
+
* result.push(data[i] * 2);
|
|
97
|
+
* }
|
|
98
|
+
* return result;
|
|
99
|
+
* }
|
|
100
|
+
* }, { iterations: 5 });
|
|
101
|
+
*
|
|
102
|
+
* results.forEach(result => {
|
|
103
|
+
* console.log(`${result.name}: ${result.avgDuration.toFixed(2)}ms`);
|
|
104
|
+
* });
|
|
105
|
+
* ```
|
|
106
|
+
*/
|
|
107
|
+
export declare function compareBenchmarks(benchmarks: Record<string, BenchmarkFunction>, options?: BenchmarkOptions): Promise<BenchmarkResult[]>;
|
|
108
|
+
/**
|
|
109
|
+
* Benchmark suite for organizing and running multiple related benchmarks
|
|
110
|
+
*/
|
|
111
|
+
export declare class BenchmarkSuite {
|
|
112
|
+
readonly suiteName: string;
|
|
113
|
+
private benchmarks;
|
|
114
|
+
private options;
|
|
115
|
+
constructor(suiteName: string, options?: BenchmarkOptions);
|
|
116
|
+
/**
|
|
117
|
+
* Add a benchmark to the suite
|
|
118
|
+
*
|
|
119
|
+
* @param name - Name of the benchmark
|
|
120
|
+
* @param fn - Function to benchmark
|
|
121
|
+
* @returns This suite instance for chaining
|
|
122
|
+
*/
|
|
123
|
+
add(name: string, fn: BenchmarkFunction): this;
|
|
124
|
+
/**
|
|
125
|
+
* Run all benchmarks in the suite
|
|
126
|
+
*
|
|
127
|
+
* @param customOptions - Override suite options for this run
|
|
128
|
+
* @returns Promise resolving to array of benchmark results
|
|
129
|
+
*/
|
|
130
|
+
run(customOptions?: BenchmarkOptions): Promise<BenchmarkResult[]>;
|
|
131
|
+
/**
|
|
132
|
+
* Get the number of benchmarks in the suite
|
|
133
|
+
*/
|
|
134
|
+
get size(): number;
|
|
135
|
+
/**
|
|
136
|
+
* Clear all benchmarks from the suite
|
|
137
|
+
*/
|
|
138
|
+
clear(): this;
|
|
139
|
+
}
|
|
140
|
+
/**
|
|
141
|
+
* Create a new benchmark suite
|
|
142
|
+
*
|
|
143
|
+
* @param suiteName - Name of the benchmark suite
|
|
144
|
+
* @param options - Default options for all benchmarks in the suite
|
|
145
|
+
* @returns New BenchmarkSuite instance
|
|
146
|
+
*
|
|
147
|
+
* @example
|
|
148
|
+
* ```typescript
|
|
149
|
+
* const suite = createBenchmarkSuite('String Operations', {
|
|
150
|
+
* iterations: 10,
|
|
151
|
+
* warmupRuns: 2
|
|
152
|
+
* });
|
|
153
|
+
*
|
|
154
|
+
* suite
|
|
155
|
+
* .add('concat', () => str1 + str2)
|
|
156
|
+
* .add('template', () => `${str1}${str2}`)
|
|
157
|
+
* .add('join', () => [str1, str2].join(''));
|
|
158
|
+
*
|
|
159
|
+
* await suite.run();
|
|
160
|
+
* ```
|
|
161
|
+
*/
|
|
162
|
+
export declare function createBenchmarkSuite(suiteName: string, options?: BenchmarkOptions): BenchmarkSuite;
|
|
163
|
+
/**
|
|
164
|
+
* Utility to measure memory usage of a function
|
|
165
|
+
*
|
|
166
|
+
* @param fn - Function to measure
|
|
167
|
+
* @param forceGC - Whether to force garbage collection before measuring
|
|
168
|
+
* @returns Memory usage in bytes
|
|
169
|
+
*
|
|
170
|
+
* @example
|
|
171
|
+
* ```typescript
|
|
172
|
+
* const memoryUsed = await measureMemory(() => {
|
|
173
|
+
* return new Array(100000).fill(0).map(x => ({ value: x }));
|
|
174
|
+
* });
|
|
175
|
+
*
|
|
176
|
+
* console.log(`Memory used: ${(memoryUsed / 1024 / 1024).toFixed(2)}MB`);
|
|
177
|
+
* ```
|
|
178
|
+
*/
|
|
179
|
+
export declare function measureMemory(fn: BenchmarkFunction, forceGC?: boolean): Promise<number>;
|
|
180
|
+
//# sourceMappingURL=benchmark.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"benchmark.d.ts","sourceRoot":"","sources":["../../src/utils/benchmark.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;;;;;;;;;;GAoBG;AAEH;;GAEG;AACH,MAAM,MAAM,eAAe,GAAG;IAC5B,4BAA4B;IAC5B,IAAI,EAAE,MAAM,CAAC;IACb,qCAAqC;IACrC,QAAQ,EAAE,MAAM,CAAC;IACjB,4BAA4B;IAC5B,YAAY,EAAE,MAAM,CAAC;IACrB,6CAA6C;IAC7C,UAAU,EAAE,MAAM,CAAC;IACnB,+BAA+B;IAC/B,UAAU,EAAE,MAAM,CAAC;IACnB,qDAAqD;IACrD,WAAW,EAAE,MAAM,CAAC;IACpB,sCAAsC;IACtC,MAAM,EAAE,MAAM,CAAC;IACf,gCAAgC;IAChC,WAAW,EAAE,MAAM,CAAC;IACpB,gCAAgC;IAChC,WAAW,EAAE,MAAM,CAAC;CACrB,CAAC;AAEF;;GAEG;AACH,MAAM,MAAM,gBAAgB,GAAG;IAC7B,+CAA+C;IAC/C,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,0DAA0D;IAC1D,UAAU,CAAC,EAAE,MAAM,CAAC;IACpB,wEAAwE;IACxE,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,sDAAsD;IACtD,aAAa,CAAC,EAAE,OAAO,CAAC;IACxB,wEAAwE;IACxE,OAAO,CAAC,EAAE,MAAM,CAAC;CAClB,CAAC;AAEF;;GAEG;AACH,MAAM,MAAM,iBAAiB,CAAC,CAAC,GAAG,OAAO,IAAI,MAAM,CAAC,GAAG,OAAO,CAAC,CAAC,CAAC,CAAC;AAqBlE;;;;;;;;;;;;;;;;GAgBG;AAEH,wBAAsB,SAAS,CAC7B,IAAI,EAAE,MAAM,EACZ,EAAE,EAAE,iBAAiB,EACrB,OAAO,GAAE,gBAAqB,GAC7B,OAAO,CAAC,eAAe,CAAC,CA4E1B;AAED;;;;;;;;;;;;;;;;;;;;;;;;GAwBG;AACH,wBAAsB,iBAAiB,CACrC,UAAU,EAAE,MAAM,CAAC,MAAM,EAAE,iBAAiB,CAAC,EAC7C,OAAO,GAAE,gBAAqB,GAC7B,OAAO,CAAC,eAAe,EAAE,CAAC,CAM5B;AAED;;GAEG;AACH,qBAAa,cAAc;aAKG,SAAS,EAAE,MAAM;IAJ7C,OAAO,CAAC,UAAU,CAAsD;IAExE,OAAO,CAAC,OAAO,CAAmB;gBAEN,SAAS,EAAE,MAAM,EAAE,OAAO,GAAE,gBAAqB;IAI7E;;;;;;OAMG;IACH,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE,EAAE,EAAE,iBAAiB,GAAG,IAAI;IAK9C;;;;;OAKG;IACG,GAAG,CAAC,aAAa,CAAC,EAAE,gBAAgB,GAAG,OAAO,CAAC,eAAe,EAAE,CAAC;IAkEvE;;OAEG;IACH,IAAI,IAAI,IAAI,MAAM,CAEjB;IAED;;OAEG;IACH,KAAK,IAAI,IAAI;CAId;AAED;;;;;;;;;;;;;;;;;;;;;GAqBG;AACH,wBAAgB,oBAAoB,CAClC,SAAS,EAAE,MAAM,EACjB,OAAO,CAAC,EAAE,gBAAgB,GACzB,cAAc,CAEhB;AAED;;;;;;;;;;;;;;;GAeG;AACH,wBAAsB,aAAa,CAAC,EAAE,EAAE,iBAAiB,EAAE,OAAO,UAAO,GAAG,OAAO,CAAC,MAAM,CAAC,CAU1F"}
|
|
@@ -0,0 +1,288 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* Benchmark utilities for performance testing
|
|
4
|
+
*
|
|
5
|
+
* Provides easy-to-use tools for measuring function performance,
|
|
6
|
+
* memory usage, and creating performance comparisons.
|
|
7
|
+
*
|
|
8
|
+
* @example
|
|
9
|
+
* ```typescript
|
|
10
|
+
* import { benchmark, createBenchmarkSuite } from './benchmark';
|
|
11
|
+
*
|
|
12
|
+
* // Simple benchmark
|
|
13
|
+
* const result = await benchmark('myFunction', () => myFunction(data));
|
|
14
|
+
* console.log(`Duration: ${result.duration}ms`);
|
|
15
|
+
*
|
|
16
|
+
* // Benchmark suite
|
|
17
|
+
* const suite = createBenchmarkSuite('Array Operations');
|
|
18
|
+
* suite.add('native map', () => data.map(x => x * 2));
|
|
19
|
+
* suite.add('custom map', () => customMap(data, x => x * 2));
|
|
20
|
+
* await suite.run();
|
|
21
|
+
* ```
|
|
22
|
+
*/
|
|
23
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
24
|
+
exports.BenchmarkSuite = void 0;
|
|
25
|
+
exports.benchmark = benchmark;
|
|
26
|
+
exports.compareBenchmarks = compareBenchmarks;
|
|
27
|
+
exports.createBenchmarkSuite = createBenchmarkSuite;
|
|
28
|
+
exports.measureMemory = measureMemory;
|
|
29
|
+
/**
|
|
30
|
+
* Force garbage collection if available
|
|
31
|
+
*/
|
|
32
|
+
function forceGarbageCollection() {
|
|
33
|
+
if (typeof global !== 'undefined' && global.gc) {
|
|
34
|
+
global.gc();
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Calculate standard deviation of an array of numbers
|
|
39
|
+
*/
|
|
40
|
+
function calculateStdDev(values) {
|
|
41
|
+
const mean = values.reduce((sum, val) => sum + val, 0) / values.length;
|
|
42
|
+
const squaredDiffs = values.map((val) => (val - mean) ** 2);
|
|
43
|
+
const avgSquaredDiff = squaredDiffs.reduce((sum, val) => sum + val, 0) / values.length;
|
|
44
|
+
return Math.sqrt(avgSquaredDiff);
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* Run a benchmark for a single function
|
|
48
|
+
*
|
|
49
|
+
* @param name - Name of the benchmark
|
|
50
|
+
* @param fn - Function to benchmark
|
|
51
|
+
* @param options - Benchmark configuration options
|
|
52
|
+
* @returns Promise resolving to benchmark results
|
|
53
|
+
*
|
|
54
|
+
* @example
|
|
55
|
+
* ```typescript
|
|
56
|
+
* const result = await benchmark('Array.map', () => {
|
|
57
|
+
* return largeArray.map(x => x * 2);
|
|
58
|
+
* }, { iterations: 10, warmupRuns: 3 });
|
|
59
|
+
*
|
|
60
|
+
* console.log(`${result.name}: ${result.avgDuration.toFixed(2)}ms avg`);
|
|
61
|
+
* ```
|
|
62
|
+
*/
|
|
63
|
+
// eslint-disable-next-line require-await
|
|
64
|
+
async function benchmark(name, fn, options = {}) {
|
|
65
|
+
const { iterations = 1, warmupRuns = 1, forceGC = false, measureMemory: shouldMeasureMemory = true, timeout = 30000, } = options;
|
|
66
|
+
// Timeout protection
|
|
67
|
+
let timeoutId;
|
|
68
|
+
const timeoutPromise = new Promise((_, reject) => {
|
|
69
|
+
timeoutId = setTimeout(() => reject(new Error(`Benchmark "${name}" timed out after ${timeout}ms`)), timeout);
|
|
70
|
+
});
|
|
71
|
+
const benchmarkPromise = async () => {
|
|
72
|
+
// Warmup runs
|
|
73
|
+
await Promise.all(Array.from({ length: warmupRuns }, async () => {
|
|
74
|
+
await fn();
|
|
75
|
+
}));
|
|
76
|
+
if (forceGC) {
|
|
77
|
+
forceGarbageCollection();
|
|
78
|
+
}
|
|
79
|
+
const durations = [];
|
|
80
|
+
const startMemory = shouldMeasureMemory ? process.memoryUsage().heapUsed : 0;
|
|
81
|
+
const overallStart = performance.now();
|
|
82
|
+
// Actual benchmark runs
|
|
83
|
+
for (let i = 0; i < iterations; i += 1) {
|
|
84
|
+
if (forceGC && i > 0) {
|
|
85
|
+
forceGarbageCollection();
|
|
86
|
+
}
|
|
87
|
+
const iterationStart = performance.now();
|
|
88
|
+
// eslint-disable-next-line no-await-in-loop
|
|
89
|
+
await fn();
|
|
90
|
+
const iterationEnd = performance.now();
|
|
91
|
+
durations.push(iterationEnd - iterationStart);
|
|
92
|
+
}
|
|
93
|
+
const overallEnd = performance.now();
|
|
94
|
+
const endMemory = shouldMeasureMemory ? process.memoryUsage().heapUsed : 0;
|
|
95
|
+
const totalDuration = overallEnd - overallStart;
|
|
96
|
+
const avgDuration = durations.length > 0 ? durations.reduce((sum, d) => sum + d, 0) / durations.length : 0;
|
|
97
|
+
const minDuration = durations.length > 0 ? Math.min(...durations) : 0;
|
|
98
|
+
const maxDuration = durations.length > 0 ? Math.max(...durations) : 0;
|
|
99
|
+
const stdDev = durations.length > 0 ? calculateStdDev(durations) : 0;
|
|
100
|
+
const opsPerSecond = totalDuration > 0 ? iterations / (totalDuration / 1000) : 0;
|
|
101
|
+
const memoryUsed = endMemory - startMemory;
|
|
102
|
+
return {
|
|
103
|
+
name,
|
|
104
|
+
duration: totalDuration,
|
|
105
|
+
opsPerSecond,
|
|
106
|
+
memoryUsed,
|
|
107
|
+
iterations,
|
|
108
|
+
avgDuration,
|
|
109
|
+
stdDev,
|
|
110
|
+
minDuration,
|
|
111
|
+
maxDuration,
|
|
112
|
+
};
|
|
113
|
+
};
|
|
114
|
+
return Promise.race([benchmarkPromise(), timeoutPromise]).finally(() => {
|
|
115
|
+
clearTimeout(timeoutId);
|
|
116
|
+
});
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* Compare multiple functions with the same benchmark configuration
|
|
120
|
+
*
|
|
121
|
+
* @param benchmarks - Object with benchmark names and functions
|
|
122
|
+
* @param options - Benchmark configuration options
|
|
123
|
+
* @returns Promise resolving to array of benchmark results
|
|
124
|
+
*
|
|
125
|
+
* @example
|
|
126
|
+
* ```typescript
|
|
127
|
+
* const results = await compareBenchmarks({
|
|
128
|
+
* 'native map': () => data.map(x => x * 2),
|
|
129
|
+
* 'for loop': () => {
|
|
130
|
+
* const result = [];
|
|
131
|
+
* for (let i = 0; i < data.length; i++) {
|
|
132
|
+
* result.push(data[i] * 2);
|
|
133
|
+
* }
|
|
134
|
+
* return result;
|
|
135
|
+
* }
|
|
136
|
+
* }, { iterations: 5 });
|
|
137
|
+
*
|
|
138
|
+
* results.forEach(result => {
|
|
139
|
+
* console.log(`${result.name}: ${result.avgDuration.toFixed(2)}ms`);
|
|
140
|
+
* });
|
|
141
|
+
* ```
|
|
142
|
+
*/
|
|
143
|
+
async function compareBenchmarks(benchmarks, options = {}) {
|
|
144
|
+
const results = await Promise.all(Object.entries(benchmarks).map(([name, fn]) => benchmark(name, fn, options)));
|
|
145
|
+
return results.sort((a, b) => a.avgDuration - b.avgDuration);
|
|
146
|
+
}
|
|
147
|
+
/**
|
|
148
|
+
* Benchmark suite for organizing and running multiple related benchmarks
|
|
149
|
+
*/
|
|
150
|
+
class BenchmarkSuite {
|
|
151
|
+
constructor(suiteName, options = {}) {
|
|
152
|
+
this.suiteName = suiteName;
|
|
153
|
+
this.benchmarks = [];
|
|
154
|
+
this.options = options;
|
|
155
|
+
}
|
|
156
|
+
/**
|
|
157
|
+
* Add a benchmark to the suite
|
|
158
|
+
*
|
|
159
|
+
* @param name - Name of the benchmark
|
|
160
|
+
* @param fn - Function to benchmark
|
|
161
|
+
* @returns This suite instance for chaining
|
|
162
|
+
*/
|
|
163
|
+
add(name, fn) {
|
|
164
|
+
this.benchmarks.push({ name, fn });
|
|
165
|
+
return this;
|
|
166
|
+
}
|
|
167
|
+
/**
|
|
168
|
+
* Run all benchmarks in the suite
|
|
169
|
+
*
|
|
170
|
+
* @param customOptions - Override suite options for this run
|
|
171
|
+
* @returns Promise resolving to array of benchmark results
|
|
172
|
+
*/
|
|
173
|
+
async run(customOptions) {
|
|
174
|
+
const options = { ...this.options, ...customOptions };
|
|
175
|
+
const results = [];
|
|
176
|
+
// eslint-disable-next-line no-console
|
|
177
|
+
console.log(`\n🚀 Running benchmark suite: ${this.suiteName}`);
|
|
178
|
+
// eslint-disable-next-line no-console
|
|
179
|
+
console.log('='.repeat(60));
|
|
180
|
+
// eslint-disable-next-line no-restricted-syntax
|
|
181
|
+
for (const { name, fn } of this.benchmarks) {
|
|
182
|
+
// eslint-disable-next-line no-console
|
|
183
|
+
console.log(`\n⏱️ Running: ${name}...`);
|
|
184
|
+
try {
|
|
185
|
+
// eslint-disable-next-line no-await-in-loop
|
|
186
|
+
const result = await benchmark(name, fn, options);
|
|
187
|
+
results.push(result);
|
|
188
|
+
// eslint-disable-next-line no-console
|
|
189
|
+
console.log(` ✅ ${result.avgDuration.toFixed(2)}ms avg (${result.opsPerSecond.toFixed(0)} ops/sec)`);
|
|
190
|
+
if (options.measureMemory) {
|
|
191
|
+
// eslint-disable-next-line no-console
|
|
192
|
+
console.log(` 💾 Memory: ${(result.memoryUsed / 1024 / 1024).toFixed(2)}MB`);
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
catch (error) {
|
|
196
|
+
// eslint-disable-next-line no-console
|
|
197
|
+
console.log(` ❌ Failed: ${error instanceof Error ? error.message : 'Unknown error'}`);
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
// Summary
|
|
201
|
+
if (results.length > 1) {
|
|
202
|
+
// eslint-disable-next-line no-console
|
|
203
|
+
console.log(`\n📊 Summary (sorted by performance):`);
|
|
204
|
+
const sortedResults = [...results].sort((a, b) => a.avgDuration - b.avgDuration);
|
|
205
|
+
const fastest = sortedResults[0];
|
|
206
|
+
sortedResults.forEach((result, index) => {
|
|
207
|
+
const relative = result.avgDuration / fastest.avgDuration;
|
|
208
|
+
let icon = ' ';
|
|
209
|
+
if (index === 0)
|
|
210
|
+
icon = '🥇';
|
|
211
|
+
else if (index === 1)
|
|
212
|
+
icon = '🥈';
|
|
213
|
+
else if (index === 2)
|
|
214
|
+
icon = '🥉';
|
|
215
|
+
// eslint-disable-next-line no-console
|
|
216
|
+
console.log(` ${icon} ${result.name.padEnd(25)} | ${result.avgDuration.toFixed(2)}ms | ${relative.toFixed(2)}x`);
|
|
217
|
+
});
|
|
218
|
+
}
|
|
219
|
+
// eslint-disable-next-line no-console
|
|
220
|
+
console.log(`\n✅ Suite completed: ${results.length}/${this.benchmarks.length} benchmarks successful\n`);
|
|
221
|
+
return results;
|
|
222
|
+
}
|
|
223
|
+
/**
|
|
224
|
+
* Get the number of benchmarks in the suite
|
|
225
|
+
*/
|
|
226
|
+
get size() {
|
|
227
|
+
return this.benchmarks.length;
|
|
228
|
+
}
|
|
229
|
+
/**
|
|
230
|
+
* Clear all benchmarks from the suite
|
|
231
|
+
*/
|
|
232
|
+
clear() {
|
|
233
|
+
this.benchmarks = [];
|
|
234
|
+
return this;
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
exports.BenchmarkSuite = BenchmarkSuite;
|
|
238
|
+
/**
|
|
239
|
+
* Create a new benchmark suite
|
|
240
|
+
*
|
|
241
|
+
* @param suiteName - Name of the benchmark suite
|
|
242
|
+
* @param options - Default options for all benchmarks in the suite
|
|
243
|
+
* @returns New BenchmarkSuite instance
|
|
244
|
+
*
|
|
245
|
+
* @example
|
|
246
|
+
* ```typescript
|
|
247
|
+
* const suite = createBenchmarkSuite('String Operations', {
|
|
248
|
+
* iterations: 10,
|
|
249
|
+
* warmupRuns: 2
|
|
250
|
+
* });
|
|
251
|
+
*
|
|
252
|
+
* suite
|
|
253
|
+
* .add('concat', () => str1 + str2)
|
|
254
|
+
* .add('template', () => `${str1}${str2}`)
|
|
255
|
+
* .add('join', () => [str1, str2].join(''));
|
|
256
|
+
*
|
|
257
|
+
* await suite.run();
|
|
258
|
+
* ```
|
|
259
|
+
*/
|
|
260
|
+
function createBenchmarkSuite(suiteName, options) {
|
|
261
|
+
return new BenchmarkSuite(suiteName, options);
|
|
262
|
+
}
|
|
263
|
+
/**
|
|
264
|
+
* Utility to measure memory usage of a function
|
|
265
|
+
*
|
|
266
|
+
* @param fn - Function to measure
|
|
267
|
+
* @param forceGC - Whether to force garbage collection before measuring
|
|
268
|
+
* @returns Memory usage in bytes
|
|
269
|
+
*
|
|
270
|
+
* @example
|
|
271
|
+
* ```typescript
|
|
272
|
+
* const memoryUsed = await measureMemory(() => {
|
|
273
|
+
* return new Array(100000).fill(0).map(x => ({ value: x }));
|
|
274
|
+
* });
|
|
275
|
+
*
|
|
276
|
+
* console.log(`Memory used: ${(memoryUsed / 1024 / 1024).toFixed(2)}MB`);
|
|
277
|
+
* ```
|
|
278
|
+
*/
|
|
279
|
+
async function measureMemory(fn, forceGC = true) {
|
|
280
|
+
if (forceGC) {
|
|
281
|
+
forceGarbageCollection();
|
|
282
|
+
}
|
|
283
|
+
const startMemory = process.memoryUsage().heapUsed;
|
|
284
|
+
await fn();
|
|
285
|
+
const endMemory = process.memoryUsage().heapUsed;
|
|
286
|
+
return endMemory - startMemory;
|
|
287
|
+
}
|
|
288
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"benchmark.js","sourceRoot":"","sources":["../../src/utils/benchmark.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Benchmark utilities for performance testing\n *\n * Provides easy-to-use tools for measuring function performance,\n * memory usage, and creating performance comparisons.\n *\n * @example\n * ```typescript\n * import { benchmark, createBenchmarkSuite } from './benchmark';\n *\n * // Simple benchmark\n * const result = await benchmark('myFunction', () => myFunction(data));\n * console.log(`Duration: ${result.duration}ms`);\n *\n * // Benchmark suite\n * const suite = createBenchmarkSuite('Array Operations');\n * suite.add('native map', () => data.map(x => x * 2));\n * suite.add('custom map', () => customMap(data, x => x * 2));\n * await suite.run();\n * ```\n */\n\n/**\n * Result of a benchmark measurement\n */\nexport type BenchmarkResult = {\n  /** Name of the benchmark */\n  name: string;\n  /** Total duration in milliseconds */\n  duration: number;\n  /** Operations per second */\n  opsPerSecond: number;\n  /** Memory used in bytes (heap difference) */\n  memoryUsed: number;\n  /** Number of iterations run */\n  iterations: number;\n  /** Average duration per iteration in milliseconds */\n  avgDuration: number;\n  /** Standard deviation of durations */\n  stdDev: number;\n  /** Minimum duration observed */\n  minDuration: number;\n  /** Maximum duration observed */\n  maxDuration: number;\n};\n\n/**\n * Configuration options for benchmarks\n */\nexport type BenchmarkOptions = {\n  /** Number of iterations to run (default: 1) */\n  iterations?: number;\n  /** Number of warmup runs before measuring (default: 1) */\n  warmupRuns?: number;\n  /** Whether to force garbage collection between runs (default: false) */\n  forceGC?: boolean;\n  /** Whether to measure memory usage (default: true) */\n  measureMemory?: boolean;\n  /** Timeout in milliseconds for the entire benchmark (default: 30000) */\n  timeout?: number;\n};\n\n/**\n * Benchmark function type\n */\nexport type BenchmarkFunction<T = unknown> = () => T | Promise<T>;\n\n/**\n * Force garbage collection if available\n */\nfunction forceGarbageCollection(): void {\n  if (typeof global !== 'undefined' && global.gc) {\n    global.gc();\n  }\n}\n\n/**\n * Calculate standard deviation of an array of numbers\n */\nfunction calculateStdDev(values: number[]): number {\n  const mean = values.reduce((sum, val) => sum + val, 0) / values.length;\n  const squaredDiffs = values.map((val) => (val - mean) ** 2);\n  const avgSquaredDiff = squaredDiffs.reduce((sum, val) => sum + val, 0) / values.length;\n  return Math.sqrt(avgSquaredDiff);\n}\n\n/**\n * Run a benchmark for a single function\n *\n * @param name - Name of the benchmark\n * @param fn - Function to benchmark\n * @param options - Benchmark configuration options\n * @returns Promise resolving to benchmark results\n *\n * @example\n * ```typescript\n * const result = await benchmark('Array.map', () => {\n *   return largeArray.map(x => x * 2);\n * }, { iterations: 10, warmupRuns: 3 });\n *\n * console.log(`${result.name}: ${result.avgDuration.toFixed(2)}ms avg`);\n * ```\n */\n// eslint-disable-next-line require-await\nexport async function benchmark(\n  name: string,\n  fn: BenchmarkFunction,\n  options: BenchmarkOptions = {},\n): Promise<BenchmarkResult> {\n  const {\n    iterations = 1,\n    warmupRuns = 1,\n    forceGC = false,\n    measureMemory: shouldMeasureMemory = true,\n    timeout = 30000,\n  } = options;\n\n  // Timeout protection\n  let timeoutId: NodeJS.Timeout;\n  const timeoutPromise = new Promise<never>((_, reject) => {\n    timeoutId = setTimeout(\n      () => reject(new Error(`Benchmark \"${name}\" timed out after ${timeout}ms`)),\n      timeout,\n    );\n  });\n\n  const benchmarkPromise = async (): Promise<BenchmarkResult> => {\n    // Warmup runs\n    await Promise.all(\n      Array.from({ length: warmupRuns }, async () => {\n        await fn();\n      }),\n    );\n\n    if (forceGC) {\n      forceGarbageCollection();\n    }\n\n    const durations: number[] = [];\n    const startMemory = shouldMeasureMemory ? process.memoryUsage().heapUsed : 0;\n    const overallStart = performance.now();\n\n    // Actual benchmark runs\n    for (let i = 0; i < iterations; i += 1) {\n      if (forceGC && i > 0) {\n        forceGarbageCollection();\n      }\n\n      const iterationStart = performance.now();\n      // eslint-disable-next-line no-await-in-loop\n      await fn();\n      const iterationEnd = performance.now();\n\n      durations.push(iterationEnd - iterationStart);\n    }\n\n    const overallEnd = performance.now();\n    const endMemory = shouldMeasureMemory ? process.memoryUsage().heapUsed : 0;\n\n    const totalDuration = overallEnd - overallStart;\n    const avgDuration =\n      durations.length > 0 ? durations.reduce((sum, d) => sum + d, 0) / durations.length : 0;\n    const minDuration = durations.length > 0 ? Math.min(...durations) : 0;\n    const maxDuration = durations.length > 0 ? Math.max(...durations) : 0;\n    const stdDev = durations.length > 0 ? calculateStdDev(durations) : 0;\n    const opsPerSecond = totalDuration > 0 ? iterations / (totalDuration / 1000) : 0;\n    const memoryUsed = endMemory - startMemory;\n\n    return {\n      name,\n      duration: totalDuration,\n      opsPerSecond,\n      memoryUsed,\n      iterations,\n      avgDuration,\n      stdDev,\n      minDuration,\n      maxDuration,\n    };\n  };\n\n  return Promise.race([benchmarkPromise(), timeoutPromise]).finally(() => {\n    clearTimeout(timeoutId);\n  });\n}\n\n/**\n * Compare multiple functions with the same benchmark configuration\n *\n * @param benchmarks - Object with benchmark names and functions\n * @param options - Benchmark configuration options\n * @returns Promise resolving to array of benchmark results\n *\n * @example\n * ```typescript\n * const results = await compareBenchmarks({\n *   'native map': () => data.map(x => x * 2),\n *   'for loop': () => {\n *     const result = [];\n *     for (let i = 0; i < data.length; i++) {\n *       result.push(data[i] * 2);\n *     }\n *     return result;\n *   }\n * }, { iterations: 5 });\n *\n * results.forEach(result => {\n *   console.log(`${result.name}: ${result.avgDuration.toFixed(2)}ms`);\n * });\n * ```\n */\nexport async function compareBenchmarks(\n  benchmarks: Record<string, BenchmarkFunction>,\n  options: BenchmarkOptions = {},\n): Promise<BenchmarkResult[]> {\n  const results = await Promise.all(\n    Object.entries(benchmarks).map(([name, fn]) => benchmark(name, fn, options)),\n  );\n\n  return results.sort((a, b) => a.avgDuration - b.avgDuration);\n}\n\n/**\n * Benchmark suite for organizing and running multiple related benchmarks\n */\nexport class BenchmarkSuite {\n  private benchmarks: Array<{ name: string; fn: BenchmarkFunction }> = [];\n\n  private options: BenchmarkOptions;\n\n  constructor(public readonly suiteName: string, options: BenchmarkOptions = {}) {\n    this.options = options;\n  }\n\n  /**\n   * Add a benchmark to the suite\n   *\n   * @param name - Name of the benchmark\n   * @param fn - Function to benchmark\n   * @returns This suite instance for chaining\n   */\n  add(name: string, fn: BenchmarkFunction): this {\n    this.benchmarks.push({ name, fn });\n    return this;\n  }\n\n  /**\n   * Run all benchmarks in the suite\n   *\n   * @param customOptions - Override suite options for this run\n   * @returns Promise resolving to array of benchmark results\n   */\n  async run(customOptions?: BenchmarkOptions): Promise<BenchmarkResult[]> {\n    const options = { ...this.options, ...customOptions };\n    const results: BenchmarkResult[] = [];\n\n    // eslint-disable-next-line no-console\n    console.log(`\\n🚀 Running benchmark suite: ${this.suiteName}`);\n    // eslint-disable-next-line no-console\n    console.log('='.repeat(60));\n\n    // eslint-disable-next-line no-restricted-syntax\n    for (const { name, fn } of this.benchmarks) {\n      // eslint-disable-next-line no-console\n      console.log(`\\n⏱️  Running: ${name}...`);\n\n      try {\n        // eslint-disable-next-line no-await-in-loop\n        const result = await benchmark(name, fn, options);\n        results.push(result);\n\n        // eslint-disable-next-line no-console\n        console.log(\n          `   ✅ ${result.avgDuration.toFixed(2)}ms avg (${result.opsPerSecond.toFixed(\n            0,\n          )} ops/sec)`,\n        );\n        if (options.measureMemory) {\n          // eslint-disable-next-line no-console\n          console.log(`   💾 Memory: ${(result.memoryUsed / 1024 / 1024).toFixed(2)}MB`);\n        }\n      } catch (error) {\n        // eslint-disable-next-line no-console\n        console.log(`   ❌ Failed: ${error instanceof Error ? error.message : 'Unknown error'}`);\n      }\n    }\n\n    // Summary\n    if (results.length > 1) {\n      // eslint-disable-next-line no-console\n      console.log(`\\n📊 Summary (sorted by performance):`);\n      const sortedResults = [...results].sort((a, b) => a.avgDuration - b.avgDuration);\n      const fastest = sortedResults[0];\n\n      sortedResults.forEach((result, index) => {\n        const relative = result.avgDuration / fastest.avgDuration;\n        let icon = '  ';\n        if (index === 0) icon = '🥇';\n        else if (index === 1) icon = '🥈';\n        else if (index === 2) icon = '🥉';\n\n        // eslint-disable-next-line no-console\n        console.log(\n          `   ${icon} ${result.name.padEnd(25)} | ${result.avgDuration.toFixed(\n            2,\n          )}ms | ${relative.toFixed(2)}x`,\n        );\n      });\n    }\n\n    // eslint-disable-next-line no-console\n    console.log(\n      `\\n✅ Suite completed: ${results.length}/${this.benchmarks.length} benchmarks successful\\n`,\n    );\n\n    return results;\n  }\n\n  /**\n   * Get the number of benchmarks in the suite\n   */\n  get size(): number {\n    return this.benchmarks.length;\n  }\n\n  /**\n   * Clear all benchmarks from the suite\n   */\n  clear(): this {\n    this.benchmarks = [];\n    return this;\n  }\n}\n\n/**\n * Create a new benchmark suite\n *\n * @param suiteName - Name of the benchmark suite\n * @param options - Default options for all benchmarks in the suite\n * @returns New BenchmarkSuite instance\n *\n * @example\n * ```typescript\n * const suite = createBenchmarkSuite('String Operations', {\n *   iterations: 10,\n *   warmupRuns: 2\n * });\n *\n * suite\n *   .add('concat', () => str1 + str2)\n *   .add('template', () => `${str1}${str2}`)\n *   .add('join', () => [str1, str2].join(''));\n *\n * await suite.run();\n * ```\n */\nexport function createBenchmarkSuite(\n  suiteName: string,\n  options?: BenchmarkOptions,\n): BenchmarkSuite {\n  return new BenchmarkSuite(suiteName, options);\n}\n\n/**\n * Utility to measure memory usage of a function\n *\n * @param fn - Function to measure\n * @param forceGC - Whether to force garbage collection before measuring\n * @returns Memory usage in bytes\n *\n * @example\n * ```typescript\n * const memoryUsed = await measureMemory(() => {\n *   return new Array(100000).fill(0).map(x => ({ value: x }));\n * });\n *\n * console.log(`Memory used: ${(memoryUsed / 1024 / 1024).toFixed(2)}MB`);\n * ```\n */\nexport async function measureMemory(fn: BenchmarkFunction, forceGC = true): Promise<number> {\n  if (forceGC) {\n    forceGarbageCollection();\n  }\n\n  const startMemory = process.memoryUsage().heapUsed;\n  await fn();\n  const endMemory = process.memoryUsage().heapUsed;\n\n  return endMemory - startMemory;\n}\n"]}
|
package/build/utils/misc.d.ts
CHANGED
|
@@ -137,16 +137,33 @@ export declare const generateRandomString: (length?: number) => string;
|
|
|
137
137
|
*/
|
|
138
138
|
export declare const flattenQueryParams: (qParams: Record<string, any | any[]>) => Record<string, any>;
|
|
139
139
|
/**
|
|
140
|
-
*
|
|
141
|
-
*
|
|
142
|
-
*
|
|
143
|
-
* @
|
|
144
|
-
|
|
145
|
-
|
|
140
|
+
* Locks a property of an object, making it immutable.
|
|
141
|
+
* @param obj - The object to lock the property of.
|
|
142
|
+
* @param key - The key of the property to lock.
|
|
143
|
+
* @returns The object with the property locked.
|
|
144
|
+
*/
|
|
145
|
+
export declare const lockProperty: <T extends object, K extends keyof T>(obj: T, key: K) => T;
|
|
146
|
+
/**
|
|
147
|
+
* Deep freezes an object, making it immutable.
|
|
148
|
+
* @param obj - The object to deep freeze.
|
|
149
|
+
* @returns {T | Readonly<T>} - If `propKey` is provided, returns the original object `T` with `obj[propKey]` deep frozen and `propKey` locked. Otherwise, returns a `Readonly<T>` version of the entire object.
|
|
146
150
|
*
|
|
147
151
|
* @example
|
|
148
152
|
* const config = deepFreeze({ api: { key: 'secret', timeout: 1000 } });
|
|
149
153
|
* // Attempting to modify config.api.key will throw an error in strict mode
|
|
150
154
|
*/
|
|
151
|
-
export declare const deepFreeze: <T
|
|
155
|
+
export declare const deepFreeze: <T extends Record<string, unknown>>(obj: T) => Readonly<T>;
|
|
156
|
+
/**
|
|
157
|
+
* Deep freezes a specific property of an object and locks that property from reassignment.
|
|
158
|
+
* @template T - The type of the object.
|
|
159
|
+
* @param {T} obj - The object containing the property to freeze.
|
|
160
|
+
* @param {keyof T} propKey - The key of the property to deep freeze and lock.
|
|
161
|
+
* @returns {T} - The original object `T` with `obj[propKey]` deep frozen and `propKey` locked.
|
|
162
|
+
*
|
|
163
|
+
* @example
|
|
164
|
+
* const myObject = { data: { value: 123 } };
|
|
165
|
+
* deepFreezeProperty(myObject, 'data');
|
|
166
|
+
* // myObject.data is now immutable, and myObject.data cannot be reassigned.
|
|
167
|
+
*/
|
|
168
|
+
export declare const deepFreezeProperty: <T extends Record<string, unknown>>(obj: T, propKey: keyof T) => T;
|
|
152
169
|
//# sourceMappingURL=misc.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"misc.d.ts","sourceRoot":"","sources":["../../src/utils/misc.ts"],"names":[],"mappings":"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
|
|
1
|
+
{"version":3,"file":"misc.d.ts","sourceRoot":"","sources":["../../src/utils/misc.ts"],"names":[],"mappings":"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"}
|