023fa30280
* wip * Update ReactionService.ts * Update ApiCallService.ts * Update timeline.ts * Update GlobalModule.ts * Update GlobalModule.ts * Update NoteEntityService.ts * wip * wip * wip * Update ApPersonService.ts * wip * Update GlobalModule.ts * Update mock-resolver.ts * Update RoleService.ts * Update activitypub.ts * Update activitypub.ts * Update activitypub.ts * Update activitypub.ts * Update activitypub.ts * clean up * Update utils.ts * Update UtilityService.ts * Revert "Update utils.ts" This reverts commit a27d4be764b78c1b5a9eac685e261fee49331d89. * Revert "Update UtilityService.ts" This reverts commit e5fd9e004c482cf099252201c0c1aa888e001430. * vuwa- * Revert "vuwa-" This reverts commit 0c3bd12472b4b9938cdff2d6f131e6800bc3724c. * Update entry.ts * Update entry.ts * Update entry.ts * Update entry.ts * Update jest.setup.ts
366 lines
8.4 KiB
TypeScript
366 lines
8.4 KiB
TypeScript
/*
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
*/
|
|
|
|
import { URLSearchParams } from 'node:url';
|
|
import * as nodemailer from 'nodemailer';
|
|
import juice from 'juice';
|
|
import { Inject, Injectable } from '@nestjs/common';
|
|
import { validate as validateEmail } from 'deep-email-validator';
|
|
import { UtilityService } from '@/core/UtilityService.js';
|
|
import { DI } from '@/di-symbols.js';
|
|
import type { Config } from '@/config.js';
|
|
import type Logger from '@/logger.js';
|
|
import type { MiMeta, UserProfilesRepository } from '@/models/_.js';
|
|
import { LoggerService } from '@/core/LoggerService.js';
|
|
import { bindThis } from '@/decorators.js';
|
|
import { HttpRequestService } from '@/core/HttpRequestService.js';
|
|
|
|
@Injectable()
|
|
export class EmailService {
|
|
private logger: Logger;
|
|
|
|
constructor(
|
|
@Inject(DI.config)
|
|
private config: Config,
|
|
|
|
@Inject(DI.meta)
|
|
private meta: MiMeta,
|
|
|
|
@Inject(DI.userProfilesRepository)
|
|
private userProfilesRepository: UserProfilesRepository,
|
|
|
|
private loggerService: LoggerService,
|
|
private utilityService: UtilityService,
|
|
private httpRequestService: HttpRequestService,
|
|
) {
|
|
this.logger = this.loggerService.getLogger('email');
|
|
}
|
|
|
|
@bindThis
|
|
public async sendEmail(to: string, subject: string, html: string, text: string) {
|
|
if (!this.meta.enableEmail) return;
|
|
|
|
const iconUrl = `${this.config.url}/static-assets/mi-white.png`;
|
|
const emailSettingUrl = `${this.config.url}/settings/email`;
|
|
|
|
const enableAuth = this.meta.smtpUser != null && this.meta.smtpUser !== '';
|
|
|
|
const transporter = nodemailer.createTransport({
|
|
host: this.meta.smtpHost,
|
|
port: this.meta.smtpPort,
|
|
secure: this.meta.smtpSecure,
|
|
ignoreTLS: !enableAuth,
|
|
proxy: this.config.proxySmtp,
|
|
auth: enableAuth ? {
|
|
user: this.meta.smtpUser,
|
|
pass: this.meta.smtpPass,
|
|
} : undefined,
|
|
} as any);
|
|
|
|
const htmlContent = `<!doctype html>
|
|
<html>
|
|
<head>
|
|
<meta charset="utf-8">
|
|
<title>${ subject }</title>
|
|
<style>
|
|
html {
|
|
background: #eee;
|
|
}
|
|
|
|
body {
|
|
padding: 16px;
|
|
margin: 0;
|
|
font-family: sans-serif;
|
|
font-size: 14px;
|
|
}
|
|
|
|
a {
|
|
text-decoration: none;
|
|
color: #86b300;
|
|
}
|
|
a:hover {
|
|
text-decoration: underline;
|
|
}
|
|
|
|
main {
|
|
max-width: 500px;
|
|
margin: 0 auto;
|
|
background: #fff;
|
|
color: #555;
|
|
}
|
|
main > header {
|
|
padding: 32px;
|
|
background: #86b300;
|
|
}
|
|
main > header > img {
|
|
max-width: 128px;
|
|
max-height: 28px;
|
|
vertical-align: bottom;
|
|
}
|
|
main > article {
|
|
padding: 32px;
|
|
}
|
|
main > article > h1 {
|
|
margin: 0 0 1em 0;
|
|
}
|
|
main > footer {
|
|
padding: 32px;
|
|
border-top: solid 1px #eee;
|
|
}
|
|
|
|
nav {
|
|
box-sizing: border-box;
|
|
max-width: 500px;
|
|
margin: 16px auto 0 auto;
|
|
padding: 0 32px;
|
|
}
|
|
nav > a {
|
|
color: #888;
|
|
}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<main>
|
|
<header>
|
|
<img src="${ this.meta.logoImageUrl ?? this.meta.iconUrl ?? iconUrl }"/>
|
|
</header>
|
|
<article>
|
|
<h1>${ subject }</h1>
|
|
<div>${ html }</div>
|
|
</article>
|
|
<footer>
|
|
<a href="${ emailSettingUrl }">${ 'Email setting' }</a>
|
|
</footer>
|
|
</main>
|
|
<nav>
|
|
<a href="${ this.config.url }">${ this.config.host }</a>
|
|
</nav>
|
|
</body>
|
|
</html>`;
|
|
|
|
const inlinedHtml = juice(htmlContent);
|
|
|
|
try {
|
|
// TODO: htmlサニタイズ
|
|
const info = await transporter.sendMail({
|
|
from: this.meta.email!,
|
|
to: to,
|
|
subject: subject,
|
|
text: text,
|
|
html: inlinedHtml,
|
|
});
|
|
|
|
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' | 'banned' | 'network' | 'blacklist';
|
|
}> {
|
|
const exist = await this.userProfilesRepository.countBy({
|
|
emailVerified: true,
|
|
email: emailAddress,
|
|
});
|
|
|
|
if (exist !== 0) {
|
|
return {
|
|
available: false,
|
|
reason: 'used',
|
|
};
|
|
}
|
|
|
|
let validated: {
|
|
valid: boolean,
|
|
reason?: string | null,
|
|
} = { valid: true, reason: null };
|
|
|
|
if (this.meta.enableActiveEmailValidation) {
|
|
if (this.meta.enableVerifymailApi && this.meta.verifymailAuthKey != null) {
|
|
validated = await this.verifyMail(emailAddress, this.meta.verifymailAuthKey);
|
|
} else if (this.meta.enableTruemailApi && this.meta.truemailInstance && this.meta.truemailAuthKey != null) {
|
|
validated = await this.trueMail(this.meta.truemailInstance, emailAddress, this.meta.truemailAuthKey);
|
|
} else {
|
|
validated = await validateEmail({
|
|
email: emailAddress,
|
|
validateRegex: true,
|
|
validateMx: true,
|
|
validateTypo: false, // TLDを見ているみたいだけどclubとか弾かれるので
|
|
validateDisposable: true, // 捨てアドかどうかチェック
|
|
validateSMTP: false, // 日本だと25ポートが殆どのプロバイダーで塞がれていてタイムアウトになるので
|
|
});
|
|
}
|
|
}
|
|
|
|
if (!validated.valid) {
|
|
const formatReason: Record<string, 'format' | 'disposable' | 'mx' | 'smtp' | 'network' | 'blacklist' | undefined> = {
|
|
regex: 'format',
|
|
disposable: 'disposable',
|
|
mx: 'mx',
|
|
smtp: 'smtp',
|
|
network: 'network',
|
|
blacklist: 'blacklist',
|
|
};
|
|
|
|
return {
|
|
available: false,
|
|
reason: validated.reason ? formatReason[validated.reason] ?? null : null,
|
|
};
|
|
}
|
|
|
|
const emailDomain: string = emailAddress.split('@')[1];
|
|
const isBanned = this.utilityService.isBlockedHost(this.meta.bannedEmailDomains, emailDomain);
|
|
|
|
if (isBanned) {
|
|
return {
|
|
available: false,
|
|
reason: 'banned',
|
|
};
|
|
}
|
|
|
|
return {
|
|
available: true,
|
|
reason: 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 Partial<{
|
|
message: string;
|
|
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[];
|
|
}>;
|
|
|
|
/* api error: when there is only one `message` attribute in the returned result */
|
|
if (Object.keys(json).length === 1 && Reflect.has(json, 'message')) {
|
|
return {
|
|
valid: false,
|
|
reason: null,
|
|
};
|
|
}
|
|
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,
|
|
};
|
|
}
|
|
|
|
private async trueMail<T>(truemailInstance: string, emailAddress: string, truemailAuthKey: string): Promise<{
|
|
valid: boolean;
|
|
reason: 'used' | 'format' | 'blacklist' | 'mx' | 'smtp' | 'network' | T | null;
|
|
}> {
|
|
const endpoint = truemailInstance + '?email=' + emailAddress;
|
|
try {
|
|
const res = await this.httpRequestService.send(endpoint, {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
Accept: 'application/json',
|
|
Authorization: truemailAuthKey,
|
|
},
|
|
});
|
|
|
|
const json = (await res.json()) as {
|
|
email: string;
|
|
success: boolean;
|
|
error?: string;
|
|
errors?: {
|
|
list_match?: string;
|
|
regex?: string;
|
|
mx?: string;
|
|
smtp?: string;
|
|
} | null;
|
|
};
|
|
|
|
if (json.email === undefined || json.errors?.regex) {
|
|
return {
|
|
valid: false,
|
|
reason: 'format',
|
|
};
|
|
}
|
|
if (json.errors?.smtp) {
|
|
return {
|
|
valid: false,
|
|
reason: 'smtp',
|
|
};
|
|
}
|
|
if (json.errors?.mx) {
|
|
return {
|
|
valid: false,
|
|
reason: 'mx',
|
|
};
|
|
}
|
|
if (!json.success) {
|
|
return {
|
|
valid: false,
|
|
reason: json.errors?.list_match as T || 'blacklist',
|
|
};
|
|
}
|
|
|
|
return {
|
|
valid: true,
|
|
reason: null,
|
|
};
|
|
} catch (error) {
|
|
return {
|
|
valid: false,
|
|
reason: 'network',
|
|
};
|
|
}
|
|
}
|
|
}
|