feat: Passkey (WebAuthn) login (#1111)

* feat(auth): passkey (WebAuthn) login — server endpoints, schema + admin toggle

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.

* feat(auth): passkey enrolment, login button + admin settings UI

PasskeysSection in account settings (add/rename/remove with a current-password step-up), a 'Sign in with a passkey' button on the login page, the admin enable + RP-ID/origins controls, and a per-user admin reset action.

* i18n(auth): passkey strings across all locales

Add login/settings/admin passkey keys to en and all 19 translated locales.
This commit is contained in:
Maurice
2026-06-05 18:54:13 +02:00
committed by GitHub
parent 247433fb2a
commit a876fb2634
83 changed files with 2421 additions and 8 deletions
+24
View File
@@ -300,6 +300,30 @@ const settings: TranslationStrings = {
'settings.notificationPreferences.ntfy': 'Ntfy',
"settings.currency": "Währung",
"settings.currencyHint": "Alle Beträge in Costs werden in diese Währung umgerechnet und angezeigt.",
'settings.passkey.title': 'Passkeys',
'settings.passkey.description':
'Melde dich schneller und phishing-resistent mit einem Passkey an — per Fingerabdruck, Gesicht, PIN oder Hardware-Schlüssel. Dein Passwort bleibt als Backup erhalten.',
'settings.passkey.notConfigured':
'Passkeys sind aktiviert, aber auf diesem Server noch nicht vollständig eingerichtet. Bitte deinen Administrator, die WebAuthn-Domain festzulegen.',
'settings.passkey.add': 'Passkey hinzufügen',
'settings.passkey.addTitle': 'Passkey hinzufügen',
'settings.passkey.passwordPrompt':
'Bestätige dein aktuelles Passwort und folge dann der Aufforderung deines Geräts.',
'settings.passkey.passwordRequired': 'Dein aktuelles Passwort wird benötigt.',
'settings.passkey.namePlaceholder': 'Name (optional, z. B. "iPhone")',
'settings.passkey.addedToast': 'Passkey hinzugefügt',
'settings.passkey.added': 'Hinzugefügt',
'settings.passkey.addError': 'Passkey konnte nicht hinzugefügt werden',
'settings.passkey.cancelled': 'Passkey-Einrichtung abgebrochen',
'settings.passkey.deleted': 'Passkey entfernt',
'settings.passkey.deleteConfirm':
'Diesen Passkey entfernen? Bestätige mit deinem Passwort.',
'settings.passkey.rename': 'Umbenennen',
'settings.passkey.defaultName': 'Passkey',
'settings.passkey.synced': 'Synchronisiert',
'settings.passkey.deviceBound': 'Dieses Gerät',
'settings.passkey.lastUsed': 'Zuletzt verwendet',
'settings.passkey.neverUsed': 'Noch nie verwendet',
};
export default settings;