2021-09-23 10:56:22 +08:00
|
|
|
|
using Infrastructure;
|
2022-01-08 21:48:09 +08:00
|
|
|
|
using Infrastructure.Extensions;
|
2021-09-23 10:56:22 +08:00
|
|
|
|
using Microsoft.AspNetCore.Http;
|
2021-08-23 16:57:25 +08:00
|
|
|
|
using Microsoft.IdentityModel.Tokens;
|
|
|
|
|
|
using Newtonsoft.Json;
|
|
|
|
|
|
using System;
|
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
using System.IdentityModel.Tokens.Jwt;
|
|
|
|
|
|
using System.Linq;
|
|
|
|
|
|
using System.Security.Claims;
|
|
|
|
|
|
using System.Text;
|
|
|
|
|
|
using ZR.Admin.WebApi.Extensions;
|
2022-03-24 18:05:52 +08:00
|
|
|
|
using ZR.Common;
|
2021-08-23 16:57:25 +08:00
|
|
|
|
using ZR.Model.System;
|
|
|
|
|
|
|
|
|
|
|
|
namespace ZR.Admin.WebApi.Framework
|
|
|
|
|
|
{
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 2020-11-20
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
public class JwtUtil
|
|
|
|
|
|
{
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 获取用户身份信息
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="httpContext"></param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public static LoginUser GetLoginUser(HttpContext httpContext)
|
|
|
|
|
|
{
|
2021-12-10 18:50:26 +08:00
|
|
|
|
string token = httpContext.GetToken();
|
2021-12-03 17:42:44 +08:00
|
|
|
|
|
2021-08-23 16:57:25 +08:00
|
|
|
|
if (!string.IsNullOrEmpty(token))
|
|
|
|
|
|
{
|
|
|
|
|
|
return ValidateJwtToken(ParseToken(token));
|
|
|
|
|
|
}
|
|
|
|
|
|
return null;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 生成token
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="claims"></param>
|
2022-01-09 17:12:35 +08:00
|
|
|
|
/// <param name="jwtSettings"></param>
|
2021-08-23 16:57:25 +08:00
|
|
|
|
/// <returns></returns>
|
2022-01-09 17:12:35 +08:00
|
|
|
|
public static string GenerateJwtToken(List<Claim> claims, JwtSettings jwtSettings)
|
2021-08-23 16:57:25 +08:00
|
|
|
|
{
|
2021-12-10 09:12:05 +08:00
|
|
|
|
var authTime = DateTime.Now;
|
|
|
|
|
|
var expiresAt = authTime.AddMinutes(jwtSettings.Expire);
|
2021-08-23 16:57:25 +08:00
|
|
|
|
var tokenHandler = new JwtSecurityTokenHandler();
|
2021-12-03 17:42:44 +08:00
|
|
|
|
var key = Encoding.ASCII.GetBytes(jwtSettings.SecretKey);
|
|
|
|
|
|
claims.Add(new Claim("Audience", jwtSettings.Audience));
|
|
|
|
|
|
claims.Add(new Claim("Issuer", jwtSettings.Issuer));
|
|
|
|
|
|
|
2021-08-23 16:57:25 +08:00
|
|
|
|
var tokenDescriptor = new SecurityTokenDescriptor
|
|
|
|
|
|
{
|
|
|
|
|
|
Subject = new ClaimsIdentity(claims),
|
2021-12-03 17:42:44 +08:00
|
|
|
|
Issuer = jwtSettings.Issuer,
|
|
|
|
|
|
Audience = jwtSettings.Audience,
|
2021-12-10 09:12:05 +08:00
|
|
|
|
IssuedAt = authTime,//token生成时间
|
|
|
|
|
|
Expires = expiresAt,
|
2021-12-10 18:50:26 +08:00
|
|
|
|
//NotBefore = authTime,
|
2021-12-03 17:42:44 +08:00
|
|
|
|
TokenType = "Bearer",
|
|
|
|
|
|
//对称秘钥,签名证书
|
2021-08-23 16:57:25 +08:00
|
|
|
|
SigningCredentials = new SigningCredentials(new SymmetricSecurityKey(key), SecurityAlgorithms.HmacSha256Signature)
|
|
|
|
|
|
};
|
|
|
|
|
|
var token = tokenHandler.CreateToken(tokenDescriptor);
|
|
|
|
|
|
return tokenHandler.WriteToken(token);
|
|
|
|
|
|
}
|
2021-12-03 17:42:44 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 验证Token
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public static TokenValidationParameters ValidParameters()
|
|
|
|
|
|
{
|
2022-02-23 18:30:17 +08:00
|
|
|
|
JwtSettings jwtSettings = new();
|
|
|
|
|
|
AppSettings.Bind("JwtSettings", jwtSettings);
|
|
|
|
|
|
|
2022-01-08 21:48:09 +08:00
|
|
|
|
if (jwtSettings == null || jwtSettings.SecretKey.IsEmpty())
|
|
|
|
|
|
{
|
2022-01-11 13:39:42 +08:00
|
|
|
|
throw new Exception("JwtSettings获取失败");
|
2022-01-08 21:48:09 +08:00
|
|
|
|
}
|
2021-12-03 17:42:44 +08:00
|
|
|
|
var key = Encoding.ASCII.GetBytes(jwtSettings.SecretKey);
|
2022-01-11 13:39:42 +08:00
|
|
|
|
|
2021-12-03 17:42:44 +08:00
|
|
|
|
var tokenDescriptor = new TokenValidationParameters
|
|
|
|
|
|
{
|
|
|
|
|
|
ValidateIssuerSigningKey = true,
|
|
|
|
|
|
ValidateIssuer = true,
|
|
|
|
|
|
ValidateAudience = true,
|
|
|
|
|
|
ValidIssuer = jwtSettings.Issuer,
|
|
|
|
|
|
ValidAudience = jwtSettings.Audience,
|
|
|
|
|
|
IssuerSigningKey = new SymmetricSecurityKey(key),
|
|
|
|
|
|
ValidateLifetime = true,//是否验证Token有效期,使用当前时间与Token的Claims中的NotBefore和Expires对比
|
2021-12-10 18:50:26 +08:00
|
|
|
|
ClockSkew = TimeSpan.FromSeconds(30)
|
|
|
|
|
|
//RequireExpirationTime = true,//过期时间
|
2021-12-03 17:42:44 +08:00
|
|
|
|
};
|
|
|
|
|
|
return tokenDescriptor;
|
|
|
|
|
|
}
|
2021-08-23 16:57:25 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// 从令牌中获取数据声明
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="token">令牌</param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public static IEnumerable<Claim> ParseToken(string token)
|
|
|
|
|
|
{
|
|
|
|
|
|
var tokenHandler = new JwtSecurityTokenHandler();
|
2021-12-03 17:42:44 +08:00
|
|
|
|
var validateParameter = ValidParameters();
|
|
|
|
|
|
token = token.Replace("Bearer ", "");
|
2021-08-23 16:57:25 +08:00
|
|
|
|
try
|
|
|
|
|
|
{
|
2021-12-03 17:42:44 +08:00
|
|
|
|
tokenHandler.ValidateToken(token, validateParameter, out SecurityToken validatedToken);
|
2021-08-23 16:57:25 +08:00
|
|
|
|
|
2021-12-03 17:42:44 +08:00
|
|
|
|
var jwtToken = tokenHandler.ReadJwtToken(token);
|
2021-08-23 16:57:25 +08:00
|
|
|
|
return jwtToken.Claims;
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
Console.WriteLine(ex.Message);
|
|
|
|
|
|
// return null if validation fails
|
|
|
|
|
|
return null;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
/// jwt token校验
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="jwtToken"></param>
|
|
|
|
|
|
/// <returns></returns>
|
2021-12-26 18:26:38 +08:00
|
|
|
|
public static LoginUser ValidateJwtToken(IEnumerable<Claim> jwtToken)
|
2021-08-23 16:57:25 +08:00
|
|
|
|
{
|
|
|
|
|
|
try
|
|
|
|
|
|
{
|
|
|
|
|
|
var userData = jwtToken.FirstOrDefault(x => x.Type == ClaimTypes.UserData).Value;
|
2022-03-24 18:05:52 +08:00
|
|
|
|
var loginUser = JsonConvert.DeserializeObject<LoginUser>(userData);
|
2022-04-10 09:46:55 +08:00
|
|
|
|
var permissions = (List<string>)CacheHelper.GetCache(GlobalConstant.UserPermKEY + loginUser?.UserId);
|
2022-03-27 14:06:34 +08:00
|
|
|
|
if (loginUser?.UserName == "admin")
|
|
|
|
|
|
{
|
|
|
|
|
|
permissions = new List<string>() { GlobalConstant.AdminPerm };
|
|
|
|
|
|
}
|
2022-03-24 18:31:05 +08:00
|
|
|
|
if (permissions == null) return null;
|
2022-03-27 14:06:34 +08:00
|
|
|
|
loginUser.Permissions = permissions;
|
2021-08-23 16:57:25 +08:00
|
|
|
|
return loginUser;
|
|
|
|
|
|
}
|
|
|
|
|
|
catch (Exception ex)
|
|
|
|
|
|
{
|
|
|
|
|
|
Console.WriteLine(ex.Message);
|
|
|
|
|
|
return null;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2022-03-24 18:05:52 +08:00
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
|
///组装Claims
|
|
|
|
|
|
/// </summary>
|
|
|
|
|
|
/// <param name="user"></param>
|
|
|
|
|
|
/// <returns></returns>
|
|
|
|
|
|
public static List<Claim> AddClaims(LoginUser user)
|
|
|
|
|
|
{
|
2022-03-27 14:06:34 +08:00
|
|
|
|
if (user?.Permissions.Count > 50)
|
|
|
|
|
|
{
|
|
|
|
|
|
user.Permissions = new List<string>();
|
|
|
|
|
|
}
|
2022-03-24 18:05:52 +08:00
|
|
|
|
var claims = new List<Claim>()
|
|
|
|
|
|
{
|
|
|
|
|
|
new Claim(ClaimTypes.PrimarySid, user.UserId.ToString()),
|
|
|
|
|
|
new Claim(ClaimTypes.Name, user.UserName),
|
|
|
|
|
|
new Claim(ClaimTypes.UserData, JsonConvert.SerializeObject(user))
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
return claims;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2021-08-23 16:57:25 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|