From 1580a4340a2b2541551da5165545b2febb7f03c6 Mon Sep 17 00:00:00 2001 From: Sebastian Markbage Date: Wed, 13 Mar 2024 14:28:18 -0400 Subject: [PATCH] Support Lazy but error if an element is passed to a Reply --- .../src/ReactFlightReplyClient.js | 112 ++++++++++++++---- .../src/__tests__/ReactFlightDOMReply-test.js | 34 ++++++ scripts/error-codes/codes.json | 3 +- 3 files changed, 126 insertions(+), 23 deletions(-) diff --git a/packages/react-client/src/ReactFlightReplyClient.js b/packages/react-client/src/ReactFlightReplyClient.js index 6bae3d1a14de5..dd4cd5527a100 100644 --- a/packages/react-client/src/ReactFlightReplyClient.js +++ b/packages/react-client/src/ReactFlightReplyClient.js @@ -14,6 +14,8 @@ import type { RejectedThenable, ReactCustomFormAction, } from 'shared/ReactTypes'; +import type {LazyComponent} from 'react/src/ReactLazy'; + import {enableRenderableContext} from 'shared/ReactFeatureFlags'; import { @@ -84,9 +86,9 @@ export type ReactServerValue = type ReactServerObject = {+[key: string]: ReactServerValue}; -// function serializeByValueID(id: number): string { -// return '$' + id.toString(16); -// } +function serializeByValueID(id: number): string { + return '$' + id.toString(16); +} function serializePromiseID(id: number): string { return '$@' + id.toString(16); @@ -206,6 +208,78 @@ export function processReply( } if (typeof value === 'object') { + switch ((value: any).$$typeof) { + case REACT_ELEMENT_TYPE: { + throw new Error( + 'React Element cannot be passed to Server Functions from the Client.' + + (__DEV__ ? describeObjectForErrorMessage(parent, key) : ''), + ); + } + case REACT_LAZY_TYPE: { + // Resolve lazy as if it wasn't here. In the future this will be encoded as a Promise. + const lazy: LazyComponent = (value: any); + const payload = lazy._payload; + const init = lazy._init; + if (formData === null) { + // Upgrade to use FormData to allow us to stream this value. + formData = new FormData(); + } + try { + const resolvedModel = init(payload); + // We always outline this as a separate part even though we could inline it + // because it ensures a more deterministic encoding. + pendingParts++; + const lazyId = nextPartId++; + const partJSON = JSON.stringify(resolvedModel, resolveToJSON); + // $FlowFixMe[incompatible-type] We know it's not null because we assigned it above. + const data: FormData = formData; + // eslint-disable-next-line react-internal/safe-string-coercion + data.append(formFieldPrefix + lazyId, partJSON); + pendingParts--; + return serializeByValueID(lazyId); + } catch (x) { + if ( + typeof x === 'object' && + x !== null && + typeof x.then === 'function' + ) { + // Suspended + pendingParts++; + const lazyId = nextPartId++; + const thenable: Thenable = (x: any); + thenable.then( + partValue => { + try { + const partJSON = JSON.stringify(partValue, resolveToJSON); + // $FlowFixMe[incompatible-type] We know it's not null because we assigned it above. + const data: FormData = formData; + // eslint-disable-next-line react-internal/safe-string-coercion + data.append(formFieldPrefix + lazyId, partJSON); + pendingParts--; + if (pendingParts === 0) { + resolve(data); + } + } catch (reason) { + reject(reason); + } + }, + reason => { + // In the future we could consider serializing this as an error + // that throws on the server instead. + reject(reason); + }, + ); + return serializeByValueID(lazyId); + } else { + // In the future we could consider serializing this as an error + // that throws on the server instead. + reject(x); + return null; + } + } + } + } + // $FlowFixMe[method-unbinding] if (typeof value.then === 'function') { // We assume that any object with a .then property is a "Thenable" type, @@ -219,14 +293,18 @@ export function processReply( const thenable: Thenable = (value: any); thenable.then( partValue => { - const partJSON = JSON.stringify(partValue, resolveToJSON); - // $FlowFixMe[incompatible-type] We know it's not null because we assigned it above. - const data: FormData = formData; - // eslint-disable-next-line react-internal/safe-string-coercion - data.append(formFieldPrefix + promiseId, partJSON); - pendingParts--; - if (pendingParts === 0) { - resolve(data); + try { + const partJSON = JSON.stringify(partValue, resolveToJSON); + // $FlowFixMe[incompatible-type] We know it's not null because we assigned it above. + const data: FormData = formData; + // eslint-disable-next-line react-internal/safe-string-coercion + data.append(formFieldPrefix + promiseId, partJSON); + pendingParts--; + if (pendingParts === 0) { + resolve(data); + } + } catch (reason) { + reject(reason); } }, reason => { @@ -294,17 +372,7 @@ export function processReply( ); } if (__DEV__) { - if ((value: any).$$typeof === REACT_ELEMENT_TYPE) { - console.error( - 'React Element cannot be passed to Server Functions from the Client.%s', - describeObjectForErrorMessage(parent, key), - ); - } else if ((value: any).$$typeof === REACT_LAZY_TYPE) { - console.error( - 'React Lazy cannot be passed to Server Functions from the Client.%s', - describeObjectForErrorMessage(parent, key), - ); - } else if ( + if ( (value: any).$$typeof === (enableRenderableContext ? REACT_CONTEXT_TYPE : REACT_PROVIDER_TYPE) ) { diff --git a/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMReply-test.js b/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMReply-test.js index 1162d1d0fe738..f173b84755730 100644 --- a/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMReply-test.js +++ b/packages/react-server-dom-webpack/src/__tests__/ReactFlightDOMReply-test.js @@ -17,6 +17,7 @@ global.TextDecoder = require('util').TextDecoder; // let serverExports; let webpackServerMap; +let React; let ReactServerDOMServer; let ReactServerDOMClient; @@ -31,6 +32,7 @@ describe('ReactFlightDOMReply', () => { const WebpackMock = require('./utils/WebpackMock'); // serverExports = WebpackMock.serverExports; webpackServerMap = WebpackMock.webpackServerMap; + React = require('react'); ReactServerDOMServer = require('react-server-dom-webpack/server.browser'); jest.resetModules(); ReactServerDOMClient = require('react-server-dom-webpack/client'); @@ -241,4 +243,36 @@ describe('ReactFlightDOMReply', () => { } expect(error.message).toBe('Connection closed.'); }); + + it('resolves a promise and includes its value', async () => { + let resolve; + const promise = new Promise(r => (resolve = r)); + const bodyPromise = ReactServerDOMClient.encodeReply({promise: promise}); + resolve('Hi'); + const result = await ReactServerDOMServer.decodeReply(await bodyPromise); + expect(await result.promise).toBe('Hi'); + }); + + it('resolves a React.lazy and includes its value', async () => { + let resolve; + const lazy = React.lazy(() => new Promise(r => (resolve = r))); + const bodyPromise = ReactServerDOMClient.encodeReply({lazy: lazy}); + resolve('Hi'); + const result = await ReactServerDOMServer.decodeReply(await bodyPromise); + expect(result.lazy).toBe('Hi'); + }); + + it('errors when called with JSX by default', async () => { + let error; + try { + await ReactServerDOMClient.encodeReply(
); + } catch (x) { + error = x; + } + expect(error).toEqual( + expect.objectContaining({ + message: expect.stringContaining(''), + }), + ); + }); }); diff --git a/scripts/error-codes/codes.json b/scripts/error-codes/codes.json index 1614f95b86b42..6131ba981b0a6 100644 --- a/scripts/error-codes/codes.json +++ b/scripts/error-codes/codes.json @@ -494,5 +494,6 @@ "506": "Functions are not valid as a child of Client Components. This may happen if you return %s instead of <%s /> from render. Or maybe you meant to call this function rather than return it.%s", "507": "Expected the last optional `callback` argument to be a function. Instead received: %s.", "508": "The first argument must be a React class instance. Instead received: %s.", - "509": "ReactDOM: Unsupported Legacy Mode API." + "509": "ReactDOM: Unsupported Legacy Mode API.", + "510": "React Element cannot be passed to Server Functions from the Client.%s" }