
Most Next.js authentication tutorials teach you how to build the plumbing. This one skips that. You don't need to write session logic, token exchange, or JWT verification yourself — and you definitely shouldn't. You need to know where auth lives in a Next.js app, how the App Router changes things, and how to ship it fast without painting yourself into a corner.
A production Next.js app needs session management that survives page refreshes, protected routes that redirect unauthenticated users without flicker, API endpoints that reject requests without a valid token, Server Actions that verify identity before running any business logic, and a sign-in/sign-out flow that handles the OAuth/OIDC roundtrip.
Let's go through each one.
Next.js 13+ separates your code into two runtimes: server and client. Auth decisions should happen on the server. If you're checking auth state in a client component to decide what to render, you're doing it in the wrong place — there will be a flash of unauthenticated content before the check resolves.
The right model:
This matters because it determines where tokens are stored and how they're validated.
Install the SDK:
npm install @monocloud/auth-nextjs
Create .env.local with your credentials from the MonoCloud dashboard:
MONOCLOUD_AUTH_TENANT_DOMAIN=https://<your-tenant-domain>
MONOCLOUD_AUTH_CLIENT_ID=<your-client-id>
MONOCLOUD_AUTH_CLIENT_SECRET=<your-client-secret>
MONOCLOUD_AUTH_SCOPES=openid profile email
MONOCLOUD_AUTH_APP_URL=http://localhost:3000
MONOCLOUD_AUTH_COOKIE_SECRET=<random-secret>
Generate a cookie secret with:
openssl rand -hex 32
In Next.js 16+, authentication middleware uses a proxy-based approach. Create proxy.ts in your project root:
import { authMiddleware } from "@monocloud/auth-nextjs";
export default authMiddleware();
export const config = {
matcher: ["/((?!_next/static|_next/image|favicon.ico|sitemap.xml|robots.txt).*)"],
};
This gives you the entire OpenID Connect flow — redirects, token exchange, cookie-based session storage, silent refresh. By default, every matched route requires authentication. Unauthenticated users are redirected to sign in.
To protect only specific routes (everything else passes through unauthenticated):
import { authMiddleware } from "@monocloud/auth-nextjs";
export default authMiddleware({
protectedRoutes: ["/dashboard", "/settings", "/api/orders"],
});
export const config = {
matcher: ["/((?!_next/static|_next/image|favicon.ico|sitemap.xml|robots.txt).*)"],
};
Note: If you need multiple configurations, custom routes, or dependency injection, you can create aMonoCloudNextClientinstance manually. For standard setups, the directauthMiddleware()import above is all you need.
protectPage() is a HOC that wraps your Server Component. The user object is injected as a prop — you never need to call getSession() manually inside a protected page:
import { MonoCloudUser, protectPage } from "@monocloud/auth-nextjs";
type Props = { user: MonoCloudUser };
function DashboardPage({ user }: Props) {
return (
<div>
<h1>Welcome, {user.name}</h1>
<p>{user.email}</p>
</div>
);
}
export default protectPage(DashboardPage);
Unauthenticated users are redirected to sign in. Your component code only runs for authenticated sessions.
To restrict to a specific group (RBAC):
export default protectPage(AdminPage, { groups: ["admin"] });
Users not in the admin group are shown an access denied message before your component renders.
protectApi() wraps your route handler. The handler signature is (req, ctx) — to read the authenticated user inside, call getSession():
import { protectApi, getSession } from "@monocloud/auth-nextjs";
import { NextResponse } from "next/server";
import { db } from "@/lib/db";
export const GET = protectApi(async (req, ctx) => {
const session = await getSession();
const orders = await db.orders.findMany({
where: { userId: session!.user.sub },
});
return NextResponse.json(orders);
});
Unauthenticated requests get a 401 before your handler runs. getSession() inside a protected handler is safe — the session is guaranteed to exist.
"use server";
import { protect, getSession } from "@monocloud/auth-nextjs";
import { db } from "@/lib/db";
export async function placeOrder(items: CartItem[]) {
await protect(); // redirects if not authenticated
const session = await getSession();
const userId = session!.user.sub;
return db.orders.create({
data: { userId, items, status: "pending" },
});
}
protect() guards the action — if the session is missing, the user is redirected to sign in and the action body never executes. Retrieve the user via getSession() when you need identity data inside the action.
import { getSession } from "@monocloud/auth-nextjs";
export default async function HomePage() {
const session = await getSession();
if (!session) {
return <p>You are not signed in.</p>;
}
return <p>Hello, {session.user.name}</p>;
}
For client-side auth state, use useAuth() from the /client subpath:
"use client";
import { useAuth } from "@monocloud/auth-nextjs/client";
export function UserMenu() {
const { user, isAuthenticated, isLoading } = useAuth();
if (isLoading) {
return <div className="h-8 w-8 animate-pulse rounded-full bg-gray-200" />;
}
if (!isAuthenticated || !user) {
return <a href="/api/auth/login">Sign in</a>;
}
return (
<div className="flex items-center gap-2">
<span className="text-sm">{user.name}</span>
<a href="/api/auth/logout" className="text-sm text-gray-500">Sign out</a>
</div>
);
}
To protect a client component with a HOC:
"use client";
import { MonoCloudUser } from "@monocloud/auth-nextjs";
import { protectClientPage } from "@monocloud/auth-nextjs/client";
type Props = { user: MonoCloudUser };
const ProfilePage = ({ user }: Props) => (
<div>
<h2>{user.name}</h2>
<p>{user.email}</p>
</div>
);
export default protectClientPage(ProfilePage);
Note: protectClientPage() runs in the browser — use protectPage() (server) for anything security-sensitive.
Use the <SignIn /> and <SignOut /> components provided by the SDK:
import { getSession } from "@monocloud/auth-nextjs";
import { SignIn, SignOut } from "@monocloud/auth-nextjs/components";
export async function Navbar() {
const session = await getSession();
return (
<nav>
{session ? (
<SignOut>Sign out</SignOut>
) : (
<SignIn>Sign in</SignIn>
)}
</nav>
);
}
The SDK registers all required auth routes automatically. No route handler file needed.
protectPage() or protectApi() for enforcement.groups: ["admin"] to protectPage() or protectApi()We built MonoCloud after learning for ourselves that authentication and user management are hard. But we knew that building an easy-to-use tool wasn't enough. Our platform isn't just about saving you a few days of coding. Instead, it's about adding the expertise you need to secure your software for the long term.
You can start building with MonoCloud today. Sign up now for your free account. Click here monocloud.com