6 min read
After talking to hundreds of developers and looking at thousands of Next.js repositories, I've noticed ten common mistakes when building with the Next.js App Router.
This post will share why these mistakes can happen, how to fix them, and some tips to help you understand the new App Router model.
Using Route Handlers with Server Components
Consider the following code for a Server Component:
export default async function Page() { let res = await fetch('http://localhost:3000/api/data'); let data = await res.json(); return <h1>{JSON.stringify(data)}</h1>;}
Fetching JSON data from a Route Handler in a Server Component.
This async
component makes a request to a Route Handler to retrieve some JSON data:
export async function GET(request: Request) { return Response.json({ data: 'Next.js' });}
A Route Handler that returns static JSON data.
There's two main issues with this approach:
Both Route Handlers and Server Components run securely on the server. You don't need the additional network hop. Instead, you can call whatever logic you intended to place inside the Route Handler directly in the Server Component. This might be an external API or any
Promise
.Since this code is running on the server with Node.js, we need to provide the absolute URL for the
fetch
versus a relative URL. In reality, we wouldn't hardcodelocalhost
here, but instead need to have some conditional check based on the environment we're in. This is unnecessary since you can call the logic directly.
Instead, prefer to do the following:
export default async function Page() { // call your async function directly let data = await getData(); // { data: 'Next.js' } // or call an external API directly let data = await fetch('https://api.vercel.app/blog') // ...}
Server Components are able to fetch data directly.
Static or dynamic Route Handlers
Route Handlers are cached by default when using the GET
method. This can often be confusing for existing Next.js developers moving from the Pages Router and API Routes.
For example, the following code will be prerendered during next build
:
export async function GET(request: Request) { return Response.json({ data: 'Next.js' });}
A Route Handler that returns static JSON data.
This JSON data will not change until another build has completed. Why is that?
You can consider Route Handlers the building blocks of pages. For a given request to a route, you want to handle it. Next.js has further abstractions on top of Route Handlers like pages and layouts. This is why Route Handlers are static by default (like pages) and share the same route segment configuration options.
This functionality unlocks some new features previously not possible with API Routes in the Pages Router. For example, you can have Route Handlers that produce JSON, or txt
files, or really any file, which can be computed and prerendered during the build. The statically generated file is then automatically cached, and even periodically updated if desired.
export async function GET(request: Request) { let res = await fetch('https://api.vercel.app/blog'); let data = await res.json(); return Response.json(data);}
Return a list of blog posts as JSON data.
Further, this means the Route Handlers are compatible with Static Exports where you can deploy your Next.js application anywhere that supports static file hosting.
Route Handlers and Client Components
You might think you need to use Route Handlers with Client Components, since they cannot be marked async
and fetch or mutate data. Rather than needing to write a fetch
and create a Route Handler, you can instead call Server Actions directly from Client Components.
'use client';
import { save } from './actions';
export function UserForm() { return ( <form action={save}> <input type="text" name="username" /> <button>Save</button> </form> );}
A form and input to save a name.
This works with both forms as well as event handlers:
'use client';
import { save } from './actions';
export function UserForm({ username }) { async function onSave(event) { event.preventDefault(); await save(username); }
return <button onClick={onSave}>Save</button>;}
Server Actions can be called from event handlers.
Using Suspense with Server Components
Consider the following Server Component. Where should Suspense be placed to define what fallback UI will be shown while the data is being fetched?
async function BlogPosts() { let data = await fetch('https://api.vercel.app/blog'); let posts = await data.json(); return ( <ul> {posts.map((post) => ( <li key={post.id}>{post.title}</li> ))} </ul> );}
export default function Page() { return ( <section> <h1>Blog Posts</h1> <BlogPosts /> </section> );}
A page which contains an async component with data fetching.
If you guessed inside of the Page
component, you were correct. The Suspense
boundary needs to be placed higher than the async
component doing the data fetching. It will not work if the boundary is inside of the async
component.
import { Suspense } from 'react';
async function BlogPosts() { let data = await fetch('https://api.vercel.app/blog'); let posts = await data.json(); return ( <ul> {posts.map((post) => ( <li key={post.id}>{post.title}</li> ))} </ul> );}
export default function Page() { return ( <section> <h1>Blog Posts</h1> <Suspense fallback={<p>Loading...</p>}> <BlogPosts /> </Suspense> </section> );}
Using Suspense with React Server Components.
In the future with Partial Prerendering, this pattern will start to become more common, including defining which components should be prerendered and which should run on-demand.
import { unstable_noStore as noStore } from 'next/cache';
async function BlogPosts() { noStore(); // This component should run dynamically let data = await fetch('https://api.vercel.app/blog'); let posts = await data.json(); return ( <ul> {posts.map((post) => ( <li key={post.id}>{post.title}</li> ))} </ul> );}
Opt-into dynamic rendering inside async components.
Using the incoming request
Since the incoming request object is not able to be accessed from a Server Component, it might not be obvious how to read parts of the incoming request. This could lead to using client hooks like useSearchParams
unnecessarily.
There are specific functions and props to the Server Component which allow you to access the incoming request. For example:
export default function Page({ params, searchParams,}: { params: { slug: string } searchParams: { [key: string]: string | string[] | undefined }}) { return <h1>My Page</h1>}
Reading parts of the URL and the search parameters.
Using Context providers with App Router
You might want to use React Context or are using an external dependency which relies on context. Two common mistakes I've seen are trying to use context with Server Components (unsupported) and the placement of the provider in the App Router.
To allow for your Server and Client Components to interleave, it's important to make your provider (or multiple providers) be a separate Client Component which takes children
as a prop and renders them. For example:
'use client';
import { createContext } from 'react';
export const ThemeContext = createContext({});
export default function ThemeProvider({ children,}: { children: React.ReactNode;}) { return <ThemeContext.Provider value="dark">{children}</ThemeContext.Provider>;}
A Client Component that uses React Context.
Then, with your provider in a separate file as a Client Component, you can import and use this component inside of your layout:
import ThemeProvider from './theme-provider';
export default function RootLayout({ children,}: { children: React.ReactNode;}) { return ( <html> <body> <ThemeProvider>{children}</ThemeProvider> </body> </html> );}
A root layout that weaves a client context provider and Server Component children.
With the provider rendered at the root, all other Client Components throughout your app will be able to consume this context. And notably, this configuration still allows for other Server Components (including the page
) lower in the tree.
Using Server and Client Components together
Many React and Next.js developers are learning how to use Server and Client Components for the first time. It's expected there might be some mistakes and opportunities to learn this new model!
For example, consider the following page:
export default function Page() { return ( <section> <h1>My Page</h1> </section> );}
A Server Component page.
This is a Server Component. While that comes with new functionality like being able to fetch data directly in the component, it also means certain client-side React functionalities aren't available.
For example, consider creating a button that is a counter. This would need to be a new Client Component file marked with the "use client"
directive at the top:
'use client';
import { useState } from 'react';
export function Counter() { const [count, setCount] = useState(0); return ( <div> <p>Count: {count}</p> <button onClick={() => setCount(count + 1)}>Increment</button> </div> );}
A Client Component button that increments a count.
You can then import and use this component in your page:
import { Counter } from './counter';
export default function Page() { return ( <section> <h1>My Page</h1> <Counter /> </section> );}
Using a Client Component from a Server Component.
The page is a Server Component and the <Counter>
is a Client Component. Great! What about components lower in the tree than the counter? Can those be Server Components? Yes, through composition:
import { Counter } from './counter';
function Message() { return <p>This is a Server Component</p>;}
export default function Page() { return ( <section> <h1>My Page</h1> <Counter> <Message /> </Counter> </section> );}
Children of a Client Component can be Server Components.
Children of a Client Component can be a Server Component! Here's the updated counter:
'use client';
import { useState } from 'react';
export function Counter({ children }: { children: React.ReactNode }) { const [count, setCount] = useState(0); return ( <div> <p>Count: {count}</p> <button onClick={() => setCount(count + 1)}>Increment</button> {children} </div> );}
The counter now accepts children and displays them.
Adding “use client”
unnecessarily
Building off the previous example, does that mean that we need to add the "use client"
directive everywhere?
When the "use client"
directive is added, you pass into the "client boundary" giving you the ability to run client-side JavaScript (i.e. using React hooks or state). Client Components are still prerendered on the server, similar to components in the Next.js Pages Router.
Since you're already in the client boundary, siblings of the <Counter>
would become Client Components. You don't need to add "use client"
to every file. This might be an approach taken for incremental adoption of the App Router, where a component high up the tree becomes a Client Component and it becomes weave child Server Components further down.
Not revalidating data after mutations
The Next.js App Router includes a complete model for fetching, caching, and revalidating data. As developers are still learning this new model, and we're continuing to make improvements based on their feedback, one common mistake I've seen is forgetting to revalidate data after a mutation.
For example, consider the following Server Component. It displays a form, which uses a Server Action to handle the submission and create a new entry in a Postgres database.
export default function Page() { async function create(formData: FormData) { 'use server';
let name = formData.get('name'); await sql`INSERT INTO users (name) VALUES (${name})`; }
return ( <form action={create}> <input name="name" type="text" /> <button type="submit">Create</button> </form> );}
A Server Action that inserts the name into a Postgres database.
After the form is submitted and the insertion happens successfully, would the data displaying the list of names automatically update? No, not unless we tell Next.js to. For example:
import { revalidatePath } from 'next/cache';
export default async function Page() { let names = await sql`SELECT * FROM users`;
async function create(formData: FormData) { 'use server';
let name = formData.get('name'); await sql`INSERT INTO users (name) VALUES (${name})`;
revalidatePath('/'); }
return ( <section> <form action={create}> <input name="name" type="text" /> <button type="submit">Create</button> </form> <ul> {names.map((name) => ( <li>{name}</li> ))} </ul> </section> );}
Revalidating data inside of a Server Action.
Redirects inside of try/catch blocks
When running server-side code, like a Server Component or a Server Action, you might want to redirect if a resource is not available or after a successful mutation.
The redirect()
function does not require you to use return redirect()
as it uses the TypeScript never
type. Further, internally this function throws a Next.js specific error. This means you should handle redirecting outside of try/catch blocks.
For example, if you are trying to redirect inside of a Server Component, it might look like this:
import { redirect } from 'next/navigation';
async function fetchTeam(id) { const res = await fetch('https://...'); if (!res.ok) return undefined; return res.json();}
export default async function Profile({ params }) { const team = await fetchTeam(params.id); if (!team) { redirect('/login'); }
// ...}
Redirecting from a Server Component.
Alternatively, if you're trying to redirect from a Client Component, this should happen inside of a Server Action and not in an event handler:
'use client';
import { navigate } from './actions';
export function ClientRedirect() { return ( <form action={navigate}> <input type="text" name="id" /> <button>Submit</button> </form> );}
Redirecting in a Client Component through a Server Action.
'use server';
import { redirect } from 'next/navigation';
export async function navigate(data: FormData) { redirect('/posts');}
A Server Action that redirects to a new route.
Conclusion
The Next.js App Router is a new approach for building React applications and there's a handful of new concepts to learn. If you've made any of these mistakes, don't feel discouraged. I've made them as well learning how the model works.
If you want to keep learning more and apply this knowledge, check out our Next.js Learn course to build a real dashboard application with the App Router.