getSessionCookie is broken in Next.js middleware

I have to manually check my cookies in middleware, which works just fine. But getSessionCookie never returns.
import { type NextRequest, NextResponse } from "next/server"

export async function middleware(request: NextRequest) {
// const sessionCookie = getSessionCookie(request) // Broken

if (!request.cookies.has("better-auth.session_token")) {
const redirectTo = request.nextUrl.pathname + request.nextUrl.search
return NextResponse.redirect(new URL(`/auth/sign-in?redirectTo=${redirectTo}`, request.url))
}

return NextResponse.next()
}

export const config = {
matcher: ["/auth/settings"]
}
import { type NextRequest, NextResponse } from "next/server"

export async function middleware(request: NextRequest) {
// const sessionCookie = getSessionCookie(request) // Broken

if (!request.cookies.has("better-auth.session_token")) {
const redirectTo = request.nextUrl.pathname + request.nextUrl.search
return NextResponse.redirect(new URL(`/auth/sign-in?redirectTo=${redirectTo}`, request.url))
}

return NextResponse.next()
}

export const config = {
matcher: ["/auth/settings"]
}
80 Replies
Christian (N0_L0gic)
What version? I've been using getSessionCookie() no prob
daveycodez
daveycodezOP4w ago
1.2.3 It can be reproduced on my Next.js starter just by uncommenting it in the Middleware
daveycodez
daveycodezOP4w ago
GitHub
GitHub - daveyplate/better-auth-nextjs-starter: Better Auth Next.js...
Better Auth Next.js starter template with PostgreSQL, Drizzle, shadcn/ui and TanStack Query - daveyplate/better-auth-nextjs-starter
daveycodez
daveycodezOP4w ago
When I use getSessionCookie I can never reach settings page even after login and refreshing page
Christian (N0_L0gic)
Okay I was aparently using 1.1.21 in the implementation I was testing
import { NextRequest, NextResponse } from 'next/server';
import { getSessionCookie } from 'better-auth';

const protectedRoutes = ['/clouds', '/dashboard', '/shared', '/sign-out'];
const publicRoutes = ['/', '/sign-in', '/sign-up'];

export async function middleware(request: NextRequest) {
const path = request.nextUrl.pathname;
const isProtectedRoute = protectedRoutes.includes(path);
const isPublicRoute = publicRoutes.includes(path);

const sessionCookie = getSessionCookie(request);

if (isProtectedRoute && !sessionCookie) {
return NextResponse.redirect(new URL('/sign-in', request.url));
}

if (isPublicRoute && sessionCookie && !path.startsWith('/dashboard')) {
return NextResponse.redirect(new URL('/dashboard', request.url));
}

return NextResponse.next();
}

export const config = {
matcher: [
/*
* Match all request paths except for the ones starting with:
* - api (API routes)
* - _next/static (static files)
* - _next/image (image optimization files)
* - favicon.ico, sitemap.xml, robots.txt (metadata files)
*/
'/((?!api|_next/static|_next/image|favicon.ico|sitemap.xml|robots.txt).*)'
]
};
import { NextRequest, NextResponse } from 'next/server';
import { getSessionCookie } from 'better-auth';

const protectedRoutes = ['/clouds', '/dashboard', '/shared', '/sign-out'];
const publicRoutes = ['/', '/sign-in', '/sign-up'];

export async function middleware(request: NextRequest) {
const path = request.nextUrl.pathname;
const isProtectedRoute = protectedRoutes.includes(path);
const isPublicRoute = publicRoutes.includes(path);

const sessionCookie = getSessionCookie(request);

if (isProtectedRoute && !sessionCookie) {
return NextResponse.redirect(new URL('/sign-in', request.url));
}

if (isPublicRoute && sessionCookie && !path.startsWith('/dashboard')) {
return NextResponse.redirect(new URL('/dashboard', request.url));
}

return NextResponse.next();
}

export const config = {
matcher: [
/*
* Match all request paths except for the ones starting with:
* - api (API routes)
* - _next/static (static files)
* - _next/image (image optimization files)
* - favicon.ico, sitemap.xml, robots.txt (metadata files)
*/
'/((?!api|_next/static|_next/image|favicon.ico|sitemap.xml|robots.txt).*)'
]
};
daveycodez
daveycodezOP4w ago
You could reduce your middleware invocations significantly if you just match the defined routes Unless those are your only pages
Christian (N0_L0gic)
Probably
daveycodez
daveycodezOP4w ago
I have to do the global check for my i18n routing
Christian (N0_L0gic)
Or what do you mean exactly with this?
daveycodez
daveycodezOP4w ago
You could change the matcher to be an array of paths that are only your protectedRoutes and your publicRoutes So any paths that are neither won't trigger middleware invocations Not sure where you're hosting but Vercel charges for middleware invocations if you overage
Christian (N0_L0gic)
Not hosting anywhere rn, but I wanted to try self-hosting on my server pc eventually
daveycodez
daveycodezOP4w ago
Self hosting should be no problem then
Christian (N0_L0gic)
Ye Btw, is the cookie actually set in the browser when you tried this? I can go to /auth/settings and get redirected to /auth/sign-in when I'm not authenticated @daveycodez (even after uncommenting the call to getSessionCookie()) Yeah it works even when authenticated I can't reproduce your error even with your template repo
daveycodez
daveycodezOP4w ago
Did you turbo build turbo start And replace the cookie check in the if
Christian (N0_L0gic)
I ran npm run dev and changed the code to:
const sessionCookie = getSessionCookie(request); // Broken

console.log(sessionCookie);

if (!sessionCookie) {
const redirectTo = request.nextUrl.pathname + request.nextUrl.search;
return NextResponse.redirect(new URL(`/auth/sign-in?redirectTo=${redirectTo}`, request.url));
}
const sessionCookie = getSessionCookie(request); // Broken

console.log(sessionCookie);

if (!sessionCookie) {
const redirectTo = request.nextUrl.pathname + request.nextUrl.search;
return NextResponse.redirect(new URL(`/auth/sign-in?redirectTo=${redirectTo}`, request.url));
}
can try turbo build turbo start
Christian (N0_L0gic)
okay nvm I can't run turbo?
No description
daveycodez
daveycodezOP4w ago
Ok try just npm run build and npm run start
Christian (N0_L0gic)
Yes still didn't get stuck
daveycodez
daveycodezOP4w ago
Hmmmmm
Christian (N0_L0gic)
when unauthenticated
daveycodez
daveycodezOP4w ago
I’ll try again
Christian (N0_L0gic)
Yeah I'm gonna try something also I might have a skill issue Yeah idk man, it works for me Oh wait something is happening now Okay so now when I go to /auth/settings, it redirects me to the login page and sets ?redirectTo=auth/settings but when I sign in it gets stuck
daveycodez
daveycodezOP4w ago
Yea So it only occurs on production not on npm run dev
Christian (N0_L0gic)
However the console log ran when I pressed the button
daveycodez
daveycodezOP4w ago
only on npm run start It's specifically broken on production
Christian (N0_L0gic)
Yeah it seems so I'm a bit confused tho
Christian (N0_L0gic)
The console.log is outputting before I get stuck (after I press the button)
No description
Christian (N0_L0gic)
So is it trying to double redirect me?
daveycodez
daveycodezOP4w ago
It's logging null for me on production Even though I'm logged in
Christian (N0_L0gic)
Yeah same Oh even when ur logged in? How did u log in? I just get stuck
daveycodez
daveycodezOP4w ago
Idk what you mean by stuck It redirects me back to the login page
Christian (N0_L0gic)
The loading state on the login button never finishes
daveycodez
daveycodezOP4w ago
But you can see logged in state at the top right of the page Oh that's weird
Christian (N0_L0gic)
I literally don't get redirected
daveycodez
daveycodezOP4w ago
That's not happening to me
Christian (N0_L0gic)
😵‍💫
daveycodez
daveycodezOP4w ago
OOHh I see that happening to me now That's because the navigate is being rewrited back to current destination
Christian (N0_L0gic)
Yeah I'm wondering if its creating infinite redirects?
daveycodez
daveycodezOP4w ago
And it stays loading til you leave page
Christian (N0_L0gic)
Right
daveycodez
daveycodezOP4w ago
Nah it's not infinite redirects I set the button to loading then I call navigate the navigate gets a rewrite so its like navigate(currentUrl) so it never unmounts
daveycodez
daveycodezOP4w ago
But yea it works perfectly in dev, and it works perfectly if I manually check the cookie Must be some specific logic to production Some sort of cookie validation?
Christian (N0_L0gic)
Amazing to debug 😍
daveycodez
daveycodezOP4w ago
I merely want to check for the presence of the cookie in middleware, don't really care about the validity
Christian (N0_L0gic)
isn't that what the getSessionCookie does? I don't know the implementation
daveycodez
daveycodezOP4w ago
I'll dig through the source
daveycodez
daveycodezOP4w ago
const name = isProduction ? __Secure-${cookiePrefix}.${cookieName} : ${cookiePrefix}.${cookieName}; It's this line
Christian (N0_L0gic)
I think some other thread here on the discord was talking about this Or something similar
daveycodez
daveycodezOP4w ago
Yea so idk. I don't want to set any URL's, everything works out of the box just fine
daveycodez
daveycodezOP4w ago
https://github.com/better-auth/better-auth/blob/f90581d7ab0493fb58d3c2654c326f6f28a684b3/packages/better-auth/src/cookies/index.ts#L250
const name = isProduction
? `__Secure-${cookiePrefix}.${cookieName}`
: `${cookiePrefix}.${cookieName}`;
const parsedCookie = parseCookies(cookies);
const sessionToken = parsedCookie.get(name);
const name = isProduction
? `__Secure-${cookiePrefix}.${cookieName}`
: `${cookiePrefix}.${cookieName}`;
const parsedCookie = parseCookies(cookies);
const sessionToken = parsedCookie.get(name);
can become
const name = `${cookiePrefix}.${cookieName}`;
const parsedCookie = parseCookies(cookies);
const sessionToken = parsedCookie.get(`__Secure-${cookiePrefix}.${cookieName}`) || parsedCookie.get(`${cookiePrefix}.${cookieName}`);
const name = `${cookiePrefix}.${cookieName}`;
const parsedCookie = parseCookies(cookies);
const sessionToken = parsedCookie.get(`__Secure-${cookiePrefix}.${cookieName}`) || parsedCookie.get(`${cookiePrefix}.${cookieName}`);
@bekacru
Christian (N0_L0gic)
Is this in your code or a suggestion to better-auth source?
daveycodez
daveycodezOP4w ago
Suggestion to modify source linked above This is only needed for presence, not validation, so checking for either is just fine instead of conditionally checking for one
Christian (N0_L0gic)
If thats the case then I agree
daveycodez
daveycodezOP4w ago
getSessionCookie was created mainly for optimistic 0 latency middleware checks You use getSession on individual routes to authenticate them or client side authentication
Christian (N0_L0gic)
So do you do getSession() and then redirect on individual routes as well?
daveycodez
daveycodezOP4w ago
Yea The middleware is for "skipping" a route
daveycodez
daveycodezOP4w ago
If the user doesn't have a cookie, don't even bother a function invocation for that route
Christian (N0_L0gic)
Makes sense
daveycodez
daveycodezOP4w ago
If the user does have a cookie, run the route, then the route will check getSession
Christian (N0_L0gic)
This is also kinda what I found on the nextjs docs
daveycodez
daveycodezOP4w ago
Yea Next.js advises against fetching in middleware For my /auth/settings page from better-auth-ui, it uses the useAuthenticate hook which authenticates client side (shows Skeletons during isPending, then redirects if no session), and then you can use middleware to skip that route entirely if they don't even have a cookie So the only time useAuthenticate even does anything is if they have an old cookie or invalid cookie
Christian (N0_L0gic)
Oh does useAuthenticate handle redirecting also?
daveycodez
daveycodezOP4w ago
Yea
daveycodez
daveycodezOP4w ago
it redirects to /auth/sign-in?redirectTo=${currentUrl} pretty much
Christian (N0_L0gic)
Makes sense
daveycodez
daveycodezOP4w ago
including search params and hash
Christian (N0_L0gic)
First time using better-auth and first time-ish using nextjs here so this is great info Ty!
daveycodez
daveycodezOP4w ago
Yea the useAuthenticate hook comes from my better-auth-ui library, not better-auth itself
Christian (N0_L0gic)
Ahh Idk if its worth switching to now as I'm approaching deadline, but I took a look at ur library before. Seems pretty nice
daveycodez
daveycodezOP4w ago
Fetch in Middleware no bueno
Ahmed
Ahmed4w ago
@bekacru please have a look at this pr https://github.com/better-auth/better-auth/pull/1716 . alot is getting confused with this function.
GitHub
fix: getSessionCookie function and improve docs. by ahmed-m-abbass ...
Improved the logic in getSessionCookie() to correctly handle secure prefix and added useSecureCookies config. closes #1487.
daveycodez
daveycodezOP4w ago
It doesn't really need useSecureCookies prop You can just check if either is present and use whichever is present https://discord.com/channels/1288403910284935179/1348847982748897380/1348917566600511498

Did you find this page helpful?