Paßwort vergessen modernisiert
Some checks failed
Code Analysis and Production Deploy / analyze (push) Failing after 6m5s
Code Analysis and Production Deploy / deploy-production (push) Has been skipped
Code Analysis and Production Deploy / deploy-test (push) Has been skipped

This commit is contained in:
Torsten Schulz (local)
2026-06-09 10:31:32 +02:00
parent a98def915e
commit 300dce9835
7 changed files with 389 additions and 80 deletions

View File

@@ -49,7 +49,7 @@ fun PasswordResetScreen(
val state by viewModel.state.collectAsState() val state by viewModel.state.collectAsState()
AuthFormPage( AuthFormPage(
title = "Passwort zurücksetzen", title = "Passwort zurücksetzen",
subtitle = "Geben Sie Ihre E-Mail-Adresse ein, um Ihr Passwort zurückzusetzen.", subtitle = "Geben Sie Ihre E-Mail-Adresse ein, um einen Reset-Link zu erhalten.",
onBack = { navController.navigate(Destinations.Login.route) }, onBack = { navController.navigate(Destinations.Login.route) },
showBackNavigation = showBackNavigation, showBackNavigation = showBackNavigation,
) { ) {
@@ -68,7 +68,7 @@ fun PasswordResetScreen(
TextButton(onClick = { navController.navigate(Destinations.Login.route) }, modifier = Modifier.fillMaxWidth()) { TextButton(onClick = { navController.navigate(Destinations.Login.route) }, modifier = Modifier.fillMaxWidth()) {
Text("Zurück zum Login") Text("Zurück zum Login")
} }
AuthNotice("Sie erhalten eine E-Mail mit einem temporären Passwort, sofern ein Konto vorhanden ist.") AuthNotice("Sie erhalten eine E-Mail mit einem Reset-Link, sofern ein Konto vorhanden ist. Ihr bisheriges Passwort bleibt bis zur Änderung gültig.")
} }
} }

View File

@@ -8,8 +8,8 @@ LOCAL_API_BASE_URL=https://harheimertc.tsschulz.de/
PRODUCTION_API_BASE_URL=https://harheimertc.de/ PRODUCTION_API_BASE_URL=https://harheimertc.de/
# Android app versioning for Play Store uploads # Android app versioning for Play Store uploads
ANDROID_VERSION_CODE=23 ANDROID_VERSION_CODE=24
ANDROID_VERSION_NAME=0.9.18 ANDROID_VERSION_NAME=0.9.19
# Temporary hotfix: disable R8 minification for release to avoid Retrofit generic signature stripping. # Temporary hotfix: disable R8 minification for release to avoid Retrofit generic signature stripping.
RELEASE_MINIFY_ENABLED=false RELEASE_MINIFY_ENABLED=false

View File

@@ -6,7 +6,7 @@
Passwort zurücksetzen Passwort zurücksetzen
</h2> </h2>
<p class="mt-2 text-sm text-gray-600"> <p class="mt-2 text-sm text-gray-600">
Geben Sie Ihre E-Mail-Adresse ein, um Ihr Passwort zurückzusetzen Geben Sie Ihre E-Mail-Adresse ein, um einen Reset-Link zu erhalten
</p> </p>
</div> </div>

View File

@@ -0,0 +1,125 @@
<template>
<div class="min-h-full flex items-center justify-center py-16 px-4 sm:px-6 lg:px-8 bg-gray-50">
<div class="max-w-md w-full space-y-8">
<div class="text-center">
<h2 class="text-3xl font-display font-bold text-gray-900">
Neues Passwort setzen
</h2>
<p class="mt-2 text-sm text-gray-600">
Vergeben Sie ein neues Passwort für Ihren Zugang.
</p>
</div>
<div class="bg-white rounded-xl shadow-lg p-8">
<form class="space-y-6" @submit.prevent="handleSubmit">
<div v-if="!token" class="bg-red-50 border border-red-200 rounded-lg p-4">
<p class="text-sm text-red-800">Der Reset-Link ist unvollständig. Fordern Sie bitte einen neuen Link an.</p>
</div>
<div>
<label for="password" class="block text-sm font-medium text-gray-700 mb-2">
Neues Passwort
</label>
<input
id="password"
v-model="password"
type="password"
required
minlength="8"
autocomplete="new-password"
class="w-full px-4 py-3 border border-gray-300 rounded-lg focus:ring-2 focus:ring-primary-600 focus:border-transparent transition-all"
>
</div>
<div>
<label for="passwordRepeat" class="block text-sm font-medium text-gray-700 mb-2">
Neues Passwort wiederholen
</label>
<input
id="passwordRepeat"
v-model="passwordRepeat"
type="password"
required
minlength="8"
autocomplete="new-password"
class="w-full px-4 py-3 border border-gray-300 rounded-lg focus:ring-2 focus:ring-primary-600 focus:border-transparent transition-all"
>
</div>
<div v-if="errorMessage" class="bg-red-50 border border-red-200 rounded-lg p-4">
<p class="text-sm text-red-800">{{ errorMessage }}</p>
</div>
<div v-if="successMessage" class="bg-green-50 border border-green-200 rounded-lg p-4">
<p class="text-sm text-green-800 flex items-center">
<Check :size="18" class="mr-2" />
{{ successMessage }}
</p>
</div>
<button
type="submit"
:disabled="isLoading || !token || Boolean(successMessage)"
class="w-full px-6 py-3 bg-primary-600 hover:bg-primary-700 disabled:bg-gray-400 text-white font-semibold rounded-lg transition-colors flex items-center justify-center"
>
<Loader2 v-if="isLoading" :size="20" class="mr-2 animate-spin" />
<span>{{ isLoading ? 'Wird gespeichert...' : 'Passwort speichern' }}</span>
</button>
<div class="text-center">
<NuxtLink to="/login" class="text-sm text-primary-600 hover:text-primary-700 font-medium">
Zurück zum Login
</NuxtLink>
</div>
</form>
</div>
</div>
</div>
</template>
<script setup>
import { computed, ref } from 'vue'
import { Check, Loader2 } from 'lucide-vue-next'
const route = useRoute()
const token = computed(() => String(route.query.token || '').trim())
const password = ref('')
const passwordRepeat = ref('')
const isLoading = ref(false)
const errorMessage = ref('')
const successMessage = ref('')
const handleSubmit = async () => {
errorMessage.value = ''
successMessage.value = ''
if (password.value.length < 8) {
errorMessage.value = 'Das Passwort muss mindestens 8 Zeichen lang sein.'
return
}
if (password.value !== passwordRepeat.value) {
errorMessage.value = 'Die Passwörter stimmen nicht überein.'
return
}
isLoading.value = true
try {
const response = await $fetch('/api/auth/reset-password/complete', {
method: 'POST',
body: { token: token.value, password: password.value }
})
successMessage.value = response?.message || 'Ihr Passwort wurde geändert.'
password.value = ''
passwordRepeat.value = ''
} catch (error) {
errorMessage.value = error?.data?.message || 'Der Reset-Link ist ungültig oder abgelaufen.'
} finally {
isLoading.value = false
}
}
useHead({
title: 'Neues Passwort setzen - Harheimer TC',
})
</script>

View File

@@ -1,10 +1,45 @@
import { readUsers, hashPassword, writeUsers, revokeRefreshSessionsForUser } from '../../utils/auth.js' import { readUsers, writeUsers } from '../../utils/auth.js'
import nodemailer from 'nodemailer' import nodemailer from 'nodemailer'
import crypto from 'crypto' import crypto from 'crypto'
import { assertRateLimit, getClientIp, registerRateLimitFailure, registerRateLimitSuccess } from '../../utils/rate-limit.js' import { assertRateLimit, getClientIp, registerRateLimitFailure, registerRateLimitSuccess } from '../../utils/rate-limit.js'
import { writeAuditLog } from '../../utils/audit-log.js' import { writeAuditLog } from '../../utils/audit-log.js'
import { maskResetEmail, normalizeResetEmail, writePasswordResetLog } from '../../utils/password-reset-log.js' import { maskResetEmail, normalizeResetEmail, writePasswordResetLog } from '../../utils/password-reset-log.js'
const RESET_TOKEN_TTL_MINUTES = Number(process.env.PASSWORD_RESET_TTL_MIN || 60)
const RESET_TOKEN_MAX_AGE_MS = RESET_TOKEN_TTL_MINUTES * 60 * 1000
function generateResetToken() {
return crypto.randomBytes(32).toString('base64url')
}
function hashResetToken(token) {
return crypto.createHash('sha256').update(String(token), 'utf8').digest('hex')
}
function getResetBaseUrl(event) {
const configured = process.env.NUXT_PUBLIC_BASE_URL
if (configured) return configured.replace(/\/$/, '')
const requestUrl = getRequestURL(event)
return `${requestUrl.protocol}//${requestUrl.host}`
}
function prunePasswordResetTokens(user) {
const now = Date.now()
user.passwordResetTokens = (Array.isArray(user.passwordResetTokens) ? user.passwordResetTokens : [])
.filter(token => !token.usedAt && new Date(token.expiresAt).getTime() > now)
.slice(-4)
}
function escapeHtml(value) {
return String(value || '')
.replace(/&/g, '&amp;')
.replace(/</g, '&lt;')
.replace(/>/g, '&gt;')
.replace(/"/g, '&quot;')
.replace(/'/g, '&#39;')
}
export default defineEventHandler(async (event) => { export default defineEventHandler(async (event) => {
const requestId = crypto.randomUUID() const requestId = crypto.randomUUID()
let emailKey = '' let emailKey = ''
@@ -34,7 +69,6 @@ export default defineEventHandler(async (event) => {
}) })
} }
// Rate Limiting (IP + Account)
await logStep('rate_limit', 'checking') await logStep('rate_limit', 'checking')
try { try {
assertRateLimit(event, { assertRateLimit(event, {
@@ -57,7 +91,6 @@ export default defineEventHandler(async (event) => {
} }
await logStep('rate_limit', 'passed') await logStep('rate_limit', 'passed')
// Find user
let users let users
try { try {
users = await readUsers() users = await readUsers()
@@ -67,7 +100,6 @@ export default defineEventHandler(async (event) => {
} }
const user = users.find(u => normalizeResetEmail(u.email) === emailKey) const user = users.find(u => normalizeResetEmail(u.email) === emailKey)
// Always return success (security: don't reveal if email exists)
if (!user) { if (!user) {
await logStep('user_lookup', 'not_found') await logStep('user_lookup', 'not_found')
await registerRateLimitFailure(event, { name: 'auth:reset:ip', keyParts: [ip] }) await registerRateLimitFailure(event, { name: 'auth:reset:ip', keyParts: [ip] })
@@ -81,82 +113,84 @@ export default defineEventHandler(async (event) => {
} }
await logStep('user_lookup', 'found', { userId: user.id }) await logStep('user_lookup', 'found', { userId: user.id })
// Generate temporary password
const tempPassword = crypto.randomBytes(8).toString('hex')
const hashedPassword = await hashPassword(tempPassword)
await logStep('temporary_password', 'generated', { userId: user.id })
// Send email with temporary password
const smtpUser = process.env.SMTP_USER const smtpUser = process.env.SMTP_USER
const smtpPass = process.env.SMTP_PASS const smtpPass = process.env.SMTP_PASS
if (!smtpUser || !smtpPass) { if (!smtpUser || !smtpPass) {
await logStep('mail_configuration', 'failed', { userId: user.id, reason: 'smtp_credentials_missing' }) await logStep('mail_configuration', 'failed', { userId: user.id, reason: 'smtp_credentials_missing' })
console.warn('SMTP-Credentials fehlen! E-Mail-Versand wird übersprungen.') console.warn('SMTP-Credentials fehlen! E-Mail-Versand wird übersprungen.')
console.warn(`SMTP_USER=${smtpUser ? 'gesetzt' : 'FEHLT'}, SMTP_PASS=${smtpPass ? 'gesetzt' : 'FEHLT'}`) console.warn(`SMTP_USER=${smtpUser ? 'gesetzt' : 'FEHLT'}, SMTP_PASS=${smtpPass ? 'gesetzt' : 'FEHLT'}`)
throw new Error('SMTP-Konfiguration fuer Passwort-Reset fehlt') throw new Error('SMTP-Konfiguration fuer Passwort-Reset fehlt')
} else {
await logStep('mail_configuration', 'passed', { userId: user.id })
const transporter = nodemailer.createTransport({
host: process.env.SMTP_HOST || 'smtp.gmail.com',
port: process.env.SMTP_PORT || 587,
secure: false,
auth: {
user: smtpUser,
pass: smtpPass
}
})
const mailOptions = {
from: process.env.SMTP_FROM || 'noreply@harheimertc.de',
to: user.email,
subject: 'Passwort zurücksetzen - Harheimer TC',
html: `
<h2>Passwort zurücksetzen</h2>
<p>Hallo ${user.name},</p>
<p>Sie haben eine Anfrage zum Zurücksetzen Ihres Passworts gestellt.</p>
<p>Ihr temporäres Passwort lautet: <strong>${tempPassword}</strong></p>
<p>Bitte melden Sie sich damit an und ändern Sie Ihr Passwort im Mitgliederbereich.</p>
<br>
<p>Falls Sie diese Anfrage nicht gestellt haben, ignorieren Sie diese E-Mail.</p>
<br>
<p>Mit sportlichen Grüßen,<br>Ihr Harheimer TC</p>
`
}
await logStep('mail_send', 'started', { userId: user.id })
try {
await transporter.sendMail(mailOptions)
} catch (error) {
await logStep('mail_send', 'failed', { userId: user.id, error })
throw error
}
await logStep('mail_send', 'completed', { userId: user.id })
} }
// Erst nach erfolgreichem Versand das zugesandte Passwort aktivieren. await logStep('mail_configuration', 'passed', { userId: user.id })
user.password = hashedPassword
user.passwordResetRequired = true const token = generateResetToken()
const tokenHash = hashResetToken(token)
const nowIso = new Date().toISOString()
const expiresAt = new Date(Date.now() + RESET_TOKEN_MAX_AGE_MS).toISOString()
prunePasswordResetTokens(user)
user.passwordResetTokens.push({
tokenHash,
createdAt: nowIso,
expiresAt,
usedAt: null
})
await logStep('reset_token', 'generated', { userId: user.id, expiresAt })
const updatedUsers = users.map(u => u.id === user.id ? user : u) const updatedUsers = users.map(u => u.id === user.id ? user : u)
let passwordStored = false let tokenStored = false
try { try {
passwordStored = await writeUsers(updatedUsers) tokenStored = await writeUsers(updatedUsers)
} catch (error) { } catch (error) {
await logStep('password_storage', 'failed', { userId: user.id, error }) await logStep('token_storage', 'failed', { userId: user.id, error })
throw error throw error
} }
if (!passwordStored) { if (!tokenStored) {
await logStep('password_storage', 'failed', { userId: user.id, reason: 'write_failed' }) await logStep('token_storage', 'failed', { userId: user.id, reason: 'write_failed' })
throw new Error('Passwort konnte nach E-Mail-Versand nicht gespeichert werden') throw new Error('Reset-Token konnte nicht gespeichert werden')
} }
await logStep('password_storage', 'completed', { userId: user.id }) await logStep('token_storage', 'completed', { userId: user.id })
const transporter = nodemailer.createTransport({
host: process.env.SMTP_HOST || 'smtp.gmail.com',
port: process.env.SMTP_PORT || 587,
secure: process.env.SMTP_SECURE === 'true',
auth: {
user: smtpUser,
pass: smtpPass
}
})
const resetUrl = `${getResetBaseUrl(event)}/passwort-zuruecksetzen?token=${encodeURIComponent(token)}`
const displayName = escapeHtml(user.name || 'Mitglied')
const mailOptions = {
from: process.env.SMTP_FROM || 'noreply@harheimertc.de',
to: user.email,
subject: 'Passwort zurücksetzen - Harheimer TC',
html: `
<h2>Passwort zurücksetzen</h2>
<p>Hallo ${displayName},</p>
<p>Sie haben eine Anfrage zum Zurücksetzen Ihres Passworts gestellt.</p>
<p>Bitte klicken Sie auf den folgenden Link und vergeben Sie dort ein neues Passwort. Der Link ist ${RESET_TOKEN_TTL_MINUTES} Minuten gültig:</p>
<p><a href="${resetUrl}">Neues Passwort setzen</a></p>
<p>Ihr bisheriges Passwort bleibt gültig, bis Sie über diesen Link ein neues Passwort gesetzt haben.</p>
<br>
<p>Falls Sie diese Anfrage nicht gestellt haben, ignorieren Sie diese E-Mail.</p>
<br>
<p>Mit sportlichen Grüßen,<br>Ihr Harheimer TC</p>
`
}
await logStep('mail_send', 'started', { userId: user.id })
try { try {
await revokeRefreshSessionsForUser(user.id, 'password_reset') await transporter.sendMail(mailOptions)
} catch (error) { } catch (error) {
await logStep('session_revocation', 'failed', { userId: user.id, error }) await logStep('mail_send', 'failed', { userId: user.id, error })
throw error throw error
} }
await logStep('session_revocation', 'completed', { userId: user.id }) await logStep('mail_send', 'completed', { userId: user.id })
registerRateLimitSuccess(event, { name: 'auth:reset:account', keyParts: [emailKey] }) registerRateLimitSuccess(event, { name: 'auth:reset:account', keyParts: [emailKey] })
await writeAuditLog('auth.reset.request', { ip, emailMasked: maskResetEmail(emailKey), userFound: true, userId: user.id, requestId }) await writeAuditLog('auth.reset.request', { ip, emailMasked: maskResetEmail(emailKey), userFound: true, userId: user.id, requestId })
@@ -168,7 +202,6 @@ export default defineEventHandler(async (event) => {
} catch (error) { } catch (error) {
await logStep('request_completed', 'failed', { error }) await logStep('request_completed', 'failed', { error })
console.error('Password-Reset-Fehler:', { requestId, code: error?.code || error?.name || 'Error' }) console.error('Password-Reset-Fehler:', { requestId, code: error?.code || error?.name || 'Error' })
// Don't reveal errors to prevent email enumeration
return { return {
success: true, success: true,
message: 'Falls ein Konto mit dieser E-Mail existiert, wurde eine E-Mail gesendet.' message: 'Falls ein Konto mit dieser E-Mail existiert, wurde eine E-Mail gesendet.'

View File

@@ -0,0 +1,82 @@
import crypto from 'crypto'
import { hashPassword, readUsers, revokeRefreshSessionsForUser, writeUsers } from '../../../utils/auth.js'
import { getClientIp } from '../../../utils/rate-limit.js'
import { writeAuditLog } from '../../../utils/audit-log.js'
import { writePasswordResetLog } from '../../../utils/password-reset-log.js'
function hashResetToken(token) {
return crypto.createHash('sha256').update(String(token), 'utf8').digest('hex')
}
function isStrongEnoughPassword(password) {
return typeof password === 'string' && password.length >= 8
}
export default defineEventHandler(async (event) => {
const requestId = crypto.randomUUID()
const ip = getClientIp(event)
const body = await readBody(event)
const token = String(body?.token || '').trim()
const password = String(body?.password || '')
const logStep = async (step, status, detail = {}) => {
try {
await writePasswordResetLog({ requestId, email: detail.email || '', ip, step, status, ...detail })
} catch (logError) {
console.error('Password-Reset-Diagnoselog-Fehler:', logError)
}
}
if (!token) {
await logStep('complete_validation', 'failed', { reason: 'token_missing' })
throw createError({ statusCode: 400, message: 'Reset-Link fehlt.' })
}
if (!isStrongEnoughPassword(password)) {
await logStep('complete_validation', 'failed', { reason: 'password_too_short' })
throw createError({ statusCode: 400, message: 'Das Passwort muss mindestens 8 Zeichen lang sein.' })
}
const users = await readUsers()
const tokenHash = hashResetToken(token)
const now = Date.now()
let matchedUser = null
let matchedToken = null
for (const user of users) {
const tokens = Array.isArray(user.passwordResetTokens) ? user.passwordResetTokens : []
const candidate = tokens.find(entry => entry.tokenHash === tokenHash)
if (candidate) {
matchedUser = user
matchedToken = candidate
break
}
}
if (!matchedUser || !matchedToken || matchedToken.usedAt || new Date(matchedToken.expiresAt).getTime() <= now) {
await logStep('complete_token', 'failed', { reason: 'invalid_or_expired' })
throw createError({ statusCode: 400, message: 'Der Reset-Link ist ungültig oder abgelaufen.' })
}
const nowIso = new Date().toISOString()
matchedUser.password = await hashPassword(password)
matchedUser.passwordResetRequired = false
matchedToken.usedAt = nowIso
matchedUser.passwordResetTokens = (Array.isArray(matchedUser.passwordResetTokens) ? matchedUser.passwordResetTokens : [])
.filter(entry => entry.usedAt || new Date(entry.expiresAt).getTime() > now)
const stored = await writeUsers(users)
if (!stored) {
await logStep('complete_password_storage', 'failed', { userId: matchedUser.id, email: matchedUser.email, reason: 'write_failed' })
throw createError({ statusCode: 500, message: 'Das neue Passwort konnte nicht gespeichert werden.' })
}
await revokeRefreshSessionsForUser(matchedUser.id, 'password_reset_completed')
await writeAuditLog('auth.reset.complete', { ip, userId: matchedUser.id, requestId })
await logStep('complete_password_storage', 'completed', { userId: matchedUser.id, email: matchedUser.email })
return {
success: true,
message: 'Ihr Passwort wurde geändert. Sie können sich jetzt mit dem neuen Passwort anmelden.'
}
})

View File

@@ -1,6 +1,7 @@
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest' import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest'
import { createEvent, mockSuccessReadBody } from './setup' import { createEvent, mockSuccessReadBody } from './setup'
import { readFileSync } from 'fs' import { readFileSync } from 'fs'
import crypto from 'crypto'
vi.mock('../server/utils/auth.js', () => { vi.mock('../server/utils/auth.js', () => {
return { return {
@@ -74,6 +75,7 @@ import logoutHandler from '../server/api/auth/logout.post.js'
import refreshHandler from '../server/api/auth/refresh.post.js' import refreshHandler from '../server/api/auth/refresh.post.js'
import registerHandler from '../server/api/auth/register.post.js' import registerHandler from '../server/api/auth/register.post.js'
import resetPasswordHandler from '../server/api/auth/reset-password.post.js' import resetPasswordHandler from '../server/api/auth/reset-password.post.js'
import completePasswordResetHandler from '../server/api/auth/reset-password/complete.post.js'
import statusHandler from '../server/api/auth/status.get.js' import statusHandler from '../server/api/auth/status.get.js'
import versionHandler from '../server/api/app/version.get.js' import versionHandler from '../server/api/app/version.get.js'
@@ -81,12 +83,15 @@ describe('Auth API Endpoints', () => {
afterEach(() => { afterEach(() => {
delete process.env.NODE_ENV delete process.env.NODE_ENV
delete process.env.APP_ENV delete process.env.APP_ENV
delete process.env.NUXT_PUBLIC_BASE_URL
delete process.env.PASSWORD_RESET_TTL_MIN
}) })
beforeEach(() => { beforeEach(() => {
// Setze SMTP-Credentials für Tests // Setze SMTP-Credentials für Tests
process.env.SMTP_USER = 'test@example.com' process.env.SMTP_USER = 'test@example.com'
process.env.SMTP_PASS = 'test-password' process.env.SMTP_PASS = 'test-password'
process.env.NUXT_PUBLIC_BASE_URL = 'https://harheimertc.de'
vi.clearAllMocks() vi.clearAllMocks()
}) })
@@ -300,7 +305,7 @@ describe('Auth API Endpoints', () => {
}) })
describe('POST /api/auth/reset-password', () => { describe('POST /api/auth/reset-password', () => {
it('prüft Pflichtfelder', async () => { it('prüft Pflichtfelder ohne öffentliche Fehlermeldung', async () => {
const event = createEvent() const event = createEvent()
mockSuccessReadBody({}) mockSuccessReadBody({})
@@ -308,18 +313,27 @@ describe('Auth API Endpoints', () => {
expect(response.success).toBe(true) expect(response.success).toBe(true)
}) })
it('aktualisiert Passwort bei vorhandenem Benutzer', async () => { it('speichert einen gehashten Reset-Token und lässt das alte Passwort unverändert', async () => {
const event = createEvent() const event = createEvent()
const user = { id: '1', email: 'user@example.com', name: 'User', password: 'hash' } const user = { id: '1', email: 'user@example.com', name: 'User', password: 'old-hash' }
mockSuccessReadBody({ email: user.email }) mockSuccessReadBody({ email: user.email })
authUtils.readUsers.mockResolvedValue([user]) authUtils.readUsers.mockResolvedValue([user])
authUtils.hashPassword.mockResolvedValue('new-hash')
authUtils.writeUsers.mockResolvedValue(true) authUtils.writeUsers.mockResolvedValue(true)
const response = await resetPasswordHandler(event) const response = await resetPasswordHandler(event)
expect(response.success).toBe(true) expect(response.success).toBe(true)
expect(authUtils.writeUsers).toHaveBeenCalled() expect(authUtils.hashPassword).not.toHaveBeenCalled()
expect(authUtils.revokeRefreshSessionsForUser).toHaveBeenCalledWith('1', 'password_reset') expect(authUtils.revokeRefreshSessionsForUser).not.toHaveBeenCalled()
const writtenUser = authUtils.writeUsers.mock.calls[0][0][0]
expect(writtenUser.password).toBe('old-hash')
expect(writtenUser.passwordResetTokens).toHaveLength(1)
expect(writtenUser.passwordResetTokens[0]).toMatchObject({ usedAt: null })
expect(writtenUser.passwordResetTokens[0].tokenHash).toMatch(/^[a-f0-9]{64}$/)
const transporter = nodemailer.default.createTransport.mock.results[0].value
expect(transporter.sendMail).toHaveBeenCalledWith(expect.objectContaining({
html: expect.stringContaining('https://harheimertc.de/passwort-zuruecksetzen?token=')
}))
expect(passwordResetLog.writePasswordResetLog).toHaveBeenCalledWith(expect.objectContaining({ expect(passwordResetLog.writePasswordResetLog).toHaveBeenCalledWith(expect.objectContaining({
email: 'user@example.com', email: 'user@example.com',
step: 'mail_send', step: 'mail_send',
@@ -332,7 +346,6 @@ describe('Auth API Endpoints', () => {
const user = { id: '1', email: 'user@example.com', name: 'User', password: 'hash' } const user = { id: '1', email: 'user@example.com', name: 'User', password: 'hash' }
mockSuccessReadBody({ email: ' User@Example.com ' }) mockSuccessReadBody({ email: ' User@Example.com ' })
authUtils.readUsers.mockResolvedValue([user]) authUtils.readUsers.mockResolvedValue([user])
authUtils.hashPassword.mockResolvedValue('new-hash')
authUtils.writeUsers.mockResolvedValue(true) authUtils.writeUsers.mockResolvedValue(true)
await resetPasswordHandler(event) await resetPasswordHandler(event)
@@ -341,11 +354,10 @@ describe('Auth API Endpoints', () => {
expect(passwordResetLog.normalizeResetEmail).toHaveBeenCalledWith(' User@Example.com ') expect(passwordResetLog.normalizeResetEmail).toHaveBeenCalledWith(' User@Example.com ')
}) })
it('ändert das Passwort nicht, wenn SMTP nicht konfiguriert ist', async () => { it('ändert nichts, wenn SMTP nicht konfiguriert ist', async () => {
const event = createEvent() const event = createEvent()
mockSuccessReadBody({ email: 'user@example.com' }) mockSuccessReadBody({ email: 'user@example.com' })
authUtils.readUsers.mockResolvedValue([{ id: '1', email: 'user@example.com', name: 'User', password: 'hash' }]) authUtils.readUsers.mockResolvedValue([{ id: '1', email: 'user@example.com', name: 'User', password: 'hash' }])
authUtils.hashPassword.mockResolvedValue('new-hash')
delete process.env.SMTP_USER delete process.env.SMTP_USER
delete process.env.SMTP_PASS delete process.env.SMTP_PASS
@@ -360,11 +372,12 @@ describe('Auth API Endpoints', () => {
})) }))
}) })
it('protokolliert einen Mailfehler ohne das Passwort zu aktivieren', async () => { it('protokolliert einen Mailfehler ohne das Passwort zu ersetzen', async () => {
const event = createEvent() const event = createEvent()
const user = { id: '1', email: 'user@example.com', name: 'User', password: 'hash' }
mockSuccessReadBody({ email: 'user@example.com' }) mockSuccessReadBody({ email: 'user@example.com' })
authUtils.readUsers.mockResolvedValue([{ id: '1', email: 'user@example.com', name: 'User', password: 'hash' }]) authUtils.readUsers.mockResolvedValue([user])
authUtils.hashPassword.mockResolvedValue('new-hash') authUtils.writeUsers.mockResolvedValue(true)
nodemailer.default.createTransport.mockReturnValueOnce({ nodemailer.default.createTransport.mockReturnValueOnce({
sendMail: vi.fn().mockRejectedValue(Object.assign(new Error('SMTP fehlgeschlagen'), { code: 'EAUTH' })) sendMail: vi.fn().mockRejectedValue(Object.assign(new Error('SMTP fehlgeschlagen'), { code: 'EAUTH' }))
}) })
@@ -372,7 +385,8 @@ describe('Auth API Endpoints', () => {
const response = await resetPasswordHandler(event) const response = await resetPasswordHandler(event)
expect(response.success).toBe(true) expect(response.success).toBe(true)
expect(authUtils.writeUsers).not.toHaveBeenCalled() expect(authUtils.hashPassword).not.toHaveBeenCalled()
expect(authUtils.writeUsers.mock.calls[0][0][0].password).toBe('hash')
expect(passwordResetLog.writePasswordResetLog).toHaveBeenCalledWith(expect.objectContaining({ expect(passwordResetLog.writePasswordResetLog).toHaveBeenCalledWith(expect.objectContaining({
step: 'mail_send', step: 'mail_send',
status: 'failed' status: 'failed'
@@ -380,6 +394,61 @@ describe('Auth API Endpoints', () => {
}) })
}) })
describe('POST /api/auth/reset-password/complete', () => {
it('setzt ein neues Passwort mit gültigem Reset-Token', async () => {
const token = 'reset-token'
const tokenHash = crypto.createHash('sha256').update(token, 'utf8').digest('hex')
const event = createEvent()
const user = {
id: '1',
email: 'user@example.com',
password: 'old-hash',
passwordResetRequired: true,
passwordResetTokens: [{ tokenHash, createdAt: new Date().toISOString(), expiresAt: new Date(Date.now() + 60000).toISOString(), usedAt: null }]
}
mockSuccessReadBody({ token, password: 'new-password' })
authUtils.readUsers.mockResolvedValue([user])
authUtils.hashPassword.mockResolvedValue('new-hash')
authUtils.writeUsers.mockResolvedValue(true)
const response = await completePasswordResetHandler(event)
expect(response.success).toBe(true)
expect(authUtils.hashPassword).toHaveBeenCalledWith('new-password')
expect(authUtils.writeUsers.mock.calls[0][0][0]).toMatchObject({
password: 'new-hash',
passwordResetRequired: false
})
expect(authUtils.writeUsers.mock.calls[0][0][0].passwordResetTokens[0].usedAt).toEqual(expect.any(String))
expect(authUtils.revokeRefreshSessionsForUser).toHaveBeenCalledWith('1', 'password_reset_completed')
})
it('weist abgelaufene Reset-Tokens zurück', async () => {
const token = 'reset-token'
const tokenHash = crypto.createHash('sha256').update(token, 'utf8').digest('hex')
const event = createEvent()
mockSuccessReadBody({ token, password: 'new-password' })
authUtils.readUsers.mockResolvedValue([{
id: '1',
email: 'user@example.com',
password: 'old-hash',
passwordResetTokens: [{ tokenHash, expiresAt: new Date(Date.now() - 60000).toISOString(), usedAt: null }]
}])
await expect(completePasswordResetHandler(event)).rejects.toMatchObject({ statusCode: 400 })
expect(authUtils.writeUsers).not.toHaveBeenCalled()
expect(authUtils.hashPassword).not.toHaveBeenCalled()
})
it('verlangt mindestens acht Zeichen für das neue Passwort', async () => {
const event = createEvent()
mockSuccessReadBody({ token: 'reset-token', password: 'kurz' })
await expect(completePasswordResetHandler(event)).rejects.toMatchObject({ statusCode: 400 })
expect(authUtils.readUsers).not.toHaveBeenCalled()
})
})
describe('GET /api/auth/status', () => { describe('GET /api/auth/status', () => {
it('liefert loggedOut, wenn kein Cookie gesetzt ist', async () => { it('liefert loggedOut, wenn kein Cookie gesetzt ist', async () => {
const event = createEvent() const event = createEvent()