feat(01-05): add middleware for session refresh and route protection
- Create updateSession helper for Supabase session management - Add main middleware with protected and auth route handling - Configure matcher to exclude static files for performance - Session refresh on every request prevents random logouts Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
49
middleware.ts
Normal file
49
middleware.ts
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
import { type NextRequest, NextResponse } from 'next/server'
|
||||||
|
import { updateSession } from '@/lib/supabase/middleware'
|
||||||
|
|
||||||
|
// Routes that require authentication
|
||||||
|
const protectedRoutes = ['/dashboard', '/settings', '/subscription']
|
||||||
|
|
||||||
|
// Routes that should redirect to dashboard if already authenticated
|
||||||
|
const authRoutes = ['/login', '/register']
|
||||||
|
|
||||||
|
export async function middleware(request: NextRequest) {
|
||||||
|
const { supabaseResponse, user } = await updateSession(request)
|
||||||
|
const { pathname } = request.nextUrl
|
||||||
|
|
||||||
|
// Check if trying to access protected route without auth
|
||||||
|
const isProtectedRoute = protectedRoutes.some(route =>
|
||||||
|
pathname.startsWith(route)
|
||||||
|
)
|
||||||
|
|
||||||
|
if (isProtectedRoute && !user) {
|
||||||
|
const redirectUrl = new URL('/login', request.url)
|
||||||
|
// Save the original URL to redirect back after login
|
||||||
|
redirectUrl.searchParams.set('redirectTo', pathname)
|
||||||
|
return NextResponse.redirect(redirectUrl)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if trying to access auth routes while already authenticated
|
||||||
|
const isAuthRoute = authRoutes.some(route =>
|
||||||
|
pathname.startsWith(route)
|
||||||
|
)
|
||||||
|
|
||||||
|
if (isAuthRoute && user) {
|
||||||
|
return NextResponse.redirect(new URL('/dashboard', request.url))
|
||||||
|
}
|
||||||
|
|
||||||
|
return supabaseResponse
|
||||||
|
}
|
||||||
|
|
||||||
|
export const config = {
|
||||||
|
matcher: [
|
||||||
|
/*
|
||||||
|
* Match all request paths except for the ones starting with:
|
||||||
|
* - _next/static (static files)
|
||||||
|
* - _next/image (image optimization files)
|
||||||
|
* - favicon.ico (favicon file)
|
||||||
|
* - public folder files
|
||||||
|
*/
|
||||||
|
'/((?!_next/static|_next/image|favicon.ico|.*\\.(?:svg|png|jpg|jpeg|gif|webp)$).*)',
|
||||||
|
],
|
||||||
|
}
|
||||||
37
src/lib/supabase/middleware.ts
Normal file
37
src/lib/supabase/middleware.ts
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
import { createServerClient } from '@supabase/ssr'
|
||||||
|
import { NextResponse, type NextRequest } from 'next/server'
|
||||||
|
|
||||||
|
export async function updateSession(request: NextRequest) {
|
||||||
|
let supabaseResponse = NextResponse.next({
|
||||||
|
request,
|
||||||
|
})
|
||||||
|
|
||||||
|
const supabase = createServerClient(
|
||||||
|
process.env.NEXT_PUBLIC_SUPABASE_URL!,
|
||||||
|
process.env.NEXT_PUBLIC_SUPABASE_ANON_KEY!,
|
||||||
|
{
|
||||||
|
cookies: {
|
||||||
|
getAll() {
|
||||||
|
return request.cookies.getAll()
|
||||||
|
},
|
||||||
|
setAll(cookiesToSet) {
|
||||||
|
cookiesToSet.forEach(({ name, value }) =>
|
||||||
|
request.cookies.set(name, value)
|
||||||
|
)
|
||||||
|
supabaseResponse = NextResponse.next({
|
||||||
|
request,
|
||||||
|
})
|
||||||
|
cookiesToSet.forEach(({ name, value, options }) =>
|
||||||
|
supabaseResponse.cookies.set(name, value, options)
|
||||||
|
)
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
// IMPORTANT: Do not remove this line
|
||||||
|
// Refreshing the auth token is crucial for keeping the session alive
|
||||||
|
const { data: { user } } = await supabase.auth.getUser()
|
||||||
|
|
||||||
|
return { supabaseResponse, user }
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user