mirror of
https://github.com/mauriceboe/TREK.git
synced 2026-06-19 21:31:46 +00:00
5b8c61d215
Add @simplewebauthn/server registration and primary (discoverable) login ceremonies under /api/auth/passkey, a webauthn_credentials + single-use webauthn_challenges schema (migration), the instance-wide passkey_login toggle (default off) enforced before auth by a guard, and require_mfa satisfaction via a verified passkey. RP ID/origin come only from server config (webauthn_rp_id/origins -> APP_URL), never request headers.
115 lines
5.2 KiB
TypeScript
115 lines
5.2 KiB
TypeScript
import { Body, Controller, Delete, Get, HttpCode, HttpException, Param, Patch, Post, Req, Res, UseGuards } from '@nestjs/common';
|
|
import type { Request, Response } from 'express';
|
|
import { RateLimitService } from './rate-limit.service';
|
|
import { JwtAuthGuard } from './jwt-auth.guard';
|
|
import { PasskeyEnabledGuard } from './passkey-enabled.guard';
|
|
import { CurrentUser } from './current-user.decorator';
|
|
import { setAuthCookie } from '../../services/cookie';
|
|
import { writeAudit, getClientIp } from '../../services/auditLog';
|
|
import * as passkey from '../../services/passkeyService';
|
|
import type { User } from '../../types';
|
|
|
|
const WINDOW = 15 * 60 * 1000;
|
|
const LOGIN_MIN_LATENCY_MS = 350;
|
|
const delay = (ms: number) => new Promise((r) => setTimeout(r, ms));
|
|
|
|
/**
|
|
* /api/auth/passkey — WebAuthn (passkey) registration, primary login and
|
|
* credential management.
|
|
*
|
|
* - register/* : authenticated, gated by the admin toggle + password re-auth.
|
|
* - login/* : UNauthenticated discoverable-credential login, gated by the
|
|
* admin toggle; mints the SAME session cookie as password login.
|
|
* - credentials : owner-scoped management — intentionally NOT toggle-gated so a
|
|
* user can always view/remove their passkeys.
|
|
*
|
|
* PasskeyEnabledGuard is listed first so a disabled feature 404s before auth.
|
|
*/
|
|
@Controller('api/auth/passkey')
|
|
export class PasskeyController {
|
|
constructor(private readonly rl: RateLimitService) {}
|
|
|
|
private limit(bucket: string, req: Request, max: number): void {
|
|
if (!this.rl.check(bucket, req.ip || 'unknown', max, WINDOW, Date.now())) {
|
|
throw new HttpException({ error: 'Too many attempts. Please try again later.' }, 429);
|
|
}
|
|
}
|
|
|
|
// ── Registration (authenticated) ──
|
|
@Post('register/options')
|
|
@HttpCode(200)
|
|
@UseGuards(PasskeyEnabledGuard, JwtAuthGuard)
|
|
async registerOptions(@CurrentUser() user: User, @Body() body: { password?: string }, @Req() req: Request) {
|
|
this.limit('mfa', req, 5);
|
|
const result = await passkey.passkeyRegisterOptions(user.id, body?.password);
|
|
if (result.error) throw new HttpException({ error: result.error }, result.status!);
|
|
return result.options;
|
|
}
|
|
|
|
@Post('register/verify')
|
|
@HttpCode(200)
|
|
@UseGuards(PasskeyEnabledGuard, JwtAuthGuard)
|
|
async registerVerify(@CurrentUser() user: User, @Body() body: unknown, @Req() req: Request) {
|
|
const result = await passkey.passkeyRegisterVerify(user.id, body as Parameters<typeof passkey.passkeyRegisterVerify>[1]);
|
|
if (result.error) throw new HttpException({ error: result.error }, result.status!);
|
|
writeAudit({ userId: user.id, action: 'user.passkey_register', ip: getClientIp(req) });
|
|
return { success: true, credential: result.credential };
|
|
}
|
|
|
|
// ── Authentication (public — primary login) ──
|
|
@Post('login/options')
|
|
@HttpCode(200)
|
|
@UseGuards(PasskeyEnabledGuard)
|
|
async loginOptions(@Req() req: Request) {
|
|
this.limit('login', req, 10);
|
|
const result = await passkey.passkeyLoginOptions();
|
|
if (result.error) throw new HttpException({ error: result.error }, result.status!);
|
|
return result.options;
|
|
}
|
|
|
|
@Post('login/verify')
|
|
@HttpCode(200)
|
|
@UseGuards(PasskeyEnabledGuard)
|
|
async loginVerify(@Body() body: unknown, @Req() req: Request, @Res({ passthrough: true }) res: Response) {
|
|
this.limit('login', req, 10);
|
|
const started = Date.now();
|
|
const result = await passkey.passkeyLoginVerify(body as Parameters<typeof passkey.passkeyLoginVerify>[0]);
|
|
if (result.auditAction) {
|
|
writeAudit({ userId: result.auditUserId ?? null, action: result.auditAction, ip: getClientIp(req) });
|
|
}
|
|
// Pad to the same floor as password login so timing can't distinguish a
|
|
// known credential from an unknown one.
|
|
const elapsed = Date.now() - started;
|
|
if (elapsed < LOGIN_MIN_LATENCY_MS) await delay(LOGIN_MIN_LATENCY_MS - elapsed);
|
|
if (result.error) throw new HttpException({ error: result.error }, result.status!);
|
|
writeAudit({ userId: result.auditUserId!, action: 'user.login', ip: getClientIp(req), details: { method: 'passkey' } });
|
|
setAuthCookie(res, result.token!, req);
|
|
return { token: result.token, user: result.user };
|
|
}
|
|
|
|
// ── Management (authenticated, owner-scoped — NOT toggle-gated) ──
|
|
@Get('credentials')
|
|
@UseGuards(JwtAuthGuard)
|
|
list(@CurrentUser() user: User) {
|
|
return { credentials: passkey.listPasskeys(user.id) };
|
|
}
|
|
|
|
@Patch('credentials/:id')
|
|
@UseGuards(JwtAuthGuard)
|
|
rename(@CurrentUser() user: User, @Param('id') id: string, @Body() body: { name?: unknown }) {
|
|
const result = passkey.renamePasskey(user.id, id, body?.name);
|
|
if (result.error) throw new HttpException({ error: result.error }, result.status!);
|
|
return { success: true };
|
|
}
|
|
|
|
@Delete('credentials/:id')
|
|
@UseGuards(JwtAuthGuard)
|
|
remove(@CurrentUser() user: User, @Param('id') id: string, @Body() body: { password?: string }, @Req() req: Request) {
|
|
this.limit('login', req, 5);
|
|
const result = passkey.deletePasskey(user.id, id, body?.password);
|
|
if (result.error) throw new HttpException({ error: result.error }, result.status!);
|
|
writeAudit({ userId: user.id, action: 'user.passkey_delete', resource: String(id), ip: getClientIp(req) });
|
|
return { success: true };
|
|
}
|
|
}
|