-
-
Notifications
You must be signed in to change notification settings - Fork 632
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Refactor serverRenderReactComponent function #1653
Changes from 7 commits
71bba14
448cd5d
fa48d6f
033fa74
044cb13
b86acd1
45b61ba
cdbdc86
d1d9d33
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -6,154 +6,180 @@ import createReactOutput from './createReactOutput'; | |
import { isPromise, isServerRenderHash } from './isServerRenderResult'; | ||
import buildConsoleReplay from './buildConsoleReplay'; | ||
import handleError from './handleError'; | ||
import type { RenderParams, RenderResult, RenderingError, ServerRenderResult } from './types'; | ||
import type { CreateReactOutputResult, RegisteredComponent, RenderParams, RenderResult, RenderingError, ServerRenderResult } from './types'; | ||
|
||
/* eslint-disable @typescript-eslint/no-explicit-any */ | ||
type RenderState = { | ||
result: null | string | Promise<string>; | ||
hasErrors: boolean; | ||
error?: RenderingError; | ||
}; | ||
|
||
function serverRenderReactComponentInternal(options: RenderParams): null | string | Promise<RenderResult> { | ||
const { name, domNodeId, trace, props, railsContext, renderingReturnsPromises, throwJsErrors } = options; | ||
type RenderOptions = { | ||
componentName: string; | ||
domNodeId?: string; | ||
trace?: boolean; | ||
renderingReturnsPromises: boolean; | ||
}; | ||
|
||
let renderResult: null | string | Promise<string> = null; | ||
let hasErrors = false; | ||
let renderingError: null | RenderingError = null; | ||
function validateComponent(componentObj: RegisteredComponent, componentName: string) { | ||
if (componentObj.isRenderer) { | ||
throw new Error(`Detected a renderer while server rendering component '${componentName}'. See https://github.com/shakacode/react_on_rails#renderer-functions`); | ||
} | ||
} | ||
AbanoubGhadban marked this conversation as resolved.
Show resolved
Hide resolved
|
||
|
||
try { | ||
const componentObj = ComponentRegistry.get(name); | ||
if (componentObj.isRenderer) { | ||
throw new Error(`\ | ||
Detected a renderer while server rendering component '${name}'. \ | ||
See https://github.com/shakacode/react_on_rails#renderer-functions`); | ||
function processServerRenderHash(result: ServerRenderResult, options: RenderOptions): RenderState { | ||
const { redirectLocation, routeError } = result; | ||
const hasErrors = !!routeError; | ||
|
||
if (hasErrors) { | ||
console.error(`React Router ERROR: ${JSON.stringify(routeError)}`); | ||
} | ||
|
||
let htmlResult: string; | ||
if (redirectLocation) { | ||
if (options.trace) { | ||
const redirectPath = redirectLocation.pathname + redirectLocation.search; | ||
console.log(`ROUTER REDIRECT: ${options.componentName} to dom node with id: ${options.domNodeId}, redirect to ${redirectPath}`); | ||
} | ||
// For redirects on server rendering, we can't stop Rails from returning the same result. | ||
// Possibly, someday, we could have the rails server redirect. | ||
AbanoubGhadban marked this conversation as resolved.
Show resolved
Hide resolved
|
||
htmlResult = ''; | ||
} else { | ||
htmlResult = result.renderedHtml as string; | ||
} | ||
|
||
return { result: htmlResult, hasErrors }; | ||
} | ||
|
||
function processPromise(result: Promise<unknown>, renderingReturnsPromises: boolean): Promise<string> | string { | ||
if (!renderingReturnsPromises) { | ||
console.error('Your render function returned a Promise, which is only supported by a node renderer, not ExecJS.'); | ||
// If the app is using server rendering with ExecJS, then the promise will not be awaited. | ||
// And when a promise is passed to JSON.stringify, it will be converted to '{}'. | ||
return '{}'; | ||
} | ||
return result as Promise<string>; | ||
} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Ensure that Promise resolves to a string in In the Consider updating the function to enforce that the promise resolves to a string, either by refining the input type or by adding runtime checks. You might update the function signature to reflect the expected resolved type: -function processPromise(result: Promise<unknown>, renderingReturnsPromises: boolean): Promise<string> | string {
+function processPromise(result: Promise<string>, renderingReturnsPromises: boolean): Promise<string> | string { If the
|
||
|
||
const reactRenderingResult = createReactOutput({ | ||
componentObj, | ||
domNodeId, | ||
trace, | ||
props, | ||
railsContext, | ||
}); | ||
|
||
const processServerRenderHash = () => { | ||
// We let the client side handle any redirect | ||
// Set hasErrors in case we want to throw a Rails exception | ||
const { redirectLocation, routeError } = reactRenderingResult as ServerRenderResult; | ||
hasErrors = !!routeError; | ||
|
||
if (hasErrors) { | ||
console.error( | ||
`React Router ERROR: ${JSON.stringify(routeError)}`, | ||
); | ||
} | ||
|
||
if (redirectLocation) { | ||
if (trace) { | ||
const redirectPath = redirectLocation.pathname + redirectLocation.search; | ||
console.log(`\ | ||
ROUTER REDIRECT: ${name} to dom node with id: ${domNodeId}, redirect to ${redirectPath}`, | ||
); | ||
} | ||
// For redirects on server rendering, we can't stop Rails from returning the same result. | ||
// Possibly, someday, we could have the rails server redirect. | ||
return ''; | ||
} | ||
return (reactRenderingResult as ServerRenderResult).renderedHtml as string; | ||
}; | ||
|
||
const processPromise = () => { | ||
if (!renderingReturnsPromises) { | ||
console.error('Your render function returned a Promise, which is only supported by a node renderer, not ExecJS.'); | ||
} | ||
return reactRenderingResult; | ||
}; | ||
|
||
const processReactElement = () => { | ||
try { | ||
return ReactDOMServer.renderToString(reactRenderingResult as ReactElement); | ||
} catch (error) { | ||
console.error(`Invalid call to renderToString. Possibly you have a renderFunction, a function that already | ||
function processReactElement(result: ReactElement): string { | ||
try { | ||
return ReactDOMServer.renderToString(result); | ||
} catch (error) { | ||
console.error(`Invalid call to renderToString. Possibly you have a renderFunction, a function that already | ||
calls renderToString, that takes one parameter. You need to add an extra unused parameter to identify this function | ||
as a renderFunction and not a simple React Function Component.`); | ||
throw error; | ||
} | ||
}; | ||
|
||
if (isServerRenderHash(reactRenderingResult)) { | ||
renderResult = processServerRenderHash(); | ||
} else if (isPromise(reactRenderingResult)) { | ||
renderResult = processPromise() as Promise<string>; | ||
} else { | ||
renderResult = processReactElement(); | ||
} | ||
} catch (e: any) { | ||
if (throwJsErrors) { | ||
throw e; | ||
} | ||
|
||
hasErrors = true; | ||
renderResult = handleError({ | ||
e, | ||
name, | ||
serverSide: true, | ||
}); | ||
renderingError = e; | ||
throw error; | ||
} | ||
} | ||
|
||
const consoleReplayScript = buildConsoleReplay(); | ||
const addRenderingErrors = (resultObject: RenderResult, renderError: RenderingError) => { | ||
// Do not use `resultObject.renderingError = renderError` because JSON.stringify will turn it into '{}'. | ||
resultObject.renderingError = { // eslint-disable-line no-param-reassign | ||
message: renderError.message, | ||
stack: renderError.stack, | ||
}; | ||
}; | ||
function processRenderingResult(result: CreateReactOutputResult, options: RenderOptions): RenderState { | ||
if (isServerRenderHash(result)) { | ||
return processServerRenderHash(result, options); | ||
AbanoubGhadban marked this conversation as resolved.
Show resolved
Hide resolved
|
||
} | ||
if (isPromise(result)) { | ||
return { result: processPromise(result, options.renderingReturnsPromises), hasErrors: false }; | ||
AbanoubGhadban marked this conversation as resolved.
Show resolved
Hide resolved
|
||
} | ||
return { result: processReactElement(result), hasErrors: false }; | ||
} | ||
|
||
if (renderingReturnsPromises) { | ||
const resolveRenderResult = async () => { | ||
let promiseResult: RenderResult; | ||
|
||
try { | ||
promiseResult = { | ||
html: await renderResult, | ||
consoleReplayScript, | ||
hasErrors, | ||
}; | ||
} catch (e: any) { | ||
if (throwJsErrors) { | ||
throw e; | ||
} | ||
promiseResult = { | ||
html: handleError({ | ||
e, | ||
name, | ||
serverSide: true, | ||
}), | ||
consoleReplayScript, | ||
hasErrors: true, | ||
}; | ||
renderingError = e; | ||
} | ||
|
||
if (renderingError !== null) { | ||
addRenderingErrors(promiseResult, renderingError); | ||
} | ||
|
||
return promiseResult; | ||
}; | ||
|
||
return resolveRenderResult(); | ||
function handleRenderingError(e: unknown, options: { componentName: string, throwJsErrors: boolean }) { | ||
if (options.throwJsErrors) { | ||
throw e; | ||
} | ||
const error = e instanceof Error ? e : new Error(String(e)); | ||
return { | ||
hasErrors: true, | ||
result: handleError({ e: error, name: options.componentName, serverSide: true }), | ||
error, | ||
}; | ||
} | ||
|
||
const result: RenderResult = { | ||
html: renderResult as string, | ||
function createResultObject(html: string | null, consoleReplayScript: string, hasErrors: boolean, error?: RenderingError): RenderResult { | ||
AbanoubGhadban marked this conversation as resolved.
Show resolved
Hide resolved
|
||
return { | ||
html, | ||
consoleReplayScript, | ||
hasErrors, | ||
renderingError: error && { message: error.message, stack: error.stack }, | ||
}; | ||
} | ||
|
||
async function createPromiseResult( | ||
renderState: RenderState & { result: Promise<string> }, | ||
consoleReplayScript: string, | ||
componentName: string, | ||
throwJsErrors: boolean | ||
): Promise<RenderResult> { | ||
try { | ||
const html = await renderState.result; | ||
return createResultObject(html, consoleReplayScript, renderState.hasErrors, renderState.error); | ||
} catch (e: unknown) { | ||
const errorRenderState = handleRenderingError(e, { componentName, throwJsErrors }); | ||
return createResultObject(errorRenderState.result, consoleReplayScript, errorRenderState.hasErrors, errorRenderState.error); | ||
} | ||
} | ||
|
||
function createFinalResult( | ||
renderState: RenderState, | ||
componentName: string, | ||
throwJsErrors: boolean | ||
): null | string | Promise<RenderResult> { | ||
// Console history is stored globally in `console.history`. | ||
// If node renderer is handling a render request that returns a promise, | ||
// It can handle another request while awaiting the promise. | ||
// To prevent cross-request console logs leakage between these requests, | ||
// we build the consoleReplayScript before awaiting any promises. | ||
// The console history is reset after the synchronous part of each request. | ||
// This causes console logs happening during async operations to not be captured. | ||
const consoleReplayScript = buildConsoleReplay(); | ||
|
||
const { result } = renderState; | ||
if (isPromise(result)) { | ||
return createPromiseResult({ ...renderState, result }, consoleReplayScript, componentName, throwJsErrors); | ||
} | ||
|
||
return JSON.stringify(createResultObject(result, consoleReplayScript, renderState.hasErrors, renderState.error)); | ||
} | ||
|
||
function serverRenderReactComponentInternal(options: RenderParams): null | string | Promise<RenderResult> { | ||
const { name: componentName, domNodeId, trace, props, railsContext, renderingReturnsPromises, throwJsErrors } = options; | ||
|
||
let renderState: RenderState = { | ||
result: null, | ||
hasErrors: false, | ||
}; | ||
|
||
if (renderingError) { | ||
addRenderingErrors(result, renderingError); | ||
try { | ||
const componentObj = ComponentRegistry.get(componentName); | ||
validateComponent(componentObj, componentName); | ||
|
||
// Renders the component or executes the render function | ||
// - If the registered component is a React element or component, it renders it | ||
// - If it's a render function, it executes the function and processes the result: | ||
// - For React elements or components, it renders them | ||
// - For promises, it returns them without awaiting (for async rendering) | ||
// - For other values (e.g., strings), it returns them directly | ||
// Note: Only synchronous operations are performed at this stage | ||
const reactRenderingResult = createReactOutput({ componentObj, domNodeId, trace, props, railsContext }); | ||
|
||
// Processes the result from createReactOutput: | ||
// 1. Converts React elements to HTML strings | ||
// 2. Returns rendered HTML from serverRenderHash | ||
// 3. Handles promises for async rendering | ||
renderState = processRenderingResult(reactRenderingResult, { componentName, domNodeId, trace, renderingReturnsPromises }); | ||
} catch (e: unknown) { | ||
renderState = handleRenderingError(e, { componentName, throwJsErrors }); | ||
} | ||
|
||
return JSON.stringify(result); | ||
// Finalize the rendering result and prepare it for server response | ||
// 1. Builds the consoleReplayScript for client-side console replay | ||
// 2. Extract the result from promise (if needed) by awaiting it | ||
// 3. Constructs a JSON object with the following properties: | ||
// - html: string | null (The rendered component HTML) | ||
// - consoleReplayScript: string (Script to replay console outputs on the client) | ||
// - hasErrors: boolean (Indicates if any errors occurred during rendering) | ||
// - renderingError: Error | null (The error object if an error occurred, null otherwise) | ||
// 4. For Promise results, it awaits resolution before creating the final JSON | ||
return createFinalResult(renderState, componentName, throwJsErrors); | ||
} | ||
|
||
const serverRenderReactComponent: typeof serverRenderReactComponentInternal = (options) => { | ||
|
@@ -165,4 +191,5 @@ const serverRenderReactComponent: typeof serverRenderReactComponentInternal = (o | |
console.history = []; | ||
} | ||
}; | ||
|
||
export default serverRenderReactComponent; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
🛠️ Refactor suggestion
Consider renaming the function and improving promise detection.
The changes to the
isPromise
function improve its flexibility and type safety. However, there are a couple of points to consider:The function name
isPromise
might be misleading as it now accepts non-promise types. Consider renaming it to something likeisPromiseLike
orcouldBePromise
to better reflect its broader input types.The current implementation might not correctly identify all promise-like objects. The
then
property check is a good start, but it's not foolproof. Consider using a more robust check.Here's a suggested improvement:
This implementation:
isPromiseLike
.testValue
is not null and is an object.then
property is a function.These changes make the function more accurate in identifying promise-like objects while maintaining its flexibility.