feat: send forgot password email
This commit is contained in:
65
a.md
65
a.md
@@ -1,65 +0,0 @@
|
|||||||
// import { NextApiRequest, NextApiResponse } from 'next';
|
|
||||||
|
|
||||||
// import crypto from 'crypto';
|
|
||||||
|
|
||||||
// import { sendResetPassword } from '@documenso/lib/mail';
|
|
||||||
// import { defaultHandler, defaultResponder } from '@documenso/lib/server';
|
|
||||||
// import prisma from '@documenso/prisma';
|
|
||||||
|
|
||||||
// async function postHandler(req: NextApiRequest, res: NextApiResponse) {
|
|
||||||
// const { email } = req.body;
|
|
||||||
// const cleanEmail = email.toLowerCase();
|
|
||||||
|
|
||||||
// if (!cleanEmail || !/.+@.+/.test(cleanEmail)) {
|
|
||||||
// res.status(400).json({ message: 'Invalid email' });
|
|
||||||
// return;
|
|
||||||
// }
|
|
||||||
|
|
||||||
// const user = await prisma.user.findFirst({
|
|
||||||
// where: {
|
|
||||||
// email: cleanEmail,
|
|
||||||
// },
|
|
||||||
// });
|
|
||||||
|
|
||||||
// if (!user) {
|
|
||||||
// return res.status(200).json({ message: 'A password reset email has been sent.' });
|
|
||||||
// }
|
|
||||||
|
|
||||||
// const existingToken = await prisma.passwordResetToken.findFirst({
|
|
||||||
// where: {
|
|
||||||
// userId: user.id,
|
|
||||||
// createdAt: {
|
|
||||||
// gte: new Date(Date.now() - 1000 _ 60 _ 60),
|
|
||||||
// },
|
|
||||||
// },
|
|
||||||
// });
|
|
||||||
|
|
||||||
// if (existingToken) {
|
|
||||||
// return res.status(200).json({ message: 'A password reset email has been sent.' });
|
|
||||||
// }
|
|
||||||
|
|
||||||
// const token = crypto.randomBytes(64).toString('hex');
|
|
||||||
// const expiry = new Date();
|
|
||||||
// expiry.setHours(expiry.getHours() + 24); // Set expiry to one hour from now
|
|
||||||
|
|
||||||
// let passwordResetToken;
|
|
||||||
// try {
|
|
||||||
// passwordResetToken = await prisma.passwordResetToken.create({
|
|
||||||
// data: {
|
|
||||||
// token,
|
|
||||||
// expiry,
|
|
||||||
// userId: user.id,
|
|
||||||
// },
|
|
||||||
// });
|
|
||||||
// } catch (error) {
|
|
||||||
// return res.status(500).json({ message: 'Something went wrong' });
|
|
||||||
// }
|
|
||||||
|
|
||||||
// await sendResetPassword(user, passwordResetToken.token);
|
|
||||||
|
|
||||||
// return res.status(200).json({ message: 'A password reset email has been sent.' });
|
|
||||||
// }
|
|
||||||
|
|
||||||
// export default defaultHandler({
|
|
||||||
// POST: Promise.resolve({ default: defaultResponder(postHandler) }),
|
|
||||||
// });
|
|
||||||
@@ -51,8 +51,6 @@ export const ForgotPasswordForm = ({ className }: ForgotPasswordFormProps) => {
|
|||||||
email,
|
email,
|
||||||
});
|
});
|
||||||
|
|
||||||
reset();
|
|
||||||
|
|
||||||
toast({
|
toast({
|
||||||
title: 'Password updated',
|
title: 'Password updated',
|
||||||
description: 'Your password has been updated successfully.',
|
description: 'Your password has been updated successfully.',
|
||||||
@@ -63,6 +61,7 @@ export const ForgotPasswordForm = ({ className }: ForgotPasswordFormProps) => {
|
|||||||
setTimeout(resolve, 2000);
|
setTimeout(resolve, 2000);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
reset();
|
||||||
router.push('/check-email');
|
router.push('/check-email');
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
53
packages/lib/server-only/auth/send-forgot-password.ts
Normal file
53
packages/lib/server-only/auth/send-forgot-password.ts
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
import { createElement } from 'react';
|
||||||
|
|
||||||
|
import { mailer } from '@documenso/email/mailer';
|
||||||
|
import { render } from '@documenso/email/render';
|
||||||
|
import { ForgotPasswordTemplate } from '@documenso/email/templates/forgot-password';
|
||||||
|
import { prisma } from '@documenso/prisma';
|
||||||
|
|
||||||
|
export interface SendForgotPasswordOptions {
|
||||||
|
userId: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const sendForgotPassword = async ({ userId }: SendForgotPasswordOptions) => {
|
||||||
|
const user = await prisma.user.findFirstOrThrow({
|
||||||
|
where: {
|
||||||
|
id: userId,
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
PasswordResetToken: {
|
||||||
|
orderBy: {
|
||||||
|
createdAt: 'desc',
|
||||||
|
},
|
||||||
|
take: 1,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!user) {
|
||||||
|
throw new Error('User not found');
|
||||||
|
}
|
||||||
|
|
||||||
|
const token = user.PasswordResetToken[0].token;
|
||||||
|
const assetBaseUrl = process.env.NEXT_PUBLIC_SITE_URL || 'http://localhost:3000';
|
||||||
|
const resetPasswordLink = `${process.env.NEXT_PUBLIC_SITE_URL}/reset/${token}`;
|
||||||
|
|
||||||
|
const template = createElement(ForgotPasswordTemplate, {
|
||||||
|
assetBaseUrl,
|
||||||
|
resetPasswordLink,
|
||||||
|
});
|
||||||
|
|
||||||
|
return await mailer.sendMail({
|
||||||
|
to: {
|
||||||
|
address: user.email,
|
||||||
|
name: user.name || '',
|
||||||
|
},
|
||||||
|
from: {
|
||||||
|
name: process.env.NEXT_PRIVATE_SMTP_FROM_NAME || 'Documenso',
|
||||||
|
address: process.env.NEXT_PRIVATE_SMTP_FROM_ADDRESS || 'noreply@documenso.com',
|
||||||
|
},
|
||||||
|
subject: 'Forgot Password?',
|
||||||
|
html: render(template),
|
||||||
|
text: render(template, { plainText: true }),
|
||||||
|
});
|
||||||
|
};
|
||||||
@@ -3,6 +3,8 @@ import crypto from 'crypto';
|
|||||||
import { prisma } from '@documenso/prisma';
|
import { prisma } from '@documenso/prisma';
|
||||||
import { TForgotPasswordFormSchema } from '@documenso/trpc/server/profile-router/schema';
|
import { TForgotPasswordFormSchema } from '@documenso/trpc/server/profile-router/schema';
|
||||||
|
|
||||||
|
import { sendForgotPassword } from '../auth/send-forgot-password';
|
||||||
|
|
||||||
export const forgotPassword = async ({ email }: TForgotPasswordFormSchema) => {
|
export const forgotPassword = async ({ email }: TForgotPasswordFormSchema) => {
|
||||||
const user = await prisma.user.findFirstOrThrow({
|
const user = await prisma.user.findFirstOrThrow({
|
||||||
where: {
|
where: {
|
||||||
@@ -31,10 +33,8 @@ export const forgotPassword = async ({ email }: TForgotPasswordFormSchema) => {
|
|||||||
const expiry = new Date();
|
const expiry = new Date();
|
||||||
expiry.setHours(expiry.getHours() + 24); // Set expiry to one hour from now
|
expiry.setHours(expiry.getHours() + 24); // Set expiry to one hour from now
|
||||||
|
|
||||||
let passwordResetToken;
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
passwordResetToken = await prisma.passwordResetToken.create({
|
await prisma.passwordResetToken.create({
|
||||||
data: {
|
data: {
|
||||||
token,
|
token,
|
||||||
expiry,
|
expiry,
|
||||||
@@ -45,8 +45,7 @@ export const forgotPassword = async ({ email }: TForgotPasswordFormSchema) => {
|
|||||||
throw new Error('Something went wrong');
|
throw new Error('Something went wrong');
|
||||||
}
|
}
|
||||||
|
|
||||||
console.log('Password reset token: ', passwordResetToken);
|
return await sendForgotPassword({
|
||||||
// send an email to user with password token
|
userId: user.id,
|
||||||
|
});
|
||||||
return passwordResetToken;
|
|
||||||
};
|
};
|
||||||
|
|||||||
Reference in New Issue
Block a user