JwtHandler.cs 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. // 麻省理工学院许可证
  2. //
  3. // 版权所有 (c) 2021-2023 zuohuaijun,大名科技(天津)有限公司 联系电话/微信:18020030720 QQ:515096995
  4. //
  5. // 特此免费授予获得本软件的任何人以处理本软件的权利,但须遵守以下条件:在所有副本或重要部分的软件中必须包括上述版权声明和本许可声明。
  6. //
  7. // 软件按“原样”提供,不提供任何形式的明示或暗示的保证,包括但不限于对适销性、适用性和非侵权的保证。
  8. // 在任何情况下,作者或版权持有人均不对任何索赔、损害或其他责任负责,无论是因合同、侵权或其他方式引起的,与软件或其使用或其他交易有关。
  9. using Admin.NET.Core;
  10. using Admin.NET.Core.Service;
  11. using Furion;
  12. using Furion.Authorization;
  13. using Furion.DataEncryption;
  14. using Microsoft.AspNetCore.Authorization;
  15. using Microsoft.AspNetCore.Http;
  16. using Microsoft.Extensions.DependencyInjection;
  17. using SqlSugar;
  18. using System;
  19. using System.Threading.Tasks;
  20. namespace Admin.NET.Web.Core
  21. {
  22. public class JwtHandler : AppAuthorizeHandler
  23. {
  24. private readonly IServiceProvider _serviceProvider;
  25. public JwtHandler(IServiceProvider serviceProvider)
  26. {
  27. _serviceProvider = serviceProvider;
  28. }
  29. /// <summary>
  30. /// 自动刷新Token
  31. /// </summary>
  32. /// <param name="context"></param>
  33. /// <returns></returns>
  34. public override async Task HandleAsync(AuthorizationHandlerContext context)
  35. {
  36. // var serviceProvider = context.GetCurrentHttpContext().RequestServices;
  37. using var serviceScope = _serviceProvider.CreateScope();
  38. // 若当前账号存在黑名单中则授权失败
  39. var sysCacheService = serviceScope.ServiceProvider.GetService<SysCacheService>();
  40. if (sysCacheService.ExistKey($"{CacheConst.KeyBlacklist}{context.User.FindFirst(ClaimConst.UserId)?.Value}"))
  41. {
  42. context.Fail();
  43. context.GetCurrentHttpContext().SignoutToSwagger();
  44. return;
  45. }
  46. var sysConfigService = serviceScope.ServiceProvider.GetService<SysConfigService>();
  47. var tokenExpire = await sysConfigService.GetTokenExpire();
  48. var refreshTokenExpire = await sysConfigService.GetRefreshTokenExpire();
  49. if (JWTEncryption.AutoRefreshToken(context, context.GetCurrentHttpContext(), tokenExpire, refreshTokenExpire))
  50. {
  51. await AuthorizeHandleAsync(context);
  52. }
  53. else
  54. {
  55. context.Fail(); // 授权失败
  56. var currentHttpContext = context.GetCurrentHttpContext();
  57. if (currentHttpContext == null)
  58. return;
  59. // 跳过由于 SignatureAuthentication 引发的失败
  60. if (currentHttpContext.Items.ContainsKey(SignatureAuthenticationDefaults.AuthenticateFailMsgKey))
  61. return;
  62. currentHttpContext.SignoutToSwagger();
  63. }
  64. }
  65. public override async Task<bool> PipelineAsync(AuthorizationHandlerContext context, DefaultHttpContext httpContext)
  66. {
  67. // 已自动验证 Jwt Token 有效性
  68. return await CheckAuthorizeAsync(httpContext);
  69. }
  70. /// <summary>
  71. /// 权限校验核心逻辑
  72. /// </summary>
  73. /// <param name="httpContext"></param>
  74. /// <returns></returns>
  75. private static async Task<bool> CheckAuthorizeAsync(DefaultHttpContext httpContext)
  76. {
  77. // 登录模式判断PC、APP
  78. if (App.User.FindFirst(ClaimConst.LoginMode)?.Value == ((int)LoginModeEnum.APP).ToString())
  79. return true;
  80. // 排除超管
  81. if (App.User.FindFirst(ClaimConst.AccountType)?.Value == ((int)AccountTypeEnum.SuperAdmin).ToString())
  82. return true;
  83. // 路由名称
  84. var routeName = httpContext.Request.Path.StartsWithSegments("/api")
  85. ? httpContext.Request.Path.Value[5..].Replace("/", ":")
  86. : httpContext.Request.Path.Value[1..].Replace("/", ":");
  87. // 获取用户拥有按钮权限集合
  88. var ownBtnPermList = await App.GetService<SysMenuService>().GetOwnBtnPermList();
  89. // 获取系统所有按钮权限集合
  90. var allBtnPermList = await App.GetService<SysMenuService>().GetAllBtnPermList();
  91. // 已拥有该按钮权限或者所有按钮集合里面不存在
  92. var exist1 = ownBtnPermList.Exists(u => routeName.Equals(u, StringComparison.CurrentCultureIgnoreCase));
  93. var exist2 = allBtnPermList.TrueForAll(u => !routeName.Equals(u, StringComparison.CurrentCultureIgnoreCase));
  94. return exist1 || exist2;
  95. }
  96. }
  97. }