refactor progress
This commit is contained in:
347
packages/email/src/EmailService.tsx
Normal file
347
packages/email/src/EmailService.tsx
Normal file
@@ -0,0 +1,347 @@
|
||||
/*
|
||||
* Copyright (C) 2026 Fluxer Contributors
|
||||
*
|
||||
* This file is part of Fluxer.
|
||||
*
|
||||
* Fluxer is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU Affero General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* Fluxer is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU Affero General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU Affero General Public License
|
||||
* along with Fluxer. If not, see <https://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
import type {IEmailI18nService} from '@fluxer/email/src/EmailI18nService';
|
||||
import type {EmailConfig, IEmailProvider, UserBouncedEmailChecker} from '@fluxer/email/src/EmailProviderTypes';
|
||||
import type {EmailTemplateVariables} from '@fluxer/email/src/email_i18n/EmailI18nTypes';
|
||||
import type {EmailTemplateKey} from '@fluxer/email/src/email_i18n/EmailI18nTypes.generated';
|
||||
import type {IEmailService} from '@fluxer/email/src/IEmailService';
|
||||
import {createLogger} from '@fluxer/logger/src/Logger';
|
||||
import {ms} from 'itty-time';
|
||||
|
||||
const logger = createLogger('email-service');
|
||||
|
||||
export class EmailService implements IEmailService {
|
||||
private readonly config: EmailConfig;
|
||||
private readonly emailI18n: IEmailI18nService;
|
||||
private readonly provider: IEmailProvider | null;
|
||||
private readonly bouncedEmailChecker: UserBouncedEmailChecker | null;
|
||||
|
||||
constructor(
|
||||
config: EmailConfig,
|
||||
emailI18n: IEmailI18nService,
|
||||
provider: IEmailProvider | null = null,
|
||||
bouncedEmailChecker: UserBouncedEmailChecker | null = null,
|
||||
) {
|
||||
this.config = config;
|
||||
this.emailI18n = emailI18n;
|
||||
this.provider = provider;
|
||||
this.bouncedEmailChecker = bouncedEmailChecker;
|
||||
}
|
||||
|
||||
async sendPasswordResetEmail(
|
||||
email: string,
|
||||
username: string,
|
||||
resetToken: string,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'password_reset', locale, {
|
||||
username,
|
||||
resetUrl: `${this.config.appBaseUrl}/reset#token=${resetToken}`,
|
||||
});
|
||||
}
|
||||
|
||||
async sendEmailVerification(
|
||||
email: string,
|
||||
username: string,
|
||||
verificationToken: string,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'email_verification', locale, {
|
||||
username,
|
||||
verifyUrl: `${this.config.appBaseUrl}/verify#token=${verificationToken}`,
|
||||
});
|
||||
}
|
||||
|
||||
async sendIpAuthorizationEmail(
|
||||
email: string,
|
||||
username: string,
|
||||
authorizationToken: string,
|
||||
ipAddress: string,
|
||||
location: string,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'ip_authorization', locale, {
|
||||
username,
|
||||
authUrl: `${this.config.appBaseUrl}/authorize-ip#token=${authorizationToken}`,
|
||||
ipAddress,
|
||||
location,
|
||||
});
|
||||
}
|
||||
|
||||
async sendAccountDisabledForSuspiciousActivityEmail(
|
||||
email: string,
|
||||
username: string,
|
||||
reason: string | null,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'account_disabled_suspicious', locale, {
|
||||
username,
|
||||
reason,
|
||||
forgotUrl: `${this.config.appBaseUrl}/forgot`,
|
||||
});
|
||||
}
|
||||
|
||||
async sendAccountTempBannedEmail(
|
||||
email: string,
|
||||
username: string,
|
||||
reason: string | null,
|
||||
durationHours: number,
|
||||
bannedUntil: Date,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'account_temp_banned', locale, {
|
||||
username,
|
||||
reason,
|
||||
durationHours,
|
||||
bannedUntil,
|
||||
termsUrl: `${this.config.marketingBaseUrl}/terms`,
|
||||
guidelinesUrl: `${this.config.marketingBaseUrl}/guidelines`,
|
||||
});
|
||||
}
|
||||
|
||||
async sendAccountScheduledForDeletionEmail(
|
||||
email: string,
|
||||
username: string,
|
||||
reason: string | null,
|
||||
deletionDate: Date,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'account_scheduled_deletion', locale, {
|
||||
username,
|
||||
reason,
|
||||
deletionDate,
|
||||
termsUrl: `${this.config.marketingBaseUrl}/terms`,
|
||||
guidelinesUrl: `${this.config.marketingBaseUrl}/guidelines`,
|
||||
});
|
||||
}
|
||||
|
||||
async sendSelfDeletionScheduledEmail(
|
||||
email: string,
|
||||
username: string,
|
||||
deletionDate: Date,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'self_deletion_scheduled', locale, {username, deletionDate});
|
||||
}
|
||||
|
||||
async sendUnbanNotification(
|
||||
email: string,
|
||||
username: string,
|
||||
reason: string,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'unban_notification', locale, {username, reason});
|
||||
}
|
||||
|
||||
async sendScheduledDeletionNotification(
|
||||
email: string,
|
||||
username: string,
|
||||
deletionDate: Date,
|
||||
reason: string,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'scheduled_deletion_notification', locale, {username, deletionDate, reason});
|
||||
}
|
||||
|
||||
async sendInactivityWarningEmail(
|
||||
email: string,
|
||||
username: string,
|
||||
deletionDate: Date,
|
||||
lastActiveDate: Date,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'inactivity_warning', locale, {
|
||||
username,
|
||||
deletionDate,
|
||||
lastActiveDate,
|
||||
loginUrl: `${this.config.appBaseUrl}/login`,
|
||||
});
|
||||
}
|
||||
|
||||
async sendHarvestCompletedEmail(
|
||||
email: string,
|
||||
username: string,
|
||||
downloadUrl: string,
|
||||
totalMessages: number,
|
||||
fileSize: number,
|
||||
expiresAt: Date,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
const fileSizeMB = Number.parseFloat((fileSize / 1024 / 1024).toFixed(2));
|
||||
return this.sendTemplatedEmail(email, 'harvest_completed', locale, {
|
||||
username,
|
||||
downloadUrl,
|
||||
totalMessages,
|
||||
fileSizeMB,
|
||||
expiresAt,
|
||||
});
|
||||
}
|
||||
|
||||
async sendGiftChargebackNotification(
|
||||
email: string,
|
||||
username: string,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'gift_chargeback_notification', locale, {username});
|
||||
}
|
||||
|
||||
async sendReportResolvedEmail(
|
||||
email: string,
|
||||
username: string,
|
||||
reportId: string,
|
||||
publicComment: string,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'report_resolved', locale, {username, reportId, publicComment});
|
||||
}
|
||||
|
||||
async sendDsaReportVerificationCode(
|
||||
email: string,
|
||||
code: string,
|
||||
expiresAt: Date,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'dsa_report_verification', locale, {code, expiresAt});
|
||||
}
|
||||
|
||||
async sendRegistrationApprovedEmail(email: string, username: string, locale: string | null = null): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'registration_approved', locale, {
|
||||
username,
|
||||
channelsUrl: `${this.config.appBaseUrl}/channels/@me`,
|
||||
});
|
||||
}
|
||||
|
||||
async sendPasswordChangeVerification(
|
||||
email: string,
|
||||
username: string,
|
||||
code: string,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'password_change_verification', locale, {
|
||||
username,
|
||||
code,
|
||||
expiresAt: new Date(Date.now() + ms('10 minutes')),
|
||||
});
|
||||
}
|
||||
|
||||
async sendEmailChangeOriginal(
|
||||
email: string,
|
||||
username: string,
|
||||
code: string,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'email_change_original', locale, {
|
||||
username,
|
||||
code,
|
||||
expiresAt: new Date(Date.now() + ms('10 minutes')),
|
||||
});
|
||||
}
|
||||
|
||||
async sendEmailChangeNew(
|
||||
email: string,
|
||||
username: string,
|
||||
code: string,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'email_change_new', locale, {
|
||||
username,
|
||||
code,
|
||||
expiresAt: new Date(Date.now() + ms('10 minutes')),
|
||||
});
|
||||
}
|
||||
|
||||
async sendEmailChangeRevert(
|
||||
email: string,
|
||||
username: string,
|
||||
newEmail: string,
|
||||
token: string,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'email_change_revert', locale, {
|
||||
username,
|
||||
revertUrl: `${this.config.appBaseUrl}/wasntme#token=${token}`,
|
||||
newEmail,
|
||||
});
|
||||
}
|
||||
|
||||
async sendDonationMagicLink(
|
||||
email: string,
|
||||
_token: string,
|
||||
manageUrl: string,
|
||||
expiresAt: Date,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'donation_magic_link', locale, {manageUrl, expiresAt});
|
||||
}
|
||||
|
||||
async sendDonationConfirmation(
|
||||
email: string,
|
||||
amountCents: number,
|
||||
currency: string,
|
||||
interval: string,
|
||||
manageUrl: string,
|
||||
locale: string | null = null,
|
||||
): Promise<boolean> {
|
||||
return this.sendTemplatedEmail(email, 'donation_confirmation', locale, {
|
||||
amount: (amountCents / 100).toFixed(2),
|
||||
currency: currency.toUpperCase(),
|
||||
interval,
|
||||
manageUrl,
|
||||
});
|
||||
}
|
||||
|
||||
private async sendTemplatedEmail<T extends EmailTemplateKey>(
|
||||
email: string,
|
||||
templateKey: T,
|
||||
locale: string | null,
|
||||
variables: EmailTemplateVariables[T],
|
||||
): Promise<boolean> {
|
||||
const result = this.emailI18n.getTemplate(templateKey, locale, variables);
|
||||
if (!result.ok) {
|
||||
logger.error({key: templateKey, locale: result.locale, error: result.error}, 'Failed to resolve email template');
|
||||
return false;
|
||||
}
|
||||
|
||||
const {subject, body} = result.value;
|
||||
|
||||
if (!this.config.enabled || !this.provider) {
|
||||
logger.info(
|
||||
{templateKey},
|
||||
`Email service disabled. Would have sent:\nTo: ${email}\nSubject: ${subject}\n\n${body}`,
|
||||
);
|
||||
return true;
|
||||
}
|
||||
|
||||
if (this.bouncedEmailChecker) {
|
||||
const bounced = await this.bouncedEmailChecker.isEmailBounced(email);
|
||||
if (bounced) {
|
||||
logger.warn({email}, 'Refusing to send email to bounced address - email marked as hard bounced');
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return this.provider.sendEmail({
|
||||
to: email,
|
||||
from: {email: this.config.fromEmail, name: this.config.fromName},
|
||||
subject,
|
||||
text: body,
|
||||
});
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user