Skip to content

Commit 9d1b321

Browse files
committedJan 7, 2023
feat!: implement default filepath inference using Error stack trace
BREAKING CHANGE: `babelOptions.filename` is now set to `filepath` by default rather than `undefined`.
1 parent 2efbe55 commit 9d1b321

File tree

2 files changed

+105
-10
lines changed

2 files changed

+105
-10
lines changed
 

‎src/plugin-tester.ts

+78-3
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,12 @@ import {
2828

2929
import type { Class } from 'type-fest';
3030

31+
const parseErrorStackRegExp =
32+
/at (?<fn>\S+) (?:.*? )?\(?(?<path>(?:\/|file:).*?)(?:\)|$)/i;
33+
34+
const parseScriptFilepathRegExp =
35+
/\/babel-plugin-tester\/(dist|src)\/(index|plugin-tester)\.(j|t)s$/;
36+
3137
export default pluginTester;
3238

3339
/**
@@ -110,8 +116,7 @@ export function pluginTester(options: PluginTesterOptions = {}) {
110116
const baseConfig: PartialPluginTesterBaseConfig = {
111117
babel: rawBaseConfig.babel || require('@babel/core'),
112118
baseBabelOptions: rawBaseConfig.babelOptions,
113-
// TODO: implement default filepath inference using Error stack trace
114-
filepath: rawBaseConfig.filepath ?? rawBaseConfig.filename,
119+
filepath: rawBaseConfig.filepath ?? rawBaseConfig.filename ?? tryInferFilepath(),
115120
endOfLine: rawBaseConfig.endOfLine,
116121
baseSetup: rawBaseConfig.setup,
117122
baseTeardown: rawBaseConfig.teardown,
@@ -159,6 +164,76 @@ export function pluginTester(options: PluginTesterOptions = {}) {
159164
return undefined;
160165
}
161166
}
167+
168+
function tryInferFilepath() {
169+
const oldStackTraceLimit = Error.stackTraceLimit;
170+
Error.stackTraceLimit = Number.POSITIVE_INFINITY;
171+
172+
try {
173+
let inferredFilepath: string | undefined = undefined;
174+
// ? Turn the V8 call stack into function names and file paths
175+
const reversedCallStack = (
176+
new Error('faux error').stack
177+
?.split('\n')
178+
.map((line) => {
179+
const { fn: functionName, path: filePath } =
180+
line.match(parseErrorStackRegExp)?.groups || {};
181+
182+
return functionName && filePath
183+
? {
184+
functionName,
185+
// ? Paranoid just in case the script name/path has colons
186+
filePath: filePath.split(':').slice(0, -2).join(':')
187+
}
188+
: undefined;
189+
})
190+
.filter(<T>(o: T): o is NonNullable<T> => Boolean(o)) || []
191+
).reverse();
192+
193+
// TODO: debug statement here displaying reversed call stack contents
194+
195+
if (reversedCallStack?.length) {
196+
// TODO: debug statements below
197+
const referenceIndex = findReferenceStackIndex(reversedCallStack);
198+
199+
if (referenceIndex) {
200+
inferredFilepath = reversedCallStack.at(referenceIndex - 1)?.filePath;
201+
}
202+
}
203+
204+
// TODO: debug statement here outputting inferredFilepath
205+
206+
return inferredFilepath;
207+
} finally {
208+
Error.stackTraceLimit = oldStackTraceLimit;
209+
}
210+
211+
function findReferenceStackIndex(
212+
reversedCallStack: { functionName: string; filePath: string }[]
213+
) {
214+
// ? Different realms might have slightly different stacks depending on
215+
// ? which file was imported. Return the first one found.
216+
return [
217+
reversedCallStack.findIndex(({ functionName, filePath }) => {
218+
return (
219+
functionName == 'defaultPluginTester' &&
220+
parseScriptFilepathRegExp.test(filePath)
221+
);
222+
}),
223+
reversedCallStack.findIndex(({ functionName, filePath }) => {
224+
return (
225+
functionName == 'pluginTester' && parseScriptFilepathRegExp.test(filePath)
226+
);
227+
}),
228+
reversedCallStack.findIndex(({ functionName, filePath }) => {
229+
return (
230+
functionName == 'resolveBaseConfig' &&
231+
parseScriptFilepathRegExp.test(filePath)
232+
);
233+
})
234+
].find((ndx) => ndx != -1);
235+
}
236+
}
162237
}
163238

164239
function normalizeTests() {
@@ -480,7 +555,7 @@ export function pluginTester(options: PluginTesterOptions = {}) {
480555
{ [$type]: 'test-object' } as const,
481556
{ babelOptions: baseBabelOptions },
482557
{
483-
babelOptions: { filename: getAbsolutePath(filepath, codeFixture) }
558+
babelOptions: { filename: getAbsolutePath(filepath, codeFixture) ?? filepath }
484559
},
485560
{ babelOptions },
486561
{

‎test/plugin-tester.test.ts

+27-7
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,8 @@ import { unstringSnapshotSerializer } from '../src/serializers/unstring-snapshot
1616
import {
1717
type PluginTesterOptions,
1818
runPluginUnderTestHere,
19-
runPresetUnderTestHere
19+
runPresetUnderTestHere,
20+
pluginTester
2021
} from '../src/index';
2122

2223
import {
@@ -45,7 +46,8 @@ import {
4546
runPluginTesterExpectThrownExceptionWhenCapturingError,
4647
getFixturePath,
4748
getFixtureContents,
48-
requireFixtureOptions
49+
requireFixtureOptions,
50+
getPendingJestTests
4951
} from './helpers';
5052

5153
import type { AnyFunction } from '@xunnamius/jest-types';
@@ -891,21 +893,39 @@ describe('tests targeting the PluginTesterOptions interface', () => {
891893
await runPluginTester(
892894
getDummyPluginOptions({
893895
tests: [simpleTest],
894-
fixtures: 'fixtures/simple'
896+
fixtures: '../fixtures/simple'
895897
})
896898
);
897899

898900
await runPluginTester(
899901
getDummyPresetOptions({
900902
tests: [simpleTest],
901-
fixtures: 'fixtures/simple'
903+
fixtures: '../fixtures/simple'
902904
})
903905
);
904906

907+
const fixtureFilename = getFixturePath('simple/fixture/code.js');
908+
const testObjectFilename = path.resolve(__dirname, './helpers/index.ts');
909+
910+
expect(transformAsyncSpy.mock.calls).toMatchObject([
911+
[expect.any(String), expect.objectContaining({ filename: fixtureFilename })],
912+
[expect.any(String), expect.objectContaining({ filename: testObjectFilename })],
913+
[expect.any(String), expect.objectContaining({ filename: fixtureFilename })],
914+
[expect.any(String), expect.objectContaining({ filename: testObjectFilename })]
915+
]);
916+
917+
jest.clearAllMocks();
918+
919+
pluginTester({
920+
plugin: () => ({ visitor: {} }),
921+
tests: [simpleTest],
922+
fixtures: 'fixtures/simple'
923+
});
924+
925+
await Promise.all(getPendingJestTests());
926+
905927
expect(transformAsyncSpy.mock.calls).toMatchObject([
906-
[expect.any(String), expect.objectContaining({ filename: __filename })],
907-
[expect.any(String), expect.objectContaining({ filename: __filename })],
908-
[expect.any(String), expect.objectContaining({ filename: __filename })],
928+
[expect.any(String), expect.objectContaining({ filename: fixtureFilename })],
909929
[expect.any(String), expect.objectContaining({ filename: __filename })]
910930
]);
911931
});

0 commit comments

Comments
 (0)
Please sign in to comment.