新增手机号登录功能

This commit is contained in:
不做码农 2023-11-19 13:11:29 +08:00
parent f859748825
commit df36de52fa
10 changed files with 193 additions and 14 deletions

View File

@ -0,0 +1,25 @@
using System;
using System.Text;
namespace ZR.Infrastructure.Helper
{
public class RandomHelper
{
/// <summary>
/// 生成n为验证码
/// </summary>
/// <param name="Length"></param>
/// <returns></returns>
public static string GenerateNum(int Length)
{
char[] constant = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9' };
StringBuilder newRandom = new(constant.Length);
Random rd = new();
for (int i = 0; i < Length; i++)
{
newRandom.Append(constant[rd.Next(constant.Length - 1)]);
}
return newRandom.ToString();
}
}
}

View File

@ -1,10 +1,13 @@
using Lazy.Captcha.Core;
using Microsoft.AspNetCore.Mvc;
using ZR.Admin.WebApi.Filters;
using ZR.Infrastructure.Helper;
using ZR.Model.System;
using ZR.Model.System.Dto;
using ZR.Service.System;
using ZR.Service.System.IService;
using ZR.ServiceCore.Model.Dto;
using ZR.ServiceCore.Services;
namespace ZR.Admin.WebApi.Controllers.System
{
@ -21,6 +24,7 @@ namespace ZR.Admin.WebApi.Controllers.System
private readonly ICaptcha SecurityCodeHelper;
private readonly ISysConfigService sysConfigService;
private readonly ISysRoleService roleService;
private readonly ISmsCodeLogService smsCodeLogService;
public SysLoginController(
ISysMenuService sysMenuService,
@ -29,6 +33,7 @@ namespace ZR.Admin.WebApi.Controllers.System
ISysPermissionService permissionService,
ISysConfigService configService,
ISysRoleService sysRoleService,
ISmsCodeLogService smsCodeLogService,
ICaptcha captcha)
{
SecurityCodeHelper = captcha;
@ -37,6 +42,7 @@ namespace ZR.Admin.WebApi.Controllers.System
this.sysLoginService = sysLoginService;
this.permissionService = permissionService;
this.sysConfigService = configService;
this.smsCodeLogService = smsCodeLogService;
roleService = sysRoleService;
}
@ -236,7 +242,7 @@ namespace ZR.Admin.WebApi.Controllers.System
{
if (dto == null) { return ToResponse(ResultCode.CUSTOM_ERROR, "扫码失败"); }
var name = App.HttpContext.GetName();
sysLoginService.CheckLockUser(name);
TokenModel tokenModel = JwtUtil.GetLoginUser(HttpContext);
@ -246,12 +252,80 @@ namespace ZR.Admin.WebApi.Controllers.System
dict.Add("status", "success");
dict.Add("token", JwtUtil.GenerateJwtToken(JwtUtil.AddClaims(tokenModel)));
CacheService.SetScanLogin(dto.Uuid, dict);
return SUCCESS(1);
}
return ToResponse(ResultCode.FAIL, "二维码已失效");
}
#endregion
/// <summary>
///
/// </summary>
/// <param name="dto"></param>
/// <returns></returns>
[HttpPost("checkMobile")]
[Log(Title = "发送短息", BusinessType = BusinessType.INSERT)]
public IActionResult CheckMobile([FromBody] PhoneLoginDto dto)
{
dto.LoginIP = HttpContextExtension.GetClientUserIp(HttpContext);
SysConfig sysConfig = sysConfigService.GetSysConfigByKey("sys.account.captchaOnOff");
if (sysConfig?.ConfigValue != "off" && !SecurityCodeHelper.Validate(dto.Uuid, dto.Code, false))
{
return ToResponse(ResultCode.CUSTOM_ERROR, "验证码错误");
}
string location = HttpContextExtension.GetIpInfo(dto.LoginIP);
var info = sysUserService.GetFirst(f => f.Phonenumber == dto.PhoneNum) ?? throw new CustomException(ResultCode.CUSTOM_ERROR, "该手机号不存在", false);
var smsCode = RandomHelper.GenerateNum(6);
var smsContent = $"验证码{smsCode}(随机验证码),有效期10分钟。";
//TODO 发送短息验证码,1分钟内允许一次
smsCodeLogService.AddSmscodeLog(new ServiceCore.Model.SmsCodeLog()
{
Userid = info.UserId,
PhoneNum = dto.PhoneNum.ParseToLong(),
AddTime = DateTime.Now,
SendType = 1,
SmsCode = smsCode,
SmsContent = smsContent,
UserIP = dto.LoginIP,
Location = location,
});
CacheService.SetPhoneCode(dto.PhoneNum, smsCode);
return SUCCESS(new { smsCode });
}
/// <summary>
/// 手机号登录
/// </summary>
/// <param name="loginBody">登录对象</param>
/// <returns></returns>
[Route("PhoneLogin")]
[HttpPost]
[Log(Title = "手机号登录")]
public IActionResult PhoneLogin([FromBody] PhoneLoginDto loginBody)
{
if (loginBody == null) { throw new CustomException("请求参数错误"); }
loginBody.LoginIP = HttpContextExtension.GetClientUserIp(HttpContext);
if (!CacheService.CheckPhoneCode(loginBody.PhoneNum, loginBody.PhoneCode))
{
return ToResponse(ResultCode.CUSTOM_ERROR, "短信验证码错误");
}
var info = sysUserService.GetFirst(f => f.Phonenumber == loginBody.PhoneNum) ?? throw new CustomException(ResultCode.CUSTOM_ERROR, "该手机号不存在", false);
sysLoginService.CheckLockUser(info.UserName);
string location = HttpContextExtension.GetIpInfo(loginBody.LoginIP);
var user = sysLoginService.PhoneLogin(loginBody, new SysLogininfor() { LoginLocation = location }, info);
List<SysRole> roles = roleService.SelectUserRoleListByUserId(user.UserId);
//权限集合 eg *:*:*,system:user:list
List<string> permissions = permissionService.GetMenuPermission(user);
TokenModel loginUser = new(user.Adapt<TokenModel>(), roles.Adapt<List<Roles>>());
CacheService.SetUserPerms(GlobalConstant.UserPermKEY + user.UserId, permissions);
return SUCCESS(JwtUtil.GenerateJwtToken(JwtUtil.AddClaims(loginUser)));
}
}
}

View File

@ -1,5 +1,6 @@
using Microsoft.AspNetCore.Mvc;
using ZR.Admin.WebApi.Filters;
using ZR.Infrastructure.Helper;
using ZR.Model.System;
using ZR.Model.System.Dto;
using ZR.Service.System.IService;

Binary file not shown.

View File

@ -0,0 +1,25 @@
using System.ComponentModel.DataAnnotations;
namespace ZR.ServiceCore.Model.Dto
{
public class PhoneLoginDto
{
/// <summary>
/// 验证码
/// </summary>
public string Code { get; set; }
/// <summary>
/// 唯一标识
/// </summary>
public string Uuid { get; set; } = "";
public string LoginIP { get; set; }
[Required(ErrorMessage = "手机号不能为空")]
public string PhoneNum { get; set; }
/// <summary>
/// 手机短信验证码
/// </summary>
//[Required(ErrorMessage = "短信验证码不能为空")]
public string PhoneCode { get; set; }
}
}

View File

@ -1,11 +1,11 @@
using System;
using ZR.Common;
using ZR.Common;
namespace ZR.Service.System
{
public class CacheService
{
private readonly static string CK_verifyScan = "verifyScan_";
private readonly static string CK_phoneSmsCode = "phone_sms_code_";
#region
public static List<string> GetUserPerms(string key)
{
@ -59,5 +59,23 @@ namespace ZR.Service.System
}
return 0;
}
public static object SetPhoneCode(string key, string val)
{
var ck = CK_phoneSmsCode + key;
return CacheHelper.SetCache(ck, val, 10);
}
public static bool CheckPhoneCode(string key, string val)
{
var ck = CK_phoneSmsCode + key;
var save_code = CacheHelper.Get(ck);
if (save_code != null && save_code.Equals(val))
{
return true;
}
return false;
}
}
}

View File

@ -1,8 +1,7 @@
using Infrastructure;
using System;
using ZR.Model;
using ZR.Model;
using ZR.Model.System;
using ZR.Model.System.Dto;
using ZR.ServiceCore.Model.Dto;
namespace ZR.Service.System.IService
{
@ -15,7 +14,14 @@ namespace ZR.Service.System.IService
/// <param name="logininfor"></param>
/// <returns></returns>
public SysUser Login(LoginBodyDto loginBody, SysLogininfor logininfor);
/// <summary>
/// 手机号登录
/// </summary>
/// <param name="loginBody"></param>
/// <param name="logininfor"></param>
/// <param name="user"></param>
/// <returns></returns>
SysUser PhoneLogin(PhoneLoginDto loginBody, SysLogininfor logininfor, SysUser user);
/// <summary>
/// 查询操作日志
/// </summary>

View File

@ -81,6 +81,6 @@ namespace ZR.Service.System.IService
SysUser Login(LoginBodyDto user);
void UpdateLoginInfo(LoginBodyDto user, long userId);
void UpdateLoginInfo(string userIP, long userId);
}
}

View File

@ -1,6 +1,5 @@
using Infrastructure;
using Infrastructure.Attribute;
using Infrastructure.Extensions;
using Microsoft.AspNetCore.Http;
using UAParser;
using ZR.Model;
@ -8,6 +7,7 @@ using ZR.Model.System;
using ZR.Model.System.Dto;
using ZR.Repository;
using ZR.Service.System.IService;
using ZR.ServiceCore.Model.Dto;
namespace ZR.Service.System
{
@ -64,10 +64,40 @@ namespace ZR.Service.System
logininfor.Status = "0";
logininfor.Msg = "登录成功";
AddLoginInfo(logininfor);
SysUserService.UpdateLoginInfo(loginBody, user.UserId);
SysUserService.UpdateLoginInfo(loginBody.LoginIP, user.UserId);
return user;
}
/// <summary>
/// 登录验证
/// </summary>
/// <param name="logininfor"></param>
/// <param name="loginBody"></param>
/// <param name="user"></param>
/// <returns></returns>
public SysUser PhoneLogin(PhoneLoginDto loginBody, SysLogininfor logininfor, SysUser user)
{
logininfor.UserName = user.UserName;
logininfor.Status = "1";
logininfor.LoginTime = DateTime.Now;
logininfor.Ipaddr = loginBody.LoginIP;
ClientInfo clientInfo = httpContextAccessor.HttpContext.GetClientInfo();
logininfor.Browser = clientInfo.ToString();
logininfor.Os = clientInfo.OS.ToString();
if (user.Status == 1)
{
logininfor.Msg = "该用户已禁用";
AddLoginInfo(logininfor);
throw new CustomException(ResultCode.LOGIN_ERROR, logininfor.Msg, false);
}
logininfor.Status = "0";
logininfor.Msg = "登录成功";
AddLoginInfo(logininfor);
SysUserService.UpdateLoginInfo(loginBody.LoginIP, user.UserId);
return user;
}
/// <summary>
/// 查询登录日志
/// </summary>

View File

@ -329,12 +329,12 @@ namespace ZR.Service
/// <summary>
/// 修改登录信息
/// </summary>
/// <param name="user"></param>
/// <param name="userIP"></param>
/// <param name="userId"></param>
/// <returns></returns>
public void UpdateLoginInfo(LoginBodyDto user, long userId)
public void UpdateLoginInfo(string userIP, long userId)
{
Update(new SysUser() { LoginIP = user.LoginIP, LoginDate = DateTime.Now, UserId = userId }, it => new { it.LoginIP, it.LoginDate });
Update(new SysUser() { LoginIP = userIP, LoginDate = DateTime.Now, UserId = userId }, it => new { it.LoginIP, it.LoginDate });
}
}
}