Files
Yi.Admin/Yi.Abp.Net8/module/rbac/Yi.Framework.Rbac.Application/Services/AccountService.cs

677 lines
25 KiB
C#
Raw Normal View History

2025-07-13 21:26:46 +08:00
using System.Net.Mail;
using System.Net.Mime;
using System.Text;
using System.Text.RegularExpressions;
2023-12-11 09:55:12 +08:00
using Lazy.Captcha.Core;
using Mapster;
2023-12-11 09:55:12 +08:00
using Microsoft.AspNetCore.Authorization;
using Microsoft.AspNetCore.Http;
2023-12-11 09:55:12 +08:00
using Microsoft.AspNetCore.Mvc;
using Microsoft.Extensions.Caching.Distributed;
2023-04-13 21:12:06 +08:00
using Microsoft.Extensions.Options;
2023-12-11 09:55:12 +08:00
using Volo.Abp.Application.Services;
using Volo.Abp.Authorization;
using Volo.Abp.Caching;
2025-07-13 21:26:46 +08:00
using Volo.Abp.Emailing;
2024-08-16 17:57:58 +08:00
using Volo.Abp.EventBus.Local;
2023-12-11 09:55:12 +08:00
using Volo.Abp.Guids;
using Volo.Abp.Uow;
using Volo.Abp.Users;
2024-08-16 17:57:58 +08:00
using Yi.Framework.Bbs.Domain.Shared.Enums;
using Yi.Framework.Bbs.Domain.Shared.Etos;
2023-12-11 09:55:12 +08:00
using Yi.Framework.Rbac.Application.Contracts.Dtos.Account;
using Yi.Framework.Rbac.Application.Contracts.IServices;
using Yi.Framework.Rbac.Domain.Entities;
using Yi.Framework.Rbac.Domain.Managers;
using Yi.Framework.Rbac.Domain.Repositories;
using Yi.Framework.Rbac.Domain.Shared.Caches;
using Yi.Framework.Rbac.Domain.Shared.Consts;
using Yi.Framework.Rbac.Domain.Shared.Dtos;
2024-10-03 01:10:32 +08:00
using Yi.Framework.Rbac.Domain.Shared.Enums;
using Yi.Framework.Rbac.Domain.Shared.Etos;
2023-12-11 09:55:12 +08:00
using Yi.Framework.Rbac.Domain.Shared.Options;
using Yi.Framework.SqlSugarCore.Abstractions;
namespace Yi.Framework.Rbac.Application.Services
2023-04-13 21:12:06 +08:00
{
2023-12-11 09:55:12 +08:00
public class AccountService : ApplicationService, IAccountService
{
2024-08-16 17:57:58 +08:00
protected ILocalEventBus LocalEventBus => LazyServiceProvider.LazyGetRequiredService<ILocalEventBus>();
2023-12-11 09:55:12 +08:00
private IDistributedCache<CaptchaPhoneCacheItem, CaptchaPhoneCacheKey> _phoneCache;
2025-07-13 21:26:46 +08:00
private IDistributedCache<CaptchaEmailCacheItem, CaptchaEmailCacheKey> _emailCache;
2023-12-11 09:55:12 +08:00
private readonly ICaptcha _captcha;
private readonly IGuidGenerator _guidGenerator;
private readonly RbacOptions _rbacOptions;
2023-12-20 21:43:16 +08:00
private readonly IAliyunManger _aliyunManger;
2025-07-13 21:26:46 +08:00
private readonly IDistributedCache<UserInfoCacheItem, UserInfoCacheKey> _userCache;
private readonly UserManager _userManager;
private readonly IHttpContextAccessor _httpContextAccessor;
private readonly IEmailSender _emailSender;
2024-10-03 01:10:32 +08:00
2023-12-11 09:55:12 +08:00
public AccountService(IUserRepository userRepository,
ICurrentUser currentUser,
IAccountManager accountManager,
2024-05-22 14:35:08 +08:00
ISqlSugarRepository<MenuAggregateRoot> menuRepository,
2023-12-11 09:55:12 +08:00
IDistributedCache<CaptchaPhoneCacheItem, CaptchaPhoneCacheKey> phoneCache,
2024-02-18 11:41:43 +08:00
IDistributedCache<UserInfoCacheItem, UserInfoCacheKey> userCache,
2023-12-11 09:55:12 +08:00
ICaptcha captcha,
IGuidGenerator guidGenerator,
2023-12-20 21:43:16 +08:00
IOptions<RbacOptions> options,
IAliyunManger aliyunManger,
2025-07-13 21:26:46 +08:00
UserManager userManager, IHttpContextAccessor httpContextAccessor,
IDistributedCache<CaptchaEmailCacheItem, CaptchaEmailCacheKey> emailCache, IEmailSender emailSender)
2023-12-11 09:55:12 +08:00
{
_userRepository = userRepository;
_currentUser = currentUser;
_accountManager = accountManager;
_menuRepository = menuRepository;
_phoneCache = phoneCache;
_captcha = captcha;
_guidGenerator = guidGenerator;
_rbacOptions = options.Value;
2023-12-20 21:43:16 +08:00
_aliyunManger = aliyunManger;
2024-02-18 11:41:43 +08:00
_userCache = userCache;
_userManager = userManager;
_httpContextAccessor = httpContextAccessor;
2025-07-13 21:26:46 +08:00
_emailCache = emailCache;
_emailSender = emailSender;
2023-12-11 09:55:12 +08:00
}
2023-04-13 21:12:06 +08:00
2023-12-11 09:55:12 +08:00
private IUserRepository _userRepository;
private ICurrentUser _currentUser;
private IAccountManager _accountManager;
2024-05-22 14:35:08 +08:00
private ISqlSugarRepository<MenuAggregateRoot> _menuRepository;
2024-08-16 17:57:58 +08:00
2023-04-13 21:12:06 +08:00
/// <summary>
2024-03-07 11:32:49 +08:00
/// 校验图片登录验证码,无需和账号绑定
2023-04-13 21:12:06 +08:00
/// </summary>
2025-06-21 13:02:38 +08:00
[RemoteService(isEnabled: false)]
private void ValidationImageCaptcha(string? uuid, string? code)
2023-04-13 21:12:06 +08:00
{
if (_rbacOptions.EnableCaptcha)
2023-04-13 21:12:06 +08:00
{
2024-03-07 11:32:49 +08:00
//登录不想要验证码 ,可不校验
2024-10-21 10:39:16 +08:00
if (!_captcha.Validate(uuid, code))
{
throw new UserFriendlyException("验证码错误");
}
2023-04-13 21:12:06 +08:00
}
}
2023-12-20 21:43:16 +08:00
2023-04-13 21:12:06 +08:00
/// <summary>
/// 登录
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
2024-01-24 11:26:44 +08:00
[AllowAnonymous]
public async Task<LoginOutputDto> PostLoginAsync(LoginInputVo input)
2023-04-13 21:12:06 +08:00
{
if (string.IsNullOrEmpty(input.Password) || string.IsNullOrEmpty(input.UserName))
{
throw new UserFriendlyException("请输入合理数据!");
}
2024-03-07 11:32:49 +08:00
//校验验证码
2025-06-21 13:02:38 +08:00
ValidationImageCaptcha(input.Uuid, input.Code);
2023-04-13 21:12:06 +08:00
2024-05-22 14:35:08 +08:00
UserAggregateRoot user = new();
2024-03-07 11:32:49 +08:00
//校验
2023-04-13 21:12:06 +08:00
await _accountManager.LoginValidationAsync(input.UserName, input.Password, x => user = x);
return await PostLoginAsync(user.Id);
}
/// <summary>
/// 提供其他服务使用根据用户id直接返回token
/// </summary>
/// <returns></returns>
[RemoteService(isEnabled: false)]
public async Task<LoginOutputDto> PostLoginAsync(Guid userId)
{
var userInfo = new UserRoleMenuDto();
//获取token
var accessToken = await _accountManager.GetTokenByUserIdAsync(userId, (info) => userInfo = info);
var refreshToken = _accountManager.CreateRefreshToken(userId);
2023-12-11 09:55:12 +08:00
//这里抛出一个登录的事件,也可以在全部流程走完,在应用层组装
if (_httpContextAccessor.HttpContext is not null)
{
var loginEntity = new LoginLogAggregateRoot().GetInfoByHttpContext(_httpContextAccessor.HttpContext);
var loginEto = loginEntity.Adapt<LoginEventArgs>();
loginEto.UserName = userInfo.User.UserName;
loginEto.UserId = userInfo.User.Id;
await LocalEventBus.PublishAsync(loginEto);
}
2024-10-03 01:10:32 +08:00
return new LoginOutputDto { Token = accessToken, RefreshToken = refreshToken };
2023-04-13 21:12:06 +08:00
}
2024-01-24 11:26:44 +08:00
/// <summary>
/// 刷新token
/// </summary>
/// <param name="refresh_token"></param>
/// <returns></returns>
[Authorize(AuthenticationSchemes = TokenTypeConst.Refresh)]
public async Task<object> PostRefreshAsync([FromQuery] string refresh_token)
{
2025-06-28 23:07:32 +08:00
var userId = CurrentUser.GetId();
2024-02-18 11:41:43 +08:00
var accessToken = await _accountManager.GetTokenByUserIdAsync(userId);
var refreshToken = _accountManager.CreateRefreshToken(userId);
return new { Token = accessToken, RefreshToken = refreshToken };
2024-01-24 11:26:44 +08:00
}
2023-12-11 09:55:12 +08:00
2023-04-13 21:12:06 +08:00
/// <summary>
/// 生成验证码
/// </summary>
/// <returns></returns>
[AllowAnonymous]
2023-12-11 09:55:12 +08:00
public async Task<CaptchaImageDto> GetCaptchaImageAsync()
2023-04-13 21:12:06 +08:00
{
2023-12-11 09:55:12 +08:00
var uuid = _guidGenerator.Create();
var captcha = _captcha.Generate(uuid.ToString());
var enableCaptcha = _rbacOptions.EnableCaptcha;
return new CaptchaImageDto { Img = captcha.Bytes, Uuid = uuid, IsEnableCaptcha = enableCaptcha };
2023-04-13 21:12:06 +08:00
}
2024-10-03 01:10:32 +08:00
/// <summary>
/// 找回密码
/// </summary>
/// <param name="input"></param>
[AllowAnonymous]
[UnitOfWork]
2024-10-04 00:00:44 +08:00
public async Task<string> PostRetrievePasswordAsync(RetrievePasswordDto input)
2024-10-03 01:10:32 +08:00
{
2025-07-13 21:26:46 +08:00
if (_rbacOptions.CaptchaType == CaptchaTypeEnum.Email)
{
throw new UserFriendlyException("当前模式,不允许手机号找回密码,请联系管理员");
}
2024-10-04 00:00:44 +08:00
//校验验证码,根据电话号码获取 value比对验证码已经uuid
await ValidationPhoneCaptchaAsync(ValidationPhoneTypeEnum.RetrievePassword, input.Phone, input.Code);
2024-10-03 01:10:32 +08:00
var entity = await _userRepository.GetFirstAsync(x => x.Phone == input.Phone);
if (entity is null)
{
throw new UserFriendlyException("该手机号码未注册");
}
await _accountManager.RestPasswordAsync(entity.Id, input.Password);
2024-10-04 00:00:44 +08:00
return entity.UserName;
2024-10-03 01:10:32 +08:00
}
2023-04-13 21:12:06 +08:00
/// <summary>
/// 注册,需要验证码通过
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
[AllowAnonymous]
[UnitOfWork]
2025-06-21 13:02:38 +08:00
public async Task<LoginOutputDto> PostRegisterAsync(RegisterDto input)
2023-04-13 21:12:06 +08:00
{
2023-12-20 21:43:16 +08:00
if (_rbacOptions.EnableRegister == false)
{
throw new UserFriendlyException("该系统暂未开放注册功能");
}
2024-08-16 17:57:58 +08:00
2025-07-13 21:26:46 +08:00
if (input.Phone is null && input.Email is null)
2024-04-29 17:50:51 +08:00
{
2025-07-13 21:26:46 +08:00
throw new UserFriendlyException("手机号和邮箱不能为空");
2024-04-29 17:50:51 +08:00
}
2025-06-21 13:02:38 +08:00
2024-10-22 12:20:53 +08:00
//临时账号
2024-10-25 17:36:23 +08:00
if (input.UserName.StartsWith("ls_"))
2024-10-22 12:20:53 +08:00
{
2024-10-25 17:36:23 +08:00
throw new UserFriendlyException("注册账号不能以ls_字符开头");
2024-10-22 12:20:53 +08:00
}
2025-06-21 13:02:38 +08:00
if (_rbacOptions.EnableCaptcha)
{
2025-07-13 21:26:46 +08:00
switch (_rbacOptions.CaptchaType)
{
case CaptchaTypeEnum.Phone:
//校验验证码,根据电话号码获取 value比对验证码已经uuid
await ValidationPhoneCaptchaAsync(ValidationPhoneTypeEnum.Register, input.Phone.Value,
input.Code);
break;
case CaptchaTypeEnum.Email:
//校验验证码,根据电子邮箱获取 value比对验证码已经uuid
await ValidationEmailCaptchaAsync(ValidationEmailTypeEnum.Register, input.Email, input.Code);
break;
}
}
2025-06-21 13:02:38 +08:00
//注册之后免再次登录直接给前端token
2025-07-13 21:26:46 +08:00
var userId = await _accountManager.RegisterAsync(input.UserName, input.Password, input.Phone, input.Email,
2025-08-27 23:42:46 +08:00
input.Nick, null);
2025-06-21 13:02:38 +08:00
return await this.PostLoginAsync(userId);
2023-04-13 21:12:06 +08:00
}
2024-10-22 12:20:53 +08:00
/// <summary>
2025-08-27 23:42:46 +08:00
/// 系统直接注册用户
2024-10-22 12:20:53 +08:00
/// 不需要验证,为了给第三方使用,例如微信小程序,后续可通过绑定操作,进行账号合并
/// </summary>
/// <param name="input"></param>
2025-06-21 13:02:38 +08:00
[RemoteService(isEnabled: false)]
2025-08-27 23:42:46 +08:00
public async Task<Guid> PostSystemRegisterAsync(RegisterDto input)
2024-10-22 12:20:53 +08:00
{
//注册领域逻辑
2025-08-27 23:42:46 +08:00
return await _accountManager.RegisterAsync(input.UserName, input.Password, input.Phone, input.Email,
input.Nick, input.Icon);
2024-10-22 12:20:53 +08:00
}
2023-04-13 21:12:06 +08:00
/// <summary>
2024-10-03 01:10:32 +08:00
/// 查询已登录的账户信息
2023-04-13 21:12:06 +08:00
/// </summary>
/// <returns></returns>
2023-12-11 09:55:12 +08:00
[Route("account")]
2023-04-13 21:12:06 +08:00
[Authorize]
2024-04-29 17:50:51 +08:00
public async Task<UserRoleMenuDto> GetAsync()
2023-04-13 21:12:06 +08:00
{
//通过鉴权jwt获取到用户的id
var userId = _currentUser.Id;
2023-12-11 09:55:12 +08:00
if (userId is null)
{
throw new UserFriendlyException("用户未登录");
}
2024-08-16 17:57:58 +08:00
2024-02-18 11:41:43 +08:00
//此处优先从缓存中获取
var output = await _userManager.GetInfoAsync(userId.Value);
2024-02-18 11:41:43 +08:00
return output;
2023-04-13 21:12:06 +08:00
}
[RemoteService(isEnabled: false)]
public async Task<UserRoleMenuDto?> GetAsync(string? userName, long? phone = null, Guid? userId = null)
{
var user = await _userRepository._DbQueryable
.WhereIF(userName is not null, x => x.UserName == userName)
.WhereIF(phone is not null, x => x.Phone == phone)
.WhereIF(userId is not null, x => x.Id == userId)
.Where(x => x.State == true)
.FirstAsync();
//该用户不存在
if (user is null)
{
return null;
}
//注意用户名大小写数据库不敏感问题
2024-10-25 20:45:35 +08:00
if (userName is not null && !user.UserName.Equals(userName))
{
throw new UserFriendlyException($"该用户名不存在或已禁用-{userName}");
}
var output = await _userManager.GetInfoAsync(user.Id);
return output;
}
2023-04-13 21:12:06 +08:00
/// <summary>
/// 获取当前登录用户的前端路由
2024-09-07 02:17:07 +08:00
/// 支持ruoyi/pure
2023-04-13 21:12:06 +08:00
/// </summary>
/// <returns></returns>
[Authorize]
2024-09-07 02:17:07 +08:00
[Route("account/Vue3Router/{routerType?}")]
2024-10-03 01:10:32 +08:00
public async Task<object> GetVue3Router([FromRoute] string? routerType)
2023-04-13 21:12:06 +08:00
{
var userId = _currentUser.Id;
2023-12-11 09:55:12 +08:00
if (_currentUser.Id is null)
{
throw new AbpAuthorizationException("用户未登录");
}
2024-08-16 17:57:58 +08:00
var data = await _userManager.GetInfoAsync(userId!.Value);
2023-04-13 21:12:06 +08:00
var menus = data.Menus.ToList();
//为超级管理员直接给全部路由
if (UserConst.Admin.Equals(data.User.UserName))
{
2024-05-22 14:35:08 +08:00
menus = ObjectMapper.Map<List<MenuAggregateRoot>, List<MenuDto>>(await _menuRepository.GetListAsync());
2023-04-13 21:12:06 +08:00
}
2024-08-16 17:57:58 +08:00
2024-09-07 02:17:07 +08:00
object output = null;
2024-10-03 01:10:32 +08:00
if (routerType is null || routerType == "ruoyi")
2024-09-07 02:17:07 +08:00
{
//将后端菜单转换成前端路由,组件级别需要过滤
output =
2025-06-21 13:02:38 +08:00
ObjectMapper
.Map<List<MenuDto>, List<MenuAggregateRoot>>(menus
.Where(x => x.MenuSource == MenuSourceEnum.Ruoyi).ToList()).Vue3RuoYiRouterBuild();
2024-09-07 02:17:07 +08:00
}
2024-10-03 01:10:32 +08:00
else if (routerType == "pure")
2024-09-07 02:17:07 +08:00
{
//将后端菜单转换成前端路由,组件级别需要过滤
output =
2025-06-21 13:02:38 +08:00
ObjectMapper
.Map<List<MenuDto>, List<MenuAggregateRoot>>(menus
.Where(x => x.MenuSource == MenuSourceEnum.Pure).ToList()).Vue3PureRouterBuild();
2024-09-07 02:17:07 +08:00
}
2024-10-03 01:10:32 +08:00
2024-09-07 02:17:07 +08:00
return output;
2023-04-13 21:12:06 +08:00
}
/// <summary>
/// 退出登录
/// </summary>
/// <returns></returns>
2024-02-18 11:41:43 +08:00
public async Task<bool> PostLogout()
2023-04-13 21:12:06 +08:00
{
2024-02-18 11:41:43 +08:00
//通过鉴权jwt获取到用户的id
var userId = _currentUser.Id;
if (userId is null)
{
return false;
// throw new UserFriendlyException("用户已退出");
2024-02-18 11:41:43 +08:00
}
2024-08-16 17:57:58 +08:00
2024-02-18 11:41:43 +08:00
await _userCache.RemoveAsync(new UserInfoCacheKey(userId.Value));
//Jwt去中心化登出只需用记录日志即可
2024-02-18 11:41:43 +08:00
return true;
2023-04-13 21:12:06 +08:00
}
/// <summary>
/// 更新密码
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
public async Task<bool> UpdatePasswordAsync(UpdatePasswordDto input)
{
if (input.OldPassword.Equals(input.NewPassword))
{
throw new UserFriendlyException("无效更新!输入的数据,新密码不能与老密码相同");
}
2024-08-16 17:57:58 +08:00
2023-12-11 09:55:12 +08:00
if (_currentUser.Id is null)
{
throw new UserFriendlyException("用户未登录");
}
2024-08-16 17:57:58 +08:00
await _accountManager.UpdatePasswordAsync(_currentUser.Id ?? Guid.Empty, input.NewPassword,
input.OldPassword);
2023-04-13 21:12:06 +08:00
return true;
}
/// <summary>
/// 重置密码
/// </summary>
/// <param name="userId"></param>
/// <param name="input"></param>
/// <returns></returns>
[HttpPut]
2023-12-11 09:55:12 +08:00
public async Task<bool> RestPasswordAsync(Guid userId, RestPasswordDto input)
2023-04-13 21:12:06 +08:00
{
2023-07-19 01:24:27 +08:00
if (string.IsNullOrEmpty(input.Password))
2023-04-13 21:12:06 +08:00
{
throw new UserFriendlyException("重置密码不能为空!");
}
2024-08-16 17:57:58 +08:00
2023-04-13 21:12:06 +08:00
await _accountManager.RestPasswordAsync(userId, input.Password);
return true;
}
/// <summary>
/// 更新头像
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
public async Task<bool> UpdateIconAsync(UpdateIconDto input)
{
Guid userId = input.UserId == null ? _currentUser.GetId() : input.UserId.Value;
2024-09-24 14:44:06 +08:00
var entity = await _userRepository.GetByIdAsync(userId);
2024-08-16 17:57:58 +08:00
if (entity.Icon == input.Icon)
{
return false;
}
2023-04-13 21:12:06 +08:00
entity.Icon = input.Icon;
await _userRepository.UpdateAsync(entity);
2024-08-16 17:57:58 +08:00
//发布更新头像任务事件
await this.LocalEventBus.PublishAsync(
2024-09-24 14:44:06 +08:00
new AssignmentEventArgs(AssignmentRequirementTypeEnum.UpdateIcon, userId), false);
2023-04-13 21:12:06 +08:00
return true;
}
2025-07-13 21:26:46 +08:00
#region
/// <summary>
/// 验证电话号码
/// </summary>
/// <param name="phone"></param>
private async Task ValidationPhone(string phone)
{
var res = Regex.IsMatch(phone, @"^\d{11}$");
if (res == false)
{
throw new UserFriendlyException("手机号码格式错误!请检查");
}
}
/// <summary>
/// 手机验证码-注册
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
[HttpPost("account/captcha-phone/register")]
[AllowAnonymous]
public async Task<object> PostCaptchaPhoneForRegisterAsync(PhoneCaptchaImageDto input)
{
return await InternalPostCaptchaPhoneAsync(ValidationPhoneTypeEnum.Register, input);
}
/// <summary>
/// 手机验证码-找回密码
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
[HttpPost("account/captcha-phone/repassword")]
public async Task<object> PostCaptchaPhoneForRetrievePasswordAsync(PhoneCaptchaImageDto input)
{
return await InternalPostCaptchaPhoneAsync(ValidationPhoneTypeEnum.RetrievePassword, input);
}
/// <summary>
/// 手机验证码-绑定
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
[HttpPost("account/captcha-phone/bind")]
[AllowAnonymous]
public async Task<object> PostCaptchaPhoneForBindAsync(PhoneCaptchaImageDto input)
{
return await InternalPostCaptchaPhoneAsync(ValidationPhoneTypeEnum.Bind, input);
}
/// <summary>
/// 手机验证码-内部使用-需通过图形验证码
/// </summary>
/// <returns></returns>
[RemoteService(isEnabled: false)]
private async Task<object> InternalPostCaptchaPhoneAsync(ValidationPhoneTypeEnum validationPhoneType,
PhoneCaptchaImageDto input)
{
//验证uuid 和 验证码
ValidationImageCaptcha(input.Uuid, input.Code);
await ValidationPhone(input.Phone);
if (validationPhoneType == ValidationPhoneTypeEnum.Register &&
await _userRepository.IsAnyAsync(x => x.Phone.ToString() == input.Phone))
{
throw new UserFriendlyException("该手机号已被注册!");
}
var value = await _phoneCache.GetAsync(new CaptchaPhoneCacheKey(validationPhoneType, input.Phone));
//防止暴刷
if (value is not null)
{
throw new UserFriendlyException($"{input.Phone}已发送过验证码10分钟后可重试");
}
//生成一个4位数的验证码
//发送短信同时生成uuid
////key 电话号码 value:验证码+uuid
var code = Guid.NewGuid().ToString().Substring(0, 4);
var uuid = Guid.NewGuid();
await _aliyunManger.SendSmsAsync(input.Phone, code);
await _phoneCache.SetAsync(new CaptchaPhoneCacheKey(validationPhoneType, input.Phone),
new CaptchaPhoneCacheItem(code),
new DistributedCacheEntryOptions { SlidingExpiration = TimeSpan.FromMinutes(10) });
return new
{
Uuid = uuid
};
}
/// <summary>
/// 校验电话验证码,需要与电话号码绑定
/// </summary>
public async Task ValidationPhoneCaptchaAsync(ValidationPhoneTypeEnum validationPhoneType, long phone,
string code)
{
var item = await _phoneCache.GetAsync(new CaptchaPhoneCacheKey(validationPhoneType, phone.ToString()));
if (item is not null && item.Code.Equals($"{code}"))
{
//成功,需要清空
await _phoneCache.RemoveAsync(new CaptchaPhoneCacheKey(validationPhoneType, phone.ToString()));
return;
}
throw new UserFriendlyException("验证码错误");
}
#endregion
#region
/// <summary>
/// 验证电子邮箱
/// </summary>
/// <param name="email"></param>
private async Task ValidationEmail(string email)
{
// 简单邮箱正则表达式
var res = Regex.IsMatch(email, @"^[^@\s]+@[^@\s]+\.[^@\s]+$");
if (res == false)
{
throw new UserFriendlyException("邮箱格式错误!请检查");
}
}
/// <summary>
/// 邮箱验证码-注册
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
[HttpPost("account/captcha-email/register")]
[AllowAnonymous]
public async Task<object> PostCaptchaEmailForRegisterAsync(PhoneCaptchaImageDto input)
{
return await InternalPostCaptchaEmailAsync(ValidationEmailTypeEnum.Register, input);
}
/// <summary>
/// 邮箱验证码-找回密码
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
[HttpPost("account/captcha-email/repassword")]
public async Task<object> PostCaptchaEmailForRetrievePasswordAsync(PhoneCaptchaImageDto input)
{
return await InternalPostCaptchaEmailAsync(ValidationEmailTypeEnum.RetrievePassword, input);
}
/// <summary>
/// 邮箱验证码-绑定
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
[HttpPost("account/captcha-email/bind")]
[AllowAnonymous]
public async Task<object> PostCaptchaEmailForBindAsync(PhoneCaptchaImageDto input)
{
return await InternalPostCaptchaEmailAsync(ValidationEmailTypeEnum.Bind, input);
}
/// <summary>
/// 邮箱验证码-内部使用-需通过图形验证码
/// </summary>
/// <returns></returns>
[RemoteService(isEnabled: false)]
private async Task<object> InternalPostCaptchaEmailAsync(ValidationEmailTypeEnum validationEmailType,
PhoneCaptchaImageDto input)
{
//验证uuid 和 验证码
ValidationImageCaptcha(input.Uuid, input.Code);
await ValidationEmail(input.Email);
if (validationEmailType == ValidationEmailTypeEnum.Register)
{
//处理大小写问题
var emailOrNull = await _userRepository._DbQueryable.Where(x => x.Email == input.Email)
.Select(x => x.Email)
.FirstAsync();
if (emailOrNull is not null && emailOrNull.Equals(input.Email))
{
throw new UserFriendlyException("该邮箱已被注册!");
}
}
var value = await _emailCache.GetAsync(new CaptchaEmailCacheKey(validationEmailType, input.Email));
//防止暴刷
if (value is not null)
{
throw new UserFriendlyException($"{input.Email}已发送过验证码10分钟后可重试");
}
//生成一个4位数的验证码
//发送邮箱同时生成uuid
////key 邮箱 value:验证码+uuid
var code = Guid.NewGuid().ToString().Substring(0, 4);
var uuid = Guid.NewGuid();
//await _aliyunManger.SendSmsAsync(input.Phone, code);
//发送邮件
await _emailSender.SendAsync(input.Email,
"意社区官方邮件",
$"欢迎加入我们,您的验证码为 {code} 该验证码10分钟内有效请勿泄露于他人。");
await _emailCache.SetAsync(new CaptchaEmailCacheKey(validationEmailType, input.Email),
new CaptchaEmailCacheItem(code),
new DistributedCacheEntryOptions { SlidingExpiration = TimeSpan.FromMinutes(10) });
return new
{
Uuid = uuid
};
}
/// <summary>
/// 校验电子邮箱验证码,需要与电子邮箱绑定
/// </summary>
public async Task ValidationEmailCaptchaAsync(ValidationEmailTypeEnum validationEmailType, string email,
string code)
{
var item = await _emailCache.GetAsync(new CaptchaEmailCacheKey(validationEmailType, email.ToString()));
if (item is not null && item.Code.Equals($"{code}"))
{
//成功,需要清空
await _emailCache.RemoveAsync(new CaptchaEmailCacheKey(validationEmailType, email.ToString()));
return;
}
throw new UserFriendlyException("验证码错误");
}
#endregion
2023-04-13 21:12:06 +08:00
}
2024-08-16 17:57:58 +08:00
}