8404165f5c
- auth: Supabase-JWT lokal verifizieren (hono/jwt, HS256) statt GoTrue- Roundtrip pro Request; JWT_SECRET in cms-env, Remote-Fallback wenn ungesetzt - dialog: comment_stats-View (group by thread) ersetzt Full-Table-Scan + JS-Aggregation bei jedem Forum-Aufruf - ops: scripts/backup-db.sh (pg_dump, rotiert) + täglicher Cron im Proxmox- Script — Dialog-Daten liegen nur in Postgres, nicht in Git - security: Rate-Limit auf Schreib-Endpunkte (/api non-GET, 60/min je Nutzer) - perf: Cache-Control (1 Woche) auf statische Assets, HTML bleibt frisch Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
73 lines
2.8 KiB
JavaScript
73 lines
2.8 KiB
JavaScript
import { verify } from 'hono/jwt';
|
|
import { supabaseAuth } from './supabase.js';
|
|
|
|
// Supabase-Tokens sind HS256-signiert. Mit dem JWT_SECRET verifizieren wir sie
|
|
// lokal (Signatur + Ablauf) — das spart pro Request den Roundtrip zu GoTrue.
|
|
// Ohne JWT_SECRET (z.B. Alt-Deploy) fällt requireAuth auf die Remote-Prüfung
|
|
// zurück. Tokens sind kurzlebig (1h) und Self-Signup ist aus → kein
|
|
// Sperr-Check nötig.
|
|
const JWT_SECRET = process.env.JWT_SECRET || '';
|
|
|
|
// Liefert ein User-Objekt {id,email,app_metadata} oder null.
|
|
async function verifyToken(token) {
|
|
if (JWT_SECRET) {
|
|
try {
|
|
const p = await verify(token, JWT_SECRET, 'HS256');
|
|
if (!p?.sub) return null;
|
|
return { id: p.sub, email: p.email || '', app_metadata: p.app_metadata || {} };
|
|
} catch { return null; }
|
|
}
|
|
const { data, error } = await supabaseAuth.auth.getUser(token);
|
|
if (error || !data?.user) return null;
|
|
return data.user;
|
|
}
|
|
|
|
// Rollen-Hierarchie: admin > editor (Redakteur) > user.
|
|
// - admin: alles (Foren verwalten, moderieren, Nutzer/Rollen, Inhalte)
|
|
// - editor: moderieren (Wortmeldungen ausblenden/löschen, Threads sperren)
|
|
// - user: im Forum mitschreiben
|
|
// Admins aus der .env (ADMIN_EMAILS=a@x,b@y) sind immer Admin (Bootstrap, damit
|
|
// man sich nicht aussperrt). Zusätzlich kann eine Rolle in app_metadata.role
|
|
// liegen (im Admin-UI vergeben).
|
|
const ADMINS = (process.env.ADMIN_EMAILS || '')
|
|
.split(',').map((s) => s.trim().toLowerCase()).filter(Boolean);
|
|
|
|
export function roleOf(user) {
|
|
const email = (user?.email || '').toLowerCase();
|
|
const meta = (user?.app_metadata?.role || '').toLowerCase();
|
|
if (ADMINS.includes(email) || meta === 'admin') return 'admin';
|
|
if (meta === 'editor') return 'editor';
|
|
return 'user';
|
|
}
|
|
|
|
// Verifiziert den Supabase-Access-Token und legt user/email/role im Kontext ab.
|
|
export async function requireAuth(c, next) {
|
|
const header = c.req.header('Authorization') || '';
|
|
const token = header.startsWith('Bearer ') ? header.slice(7) : null;
|
|
if (!token) return c.json({ error: 'Nicht eingeloggt' }, 401);
|
|
|
|
const user = await verifyToken(token);
|
|
if (!user) return c.json({ error: 'Ungültiges Token' }, 401);
|
|
|
|
const email = (user.email || '').toLowerCase();
|
|
const role = roleOf(user);
|
|
c.set('user', user);
|
|
c.set('email', email);
|
|
c.set('role', role);
|
|
c.set('isAdmin', role === 'admin');
|
|
c.set('canModerate', role === 'admin' || role === 'editor');
|
|
await next();
|
|
}
|
|
|
|
// Nur Admins (nach requireAuth einsetzen).
|
|
export async function requireAdmin(c, next) {
|
|
if (!c.get('isAdmin')) return c.json({ error: 'Nur für Admins' }, 403);
|
|
await next();
|
|
}
|
|
|
|
// Admins + Redakteure — fürs Moderieren (nach requireAuth einsetzen).
|
|
export async function requireModerator(c, next) {
|
|
if (!c.get('canModerate')) return c.json({ error: 'Nur für Moderation' }, 403);
|
|
await next();
|
|
}
|