React Server Side Rendering and Server Components
Adding Server-Side Rendering
SSR focuses on initial page load, sending pre-rendered HTML to the client that must then be hydrated with downloaded JavaScript before it behaves as a typical React app. SSR also only happens one time: when directly navigating to a page.
Debunking the Myth: SSR Isn’t Expensive: https://t3.gg/blog/post/ssr-is-not-expensive
- SSR Overhead Is Minimal
- SSR Can Actually Reduce Other Cloud Costs
- SSR Delivers a Better User Experience
Let’s create a simple React component App. We will render this component on the server-side and hydrate it on the client-side.
// client/components/App/index.js
import React from 'react'
const App = () => (
<>
<div>Hello World</div>
<button onClick={e => alert('Hello You!')}>Say Hello</button>
</>
);
export default App;
Then create an Express server and define a route that serves an HTML page when a user visits http://localhost:3000
.
// server/index.js
import React from 'react'
import ReactDOMServer from 'react-dom/server'
import express from 'express'
import App from '../client/components/App'
const app = express()
const port = 3000
// This is the local static server that serves the client-side bundles.
const cdnHost = `http://localhost:5000`;
app.get('/', (req, res) => {
// This turns the React component App into an HTML string
const jsx = ReactDOMServer.renderToString(<App />)
const clientBundleStyle = `<link rel="stylesheet" href="${cdnHost}/styles/bundle.css">`
// This loads the JS code to “hydrate” the markup with interactivity.
const clientBundleScript = `<script src="${cdnHost}/scripts/bundle.js"></script>`
res.send(`
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>My SSR App</title>
${clientBundleStyle}
</head>
<body>
<div id='ssr-app'>${jsx}</div>
${clientBundleScript}
</body>
</html>
`)
})
app.listen(port, () => {
console.log(`App listening on http://localhost:${port}`)
})
In the client-side entry point, we will “hydrate” the React component that was SSR-ed into the root DOM container with the ID “ssr-app”.
// ./client/index.js
import React from 'react';
import ReactDOM from 'react-dom';
import App from './components/App';
ReactDOM.hydrate(<App />, document.getElementById('ssr-app'));
It’s extremely important that SSR React output (HTML) and CSR React output (HTML) are matching, otherwise React will not be able to render and attach event listeners properly. If you follow semantic HTML principles, most of your app should work even before React has hydrated. Links can be followed, forms can be submitted, accordions can be expanded and collapsed (using <details>
and <summary>
). For most projects, it’s fine if it takes a few seconds for React to hydrate.
For Vue server-side rendering, check out https://vuejs.org/guide/scaling-up/ssr.html
Vite provides built-in support for Vue server-side rendering, but it is intentionally low-level (for library and framework authors). If you wish to go directly with Vite, check out vite-plugin-ssr, a community plugin that abstracts away many challenging details for you.
React hydration error
While rendering your application, there was a difference between the React tree that was pre-rendered from the server and the React tree that was rendered during the first render in the browser (hydration).
When the React app runs on the client for the first time, it builds up a mental picture of what the DOM should look like, by mounting all of your components. Then it squints at the DOM nodes already on the page, and tries to fit the two together. We’re rendering one thing on the server, but then telling React to expect something else on the client.
To avoid issues, we need to ensure that the hydrated app matches the original HTML. When the React app adopts the DOM during hydration, useEffect
hasn’t been called yet, and so we’re meeting React’s expectation. Immediately after this comparison, we trigger a re-render, and this allows React to do a proper reconciliation. It’ll notice that there’s some new content to render here.
'use client';
import React, { useState, useEffect } from 'react';
interface ClientOnlyProps {
children: React.ReactNode;
}
const ClientOnly: React.FC<ClientOnlyProps> = ({
children
}) => {
const [hasMounted, setHasMounted] = useState(false);
// `useEffect` doesn't run during SSR. It runs until the JavaScript is downloaded.
useEffect(() => {
setHasMounted(true);
}, [])
if (!hasMounted) return null;
return (
<>
{children}
</>
);
};
export default ClientOnly;
Understand the “children pattern”
React components re-render themselves and all their children when the state is updated. In this case, on every mouse move the state of MovingComponent
is updated, its re-render is triggered, and as a result, ChildComponent
will re-render as well.
const MovingComponent = () => {
const [state, setState] = useState({ x: 100, y: 100 });
return (
<div
onMouseMove={(e) => setState({ x: e.clientX - 20, y: e.clientY - 20 })}
style={{ left: state.x, top: state.y }}
>
<ChildComponent />
</div>
);
};
The way to fight this, other than React.memo
, is to extract ChildComponent
outside and pass it as children. React “children” is just a prop. When children are passed through props, React doesn’t recreate them on each render. The child component’s element is created when the JSX is evaluated. Once created, it’s just passed down as a prop reference that stays stable across re-renders.
// https://www.developerway.com/posts/react-elements-children-parents
const MovingComponent = ({ children }) => {
const [state, setState] = useState({ x: 100, y: 100 });
return (
<div
onMouseMove={(e) => setState({ x: e.clientX - 20, y: e.clientY - 20 })}
style={{ left: state.x, top: state.y }}>
// children now will not be re-rendered!
{children}
</div>
);
};
const SomeOutsideComponent = () => {
return (
<MovingComponent>
<ChildComponent />
</MovingComponent>
);
};
React.memo
is a higher order component that accepts another component as a prop. It will only render the component if there is any change in the props. (Hey React, I know that this component is pure. You don’t need to re-render it unless its props change.)
useMemo
is used to memoize a calculation result, which focuses on avoiding heavy calculation.
useCallback
will return a memoized version of the callback that only changes if one of the inputs has changed. This is useful when passing callbacks to optimized child components that rely on reference equality to prevent unnecessary renders. Note that useCallback(fn, deps)
is equivalent to useMemo(() => fn, deps)
.
const PageMemoized = React.memo(Page);
const App = () => {
const [state, setState] = useState(1);
const onClick = useCallback(() => {
console.log('Do something on click');
}, []);
return (
// will NOT re-render because onClick is memoized
<PageMemoized onClick={onClick} />
// WILL re-render because value is not memoized
<PageMemoized onClick={onClick} value={[1, 2, 3]} />
);
};
useCallback
anduseMemo
for props don’t prevent re-renders by themselves. You can probably remove 90% of alluseMemo
anduseCallback
in your app right now, and the app will be fine and might even become slightly faster.
Add File-System Based Routing and Data Fetching into the server
Learn from https://www.youtube.com/watch?v=3RzhNYhjVAw&t=460s
- server side rendering
- file based routing
- fetch data as early as possible (from client
useEffect
to the server before rendering) - renderToString vs. renderToPipeableStream
import React from "react";
import express from "express";
// renderToPipeableStream is better
import { renderToString } from "react-dom/server";
import { readdirSync } from "fs";
import { join } from "path";
const app = express();
app.use(express.static("./dist"));
const pages = readdirSync(join(process.cwd(), "pages")).map(
file => file.split(".")[0]
);
pages.forEach((page) => {
app.get(`/${page}`, async (req, res) => {
const mod = await import(`./pages/${page}`);
// This is why nextjs needs default export
const Component = mod.default;
let props = {};
// getServerSideProps: Data Fetching (Server-Side) before rendering (only works at the page level)
// export const gSSP = async () => await getStuff();
if (mod.gSSP) {
props = await mod.gSSP(req);
}
res.send(`
<!DOCTYPE html>
<html>
<head>
<title>SSR</title>
</head>
<body>
<div id='root'>${renderToString(<Component {...props} />)}</div>
<script src="/client.js"></script>
</body>
</html>
`)
// replace `res.send` with `renderToPipeableStream` to use React concurrent features
// const { pipe } = renderToPipeableStream(<Component {...props} />, { ... })
// pipe(res)
});
});
app.listen(3000, () => {
console.info("App is live on port 3000");
});
Write React Server Components from Scratch
Before React Server Components, all React components are “client” components — they are all run in the browser. RSC makes it possible for some components to be rendered by the server, and some components to be rendered by the browser. Server Components are not a replacement for SSR. They render exclusively on the server. Their code isn’t included in the JS bundle, and so they never hydrate or re-render. With only SSR, we haven’t been able to do server-exclusive work within our components (e.g. access database), because that same code would re-run in the browser.
- Server Component: Fetch data; Access backend resources directly; Keep large dependencies on the server.
- Client Component: Add interactivity and event listeners (
onClick()
); Use State and Lifecycle Effects (useState()
,useEffect()
); Use browser-only APIs.
A common misconception here is that components with "use client"
only run in browser. Client components still get pre-rendered to the initial HTML on the server (SSR). The "use client"
doesn’t mean the component is “client only”, it means that we send the code for this component to the client and hydrate it.
Btw, wrapping your root layout in the client component does not automatically turn your entire app into a client rendering. The client component gets hydrated on the client. The children can stay server components (do not hydrate on the client).
"use client";
export function Providers({ children }: { children: React.ReactNode }) {
const theme = ...
return <ThemeProvider theme={theme}>{children}</ThemeProvider>
}
What is RSC wire format?
The server sends a serialized tree (similar to JSON but with “holes”) to the browser, and the browser can do the work of deserializing it, filling the client placeholders with the actual client components, and rendering the end result.Essentially, we’re telling React “Hey, I know you’re missing the server component code, but don’t worry: here’s what it rendered”. We send along the rendered value, the virtual representation that was generated by the server. When React loads on the client, it re-uses that description instead of re-generating it.
How is the Next.js App Router related to Server Components?
If a page uses server-side rendering, the page HTML is generated on each request. Next.js used to allow us to do so by usinggetServerSideProps
, which will be called by the server on every request.Next.js 13.4 introduced the App Router with new features, conventions, and support for React Server Components. Components in the app directory are React Server Components by default.
"use client"
directive used to mark components as Client Components. Server and Client Components can be interleaved in the same component tree, with React handling the merging of both environments.Next.js App Router wanted to solve the client-server waterfalls (move client-server REST fetches to the server using React Server Components in a single roundtrip). This meant the server had to sometimes be dynamic, sacrificing the great initial loading performance. Next uses partial prerendering (https://partialprerendering.com), where the static “shell” of a dynamic page is prerendered at build time and served immediately on request, with the dynamic parts streamed in, to solve this tradeoff.
Compare for server side rendering and server components:
- https://github.com/TejasQ/makeshift-next.js/tree/spoiled
- https://github.com/TejasQ/react-server-components-from-scratch/tree/spoild
// React Components (used in server-side rendering)
import React from "react";
export const getStuff = async () => {
return fetch("https://dog.ceo/api/breeds/list/all")
.then((response) => response.json())
.then((data) => {
return Object.keys(data.message);
});
};
// This code runs on the server and on the client
const Breeds = ({ initialBreeds }) => {
const [breeds, setBreeds] = React.useState(initialBreeds);
React.useEffect(() => {
getStuff().then(setBreeds);
}, []);
return (
<ul>
{breeds.map((breed) => (
<li key={breed}>
<a href={`/detail?breed=${breed}`}>{breed}</a>
</li>
))}
</ul>
);
};
export default Breeds;
// This code only runs on the server (and only works at the route level)
export const gSSP = async () => {
const data = await getStuff();
return {
initialBreeds: data,
};
};
// React Server Components, notice the difference from above
// The big difference is that we've never before had a way
// to run server-exclusive code inside our components.
import React from "react";
const List = async () => {
const breeds = await fetch("https://dog.ceo/api/breeds/list/all")
.then((r) => r.json())
.then((data) => Object.keys(data.message));
return (
<ul>
{breeds.map((breed) => (
<li key={breed}>
<a href={`/detail?breed=${breed}`}>{breed}</a>
</li>
))}
</ul>
);
};
export default List;
Besides a <script>
tag that loads up the JS bundle (client components used in our application), we also tells React “Hey, I know you’re missing the server components code, but don’t worry: here’s what it rendered”.
app.get("/:path", async (req, res) => {
const page = await import(
join(process.cwd(), "dist", "pages", req.params.path)
);
const Component = page.default;
// `createReactTree` is a method to turn jsx into "a big object" that React can recognize
// {
// $$typeof: Symbol("react.element"),
// type: "div", // -> typeof jsx.type === "string"
// props: { title: "oh my" },
// ...
// }
// {
// $$typeof: Symbol.for("react.element"),
// type: MyComponent // -> typeof jsx.type === "function"
// props: { children: "oh my" },
// ...
// }
//
// https://github.com/TejasQ/react-server-components-from-scratch/blob/spoild/server.tsx#L37
const clientJsx = await createReactTree(
<Layout bgColor="white">
<Component {...req.query} />
</Layout>
);
const html = `${renderToString(clientJsx)}
<script>
window.__INITIAL_CLIENT_JSX_STRING__=\`${JSON.stringify(clientJsx, escapeJsx)}\`;
</script>
<script src="/client.js" type="module"></script>`;
res.end(html);
});
const escapeJsx = (key, value) => {
// A Symbol value doesn't "survive" JSON serialization
// We're going to substutute `Symbol.for('react.element')` with a special string like "$RE"
if (value === Symbol.for("react.element")) {
return "$RE"; // Could be arbitrary. I picked RE for React Element.
} else if (typeof value === "string" && value.startsWith("$")) {
// To avoid clashes, prepend an extra $ to any string already starting with $.
return "$" + value;
} else {
return value;
}
}
Note that if we directly send renderToString(<Component>)
to the client, React will complain “Error: Objects are not valid as a React child (found: [object Promise])”. The code doesn’t support RSC yet. We need transform JSX into an object that client React can recognize, which the function createReactTree
does.
async function createReactTree(jsx) {
if (
typeof jsx === "string" ||
typeof jsx === "number" ||
typeof jsx === "boolean" ||
jsx == null
) {
return jsx;
} else if (Array.isArray(jsx)) {
return Promise.all(jsx.map((child) => renderJSXToClientJSX(child)));
} else if (jsx != null && typeof jsx === "object") {
if (jsx.$$typeof === Symbol.for("react.element")) {
if (typeof jsx.type === "string") {
return {
...jsx,
props: await renderJSXToClientJSX(jsx.props),
};
} else if (typeof jsx.type === "function") {
const Component = jsx.type;
const props = jsx.props;
const returnedJsx = await Component(props);
return renderJSXToClientJSX(returnedJsx);
} else throw new Error("Not implemented.");
} else {
return Object.fromEntries(
await Promise.all(
Object.entries(jsx).map(async ([propName, value]) => [
propName,
await renderJSXToClientJSX(value),
])
)
);
}
} else throw new Error("Not implemented");
}
The goal on the client is to reconstruct the React element tree. It is much easier to accomplish this from this format than from html, where we’d have to parse the HTML to create the React elements. Note that the reconstruction of the React element tree is important, as this allows us to merge subsequent changes to the React tree with minimal commits to the DOM.
// We need to hydrate the root with the initial client JSX on the client.
const root = hydrateRoot(document, getInitialClientJSX());
function getInitialClientJSX() {
const clientJSX = JSON.parse(window.__INITIAL_CLIENT_JSX_STRING__, reviveJSX);
return clientJSX;
}
// On the client, we'll replace "$RE" back with `Symbol.for('react.element')`
function reviveJSX(key, value) {
if (value === "$RE") {
return Symbol.for("react.element");
} else if (typeof value === "string" && value.startsWith("$$")) {
// This is a string starting with $. Remove the extra $ added by the server.
return value.slice(1);
} else {
return value;
}
}
Must-read articles on React Server Components:
- https://www.joshwcomeau.com/react/server-components
- https://github.com/reactwg/server-components/discussions/5
- https://vercel.com/blog/understanding-react-server-components
- https://www.youtube.com/watch?v=CvAySC5ex9c
React Server Actions
React Server Actions allow you to run asynchronous code directly on the server. They eliminate the need to create API endpoints to mutate your data. Instead, you write asynchronous functions that execute on the server and can be invoked from your Client or Server Components. (Server actions let us put our API endpoint back into the component boundary in the same way that server components let us move getServerSideProps
into the component boundary.)
An advantage of invoking a Server Action within a Server Component is progressive enhancement - forms work even if JavaScript is disabled on the client.
// Server Component
export default function Page() {
async function create(formData: FormData) {
'use server';
// Logic to mutate data...
}
return <form action={create}>...</form>;
}
Behind the scenes, Server Actions create a POST API endpoint. This is why you don’t need to create API endpoints manually when using Server Actions. Server actions are different from regular server-side code. They are specifically designed to be invoked from the client-side, usually through form submissions or other user interactions. If you need to “expose” server functions to the client, you can use "use server"
.
- Next.js creates a unique identifier for each server action. This identifier links the client-side request to the correct server-side function.
- Next.js automatically generates an API endpoint for each server action. These endpoints are created during the compilation process and are not visible in your codebase. The generated endpoints handle the incoming requests from the client and route them to the corresponding server action. The request includes a special header called “Next-Action” which contains the unique identifier of the server action.
- Server Actions integrate with Next.js’ caching and revalidation architecture.
revalidatePath
accepts a relative URL string where it will clear the cache and revalidate the data for that path after a server action.
Server Actions have officially been renamed to Server Functions. Until September 2024, we referred to all Server Functions as “Server Actions”. If a Server Function is passed to an action prop or called from inside an action then it is a Server Action, but not all Server Functions are Server Actions. The naming in this documentation has been updated to reflect that Server Functions can be used for multiple purposes.