import createMiddleware from 'next-intl/middleware'; import type { NextRequest } from 'next/server'; const i18nMiddleware = createMiddleware({ locales: ['en', 'de'], defaultLocale: 'en', // Wir nutzen überall Locale-Präfixe (`/en`, `/de`) localePrefix: 'always' }); export default function proxy(request: NextRequest) { // 1. i18n-Routing const response = i18nMiddleware(request); // 2. Security-Header ergänzen const headers = response.headers; headers.set('X-Frame-Options', 'SAMEORIGIN'); headers.set('X-XSS-Protection', '1; mode=block'); headers.set('X-Content-Type-Options', 'nosniff'); headers.set('Referrer-Policy', 'strict-origin-when-cross-origin'); headers.set('Permissions-Policy', 'camera=(), microphone=(), geolocation=()'); // Extract Plausible domain from script URL for CSP const plausibleScriptSrc = process.env.NEXT_PUBLIC_PLAUSIBLE_SCRIPT_SRC || 'https://plausible.example.com/js/script.js'; let plausibleOrigin = 'https://plausible.example.com'; try { const url = new URL(plausibleScriptSrc); plausibleOrigin = url.origin; } catch { // If URL parsing fails, try to extract domain manually const match = plausibleScriptSrc.match(/https?:\/\/([^/]+)/); if (match) { plausibleOrigin = `https://${match[1]}`; } } // Get other service URLs from environment (only add to CSP if configured) const gotifyUrl = process.env.GOTIFY_URL; const openrouterUrl = process.env.NEXT_PUBLIC_OPENROUTER_URL || 'https://openrouter.ai'; // Build CSP dynamically based on environment variables const connectSrcParts = ["'self'", openrouterUrl, plausibleOrigin]; if (gotifyUrl && !gotifyUrl.includes('example.com')) { connectSrcParts.push(gotifyUrl); } const csp = [ "default-src 'self'", `script-src 'self' 'unsafe-inline' 'unsafe-eval' ${plausibleOrigin}`, "style-src 'self' 'unsafe-inline'", "img-src 'self' data: blob:", "font-src 'self' data:", `connect-src ${connectSrcParts.join(' ')}`, "media-src 'self' blob:", "frame-ancestors 'self'", ].join('; '); headers.set('Content-Security-Policy', csp); return response; } export const config = { // Empfohlener Matcher aus der next-intl Doku: // alle Routen außer _next, API und statischen Dateien matcher: ['/((?!api|_next|.*\\..*).*)'] };