Skip to content

Commit db535ea

Browse files
authoredMar 11, 2025··
feat: add support for setup/cleanup functions during each test run (#563)
1 parent b34459f commit db535ea

10 files changed

+121
-2
lines changed
 

‎.changeset/silent-cups-fail.md

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
---
2+
'reassure': minor
3+
'@callstack/reassure-measure': minor
4+
---
5+
6+
feat: add support for setup/cleanup functions during each test run

‎README.md

+12
Original file line numberDiff line numberDiff line change
@@ -374,6 +374,8 @@ interface MeasureRendersOptions {
374374
wrapper?: React.ComponentType<{ children: ReactElement }>;
375375
scenario?: (view?: RenderResult) => Promise<any>;
376376
writeFile?: boolean;
377+
beforeEach?: () => Promise<void> | void;
378+
afterEach?: () => Promise<void> | void;
377379
}
378380
```
379381

@@ -383,6 +385,8 @@ interface MeasureRendersOptions {
383385
- **`wrapper`**: React component, such as a `Provider`, which the `ui` will be wrapped with. Note: the render duration of the `wrapper` itself is excluded from the results; only the wrapped component is measured.
384386
- **`scenario`**: a custom async function, which defines user interaction within the UI by utilising RNTL or RTL functions
385387
- **`writeFile`**: should write output to file (default `true`)
388+
- **`beforeEach`**: function to execute before each test run.
389+
- **`afterEach`**: function to execute after each test run.
386390

387391
#### `measureFunction` function
388392

@@ -403,13 +407,17 @@ interface MeasureFunctionOptions {
403407
warmupRuns?: number;
404408
removeOutliers?: boolean;
405409
writeFile?: boolean;
410+
beforeEach?: () => Promise<void> | void;
411+
afterEach?: () => Promise<void> | void;
406412
}
407413
```
408414

409415
- **`runs`**: number of runs per series for the particular test
410416
- **`warmupRuns`**: number of additional warmup runs that will be done and discarded before the actual runs
411417
- **`removeOutliers`**: should remove statistical outlier results (default: `true`)
412418
- **`writeFile`**: should write output to file (default `true`)
419+
- **`beforeEach`**: function to execute before each test run.
420+
- **`afterEach`**: function to execute after each test run.
413421

414422
#### `measureAsyncFunction` function
415423

@@ -432,13 +440,17 @@ interface MeasureAsyncFunctionOptions {
432440
warmupRuns?: number;
433441
removeOutliers?: boolean;
434442
writeFile?: boolean;
443+
beforeEach?: () => Promise<void> | void;
444+
afterEach?: () => Promise<void> | void;
435445
}
436446
```
437447

438448
- **`runs`**: number of runs per series for the particular test
439449
- **`warmupRuns`**: number of additional warmup runs that will be done and discarded before the actual runs
440450
- **`removeOutliers`**: should remove statistical outlier results (default: `true`)
441451
- **`writeFile`**: should write output to file (default `true`)
452+
- **`beforeEach`**: function to execute before each test run.
453+
- **`afterEach`**: function to execute after each test run.
442454

443455
### Configuration
444456

‎docusaurus/docs/api.md

+12
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,8 @@ interface MeasureRendersOptions {
5454
wrapper?: React.ComponentType<{ children: ReactElement }>;
5555
scenario?: (view?: RenderResult) => Promise<any>;
5656
writeFile?: boolean;
57+
beforeEach?: () => Promise<void> | void;
58+
afterEach?: () => Promise<void> | void;
5759
}
5860
```
5961
@@ -63,6 +65,8 @@ interface MeasureRendersOptions {
6365
- **`wrapper`**: React component, such as a `Provider`, which the `ui` will be wrapped with. Note: the render duration of the `wrapper` itself is excluded from the results, only the wrapped component is measured.
6466
- **`scenario`**: a custom async function, which defines user interaction within the ui by utilized RNTL functions
6567
- **`writeFile`**: should write output to file (default `true`)
68+
- **`beforeEach`**: function to execute before each test run.
69+
- **`afterEach`**: function to execute after each test run.
6670
6771
### `measureFunction` function {#measure-function}
6872
@@ -95,13 +99,17 @@ interface MeasureFunctionOptions {
9599
warmupRuns?: number;
96100
removeOutliers?: boolean;
97101
writeFile?: boolean;
102+
beforeEach?: () => Promise<void> | void;
103+
afterEach?: () => Promise<void> | void;
98104
}
99105
```
100106
101107
- **`runs`**: number of runs per series for the particular test
102108
- **`warmupRuns`**: number of additional warmup runs that will be done and discarded before the actual runs
103109
- **`removeOutliers`**: should remove statistical outlier results (default: `true`)
104110
- **`writeFile`**: should write output to file (default `true`)
111+
- **`beforeEach`**: function to execute before each test run.
112+
- **`afterEach`**: function to execute after each test run.
105113
106114
### `measureAsyncFunction` function {#measure-async-function}
107115
@@ -142,13 +150,17 @@ interface MeasureAsyncFunctionOptions {
142150
warmupRuns?: number;
143151
removeOutliers?: boolean;
144152
writeFile?: boolean;
153+
beforeEach?: () => Promise<void> | void;
154+
afterEach?: () => Promise<void> | void;
145155
}
146156
```
147157
148158
- **`runs`**: number of runs per series for the particular test
149159
- **`warmupRuns`**: number of additional warmup runs that will be done and discarded before the actual runs
150160
- **`removeOutliers`**: should remove statistical outlier results (default: `true`)
151161
- **`writeFile`**: (default `true`) should write output to file
162+
- **`beforeEach`**: function to execute before each test run.
163+
- **`afterEach`**: function to execute after each test run.
152164
153165
## Configuration
154166

‎packages/measure/src/__tests__/measure-async-function.test.tsx

+20
Original file line numberDiff line numberDiff line change
@@ -45,6 +45,26 @@ test('measureAsyncFunction applies "warmupRuns" option', async () => {
4545
expect(results.stdevCount).toBe(0);
4646
});
4747

48+
test('measureAsyncFunction executes setup and cleanup functions for each run', async () => {
49+
const fn = jest.fn(() => Promise.resolve().then(() => fib(5)));
50+
const beforeFn = jest.fn();
51+
const afterFn = jest.fn();
52+
const results = await measureAsyncFunction(fn, {
53+
runs: 10,
54+
warmupRuns: 1,
55+
writeFile: false,
56+
beforeEach: beforeFn,
57+
afterEach: afterFn,
58+
});
59+
60+
expect(beforeFn).toHaveBeenCalledTimes(11);
61+
expect(fn).toHaveBeenCalledTimes(11);
62+
expect(afterFn).toHaveBeenCalledTimes(11);
63+
expect(results.runs).toBe(10);
64+
expect(results.durations.length + (results.outlierDurations?.length ?? 0)).toBe(10);
65+
expect(results.counts).toHaveLength(10);
66+
});
67+
4868
const errorsToIgnore = ['❌ Measure code is running under incorrect Node.js configuration.'];
4969
const realConsole = jest.requireActual('console') as Console;
5070

‎packages/measure/src/__tests__/measure-function.test.tsx

+20
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,26 @@ test('measureFunction applies "warmupRuns" option', async () => {
5757
expect(results.stdevCount).toBe(0);
5858
});
5959

60+
test('measureFunction executes setup and cleanup functions for each run', async () => {
61+
const fn = jest.fn(() => fib(5));
62+
const beforeFn = jest.fn();
63+
const afterFn = jest.fn();
64+
const results = await measureFunction(fn, {
65+
runs: 10,
66+
warmupRuns: 1,
67+
writeFile: false,
68+
beforeEach: beforeFn,
69+
afterEach: afterFn,
70+
});
71+
72+
expect(beforeFn).toHaveBeenCalledTimes(11);
73+
expect(fn).toHaveBeenCalledTimes(11);
74+
expect(afterFn).toHaveBeenCalledTimes(11);
75+
expect(results.runs).toBe(10);
76+
expect(results.durations.length + (results.outlierDurations?.length ?? 0)).toBe(10);
77+
expect(results.counts).toHaveLength(10);
78+
});
79+
6080
const errorsToIgnore = ['❌ Measure code is running under incorrect Node.js configuration.'];
6181
const realConsole = jest.requireActual('console') as Console;
6282

‎packages/measure/src/__tests__/measure-renders.test.tsx

+21
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,27 @@ test('measureRenders applies "warmupRuns" option', async () => {
3838
expect(results.stdevCount).toBe(0);
3939
});
4040

41+
test('measureRenders executes setup and cleanup functions for each run', async () => {
42+
const scenario = jest.fn(() => Promise.resolve(null));
43+
const beforeFn = jest.fn();
44+
const afterFn = jest.fn();
45+
const results = await measureRenders(<View />, {
46+
runs: 10,
47+
warmupRuns: 1,
48+
scenario,
49+
writeFile: false,
50+
beforeEach: beforeFn,
51+
afterEach: afterFn,
52+
});
53+
54+
expect(beforeFn).toHaveBeenCalledTimes(11);
55+
expect(scenario).toHaveBeenCalledTimes(11);
56+
expect(afterFn).toHaveBeenCalledTimes(11);
57+
expect(results.runs).toBe(10);
58+
expect(results.durations.length + (results.outlierDurations?.length ?? 0)).toBe(10);
59+
expect(results.counts).toHaveLength(10);
60+
});
61+
4162
test('measureRenders should log error when running under incorrect node flags', async () => {
4263
jest.spyOn(realConsole, 'error').mockImplementation((message) => {
4364
if (!errorsToIgnore.some((error) => message.includes(error))) {

‎packages/measure/src/measure-async-function.tsx

+4
Original file line numberDiff line numberDiff line change
@@ -31,10 +31,14 @@ async function measureAsyncFunctionInternal(
3131

3232
const runResults: RunResult[] = [];
3333
for (let i = 0; i < runs + warmupRuns; i += 1) {
34+
await options?.beforeEach?.();
35+
3436
const timeStart = getCurrentTime();
3537
await fn();
3638
const timeEnd = getCurrentTime();
3739

40+
await options?.afterEach?.();
41+
3842
const duration = timeEnd - timeStart;
3943
runResults.push({ duration, count: 1 });
4044
}

‎packages/measure/src/measure-function.tsx

+8-2
Original file line numberDiff line numberDiff line change
@@ -8,10 +8,12 @@ export interface MeasureFunctionOptions {
88
warmupRuns?: number;
99
removeOutliers?: boolean;
1010
writeFile?: boolean;
11+
beforeEach?: () => Promise<void> | void;
12+
afterEach?: () => Promise<void> | void;
1113
}
1214

1315
export async function measureFunction(fn: () => void, options?: MeasureFunctionOptions): Promise<MeasureResults> {
14-
const stats = measureFunctionInternal(fn, options);
16+
const stats = await measureFunctionInternal(fn, options);
1517

1618
if (options?.writeFile !== false) {
1719
await writeTestStats(stats, 'function');
@@ -20,7 +22,7 @@ export async function measureFunction(fn: () => void, options?: MeasureFunctionO
2022
return stats;
2123
}
2224

23-
function measureFunctionInternal(fn: () => void, options?: MeasureFunctionOptions): MeasureResults {
25+
async function measureFunctionInternal(fn: () => void, options?: MeasureFunctionOptions): Promise<MeasureResults> {
2426
const runs = options?.runs ?? config.runs;
2527
const warmupRuns = options?.warmupRuns ?? config.warmupRuns;
2628
const removeOutliers = options?.removeOutliers ?? config.removeOutliers;
@@ -29,10 +31,14 @@ function measureFunctionInternal(fn: () => void, options?: MeasureFunctionOption
2931

3032
const runResults: RunResult[] = [];
3133
for (let i = 0; i < runs + warmupRuns; i += 1) {
34+
await options?.beforeEach?.();
35+
3236
const timeStart = getCurrentTime();
3337
fn();
3438
const timeEnd = getCurrentTime();
3539

40+
await options?.afterEach?.();
41+
3642
const duration = timeEnd - timeStart;
3743
runResults.push({ duration, count: 1 });
3844
}

‎packages/measure/src/measure-renders.tsx

+6
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,8 @@ export interface MeasureRendersOptions {
2020
wrapper?: React.ComponentType<{ children: React.ReactElement }>;
2121
scenario?: (screen: any) => Promise<any>;
2222
writeFile?: boolean;
23+
beforeEach?: () => Promise<void> | void;
24+
afterEach?: () => Promise<void> | void;
2325
}
2426

2527
export async function measureRenders(
@@ -69,6 +71,8 @@ async function measureRendersInternal(
6971
let initialRenderCount = 0;
7072

7173
for (let iteration = 0; iteration < runs + warmupRuns; iteration += 1) {
74+
await options?.beforeEach?.();
75+
7276
let duration = 0;
7377
let count = 0;
7478
let renderResult: any = null;
@@ -108,6 +112,8 @@ async function measureRendersInternal(
108112
cleanup();
109113
global.gc?.();
110114

115+
await options?.afterEach?.();
116+
111117
runResults.push({ duration, count });
112118
}
113119

‎packages/reassure/README.md

+12
Original file line numberDiff line numberDiff line change
@@ -374,6 +374,8 @@ interface MeasureRendersOptions {
374374
wrapper?: React.ComponentType<{ children: ReactElement }>;
375375
scenario?: (view?: RenderResult) => Promise<any>;
376376
writeFile?: boolean;
377+
beforeEach?: () => Promise<void> | void;
378+
afterEach?: () => Promise<void> | void;
377379
}
378380
```
379381

@@ -383,6 +385,8 @@ interface MeasureRendersOptions {
383385
- **`wrapper`**: React component, such as a `Provider`, which the `ui` will be wrapped with. Note: the render duration of the `wrapper` itself is excluded from the results; only the wrapped component is measured.
384386
- **`scenario`**: a custom async function, which defines user interaction within the UI by utilising RNTL or RTL functions
385387
- **`writeFile`**: should write output to file (default `true`)
388+
- **`beforeEach`**: function to execute before each test run.
389+
- **`afterEach`**: function to execute after each test run.
386390

387391
#### `measureFunction` function
388392

@@ -403,13 +407,17 @@ interface MeasureFunctionOptions {
403407
warmupRuns?: number;
404408
removeOutliers?: boolean;
405409
writeFile?: boolean;
410+
beforeEach?: () => Promise<void> | void;
411+
afterEach?: () => Promise<void> | void;
406412
}
407413
```
408414

409415
- **`runs`**: number of runs per series for the particular test
410416
- **`warmupRuns`**: number of additional warmup runs that will be done and discarded before the actual runs
411417
- **`removeOutliers`**: should remove statistical outlier results (default: `true`)
412418
- **`writeFile`**: should write output to file (default `true`)
419+
- **`beforeEach`**: function to execute before each test run.
420+
- **`afterEach`**: function to execute after each test run.
413421

414422
#### `measureAsyncFunction` function
415423

@@ -432,13 +440,17 @@ interface MeasureAsyncFunctionOptions {
432440
warmupRuns?: number;
433441
removeOutliers?: boolean;
434442
writeFile?: boolean;
443+
beforeEach?: () => Promise<void> | void;
444+
afterEach?: () => Promise<void> | void;
435445
}
436446
```
437447

438448
- **`runs`**: number of runs per series for the particular test
439449
- **`warmupRuns`**: number of additional warmup runs that will be done and discarded before the actual runs
440450
- **`removeOutliers`**: should remove statistical outlier results (default: `true`)
441451
- **`writeFile`**: should write output to file (default `true`)
452+
- **`beforeEach`**: function to execute before each test run.
453+
- **`afterEach`**: function to execute after each test run.
442454

443455
### Configuration
444456

0 commit comments

Comments
 (0)
Please sign in to comment.