57 lines
1.9 KiB
TypeScript
57 lines
1.9 KiB
TypeScript
import { type ActionFunction, json, redirect } from "@remix-run/node";
|
|
import { TokenType } from "@prisma/client";
|
|
|
|
import db from "~/utils/db.server";
|
|
import logger from "~/utils/logger.server";
|
|
import { type FormError, validate } from "~/utils/validation.server";
|
|
import { authenticate, hashPassword } from "~/utils/auth.server";
|
|
import { ResetPasswordError } from "~/utils/errors";
|
|
import { hashToken } from "~/utils/token.server";
|
|
import { ResetPassword } from "../validations";
|
|
|
|
export type ResetPasswordActionData = { errors: FormError<typeof ResetPassword> };
|
|
|
|
const action: ActionFunction = async ({ request }) => {
|
|
const searchParams = new URL(request.url).searchParams;
|
|
const token = searchParams.get("token");
|
|
if (!token) {
|
|
return redirect("/forgot-password");
|
|
}
|
|
|
|
const formData = Object.fromEntries(await request.formData());
|
|
const validation = validate(ResetPassword, { ...formData, token });
|
|
if (validation.errors) {
|
|
return json<ResetPasswordActionData>({ errors: validation.errors });
|
|
}
|
|
|
|
const hashedToken = hashToken(token);
|
|
const savedToken = await db.token.findFirst({
|
|
where: { hashedToken, type: TokenType.RESET_PASSWORD },
|
|
include: { user: true },
|
|
});
|
|
if (!savedToken) {
|
|
logger.warn(`No token found with hashedToken=${hashedToken}`);
|
|
throw new ResetPasswordError();
|
|
}
|
|
|
|
await db.token.delete({ where: { id: savedToken.id } });
|
|
|
|
if (savedToken.expiresAt < new Date()) {
|
|
logger.warn(`Token with hashedToken=${hashedToken} is expired since ${savedToken.expiresAt.toUTCString()}`);
|
|
throw new ResetPasswordError();
|
|
}
|
|
|
|
const password = validation.data.password.trim();
|
|
const hashedPassword = await hashPassword(password);
|
|
const { email } = await db.user.update({
|
|
where: { id: savedToken.userId },
|
|
data: { hashedPassword },
|
|
});
|
|
|
|
await db.session.deleteMany({ where: { userId: savedToken.userId } });
|
|
|
|
return authenticate({ email, password, request });
|
|
};
|
|
|
|
export default action;
|