UNPKG

next

Version:

The React Framework

313 lines (312 loc) 16.2 kB
'use client'; // TODO: Explicitly import from client.browser // eslint-disable-next-line import/no-extraneous-dependencies import { createFromReadableStream as createFromReadableStreamBrowser, createFromFetch as createFromFetchBrowser } from 'react-server-dom-webpack/client'; import { NEXT_ROUTER_STATE_TREE_HEADER, NEXT_RSC_UNION_QUERY, NEXT_URL, RSC_HEADER, RSC_CONTENT_TYPE_HEADER, NEXT_HMR_REFRESH_HEADER, NEXT_DID_POSTPONE_HEADER, NEXT_ROUTER_STALE_TIME_HEADER, NEXT_HTML_REQUEST_ID_HEADER, NEXT_REQUEST_ID_HEADER } from '../app-router-headers'; import { callServer } from '../../app-call-server'; import { findSourceMapURL } from '../../app-find-source-map-url'; import { normalizeFlightData, prepareFlightRouterStateForRequest } from '../../flight-data-helpers'; import { getAppBuildId } from '../../app-build-id'; import { setCacheBustingSearchParam } from './set-cache-busting-search-param'; import { getRenderedSearch, urlToUrlWithoutFlightMarker } from '../../route-params'; import { getDeploymentId } from '../../../shared/lib/deployment-id'; const createFromReadableStream = createFromReadableStreamBrowser; const createFromFetch = createFromFetchBrowser; let createDebugChannel; if (process.env.NODE_ENV !== 'production' && process.env.__NEXT_REACT_DEBUG_CHANNEL) { createDebugChannel = require('../../dev/debug-channel').createDebugChannel; } function doMpaNavigation(url) { return urlToUrlWithoutFlightMarker(new URL(url, location.origin)).toString(); } let isPageUnloading = false; if (typeof window !== 'undefined') { // Track when the page is unloading, e.g. due to reloading the page or // performing hard navigations. This allows us to suppress error logging when // the browser cancels in-flight requests during page unload. window.addEventListener('pagehide', ()=>{ isPageUnloading = true; }); // Reset the flag on pageshow, e.g. when navigating back and the JavaScript // execution context is restored by the browser. window.addEventListener('pageshow', ()=>{ isPageUnloading = false; }); } /** * Fetch the flight data for the provided url. Takes in the current router state * to decide what to render server-side. */ export async function fetchServerResponse(url, options) { const { flightRouterState, nextUrl } = options; const headers = { // Enable flight response [RSC_HEADER]: '1', // Provide the current router state [NEXT_ROUTER_STATE_TREE_HEADER]: prepareFlightRouterStateForRequest(flightRouterState, options.isHmrRefresh) }; if (process.env.NODE_ENV === 'development' && options.isHmrRefresh) { headers[NEXT_HMR_REFRESH_HEADER] = '1'; } if (nextUrl) { headers[NEXT_URL] = nextUrl; } // In static export mode, we need to modify the URL to request the .txt file, // but we should preserve the original URL for the canonical URL and error handling. const originalUrl = url; try { if (process.env.NODE_ENV === 'production') { if (process.env.__NEXT_CONFIG_OUTPUT === 'export') { // In "output: export" mode, we can't rely on headers to distinguish // between HTML and RSC requests. Instead, we append an extra prefix // to the request. url = new URL(url); if (url.pathname.endsWith('/')) { url.pathname += 'index.txt'; } else { url.pathname += '.txt'; } } } // Typically, during a navigation, we decode the response using Flight's // `createFromFetch` API, which accepts a `fetch` promise. // TODO: Remove this check once the old PPR flag is removed const isLegacyPPR = process.env.__NEXT_PPR && !process.env.__NEXT_CACHE_COMPONENTS; const shouldImmediatelyDecode = !isLegacyPPR; const res = await createFetch(url, headers, 'auto', shouldImmediatelyDecode); const responseUrl = urlToUrlWithoutFlightMarker(new URL(res.url)); const canonicalUrl = res.redirected ? responseUrl : originalUrl; const contentType = res.headers.get('content-type') || ''; const interception = !!res.headers.get('vary')?.includes(NEXT_URL); const postponed = !!res.headers.get(NEXT_DID_POSTPONE_HEADER); const staleTimeHeaderSeconds = res.headers.get(NEXT_ROUTER_STALE_TIME_HEADER); const staleTime = staleTimeHeaderSeconds !== null ? parseInt(staleTimeHeaderSeconds, 10) * 1000 : -1; let isFlightResponse = contentType.startsWith(RSC_CONTENT_TYPE_HEADER); if (process.env.NODE_ENV === 'production') { if (process.env.__NEXT_CONFIG_OUTPUT === 'export') { if (!isFlightResponse) { isFlightResponse = contentType.startsWith('text/plain'); } } } // If fetch returns something different than flight response handle it like a mpa navigation // If the fetch was not 200, we also handle it like a mpa navigation if (!isFlightResponse || !res.ok || !res.body) { // in case the original URL came with a hash, preserve it before redirecting to the new URL if (url.hash) { responseUrl.hash = url.hash; } return doMpaNavigation(responseUrl.toString()); } // We may navigate to a page that requires a different Webpack runtime. // In prod, every page will have the same Webpack runtime. // In dev, the Webpack runtime is minimal for each page. // We need to ensure the Webpack runtime is updated before executing client-side JS of the new page. // TODO: This needs to happen in the Flight Client. // Or Webpack needs to include the runtime update in the Flight response as // a blocking script. if (process.env.NODE_ENV !== 'production' && !process.env.TURBOPACK) { await require('../../dev/hot-reloader/app/hot-reloader-app').waitForWebpackRuntimeHotUpdate(); } let flightResponsePromise = res.flightResponse; if (flightResponsePromise === null) { // Typically, `createFetch` would have already started decoding the // Flight response. If it hasn't, though, we need to decode it now. // TODO: This should only be reachable if legacy PPR is enabled (i.e. PPR // without Cache Components). Remove this branch once legacy PPR // is deleted. const flightStream = postponed ? createUnclosingPrefetchStream(res.body) : res.body; flightResponsePromise = createFromNextReadableStream(flightStream, headers); } const flightResponse = await flightResponsePromise; if (getAppBuildId() !== flightResponse.b) { return doMpaNavigation(res.url); } const normalizedFlightData = normalizeFlightData(flightResponse.f); if (typeof normalizedFlightData === 'string') { return doMpaNavigation(normalizedFlightData); } return { flightData: normalizedFlightData, canonicalUrl: canonicalUrl, renderedSearch: getRenderedSearch(res), couldBeIntercepted: interception, prerendered: flightResponse.S, postponed, staleTime, debugInfo: flightResponsePromise._debugInfo ?? null }; } catch (err) { if (!isPageUnloading) { console.error(`Failed to fetch RSC payload for ${originalUrl}. Falling back to browser navigation.`, err); } // If fetch fails handle it like a mpa navigation // TODO-APP: Add a test for the case where a CORS request fails, e.g. external url redirect coming from the response. // See https://github.com/vercel/next.js/issues/43605#issuecomment-1451617521 for a reproduction. return originalUrl.toString(); } } export async function createFetch(url, headers, fetchPriority, shouldImmediatelyDecode, signal) { // TODO: In output: "export" mode, the headers do nothing. Omit them (and the // cache busting search param) from the request so they're // maximally cacheable. if (process.env.__NEXT_TEST_MODE && fetchPriority !== null) { headers['Next-Test-Fetch-Priority'] = fetchPriority; } const deploymentId = getDeploymentId(); if (deploymentId) { headers['x-deployment-id'] = deploymentId; } if (process.env.NODE_ENV !== 'production') { if (self.__next_r) { headers[NEXT_HTML_REQUEST_ID_HEADER] = self.__next_r; } // Create a new request ID for the server action request. The server uses // this to tag debug information sent via WebSocket to the client, which // then routes those chunks to the debug channel associated with this ID. headers[NEXT_REQUEST_ID_HEADER] = crypto.getRandomValues(new Uint32Array(1))[0].toString(16); } const fetchOptions = { // Backwards compat for older browsers. `same-origin` is the default in modern browsers. credentials: 'same-origin', headers, priority: fetchPriority || undefined, signal }; // `fetchUrl` is slightly different from `url` because we add a cache-busting // search param to it. This should not leak outside of this function, so we // track them separately. let fetchUrl = new URL(url); setCacheBustingSearchParam(fetchUrl, headers); let fetchPromise = fetch(fetchUrl, fetchOptions); // Immediately pass the fetch promise to the Flight client so that the debug // info includes the latency from the client to the server. The internal timer // in React starts as soon as `createFromFetch` is called. // // The only case where we don't do this is during a prefetch, because we have // to do some extra processing of the response stream (see // `createUnclosingPrefetchStream`). But this is fine, because a top-level // prefetch response never blocks a navigation; if it hasn't already been // written into the cache by the time the navigation happens, the router will // go straight to a dynamic request. let flightResponsePromise = shouldImmediatelyDecode ? createFromNextFetch(fetchPromise, headers) : null; let browserResponse = await fetchPromise; // If the server responds with a redirect (e.g. 307), and the redirected // location does not contain the cache busting search param set in the // original request, the response is likely invalid — when following the // redirect, the browser forwards the request headers, but since the cache // busting search param is missing, the server will reject the request due to // a mismatch. // // Ideally, we would be able to intercept the redirect response and perform it // manually, instead of letting the browser automatically follow it, but this // is not allowed by the fetch API. // // So instead, we must "replay" the redirect by fetching the new location // again, but this time we'll append the cache busting search param to prevent // a mismatch. // // TODO: We can optimize Next.js's built-in middleware APIs by returning a // custom status code, to prevent the browser from automatically following it. // // This does not affect Server Action-based redirects; those are encoded // differently, as part of the Flight body. It only affects redirects that // occur in a middleware or a third-party proxy. let redirected = browserResponse.redirected; if (process.env.__NEXT_CLIENT_VALIDATE_RSC_REQUEST_HEADERS) { // This is to prevent a redirect loop. Same limit used by Chrome. const MAX_REDIRECTS = 20; for(let n = 0; n < MAX_REDIRECTS; n++){ if (!browserResponse.redirected) { break; } const responseUrl = new URL(browserResponse.url, fetchUrl); if (responseUrl.origin !== fetchUrl.origin) { break; } if (responseUrl.searchParams.get(NEXT_RSC_UNION_QUERY) === fetchUrl.searchParams.get(NEXT_RSC_UNION_QUERY)) { break; } // The RSC request was redirected. Assume the response is invalid. // // Append the cache busting search param to the redirected URL and // fetch again. // TODO: We should abort the previous request. fetchUrl = new URL(responseUrl); setCacheBustingSearchParam(fetchUrl, headers); fetchPromise = fetch(fetchUrl, fetchOptions); flightResponsePromise = shouldImmediatelyDecode ? createFromNextFetch(fetchPromise, headers) : null; browserResponse = await fetchPromise; // We just performed a manual redirect, so this is now true. redirected = true; } } // Remove the cache busting search param from the response URL, to prevent it // from leaking outside of this function. const responseUrl = new URL(browserResponse.url, fetchUrl); responseUrl.searchParams.delete(NEXT_RSC_UNION_QUERY); const rscResponse = { url: responseUrl.href, // This is true if any redirects occurred, either automatically by the // browser, or manually by us. So it's different from // `browserResponse.redirected`, which only tells us whether the browser // followed a redirect, and only for the last response in the chain. redirected, // These can be copied from the last browser response we received. We // intentionally only expose the subset of fields that are actually used // elsewhere in the codebase. ok: browserResponse.ok, headers: browserResponse.headers, body: browserResponse.body, status: browserResponse.status, // This is the exact promise returned by `createFromFetch`. It contains // debug information that we need to transfer to any derived promises that // are later rendered by React. flightResponse: flightResponsePromise }; return rscResponse; } export function createFromNextReadableStream(flightStream, requestHeaders) { return createFromReadableStream(flightStream, { callServer, findSourceMapURL, debugChannel: createDebugChannel && createDebugChannel(requestHeaders) }); } function createFromNextFetch(promiseForResponse, requestHeaders) { return createFromFetch(promiseForResponse, { callServer, findSourceMapURL, debugChannel: createDebugChannel && createDebugChannel(requestHeaders) }); } function createUnclosingPrefetchStream(originalFlightStream) { // When PPR is enabled, prefetch streams may contain references that never // resolve, because that's how we encode dynamic data access. In the decoded // object returned by the Flight client, these are reified into hanging // promises that suspend during render, which is effectively what we want. // The UI resolves when it switches to the dynamic data stream // (via useDeferredValue(dynamic, static)). // // However, the Flight implementation currently errors if the server closes // the response before all the references are resolved. As a cheat to work // around this, we wrap the original stream in a new stream that never closes, // and therefore doesn't error. const reader = originalFlightStream.getReader(); return new ReadableStream({ async pull (controller) { while(true){ const { done, value } = await reader.read(); if (!done) { // Pass to the target stream and keep consuming the Flight response // from the server. controller.enqueue(value); continue; } // The server stream has closed. Exit, but intentionally do not close // the target stream. return; } } }); } //# sourceMappingURL=fetch-server-response.js.map