2025-01-15 01:05:34 +08:00
|
|
|
import { Inject, Provide, Scope, ScopeEnum } from '@midwayjs/core';
|
2025-11-27 01:59:22 +08:00
|
|
|
import { cache, isDev, randomNumber, simpleNanoId } from '@certd/basic';
|
2024-12-11 11:30:32 +08:00
|
|
|
import { SysSettingsService, SysSiteInfo } from '@certd/lib-server';
|
2024-11-28 17:36:45 +08:00
|
|
|
import { SmsServiceFactory } from '../sms/factory.js';
|
|
|
|
|
import { ISmsService } from '../sms/api.js';
|
2025-09-27 08:29:22 +08:00
|
|
|
import { CodeErrorException } from '@certd/lib-server';
|
2024-11-28 17:36:45 +08:00
|
|
|
import { EmailService } from './email-service.js';
|
2024-12-22 14:00:46 +08:00
|
|
|
import { AccessService } from '@certd/lib-server';
|
|
|
|
|
import { AccessSysGetter } from '@certd/lib-server';
|
2024-12-11 11:30:32 +08:00
|
|
|
import { isComm } from '@certd/plus-core';
|
2025-09-13 23:01:14 +08:00
|
|
|
import { CaptchaService } from "./captcha-service.js";
|
2023-01-29 13:44:19 +08:00
|
|
|
|
|
|
|
|
// {data: '<svg.../svg>', text: 'abcd'}
|
|
|
|
|
/**
|
|
|
|
|
*/
|
|
|
|
|
@Provide()
|
2025-01-15 01:05:34 +08:00
|
|
|
@Scope(ScopeEnum.Request, { allowDowngrade: true })
|
2023-01-29 13:44:19 +08:00
|
|
|
export class CodeService {
|
|
|
|
|
@Inject()
|
2024-11-28 17:36:45 +08:00
|
|
|
sysSettingsService: SysSettingsService;
|
|
|
|
|
@Inject()
|
|
|
|
|
emailService: EmailService;
|
|
|
|
|
|
|
|
|
|
@Inject()
|
|
|
|
|
accessService: AccessService;
|
2023-01-29 13:44:19 +08:00
|
|
|
|
2025-09-13 23:01:14 +08:00
|
|
|
@Inject()
|
|
|
|
|
captchaService: CaptchaService;
|
2023-01-29 13:44:19 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2025-09-13 23:01:14 +08:00
|
|
|
async checkCaptcha(body:any) {
|
|
|
|
|
return await this.captchaService.doValidate({form:body})
|
2023-01-29 13:44:19 +08:00
|
|
|
}
|
|
|
|
|
/**
|
|
|
|
|
*/
|
2025-07-24 16:56:22 +08:00
|
|
|
async sendSmsCode(
|
|
|
|
|
phoneCode = '86',
|
|
|
|
|
mobile: string,
|
|
|
|
|
opts?: {
|
|
|
|
|
duration?: number,
|
2025-08-09 16:41:57 +08:00
|
|
|
verificationType?: string,
|
|
|
|
|
verificationCodeLength?: number,
|
2025-07-24 16:56:22 +08:00
|
|
|
},
|
|
|
|
|
) {
|
2024-12-01 03:09:29 +08:00
|
|
|
if (!mobile) {
|
2024-12-01 03:02:59 +08:00
|
|
|
throw new Error('手机号不能为空');
|
|
|
|
|
}
|
2024-11-28 17:36:45 +08:00
|
|
|
|
2025-08-09 16:41:57 +08:00
|
|
|
const verificationCodeLength = Math.floor(Math.max(Math.min(opts?.verificationCodeLength || 4, 8), 4));
|
|
|
|
|
const duration = Math.floor(Math.max(Math.min(opts?.duration || 5, 15), 1));
|
2025-07-24 16:56:22 +08:00
|
|
|
|
2024-11-28 17:36:45 +08:00
|
|
|
const sysSettings = await this.sysSettingsService.getPrivateSettings();
|
|
|
|
|
if (!sysSettings.sms?.config?.accessId) {
|
|
|
|
|
throw new Error('当前站点还未配置短信');
|
|
|
|
|
}
|
|
|
|
|
const smsType = sysSettings.sms.type;
|
|
|
|
|
const smsConfig = sysSettings.sms.config;
|
2025-08-28 17:35:17 +08:00
|
|
|
const sender: ISmsService = await SmsServiceFactory.createSmsService(smsType);
|
2024-11-28 17:36:45 +08:00
|
|
|
const accessGetter = new AccessSysGetter(this.accessService);
|
|
|
|
|
sender.setCtx({
|
|
|
|
|
accessService: accessGetter,
|
|
|
|
|
config: smsConfig,
|
|
|
|
|
});
|
2025-08-09 16:41:57 +08:00
|
|
|
const smsCode = randomNumber(verificationCodeLength);
|
2024-11-28 17:36:45 +08:00
|
|
|
await sender.sendSmsCode({
|
|
|
|
|
mobile,
|
|
|
|
|
code: smsCode,
|
|
|
|
|
phoneCode,
|
|
|
|
|
});
|
|
|
|
|
|
2025-09-13 23:01:14 +08:00
|
|
|
const key = this.buildSmsCodeKey(phoneCode, mobile, opts?.verificationType);
|
2024-11-28 17:36:45 +08:00
|
|
|
cache.set(key, smsCode, {
|
2025-07-24 16:56:22 +08:00
|
|
|
ttl: duration * 60 * 1000, //5分钟
|
2024-11-28 17:36:45 +08:00
|
|
|
});
|
2024-11-30 01:57:09 +08:00
|
|
|
return smsCode;
|
2023-01-29 13:44:19 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2025-07-24 16:56:22 +08:00
|
|
|
*
|
|
|
|
|
* @param email 收件邮箱
|
|
|
|
|
* @param opts title标题 content内容模版 duration有效时间单位分钟 verificationType验证类型
|
2023-01-29 13:44:19 +08:00
|
|
|
*/
|
2025-07-24 16:56:22 +08:00
|
|
|
async sendEmailCode(
|
|
|
|
|
email: string,
|
|
|
|
|
opts?: {
|
|
|
|
|
title?: string,
|
|
|
|
|
content?: string,
|
|
|
|
|
duration?: number,
|
2025-08-09 16:41:57 +08:00
|
|
|
verificationType?: string,
|
|
|
|
|
verificationCodeLength?: number,
|
2025-07-24 16:56:22 +08:00
|
|
|
},
|
|
|
|
|
) {
|
2024-12-01 03:02:59 +08:00
|
|
|
if (!email) {
|
|
|
|
|
throw new Error('Email不能为空');
|
|
|
|
|
}
|
2025-09-13 23:01:14 +08:00
|
|
|
|
2024-11-28 17:36:45 +08:00
|
|
|
|
2024-12-11 11:30:32 +08:00
|
|
|
let siteTitle = 'Certd';
|
|
|
|
|
if (isComm()) {
|
|
|
|
|
const siteInfo = await this.sysSettingsService.getSetting<SysSiteInfo>(SysSiteInfo);
|
|
|
|
|
if (siteInfo) {
|
|
|
|
|
siteTitle = siteInfo.title || siteTitle;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2025-08-09 16:41:57 +08:00
|
|
|
const verificationCodeLength = Math.floor(Math.max(Math.min(opts?.verificationCodeLength || 4, 8), 4));
|
|
|
|
|
const duration = Math.floor(Math.max(Math.min(opts?.duration || 5, 15), 1));
|
|
|
|
|
|
|
|
|
|
const code = randomNumber(verificationCodeLength);
|
2025-07-24 16:56:22 +08:00
|
|
|
|
2025-12-12 23:39:09 +08:00
|
|
|
const templateData = {
|
|
|
|
|
code, duration, siteTitle
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const titleTemplate = opts?.title?
|
|
|
|
|
|
2025-07-24 16:56:22 +08:00
|
|
|
const title = `【${siteTitle}】${!!opts?.title ? opts.title : '验证码'}`;
|
2025-12-12 23:39:09 +08:00
|
|
|
const content = !!opts.content ? this.compile(opts.content)(templateData) : `您的验证码是${code},请勿泄露`;
|
2025-07-24 16:56:22 +08:00
|
|
|
|
2024-11-28 17:36:45 +08:00
|
|
|
await this.emailService.send({
|
2025-07-24 16:56:22 +08:00
|
|
|
subject: title,
|
|
|
|
|
content: content,
|
2024-11-28 17:36:45 +08:00
|
|
|
receivers: [email],
|
|
|
|
|
});
|
|
|
|
|
|
2025-09-13 23:01:14 +08:00
|
|
|
const key = this.buildEmailCodeKey(email,opts?.verificationType);
|
2024-11-28 17:36:45 +08:00
|
|
|
cache.set(key, code, {
|
2025-07-24 16:56:22 +08:00
|
|
|
ttl: duration * 60 * 1000, //5分钟
|
2024-11-28 17:36:45 +08:00
|
|
|
});
|
2024-11-30 01:57:09 +08:00
|
|
|
return code;
|
2024-11-28 17:36:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* checkSms
|
|
|
|
|
*/
|
2025-09-13 23:01:14 +08:00
|
|
|
async checkSmsCode(opts: { mobile: string; phoneCode: string; smsCode: string; verificationType?: string; throwError: boolean; maxErrorCount?: number }) {
|
|
|
|
|
const key = this.buildSmsCodeKey(opts.phoneCode, opts.mobile, opts.verificationType);
|
|
|
|
|
return this.checkValidateCode("sms",key, opts.smsCode, opts.throwError, opts.maxErrorCount);
|
|
|
|
|
|
2024-11-28 17:36:45 +08:00
|
|
|
}
|
|
|
|
|
|
2025-09-13 23:01:14 +08:00
|
|
|
buildSmsCodeKey(phoneCode: string, mobile: string, verificationType?: string) {
|
|
|
|
|
return ['sms', verificationType, phoneCode, mobile].filter(item => !!item).join(':');
|
2024-11-28 17:36:45 +08:00
|
|
|
}
|
|
|
|
|
|
2025-09-13 23:01:14 +08:00
|
|
|
buildEmailCodeKey(email: string, verificationType?: string) {
|
|
|
|
|
return ['email', verificationType, email].filter(item => !!item).join(':');
|
2024-11-28 17:36:45 +08:00
|
|
|
}
|
2025-09-13 23:01:14 +08:00
|
|
|
checkValidateCode(type:string,key: string, userCode: string, throwError = true, maxErrorCount = 3) {
|
2025-08-09 16:41:57 +08:00
|
|
|
// 记录异常次数key
|
2025-09-13 23:01:14 +08:00
|
|
|
if (isDev() && userCode==="1234567") {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2025-08-09 16:41:57 +08:00
|
|
|
const err_num_key = key + ':err_num';
|
2025-09-13 23:01:14 +08:00
|
|
|
//验证邮件验证码
|
2024-11-28 17:36:45 +08:00
|
|
|
const code = cache.get(key);
|
|
|
|
|
if (code == null || code !== userCode) {
|
2025-08-09 16:41:57 +08:00
|
|
|
let maxRetryCount = false;
|
2025-09-13 23:01:14 +08:00
|
|
|
if (!!code && maxErrorCount > 0) {
|
2025-08-09 16:41:57 +08:00
|
|
|
const err_num = cache.get(err_num_key) || 0
|
2025-09-13 23:01:14 +08:00
|
|
|
if(err_num >= maxErrorCount - 1) {
|
2025-08-09 16:41:57 +08:00
|
|
|
maxRetryCount = true;
|
|
|
|
|
cache.delete(key);
|
|
|
|
|
cache.delete(err_num_key);
|
|
|
|
|
} else {
|
|
|
|
|
cache.set(err_num_key, err_num + 1, {
|
|
|
|
|
ttl: 30 * 60 * 1000
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-11-28 17:36:45 +08:00
|
|
|
if (throwError) {
|
2025-09-13 23:01:14 +08:00
|
|
|
const label = type ==='sms' ? '手机' : '邮箱';
|
|
|
|
|
throw new CodeErrorException(!maxRetryCount ? `${label}验证码错误`: `${label}验证码错误请获取新的验证码`);
|
2024-11-28 17:36:45 +08:00
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
cache.delete(key);
|
2025-08-09 16:41:57 +08:00
|
|
|
cache.delete(err_num_key);
|
2024-11-28 17:36:45 +08:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-13 23:01:14 +08:00
|
|
|
checkEmailCode(opts: { validateCode: string; email: string; verificationType?: string; throwError: boolean; maxErrorCount?: number }) {
|
|
|
|
|
const key = this.buildEmailCodeKey(opts.email, opts.verificationType);
|
|
|
|
|
return this.checkValidateCode('email',key, opts.validateCode, opts.throwError, opts.maxErrorCount);
|
2023-01-29 13:44:19 +08:00
|
|
|
}
|
2025-07-24 16:56:22 +08:00
|
|
|
|
|
|
|
|
compile(templateString: string) {
|
|
|
|
|
return new Function(
|
|
|
|
|
"data",
|
|
|
|
|
` with(data || {}) {
|
|
|
|
|
return \`${templateString}\`;
|
|
|
|
|
}
|
|
|
|
|
`
|
|
|
|
|
);
|
|
|
|
|
}
|
2025-11-27 01:59:22 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
buildValidationValueKey(code:string) {
|
|
|
|
|
return `validationValue:${code}`;
|
|
|
|
|
}
|
|
|
|
|
setValidationValue(value:any) {
|
|
|
|
|
const randomCode = simpleNanoId(12);
|
|
|
|
|
const key = this.buildValidationValueKey(randomCode);
|
|
|
|
|
cache.set(key, value, {
|
|
|
|
|
ttl: 5 * 60 * 1000, //5分钟
|
|
|
|
|
});
|
|
|
|
|
return randomCode;
|
|
|
|
|
}
|
|
|
|
|
getValidationValue(code:string) {
|
|
|
|
|
return cache.get(this.buildValidationValueKey(code));
|
|
|
|
|
}
|
2023-01-29 13:44:19 +08:00
|
|
|
}
|