/* * SPDX-FileCopyrightText: syuilo and other misskey contributors * SPDX-License-Identifier: AGPL-3.0-only */ import * as nodemailer from 'nodemailer'; import { Inject, Injectable } from '@nestjs/common'; import { validate as validateEmail } from 'deep-email-validator'; import { MetaService } from '@/core/MetaService.js'; import { DI } from '@/di-symbols.js'; import type { Config } from '@/config.js'; import type Logger from '@/logger.js'; import type { UserProfilesRepository } from '@/models/_.js'; import { LoggerService } from '@/core/LoggerService.js'; import { bindThis } from '@/decorators.js'; import {URLSearchParams} from "node:url"; import { HttpRequestService } from '@/core/HttpRequestService.js'; import {SubOutputFormat} from "deep-email-validator/dist/output/output.js"; @Injectable() export class EmailService { private logger: Logger; constructor( @Inject(DI.config) private config: Config, @Inject(DI.userProfilesRepository) private userProfilesRepository: UserProfilesRepository, private metaService: MetaService, private loggerService: LoggerService, private httpRequestService: HttpRequestService, ) { this.logger = this.loggerService.getLogger('email'); } @bindThis public async sendEmail(to: string, subject: string, html: string, text: string) { const meta = await this.metaService.fetch(true); const iconUrl = `${this.config.url}/static-assets/mi-white.png`; const emailSettingUrl = `${this.config.url}/settings/email`; const enableAuth = meta.smtpUser != null && meta.smtpUser !== ''; const transporter = nodemailer.createTransport({ host: meta.smtpHost, port: meta.smtpPort, secure: meta.smtpSecure, ignoreTLS: !enableAuth, proxy: this.config.proxySmtp, auth: enableAuth ? { user: meta.smtpUser, pass: meta.smtpPass, } : undefined, } as any); try { // TODO: htmlサニタイズ const info = await transporter.sendMail({ from: meta.email!, to: to, subject: subject, text: text, html: ` ${ subject }

${ subject }

${ html }
`, }); this.logger.info(`Message sent: ${info.messageId}`); } catch (err) { this.logger.error(err as Error); throw err; } } @bindThis public async validateEmailForAccount(emailAddress: string): Promise<{ available: boolean; reason: null | 'used' | 'format' | 'disposable' | 'mx' | 'smtp'; }> { const meta = await this.metaService.fetch(); const exist = await this.userProfilesRepository.countBy({ emailVerified: true, email: emailAddress, }); const verifymailApi = meta.enableVerifymailApi && meta.verifymailAuthKey != null; let validated; if (meta.enableActiveEmailValidation) { if (verifymailApi) { validated = await this.verifyMail(emailAddress, meta.verifymailAuthKey); } else { validated = meta.enableActiveEmailValidation ? await validateEmail({ email: emailAddress, validateRegex: true, validateMx: true, validateTypo: false, // TLDを見ているみたいだけどclubとか弾かれるので validateDisposable: true, // 捨てアドかどうかチェック validateSMTP: false, // 日本だと25ポートが殆どのプロバイダーで塞がれていてタイムアウトになるので }) : { valid: true, reason: null }; } } else { validated = { valid: true, reason: null }; } const available = exist === 0 && validated.valid; return { available, reason: available ? null : exist !== 0 ? 'used' : validated.reason === 'regex' ? 'format' : validated.reason === 'disposable' ? 'disposable' : validated.reason === 'mx' ? 'mx' : validated.reason === 'smtp' ? 'smtp' : null, }; } private async verifyMail(emailAddress: string, verifymailAuthKey: string): Promise<{ valid: boolean; reason: 'used' | 'format' | 'disposable' | 'mx' | 'smtp' | null; }> { const endpoint = 'https://verifymail.io/api/' + emailAddress + '?key=' + verifymailAuthKey; const res = await this.httpRequestService.send(endpoint, { method: 'GET', headers: { 'Content-Type': 'application/x-www-form-urlencoded', Accept: 'application/json, */*', }, }); const json = (await res.json()) as { block: boolean; catch_all: boolean; deliverable_email: boolean; disposable: boolean; domain: string; email_address: string; email_provider: string; mx: boolean; mx_fallback: boolean; mx_host: string[]; mx_ip: string[]; mx_priority: { [key: string]: number }; privacy: boolean; related_domains: string[]; }; if (json.email_address === undefined) { return { valid: false, reason: 'format', }; } if (json.deliverable_email !== undefined && !json.deliverable_email) { return { valid: false, reason: 'smtp', }; } if (json.disposable) { return { valid: false, reason: 'disposable', }; } if (json.mx !== undefined && !json.mx) { return { valid: false, reason: 'mx', }; } return { valid: true, reason: null, }; } }