Files
dockhand/routes/api/users/[id]/mfa/+server.ts
Jarek Krochmalski 62e3c6439e Initial commit
2025-12-28 21:16:03 +01:00

97 lines
2.7 KiB
TypeScript

import { json } from '@sveltejs/kit';
import type { RequestHandler } from '@sveltejs/kit';
import { isEnterprise } from '$lib/server/license';
import {
validateSession,
checkPermission,
generateMfaSetup,
verifyAndEnableMfa,
disableMfa
} from '$lib/server/auth';
// POST /api/users/[id]/mfa - Setup MFA (generate QR code)
export const POST: RequestHandler = async ({ params, request, cookies }) => {
// Check enterprise license
if (!(await isEnterprise())) {
return json({ error: 'Enterprise license required' }, { status: 403 });
}
const currentUser = await validateSession(cookies);
if (!params.id) {
return json({ error: 'User ID is required' }, { status: 400 });
}
const userId = parseInt(params.id);
// Users can only setup MFA for themselves, or admins can do it for others
if (!currentUser || (currentUser.id !== userId && !currentUser.isAdmin)) {
return json({ error: 'Permission denied' }, { status: 403 });
}
try {
const body = await request.json().catch(() => ({}));
// Check if this is a verification request
if (body.action === 'verify') {
if (!body.token) {
return json({ error: 'MFA token is required' }, { status: 400 });
}
const success = await verifyAndEnableMfa(userId, body.token);
if (!success) {
return json({ error: 'Invalid MFA code' }, { status: 400 });
}
return json({ success: true, message: 'MFA enabled successfully' });
}
// Generate new MFA setup
const setup = await generateMfaSetup(userId);
if (!setup) {
return json({ error: 'User not found' }, { status: 404 });
}
return json({
secret: setup.secret,
qrDataUrl: setup.qrDataUrl
});
} catch (error) {
console.error('MFA setup error:', error);
return json({ error: 'Failed to setup MFA' }, { status: 500 });
}
};
// DELETE /api/users/[id]/mfa - Disable MFA
export const DELETE: RequestHandler = async ({ params, cookies }) => {
// Check enterprise license
if (!(await isEnterprise())) {
return json({ error: 'Enterprise license required' }, { status: 403 });
}
const currentUser = await validateSession(cookies);
if (!params.id) {
return json({ error: 'User ID is required' }, { status: 400 });
}
const userId = parseInt(params.id);
// Users can only disable their own MFA, or admins can do it for others
if (!currentUser || (currentUser.id !== userId && !currentUser.isAdmin)) {
return json({ error: 'Permission denied' }, { status: 403 });
}
try {
const success = await disableMfa(userId);
if (!success) {
return json({ error: 'User not found' }, { status: 404 });
}
return json({ success: true, message: 'MFA disabled successfully' });
} catch (error) {
console.error('MFA disable error:', error);
return json({ error: 'Failed to disable MFA' }, { status: 500 });
}
};