From d5223bee11d50cf40f51872b01a899cb15633663 Mon Sep 17 00:00:00 2001 From: ST-DDT Date: Sun, 27 Oct 2024 00:07:16 +0200 Subject: [PATCH 1/3] test: patch and filter seeded run stacktraces --- test/support/seeded-runs.ts | 51 ++++++++++++++++++++++++++++++++++--- vitest.config.ts | 10 ++++++++ 2 files changed, 57 insertions(+), 4 deletions(-) diff --git a/test/support/seeded-runs.ts b/test/support/seeded-runs.ts index dee0051c11b..cf3ff1d1179 100644 --- a/test/support/seeded-runs.ts +++ b/test/support/seeded-runs.ts @@ -134,11 +134,13 @@ class TestGenerator< * * @param method The method name to call. * @param args The arguments to call it with. + * @param extraStackFrames Additional stack frames to add into the stacktrace. * @param repetitions The number of times to call it. */ private callAndVerify>( method: TMethodName, args: Parameters, + extraStackFrames: string[], repetitions: number = 1 ): void { this.setup(); @@ -148,8 +150,12 @@ class TestGenerator< } for (let i = 0; i < repetitions; i++) { - const value = callable(...args); - expect(value).toMatchSnapshot(); + try { + const value = callable(...args); + expect(value).toMatchSnapshot(); + } catch (error: unknown) { + throw patchExtraStackFrames(error, extraStackFrames); + } } } @@ -182,10 +188,12 @@ class TestGenerator< */ itRepeated(method: NoArgsMethodOf, repetitions: number): this { this.expectNotTested(method); + const extraStackFrames = collectExtraStackFrames(); vi_it(method, () => this.callAndVerify( method, [] as unknown as Parameters]>, + extraStackFrames, repetitions ) ); @@ -233,7 +241,8 @@ class TestGenerator< const tester: MethodTester = { it(name: string, ...args: Parameters) { expectVariantNotTested(name); - vi_it(name, () => callAndVerify(method, args)); + const extraStackFrames = collectExtraStackFrames(name.length + 7); + vi_it(name, () => callAndVerify(method, args, extraStackFrames)); return tester; }, itRepeated( @@ -242,7 +251,10 @@ class TestGenerator< ...args: Parameters ) { expectVariantNotTested(name); - vi_it(name, () => callAndVerify(method, args, repetitions)); + const extraStackFrames = collectExtraStackFrames(name.length + 7); + vi_it(name, () => + callAndVerify(method, args, extraStackFrames, repetitions) + ); return tester; }, }; @@ -288,6 +300,37 @@ class TestGenerator< } } +function collectExtraStackFrames(extraOffset: number = 0): string[] { + const stack = new Error('collect').stack; + if (stack == null) { + return []; + } + + return stack + .split('\n') + .filter((e) => e.replaceAll('\\', '/').includes('/test/')) + .filter((e) => !e.replaceAll('\\', '/').includes('/test/support/')) + .map((e) => + e.replace(/:(\d+)$/, (_, value: string) => `:${+value + extraOffset}`) + ); +} + +function patchExtraStackFrames( + error: unknown, + extraStackFrames: string[] +): unknown { + if (error instanceof Error && error.stack != null) { + const stack = error.stack.split('\n'); + const index = stack.findLastIndex((e) => + e.replaceAll('\\', '/').includes('/test/support/') + ); + stack.splice(index + 1, 0, ...extraStackFrames); + error.stack = stack.join('\n'); + } + + return error; +} + /** * Simple interface for a test generator for a given method. */ diff --git a/vitest.config.ts b/vitest.config.ts index ad95ef967cf..94991a6f777 100644 --- a/vitest.config.ts +++ b/vitest.config.ts @@ -20,6 +20,16 @@ export default defineConfig({ seed: VITEST_SEQUENCE_SEED, shuffle: true, }, + onStackTrace(_, { file }) { + if ( + file.includes('/src/internal/locale-proxy') || + file.includes('/test/support/') + ) { + return false; + } + + return true; + }, typecheck: { enabled: true, include: ['test/**/*.spec-d.ts'], From 4296be8f19f39e13c1966520de4ba21f67c2ca88 Mon Sep 17 00:00:00 2001 From: ST-DDT Date: Sun, 27 Oct 2024 01:05:05 +0200 Subject: [PATCH 2/3] chore: lazy stackframes + docs --- test/support/seeded-runs.ts | 41 ++++++++++++++++++++++++++----------- 1 file changed, 29 insertions(+), 12 deletions(-) diff --git a/test/support/seeded-runs.ts b/test/support/seeded-runs.ts index cf3ff1d1179..30f5e39ddc3 100644 --- a/test/support/seeded-runs.ts +++ b/test/support/seeded-runs.ts @@ -140,7 +140,7 @@ class TestGenerator< private callAndVerify>( method: TMethodName, args: Parameters, - extraStackFrames: string[], + extraStackFrames: () => string[], repetitions: number = 1 ): void { this.setup(); @@ -300,31 +300,48 @@ class TestGenerator< } } -function collectExtraStackFrames(extraOffset: number = 0): string[] { +/** + * Lazily collects the current call stack with an additional offset. + * + * Vitest's stacktraces only contain the stacktrace from inside `it(name, () => { here })`. + * This method collects the location where the `it` block is created instead of executed. + * The stack frames can then later be added to the error stack to provide a more accurate location. + * + * @param extraOffset The additional offset to add to the column numbers to account for the name of the test. + */ +function collectExtraStackFrames(extraOffset: number = 0): () => string[] { const stack = new Error('collect').stack; if (stack == null) { - return []; + return () => []; } - return stack - .split('\n') - .filter((e) => e.replaceAll('\\', '/').includes('/test/')) - .filter((e) => !e.replaceAll('\\', '/').includes('/test/support/')) - .map((e) => - e.replace(/:(\d+)$/, (_, value: string) => `:${+value + extraOffset}`) - ); + return () => + stack + .split('\n') + .map((e) => e.replaceAll('\\', '/')) + .filter((e) => e.includes('/test/')) // exclude node_modules + .filter((e) => !e.includes('/test/support/')) // exclude this file + .map((e) => + e.replace(/:(\d+)$/, (_, column: string) => `:${+column + extraOffset}`) + ); } +/** + * Modifies the error stack to include the given additional stack frames. + * + * @param error The error to modify. + * @param extraStackFrames The additional stack frames to add after this file. + */ function patchExtraStackFrames( error: unknown, - extraStackFrames: string[] + extraStackFrames: () => string[] ): unknown { if (error instanceof Error && error.stack != null) { const stack = error.stack.split('\n'); const index = stack.findLastIndex((e) => e.replaceAll('\\', '/').includes('/test/support/') ); - stack.splice(index + 1, 0, ...extraStackFrames); + stack.splice(index + 1, 0, ...extraStackFrames()); error.stack = stack.join('\n'); } From ff61ddc74362039fb64442b61fbcf11f95dafd05 Mon Sep 17 00:00:00 2001 From: ST-DDT Date: Sun, 27 Oct 2024 19:34:21 +0100 Subject: [PATCH 3/3] chore: apply review suggestions --- test/support/seeded-runs.ts | 18 ++++++++++++------ 1 file changed, 12 insertions(+), 6 deletions(-) diff --git a/test/support/seeded-runs.ts b/test/support/seeded-runs.ts index 30f5e39ddc3..66d6df3c4dd 100644 --- a/test/support/seeded-runs.ts +++ b/test/support/seeded-runs.ts @@ -241,7 +241,9 @@ class TestGenerator< const tester: MethodTester = { it(name: string, ...args: Parameters) { expectVariantNotTested(name); - const extraStackFrames = collectExtraStackFrames(name.length + 7); + const extraStackFrames = collectExtraStackFrames( + /* t. */ `it('${name}', `.length // ...args) + ); vi_it(name, () => callAndVerify(method, args, extraStackFrames)); return tester; }, @@ -251,7 +253,9 @@ class TestGenerator< ...args: Parameters ) { expectVariantNotTested(name); - const extraStackFrames = collectExtraStackFrames(name.length + 7); + const extraStackFrames = collectExtraStackFrames( + /* t. */ `itRepeated('${name}', ${repetitions}, `.length // ...args) + ); vi_it(name, () => callAndVerify(method, args, extraStackFrames, repetitions) ); @@ -318,7 +322,7 @@ function collectExtraStackFrames(extraOffset: number = 0): () => string[] { return () => stack .split('\n') - .map((e) => e.replaceAll('\\', '/')) + .map((e) => e.replaceAll('\\', '/')) // Windows to Linux paths .filter((e) => e.includes('/test/')) // exclude node_modules .filter((e) => !e.includes('/test/support/')) // exclude this file .map((e) => @@ -327,10 +331,10 @@ function collectExtraStackFrames(extraOffset: number = 0): () => string[] { } /** - * Modifies the error stack to include the given additional stack frames. + * Modifies the error stack to include the given additional stack frames after the last occurrence of this file. * * @param error The error to modify. - * @param extraStackFrames The additional stack frames to add after this file. + * @param extraStackFrames The additional stack frames to add. */ function patchExtraStackFrames( error: unknown, @@ -339,7 +343,9 @@ function patchExtraStackFrames( if (error instanceof Error && error.stack != null) { const stack = error.stack.split('\n'); const index = stack.findLastIndex((e) => - e.replaceAll('\\', '/').includes('/test/support/') + e + .replaceAll('\\', '/') // Windows to Linux paths + .includes('/test/support/') ); stack.splice(index + 1, 0, ...extraStackFrames()); error.stack = stack.join('\n');