Merge pull request '🍒 新增Session扩展属性,方便业务层扩展;部分功能优化' (#427) from jasondom/Admin.NET.Pro:v2-1 into v2
Reviewed-on: https://code.adminnet.top/Admin.NET/Admin.NET.Pro/pulls/427
This commit is contained in:
commit
bff05388f0
@ -16,15 +16,35 @@ namespace Admin.NET.Application;
|
|||||||
public class AppUserManager(
|
public class AppUserManager(
|
||||||
SysCacheService sysCacheService,
|
SysCacheService sysCacheService,
|
||||||
IHttpContextAccessor httpContextAccessor)
|
IHttpContextAccessor httpContextAccessor)
|
||||||
: UserManager(sysCacheService, httpContextAccessor)
|
: UserManager(sysCacheService, httpContextAccessor), IUserSessionExtProps
|
||||||
{
|
{
|
||||||
// 扩展属性
|
/// <summary>
|
||||||
}
|
/// 工号
|
||||||
|
/// </summary>
|
||||||
|
public string JobNum => base.GetExtProp<string>(nameof(JobNum));
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 用户会话信息
|
/// 最新登录Ip
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public partial class UserSessionDao
|
public string LastLoginIp => base.GetExtProp<string>(nameof(LastLoginIp));
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 最新登录Ip
|
||||||
|
/// </summary>
|
||||||
|
public string LastLoginDevice => base.GetExtProp<string>(nameof(LastLoginDevice));
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 初始化扩展属性
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="user"></param>
|
||||||
|
/// <returns></returns>
|
||||||
|
public Dictionary<string, object> GetInitExtProps(SysUser user)
|
||||||
{
|
{
|
||||||
// 扩展属性
|
return new()
|
||||||
|
{
|
||||||
|
{ nameof(JobNum), user.JobNum },
|
||||||
|
{ nameof(LastLoginIp), user.LastLoginIp },
|
||||||
|
{ nameof(LastLoginDevice), user.LastLoginDevice },
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@ -38,3 +38,14 @@ public interface IOrgIdFilter
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
long? CreateOrgId { get; set; }
|
long? CreateOrgId { get; set; }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 应用Id接口过滤器
|
||||||
|
/// </summary>
|
||||||
|
public interface IAppIdFilter
|
||||||
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// 系统层应用Id
|
||||||
|
/// </summary>
|
||||||
|
long? AppId { get; set; }
|
||||||
|
}
|
||||||
@ -229,6 +229,7 @@ public class SysAuthService : IDynamicApiController, ITransient
|
|||||||
await App.GetRequiredService<SysOnlineUserService>().SingleLogin(user.Id, loginMode);
|
await App.GetRequiredService<SysOnlineUserService>().SingleLogin(user.Id, loginMode);
|
||||||
|
|
||||||
// 生成Token令牌
|
// 生成Token令牌
|
||||||
|
user.TokenVersion += 1;
|
||||||
var tokenExpire = await _sysConfigService.GetTokenExpire();
|
var tokenExpire = await _sysConfigService.GetTokenExpire();
|
||||||
var accessToken = JWTEncryption.Encrypt(new Dictionary<string, object>
|
var accessToken = JWTEncryption.Encrypt(new Dictionary<string, object>
|
||||||
{
|
{
|
||||||
@ -245,10 +246,12 @@ public class SysAuthService : IDynamicApiController, ITransient
|
|||||||
RealName = user.RealName,
|
RealName = user.RealName,
|
||||||
AccountType = user.AccountType,
|
AccountType = user.AccountType,
|
||||||
OrgId = user.OrgId,
|
OrgId = user.OrgId,
|
||||||
|
OrgCode = user.SysOrg?.Code,
|
||||||
OrgName = user.SysOrg?.Name,
|
OrgName = user.SysOrg?.Name,
|
||||||
OrgType = user.SysOrg?.Type,
|
OrgType = user.SysOrg?.Type,
|
||||||
OrgLevel = user.SysOrg?.Level,
|
OrgLevel = user.SysOrg?.Level,
|
||||||
TokenVersion = user.TokenVersion,
|
TokenVersion = user.TokenVersion,
|
||||||
|
ExtProps = App.GetServices<IUserSessionExtProps>().SelectMany(u => u.GetInitExtProps(user)).ToDictionary(u => u.Key, u => u.Value)
|
||||||
}, TimeSpan.FromMinutes(tokenExpire));
|
}, TimeSpan.FromMinutes(tokenExpire));
|
||||||
|
|
||||||
// 生成刷新Token令牌
|
// 生成刷新Token令牌
|
||||||
@ -262,7 +265,6 @@ public class SysAuthService : IDynamicApiController, ITransient
|
|||||||
// ke.global.setAllHeader('Authorization', 'Bearer ' + ke.response.headers['access-token']);
|
// ke.global.setAllHeader('Authorization', 'Bearer ' + ke.response.headers['access-token']);
|
||||||
|
|
||||||
// 更新用户登录信息
|
// 更新用户登录信息
|
||||||
user.TokenVersion += 1;
|
|
||||||
user.LastLoginIp = _httpContextAccessor.HttpContext.GetRemoteIpAddressToIPv4(true);
|
user.LastLoginIp = _httpContextAccessor.HttpContext.GetRemoteIpAddressToIPv4(true);
|
||||||
(user.LastLoginAddress, double? longitude, double? latitude) = CommonHelper.GetIpAddress(user.LastLoginIp);
|
(user.LastLoginAddress, double? longitude, double? latitude) = CommonHelper.GetIpAddress(user.LastLoginIp);
|
||||||
user.LastLoginTime = DateTime.Now;
|
user.LastLoginTime = DateTime.Now;
|
||||||
@ -364,7 +366,7 @@ public class SysAuthService : IDynamicApiController, ITransient
|
|||||||
|
|
||||||
// 写入Token黑名单
|
// 写入Token黑名单
|
||||||
var tokenExpire = await _sysConfigService.GetTokenExpire();
|
var tokenExpire = await _sysConfigService.GetTokenExpire();
|
||||||
_sysCacheService.Set($"{CacheConst.KeyTokenBlacklist}:{userId}:{version}", _userManager.Account, TimeSpan.FromMinutes(tokenExpire));
|
_sysCacheService.Set($"{CacheConst.KeyTokenBlacklist}{userId}:{version}", _userManager.Account, TimeSpan.FromMinutes(tokenExpire));
|
||||||
|
|
||||||
// 发布系统退出事件
|
// 发布系统退出事件
|
||||||
await _eventPublisher.PublishAsync(UserEventTypeEnum.Logout, _userManager);
|
await _eventPublisher.PublishAsync(UserEventTypeEnum.Logout, _userManager);
|
||||||
|
|||||||
@ -0,0 +1,15 @@
|
|||||||
|
// Admin.NET 项目的版权、商标、专利和其他相关权利均受相应法律法规的保护。使用本项目应遵守相关法律法规和许可证的要求。
|
||||||
|
//
|
||||||
|
// 本项目主要遵循 MIT 许可证和 Apache 许可证(版本 2.0)进行分发和使用。许可证位于源代码树根目录中的 LICENSE-MIT 和 LICENSE-APACHE 文件。
|
||||||
|
//
|
||||||
|
// 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
|
||||||
|
|
||||||
|
namespace Admin.NET.Core;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 用户Session属性扩展接口
|
||||||
|
/// </summary>
|
||||||
|
public interface IUserSessionExtProps
|
||||||
|
{
|
||||||
|
Dictionary<string, object> GetInitExtProps(SysUser user);
|
||||||
|
}
|
||||||
@ -4,6 +4,8 @@
|
|||||||
//
|
//
|
||||||
// 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
|
// 不得利用本项目从事危害国家安全、扰乱社会秩序、侵犯他人合法权益等法律法规禁止的活动!任何基于本项目二次开发而产生的一切法律纠纷和责任,我们不承担任何责任!
|
||||||
|
|
||||||
|
using System.Text.Json;
|
||||||
|
|
||||||
namespace Admin.NET.Core;
|
namespace Admin.NET.Core;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
@ -18,14 +20,14 @@ public class UserManager(
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
[System.Text.Json.Serialization.JsonIgnore]
|
[System.Text.Json.Serialization.JsonIgnore]
|
||||||
[Newtonsoft.Json.JsonIgnore]
|
[Newtonsoft.Json.JsonIgnore]
|
||||||
private UserSessionDao Session { get; set; }
|
protected virtual UserSessionDao _session { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 代理对象
|
/// 代理对象
|
||||||
/// </summary>
|
/// </summary>
|
||||||
[System.Text.Json.Serialization.JsonIgnore]
|
[System.Text.Json.Serialization.JsonIgnore]
|
||||||
[Newtonsoft.Json.JsonIgnore]
|
[Newtonsoft.Json.JsonIgnore]
|
||||||
protected UserSessionDao session => Session ??= sysCacheService.Get<UserSessionDao>(CacheConst.KeyUserSession + UserId);
|
protected virtual UserSessionDao Session => _session ??= sysCacheService.Get<UserSessionDao>(CacheConst.KeyUserSession + UserId);
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 用户Id
|
/// 用户Id
|
||||||
@ -37,62 +39,67 @@ public class UserManager(
|
|||||||
/// <summary>
|
/// <summary>
|
||||||
/// 应用Id
|
/// 应用Id
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public override long? AppId => session?.AppId;
|
public override long? AppId => Session?.AppId;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 租户Id
|
/// 租户Id
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public override long? TenantId => session?.TenantId;
|
public override long? TenantId => Session?.TenantId;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 用户账号
|
/// 用户账号
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public override string Account => session?.Account;
|
public override string Account => Session?.Account;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 真实姓名
|
/// 真实姓名
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public override string RealName => session?.RealName;
|
public override string RealName => Session?.RealName;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 昵称
|
/// 昵称
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public override string NickName => session?.NickName;
|
public override string NickName => Session?.NickName;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 账号类型
|
/// 账号类型
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public override AccountTypeEnum? AccountType => session?.AccountType;
|
public override AccountTypeEnum? AccountType => Session?.AccountType;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 组织机构Id
|
/// 组织机构Id
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public override long OrgId => session?.OrgId ?? 0;
|
public override long OrgId => Session?.OrgId ?? 0;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 组织机构名称
|
/// 组织机构名称
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public override string OrgName => session?.OrgName;
|
public override string OrgName => Session?.OrgName;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 组织机构Id
|
/// 组织机构Id
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public override string OrgType => session?.OrgType;
|
public override string OrgType => Session?.OrgType;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 组织机构级别
|
/// 组织机构级别
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public override int? OrgLevel => session?.OrgLevel;
|
public override int? OrgLevel => Session?.OrgLevel;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 登录模式
|
/// 登录模式
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public override LoginModeEnum? LoginMode => session?.LoginMode;
|
public override LoginModeEnum? LoginMode => Session?.LoginMode;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 微信OpenId
|
/// 微信OpenId
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public override string OpenId => session?.OpenId;
|
public override string OpenId => Session?.OpenId;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 扩展属性
|
||||||
|
/// </summary>
|
||||||
|
public override Dictionary<string, dynamic> ExtProps => Session?.ExtProps;
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 用户Session是否存在
|
/// 用户Session是否存在
|
||||||
@ -125,4 +132,21 @@ public class UserManager(
|
|||||||
{
|
{
|
||||||
return sysCacheService.Get<UserSessionDao>(CacheConst.KeyUserSession + userId);
|
return sysCacheService.Get<UserSessionDao>(CacheConst.KeyUserSession + userId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 获取扩展属性
|
||||||
|
/// </summary>
|
||||||
|
public T GetExtProp<T>(string key, T defaultValue = default)
|
||||||
|
{
|
||||||
|
if (ExtProps == null || !ExtProps.TryGetValue(key, out var value) || value is null) return defaultValue;
|
||||||
|
if (value is JsonElement element) return element.Deserialize<T>();
|
||||||
|
try
|
||||||
|
{
|
||||||
|
return (T)Convert.ChangeType(value, typeof(T));
|
||||||
|
}
|
||||||
|
catch
|
||||||
|
{
|
||||||
|
return defaultValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@ -65,6 +65,11 @@ public partial class UserSessionDao
|
|||||||
/// </summary>
|
/// </summary>
|
||||||
public virtual long OrgId { get; set; }
|
public virtual long OrgId { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 组织机构编码
|
||||||
|
/// </summary>
|
||||||
|
public virtual string OrgCode { get; set; }
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// 组织机构名称
|
/// 组织机构名称
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -94,4 +99,9 @@ public partial class UserSessionDao
|
|||||||
/// token版本
|
/// token版本
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public virtual long TokenVersion { get; set; }
|
public virtual long TokenVersion { get; set; }
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 扩展属性
|
||||||
|
/// </summary>
|
||||||
|
public virtual Dictionary<string, object> ExtProps { get; set; }
|
||||||
}
|
}
|
||||||
@ -62,7 +62,7 @@ namespace Admin.NET.Web.Core
|
|||||||
// 验证Token黑名单
|
// 验证Token黑名单
|
||||||
var userId = httpContext.User.FindFirst(ClaimConst.UserId)?.Value;
|
var userId = httpContext.User.FindFirst(ClaimConst.UserId)?.Value;
|
||||||
var version = httpContext.User.FindFirst(ClaimConst.TokenVersion)?.Value;
|
var version = httpContext.User.FindFirst(ClaimConst.TokenVersion)?.Value;
|
||||||
if (sysCacheService.ExistKey($"{CacheConst.KeyTokenBlacklist}:{userId}:{version}") || !sysCacheService.ExistKey($"{CacheConst.KeyUserSession}{userId}"))
|
if (sysCacheService.ExistKey($"{CacheConst.KeyTokenBlacklist}{userId}:{version}") || !sysCacheService.ExistKey($"{CacheConst.KeyUserSession}{userId}"))
|
||||||
{
|
{
|
||||||
context.Fail(new AuthorizationFailureReason(this, "令牌已失效,请重新登录。"));
|
context.Fail(new AuthorizationFailureReason(this, "令牌已失效,请重新登录。"));
|
||||||
context.StatusCode(StatusCodes.Status401Unauthorized);
|
context.StatusCode(StatusCodes.Status401Unauthorized);
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user