Common mistakes with the Next.js App Router and how to fix them

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:

app/page.tsx
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:

app/api/data/route.ts
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:

  1. 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.

  2. 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 hardcode localhost 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:

app/page.tsx
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:

app/api/data/route.ts
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.

app/api/data/route.ts
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.

app/user-form.tsx
'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:

app/user-form.tsx
'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?

app/page.tsx
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.

app/page.tsx
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:

app/blog/[slug]/page.tsx
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:

app/theme-provider.tsx
'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:

app/layout.tsx
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:

app/page.tsx
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:

app/counter.tsx
'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:

app/page.tsx
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:

app/page.tsx
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:

app/counter.tsx
'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.

app/page.tsx
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:

app/page.tsx
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:

app/page.tsx
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:

app/client-redirect.tsx
'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.

app/actions.ts
'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.