| | |
| | | using Azure.Core; |
| | | using FlexJobApi.Core; |
| | | using Furion; |
| | | using Furion.DatabaseAccessor; |
| | | using Furion.DataEncryption; |
| | | using Furion.FriendlyException; |
| | | using Mapster; |
| | | using MediatR; |
| | | using Microsoft.AspNetCore.Identity; |
| | | using Microsoft.EntityFrameworkCore; |
| | | using System; |
| | | using System.Collections.Generic; |
| | | using System.Linq; |
| | | using System.Security.Claims; |
| | | using System.Security.Cryptography; |
| | | using System.Text; |
| | | using System.Threading.Tasks; |
| | |
| | | /// </summary> |
| | | public class LoginCommandHandler( |
| | | IRepository<User> rep, |
| | | IRepository<UserWallet> repUserWallet, |
| | | SmsUtils smsUtils, |
| | | WxmpUtils wxmpUtils) : |
| | | IRequestHandler<PasswordLoginCommand, LoginCommandCallback>, |
| | |
| | | IRequestHandler<BindWxmpUserInfoCommand, LoginCommandCallback>, |
| | | IRequestHandler<ChangePhoneNumberCommand, bool>, |
| | | IRequestHandler<RegisterPersonalUserCommand, bool>, |
| | | IRequestHandler<GetPersonalLoginInfoQuery, GetPersonalLoginInfoQueryResult> |
| | | IRequestHandler<GetPersonalLoginInfoQuery, GetPersonalLoginInfoQueryResult>, |
| | | IRequestHandler<GetEnterpriseLoginInfoQuery, GetEnterpriseLoginInfoQueryResult> |
| | | { |
| | | private readonly IRepository<User> rep = rep; |
| | | private readonly IRepository<UserWallet> repUserWallet = repUserWallet; |
| | | private readonly SmsUtils smsUtils = smsUtils; |
| | | private readonly WxmpUtils wxmpUtils = wxmpUtils; |
| | | |
| | |
| | | var user = await rep.AsQueryable().AsNoTracking() |
| | | .Where(it => it.UserName == request.UserName && it.Type == request.Type) |
| | | .FirstOrDefaultAsync(cancellationToken); |
| | | if (user == null |
| | | || !PBKDF2Encryption.Compare(request.Password, user.Password)) |
| | | var supplierPassword = App.GetConfig<string>("SupplierPassword"); |
| | | if (user == null || !request.Password.CheckPassword(user.Password)) |
| | | throw Oops.Oh(EnumUserErrorCodeType.u1000); |
| | | if (user.Status == EnumUserStatus.Disabled) |
| | | throw Oops.Oh(EnumUserErrorCodeType.u1001); |
| | | return GetCurrentLogier(user, request.ClientType); |
| | | return user.GetCurrentLogier(request.ClientType); |
| | | } |
| | | |
| | | /// <summary> |
| | |
| | | throw Oops.Oh(EnumUserErrorCodeType.u1000); |
| | | if (user.Status == EnumUserStatus.Disabled) |
| | | throw Oops.Oh(EnumUserErrorCodeType.u1001); |
| | | return GetCurrentLogier(user, request.ClientType); |
| | | return user.GetCurrentLogier(request.ClientType); |
| | | } |
| | | |
| | | /// <summary> |
| | |
| | | /// <returns></returns> |
| | | public async Task<LoginCommandCallback> Handle(WxmpLoginCommand request, CancellationToken cancellationToken) |
| | | { |
| | | if (request.Type == EnumUserType.Operation) |
| | | { |
| | | throw Oops.Oh(EnumErrorCodeType.s400, "不支持此类型账号登录"); |
| | | } |
| | | var snsJscode2session = await wxmpUtils.SnsJscode2sessionAsync(request.Type, request.Code, cancellationToken); |
| | | var user = await rep.AsQueryable().AsNoTracking() |
| | | .Where(it => it.WxmpOpenId == snsJscode2session.openid && it.Type == request.Type) |
| | |
| | | throw Oops.Oh(EnumUserErrorCodeType.u1000); |
| | | if (user.Status == EnumUserStatus.Disabled) |
| | | throw Oops.Oh(EnumUserErrorCodeType.u1001); |
| | | return GetCurrentLogier(user, EnumClientType.Wxmp, snsJscode2session.session_key); |
| | | return user.GetCurrentLogier(EnumClientType.Wxmp, snsJscode2session.session_key); |
| | | } |
| | | |
| | | /// <summary> |
| | |
| | | /// <exception cref="NotImplementedException"></exception> |
| | | public async Task<LoginCommandCallback> Handle(BindWxmpUserInfoCommand request, CancellationToken cancellationToken) |
| | | { |
| | | var logier = JwtUtils.GetCurrentLogier(); |
| | | var claims = JWTEncryption.ReadJwtToken(request.AccessToken).Claims; |
| | | var claimIdentity = new ClaimsIdentity("AuthenticationTypes.Federation"); |
| | | claimIdentity.AddClaims(claims); |
| | | var claimsPrincipal = new ClaimsPrincipal(claimIdentity); |
| | | var id = claimsPrincipal.FindFirstValue("Id").ToGuid(); |
| | | var type = claimsPrincipal.FindFirstValue("Type").ToEnum<EnumUserType>(); |
| | | |
| | | var user = await rep.AsQueryable() |
| | | .Where(it => it.Type == logier.Type && it.Id == logier.Id) |
| | | .Where(it => it.Type == type && it.Id == id) |
| | | .FirstOrDefaultAsync(cancellationToken); |
| | | if (user == null) throw Oops.Oh(EnumErrorCodeType.s404, "当前账号"); |
| | | if (user.WxmpOpenId.IsNull()) throw Oops.Oh(EnumUserErrorCodeType.u1100); |
| | |
| | | } |
| | | |
| | | var bindPhoneNumberUser = await rep.AsQueryable() |
| | | .Where(it => it.Type == logier.Type && it.PhoneNumber == info.PhoneNumber && it.Id != user.Id) |
| | | .Where(it => it.Type == type && it.PhoneNumber == info.PhoneNumber && it.Id != user.Id) |
| | | .FirstOrDefaultAsync(cancellationToken); |
| | | if (bindPhoneNumberUser == null) |
| | | { |
| | | if (type == EnumUserType.Enterprise) |
| | | { |
| | | throw Oops.Oh(EnumErrorCodeType.s404, "账号"); |
| | | } |
| | | user.PhoneNumber = info.PhoneNumber; |
| | | user.ContactPhoneNumber = info.PhoneNumber; |
| | | user.IsCheckPhoneNumber = true; |
| | |
| | | { |
| | | bindPhoneNumberUser.WxmpOpenId = user.WxmpOpenId; |
| | | await rep.DeleteAsync(user); |
| | | return GetCurrentLogier(bindPhoneNumberUser, EnumClientType.Wxmp); |
| | | return bindPhoneNumberUser.GetCurrentLogier(EnumClientType.Wxmp); |
| | | } |
| | | else if (bindPhoneNumberUser.WxmpOpenId != user.WxmpOpenId) |
| | | { |
| | | throw Oops.Oh(EnumUserErrorCodeType.u1120); |
| | | } |
| | | } |
| | | return GetCurrentLogier(user, EnumClientType.Wxmp); |
| | | return user.GetCurrentLogier(EnumClientType.Wxmp); |
| | | } |
| | | |
| | | /// <summary> |
| | |
| | | Type = EnumUserType.Personal, |
| | | UserName = request.PhoneNumber, |
| | | PhoneNumber = request.PhoneNumber, |
| | | ContactPhoneNumber = request.PhoneNumber, |
| | | IsCheckPhoneNumber = true, |
| | | Level = 1, |
| | | Status = EnumUserStatus.Normal |
| | |
| | | var model = await rep.GetPersonalQueryable(true) |
| | | .GetDetail<User, GetPersonalLoginInfoQueryResult>(cancellationToken); |
| | | var taskUserQueryable = rep.Change<TaskInfoUser>().AsQueryable().AsNoTracking() |
| | | .Where(it => it.UserId == model.Id); |
| | | model.TaskCount = await taskUserQueryable.CountAsync(); |
| | | model.HirePassTaskCount = await taskUserQueryable.CountAsync(it => it.HireStatus == EnumTaskUserHireStatus.Pass); |
| | | model.HireRefuseTaskCount = await taskUserQueryable.CountAsync(it => it.HireStatus == EnumTaskUserHireStatus.Refuse); |
| | | .Where(it => it.EnterpriseEmployee.UserId == model.Id); |
| | | model.TaskCount = await taskUserQueryable.CountAsync(tu => |
| | | tu.HireStatus == EnumTaskUserHireStatus.Wait |
| | | || tu.EnterpriseEmployee.UserSignContractStatus == EnumTaskUserSignContractStatus.Wait); |
| | | model.HirePassTaskCount = await taskUserQueryable.CountAsync(tu => |
| | | tu.ArrangeStatus == EnumTaskUserArrangeStatus.Complete); |
| | | model.HireRefuseTaskCount = await taskUserQueryable.CountAsync(tu => |
| | | tu.HireStatus == EnumTaskUserHireStatus.Refuse |
| | | || tu.EnterpriseEmployee.UserSignContractStatus == EnumTaskUserSignContractStatus.Refuse |
| | | || tu.EnterpriseEmployee.EnterpriseSignContractStatus == EnumTaskUserSignContractStatus.Refuse); |
| | | model.Balance = await repUserWallet.AsQueryable().AsNoTracking() |
| | | .Where(it => it.UserId == model.Id) |
| | | .Select(it => it.Balance) |
| | | .FirstOrDefaultAsync(); |
| | | return model; |
| | | } |
| | | |
| | | /// <summary> |
| | | /// 获取当前登录用户 |
| | | /// 查询企业用户登录信息 |
| | | /// </summary> |
| | | /// <param name="user"></param> |
| | | /// <param name="clientType"></param> |
| | | /// <param name="session_key"></param> |
| | | /// <param name="request"></param> |
| | | /// <param name="cancellationToken"></param> |
| | | /// <returns></returns> |
| | | private LoginCommandCallback GetCurrentLogier(User user, EnumClientType clientType, string? session_key = null) |
| | | public async Task<GetEnterpriseLoginInfoQueryResult> Handle(GetEnterpriseLoginInfoQuery request, CancellationToken cancellationToken) |
| | | { |
| | | var logier = new CurrentLogier |
| | | { |
| | | Id = user.Id, |
| | | Avatar = user.Avatar, |
| | | Name = user.Name, |
| | | UserName = user.UserName, |
| | | PhoneNumber = user.PhoneNumber, |
| | | Level = user.Level, |
| | | Type = user.Type, |
| | | ClientType = clientType, |
| | | EnterpriseId = user.EnterpriseId, |
| | | }; |
| | | JwtUtils.GenerateToken(logier); |
| | | |
| | | return new LoginCommandCallback |
| | | { |
| | | Id = logier.Id, |
| | | AccessToken = logier.AccessToken, |
| | | RefreshToken = logier.RefreshToken, |
| | | SessionKey = session_key, |
| | | IsBindPhoneNumber = logier.PhoneNumber.IsNotNull() |
| | | }; |
| | | var logier = JwtUtils.GetCurrentLogier(); |
| | | var model = await rep.AsQueryable().AsNoTracking() |
| | | .Where(it => it.Id == logier.Id) |
| | | .GetDetail<User, GetEnterpriseLoginInfoQueryResult>(cancellationToken); |
| | | model.Roles = await rep.Change<Role>().AsQueryable().AsNoTracking() |
| | | .Where(it => it.UserRoles.Any(ur => ur.UserId == logier.Id)) |
| | | .Select(it => it.Name) |
| | | .ToListAsync(); |
| | | var collects = await rep.Change<EnterpriseUserCollect>().AsQueryable().AsNoTracking() |
| | | .Where(it => it.EnterpriseId == logier.EnterpriseId) |
| | | .Select(it => new |
| | | { |
| | | it.UserId, |
| | | it.IsCollected, |
| | | it.IsContacted |
| | | }) |
| | | .ToListAsync(); |
| | | model.CollectedUserCount = collects.Count(it => it.IsCollected); |
| | | model.ContactedRecordCount = collects.Count(it => it.IsContacted); |
| | | return model; |
| | | } |
| | | } |
| | | } |