Files
smart-home-tablet/middleware.ts
Cosmo 6199db2977
All checks were successful
Deploy / deploy (push) Successful in 1m27s
feat: LLM provider switcher (Claude/Groq) in settings tab
2026-05-01 12:42:24 +00:00

39 lines
1.5 KiB
TypeScript

import { NextResponse } from 'next/server'
import type { NextRequest } from 'next/server'
export async function middleware(request: NextRequest) {
const { pathname } = request.nextUrl
// Only protect API routes. /api/voice/event, /api/voice/tools/*, /api/voice/timer
// have their own bearer-token auth (VOICE_API_KEY) and bypass the cookie check.
const isVoiceBearer =
pathname === '/api/voice/event' ||
pathname.startsWith('/api/voice/tools/') ||
pathname === '/api/voice/timer'
if (!pathname.startsWith('/api/') || pathname.startsWith('/api/auth') || pathname.startsWith('/api/spotify') || pathname.startsWith('/api/settings') || isVoiceBearer) {
return NextResponse.next()
}
// Internal loopback bypass: tool endpoints shell out to other API routes.
// They pass x-voice-internal with the same VOICE_API_KEY — safe because
// only processes on the same host (the tablet container itself) know the key.
const internal = request.headers.get('x-voice-internal')
if (internal && internal === process.env.VOICE_API_KEY) {
return NextResponse.next()
}
// Check auth by forwarding to auth check
const token = request.cookies.get('auth_token')?.value
if (!token) {
return NextResponse.json({ error: 'unauthorized' }, { status: 401 })
}
// Let the request through — individual API routes can do further validation if needed
// The auth cookie existence is sufficient since it is httpOnly and set by server
return NextResponse.next()
}
export const config = {
matcher: ['/api/:path*'],
}