UNPKG

next

Version:

The React Framework

362 lines (361 loc) • 15.5 kB
import { getOriginalCodeFrame, ignoreListAnonymousStackFramesIfSandwiched } from '../../next-devtools/server/shared'; import { middlewareResponse } from '../../next-devtools/server/middleware-response'; import path from 'path'; import { openFileInEditor } from '../../next-devtools/server/launch-editor'; import { SourceMapConsumer } from 'next/dist/compiled/source-map08'; import { devirtualizeReactServerURL, findApplicableSourceMapPayload } from '../lib/source-maps'; import { findSourceMap } from 'node:module'; import { fileURLToPath, pathToFileURL } from 'node:url'; import { inspect } from 'node:util'; function shouldIgnorePath(modulePath) { return modulePath.includes('node_modules') || // Only relevant for when Next.js is symlinked e.g. in the Next.js monorepo modulePath.includes('next/dist') || modulePath.startsWith('node:'); } const currentSourcesByFile = new Map(); /** * @returns 1-based lines and 1-based columns */ async function batchedTraceSource(project, frame) { const file = frame.file ? decodeURIComponent(frame.file) : undefined; if (!file) return; // For node internals they cannot traced the actual source code with project.traceSource, // we need an early return to indicate it's ignored to avoid the unknown scheme error from `project.traceSource`. if (file.startsWith('node:')) { return { frame: { file, line1: frame.line ?? null, column1: frame.column ?? null, methodName: frame.methodName ?? '<unknown>', ignored: true, arguments: [] }, source: null }; } const currentDirectoryFileUrl = pathToFileURL(process.cwd()).href; const sourceFrame = await project.traceSource(frame, currentDirectoryFileUrl); if (!sourceFrame) { return { frame: { file, line1: frame.line ?? null, column1: frame.column ?? null, methodName: frame.methodName ?? '<unknown>', ignored: shouldIgnorePath(file), arguments: [] }, source: null }; } let source = null; const originalFile = sourceFrame.originalFile; // Don't look up source for node_modules or internals. These can often be large bundled files. const ignored = shouldIgnorePath(originalFile ?? sourceFrame.file) || // isInternal means resource starts with turbopack:///[turbopack] !!sourceFrame.isInternal; if (originalFile && !ignored) { let sourcePromise = currentSourcesByFile.get(originalFile); if (!sourcePromise) { sourcePromise = project.getSourceForAsset(originalFile); currentSourcesByFile.set(originalFile, sourcePromise); setTimeout(()=>{ // Cache file reads for 100ms, as frames will often reference the same // files and can be large. currentSourcesByFile.delete(originalFile); }, 100); } source = await sourcePromise; } // TODO: get ignoredList from turbopack source map const ignorableFrame = { file: sourceFrame.file, line1: sourceFrame.line ?? null, column1: sourceFrame.column ?? null, methodName: // We ignore the sourcemapped name since it won't be the correct name. // The callsite will point to the column of the variable name instead of the // name of the enclosing function. // TODO(NDX-531): Spy on prepareStackTrace to get the enclosing line number for method name mapping. frame.methodName ?? '<unknown>', ignored, arguments: [] }; return { frame: ignorableFrame, source }; } function parseFile(fileParam) { if (!fileParam) { return undefined; } return devirtualizeReactServerURL(fileParam); } function createStackFrames(body) { const { frames, isServer } = body; return frames.map((frame)=>{ const file = parseFile(frame.file); if (!file) { return undefined; } return { file, methodName: frame.methodName ?? '<unknown>', line: frame.line1 ?? undefined, column: frame.column1 ?? undefined, isServer }; }).filter((f)=>f !== undefined); } function createStackFrame(searchParams) { const file = parseFile(searchParams.get('file')); if (!file) { return undefined; } return { file, methodName: searchParams.get('methodName') ?? '<unknown>', line: parseInt(searchParams.get('line1') ?? '0', 10) || undefined, column: parseInt(searchParams.get('column1') ?? '0', 10) || undefined, isServer: searchParams.get('isServer') === 'true' }; } /** * @returns 1-based lines and 1-based columns */ async function nativeTraceSource(frame) { const sourceURL = frame.file; let sourceMapPayload; try { var _findSourceMap; sourceMapPayload = (_findSourceMap = findSourceMap(sourceURL)) == null ? void 0 : _findSourceMap.payload; } catch (cause) { throw Object.defineProperty(new Error(`${sourceURL}: Invalid source map. Only conformant source maps can be used to find the original code.`, { cause }), "__NEXT_ERROR_CODE", { value: "E635", enumerable: false, configurable: true }); } if (sourceMapPayload !== undefined) { let consumer; try { consumer = await new SourceMapConsumer(sourceMapPayload); } catch (cause) { throw Object.defineProperty(new Error(`${sourceURL}: Invalid source map. Only conformant source maps can be used to find the original code.`, { cause }), "__NEXT_ERROR_CODE", { value: "E635", enumerable: false, configurable: true }); } let traced; try { const originalPosition = consumer.originalPositionFor({ line: frame.line ?? 1, // 0-based columns out requires 0-based columns in. column: (frame.column ?? 1) - 1 }); if (originalPosition.source === null) { traced = null; } else { const sourceContent = consumer.sourceContentFor(originalPosition.source, /* returnNullOnMissing */ true) ?? null; traced = { originalPosition, sourceContent }; } } finally{ consumer.destroy(); } if (traced !== null) { var // We ignore the sourcemapped name since it won't be the correct name. // The callsite will point to the column of the variable name instead of the // name of the enclosing function. // TODO(NDX-531): Spy on prepareStackTrace to get the enclosing line number for method name mapping. _frame_methodName_replace, _frame_methodName; const { originalPosition, sourceContent } = traced; const applicableSourceMap = findApplicableSourceMapPayload((frame.line ?? 1) - 1, (frame.column ?? 1) - 1, sourceMapPayload); // TODO(veil): Upstream a method to sourcemap consumer that immediately says if a frame is ignored or not. let ignored = false; if (applicableSourceMap === undefined) { console.error('No applicable source map found in sections for frame', frame); } else { var _applicableSourceMap_ignoreList; // TODO: O(n^2). Consider moving `ignoreList` into a Set const sourceIndex = applicableSourceMap.sources.indexOf(originalPosition.source); ignored = ((_applicableSourceMap_ignoreList = applicableSourceMap.ignoreList) == null ? void 0 : _applicableSourceMap_ignoreList.includes(sourceIndex)) ?? // When sourcemap is not available, fallback to checking `frame.file`. // e.g. In pages router, nextjs server code is not bundled into the page. shouldIgnorePath(frame.file); } const originalStackFrame = { methodName: ((_frame_methodName = frame.methodName) == null ? void 0 : (_frame_methodName_replace = _frame_methodName.replace('__WEBPACK_DEFAULT_EXPORT__', 'default')) == null ? void 0 : _frame_methodName_replace.replace('__webpack_exports__.', '')) || '<unknown>', file: originalPosition.source, line1: originalPosition.line, column1: originalPosition.column === null ? null : originalPosition.column + 1, // TODO: c&p from async createOriginalStackFrame but why not frame.arguments? arguments: [], ignored }; return { frame: originalStackFrame, source: sourceContent }; } } return undefined; } async function createOriginalStackFrame(project, projectPath, frame) { const traced = await nativeTraceSource(frame) ?? // TODO(veil): When would the bundler know more than native? // If it's faster, try the bundler first and fall back to native later. await batchedTraceSource(project, frame); if (!traced) { return null; } let normalizedStackFrameLocation = traced.frame.file; if (normalizedStackFrameLocation !== null && normalizedStackFrameLocation.startsWith('file://')) { normalizedStackFrameLocation = path.relative(projectPath, fileURLToPath(normalizedStackFrameLocation)); } return { originalStackFrame: { arguments: traced.frame.arguments, file: normalizedStackFrameLocation, line1: traced.frame.line1, column1: traced.frame.column1, ignored: traced.frame.ignored, methodName: traced.frame.methodName }, originalCodeFrame: getOriginalCodeFrame(traced.frame, traced.source) }; } export function getOverlayMiddleware({ project, projectPath, isSrcDir }) { return async function(req, res, next) { const { pathname, searchParams } = new URL(req.url, 'http://n'); if (pathname === '/__nextjs_original-stack-frames') { if (req.method !== 'POST') { return middlewareResponse.badRequest(res); } const body = await new Promise((resolve, reject)=>{ let data = ''; req.on('data', (chunk)=>{ data += chunk; }); req.on('end', ()=>resolve(data)); req.on('error', reject); }); const request = JSON.parse(body); const result = await getOriginalStackFrames({ project, projectPath, frames: request.frames, isServer: request.isServer, isEdgeServer: request.isEdgeServer, isAppDirectory: request.isAppDirectory }); ignoreListAnonymousStackFramesIfSandwiched(result); return middlewareResponse.json(res, result); } else if (pathname === '/__nextjs_launch-editor') { const isAppRelativePath = searchParams.get('isAppRelativePath') === '1'; let openEditorResult; if (isAppRelativePath) { const relativeFilePath = searchParams.get('file') || ''; const appPath = path.join('app', isSrcDir ? 'src' : '', relativeFilePath); openEditorResult = await openFileInEditor(appPath, 1, 1, projectPath); } else { const frame = createStackFrame(searchParams); if (!frame) return middlewareResponse.badRequest(res); openEditorResult = await openFileInEditor(frame.file, frame.line ?? 1, frame.column ?? 1, projectPath); } if (openEditorResult.error) { return middlewareResponse.internalServerError(res, openEditorResult.error); } if (!openEditorResult.found) { return middlewareResponse.notFound(res); } return middlewareResponse.noContent(res); } return next(); }; } export function getSourceMapMiddleware(project) { return async function(req, res, next) { const { pathname, searchParams } = new URL(req.url, 'http://n'); if (pathname !== '/__nextjs_source-map') { return next(); } let filename = searchParams.get('filename'); if (!filename) { return middlewareResponse.badRequest(res); } let nativeSourceMap; try { nativeSourceMap = findSourceMap(filename); } catch (cause) { return middlewareResponse.internalServerError(res, Object.defineProperty(new Error(`${filename}: Invalid source map. Only conformant source maps can be used to find the original code.`, { cause }), "__NEXT_ERROR_CODE", { value: "E635", enumerable: false, configurable: true })); } if (nativeSourceMap !== undefined) { const sourceMapPayload = nativeSourceMap.payload; return middlewareResponse.json(res, sourceMapPayload); } try { // Turbopack chunk filenames might be URL-encoded. filename = decodeURI(filename); } catch { return middlewareResponse.badRequest(res); } if (path.isAbsolute(filename)) { filename = pathToFileURL(filename).href; } try { const sourceMapString = await project.getSourceMap(filename); if (sourceMapString) { return middlewareResponse.jsonString(res, sourceMapString); } } catch (cause) { return middlewareResponse.internalServerError(res, Object.defineProperty(new Error(`Failed to get source map for '${filename}'. This is a bug in Next.js`, { cause }), "__NEXT_ERROR_CODE", { value: "E719", enumerable: false, configurable: true })); } middlewareResponse.noContent(res); }; } export async function getOriginalStackFrames({ project, projectPath, frames, isServer, isEdgeServer, isAppDirectory }) { const stackFrames = createStackFrames({ frames, isServer, isEdgeServer, isAppDirectory }); return Promise.all(stackFrames.map(async (frame)=>{ try { const stackFrame = await createOriginalStackFrame(project, projectPath, frame); if (stackFrame === null) { return { status: 'rejected', reason: 'Failed to create original stack frame' }; } return { status: 'fulfilled', value: stackFrame }; } catch (error) { return { status: 'rejected', reason: inspect(error, { colors: false }) }; } })); } //# sourceMappingURL=middleware-turbopack.js.map