mirror of
https://github.com/documenso/documenso.git
synced 2025-11-12 15:53:02 +10:00
70 lines
1.7 KiB
TypeScript
70 lines
1.7 KiB
TypeScript
import crypto from 'crypto';
|
|
import { DateTime } from 'luxon';
|
|
|
|
import { prisma } from '@documenso/prisma';
|
|
|
|
import { ONE_HOUR } from '../../constants/time';
|
|
import { sendConfirmationEmail } from '../auth/send-confirmation-email';
|
|
import { getMostRecentVerificationTokenByUserId } from './get-most-recent-verification-token-by-user-id';
|
|
|
|
const IDENTIFIER = 'confirmation-email';
|
|
|
|
type SendConfirmationTokenOptions = { email: string; force?: boolean };
|
|
|
|
export const sendConfirmationToken = async ({
|
|
email,
|
|
force = false,
|
|
}: SendConfirmationTokenOptions) => {
|
|
const token = crypto.randomBytes(20).toString('hex');
|
|
|
|
const user = await prisma.user.findFirst({
|
|
where: {
|
|
email: email,
|
|
},
|
|
});
|
|
|
|
if (!user) {
|
|
throw new Error('User not found');
|
|
}
|
|
|
|
if (user.emailVerified) {
|
|
throw new Error('Email verified');
|
|
}
|
|
|
|
const mostRecentToken = await getMostRecentVerificationTokenByUserId({ userId: user.id });
|
|
|
|
// If we've sent a token in the last 5 minutes, don't send another one
|
|
if (
|
|
!force &&
|
|
mostRecentToken?.createdAt &&
|
|
DateTime.fromJSDate(mostRecentToken.createdAt).diffNow('minutes').minutes > -5
|
|
) {
|
|
return;
|
|
}
|
|
|
|
const createdToken = await prisma.verificationToken.create({
|
|
data: {
|
|
identifier: IDENTIFIER,
|
|
token: token,
|
|
expires: new Date(Date.now() + ONE_HOUR),
|
|
user: {
|
|
connect: {
|
|
id: user.id,
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
if (!createdToken) {
|
|
throw new Error(`Failed to create the verification token`);
|
|
}
|
|
|
|
try {
|
|
await sendConfirmationEmail({ userId: user.id });
|
|
|
|
return { success: true };
|
|
} catch (err) {
|
|
throw new Error(`Failed to send the confirmation email`);
|
|
}
|
|
};
|