feat(backend): add cooldown and cancel-cooldown endpoints

This commit is contained in:
chahinebrini 2026-06-16 20:21:00 +02:00
parent b9bd577e47
commit 4ee6849f3f
2 changed files with 81 additions and 0 deletions

View File

@ -0,0 +1,30 @@
/**
* POST /api/magic/devices/:deviceId/cancel-cooldown
*
* Removes an active Magic device cooldown immediately.
*/
export default defineEventHandler(async (event) => {
const user = await requireUser(event);
const deviceId = getRouterParam(event, "deviceId");
if (!deviceId) {
throw createError({ statusCode: 400, message: "deviceId required" });
}
const db = usePrisma();
const device = await db.userDevice.findUnique({
where: { userId_deviceId: { userId: user.id, deviceId } },
select: { id: true, magicEnrolledAt: true, magicRevokedAt: true },
});
if (!device || !device.magicEnrolledAt || device.magicRevokedAt) {
throw createError({ statusCode: 404, message: "Magic-Binding nicht gefunden" });
}
await db.userDevice.update({
where: { id: device.id },
data: { magicCooldownUntil: null },
});
return { success: true, data: { cooldownUntil: null } };
});

View File

@ -0,0 +1,51 @@
/**
* POST /api/magic/devices/:deviceId/cooldown
*
* Sets a temporary cooldown on a Magic device binding until the given
* duration (in minutes, 11440) elapses.
*/
export default defineEventHandler(async (event) => {
const user = await requireUser(event);
const deviceId = getRouterParam(event, "deviceId");
if (!deviceId) {
throw createError({ statusCode: 400, message: "deviceId required" });
}
const body = await readBody(event);
const raw = body?.durationMinutes;
if (
typeof raw !== "number" ||
!Number.isFinite(raw) ||
!Number.isInteger(raw) ||
raw < 1 ||
raw > 1440
) {
throw createError({
statusCode: 400,
message: "durationMinutes must be an integer between 1 and 1440",
});
}
const durationMinutes = raw;
const cooldownUntil = new Date(Date.now() + durationMinutes * 60 * 1000);
const db = usePrisma();
const device = await db.userDevice.findUnique({
where: { userId_deviceId: { userId: user.id, deviceId } },
select: { id: true, magicEnrolledAt: true, magicRevokedAt: true },
});
if (!device || !device.magicEnrolledAt || device.magicRevokedAt) {
throw createError({ statusCode: 404, message: "Magic-Binding nicht gefunden" });
}
await db.userDevice.update({
where: { id: device.id },
data: { magicCooldownUntil: cooldownUntil },
});
return {
success: true,
data: { cooldownUntil: cooldownUntil.toISOString() },
};
});