using JinianNet.JNTemplate; using JinianNet.JNTemplate.Nodes; using Microsoft.AspNetCore.SignalR; using Microsoft.Extensions.Caching.Memory; using Microsoft.IdentityModel.Tokens; using Newtonsoft.Json; using System.IdentityModel.Tokens.Jwt; using System.Security.Claims; using System.Text; using ZR.Admin.WebApi.Hubs; using ZR.Model.System.Dto; namespace ZR.Admin.WebApi.Framework { /// /// 2020-11-20 /// public class JwtUtil { /// /// 获取用户身份信息 /// /// /// public static LoginUser GetLoginUser(HttpContext httpContext) { string token = httpContext.GetToken(); if (!string.IsNullOrEmpty(token)) { return ValidateJwtToken(ParseToken(token)); } return null; } /// /// 生成token /// /// /// public static string GenerateJwtToken(List claims) { JwtSettings jwtSettings = new(); AppSettings.Bind("JwtSettings", jwtSettings); var authTime = DateTime.Now; var expiresAt = authTime.AddMinutes(jwtSettings.Expire); var tokenHandler = new JwtSecurityTokenHandler(); var key = Encoding.ASCII.GetBytes(jwtSettings.SecretKey); claims.Add(new Claim("Audience", jwtSettings.Audience)); claims.Add(new Claim("Issuer", jwtSettings.Issuer)); var tokenDescriptor = new SecurityTokenDescriptor { Subject = new ClaimsIdentity(claims), Issuer = jwtSettings.Issuer, Audience = jwtSettings.Audience, IssuedAt = authTime,//token生成时间 Expires = expiresAt, //NotBefore = authTime, TokenType = "Bearer", //对称秘钥,签名证书 SigningCredentials = new SigningCredentials(new SymmetricSecurityKey(key), SecurityAlgorithms.HmacSha256Signature) }; var token = tokenHandler.CreateToken(tokenDescriptor); return tokenHandler.WriteToken(token); } /// /// 验证Token /// /// public static TokenValidationParameters ValidParameters() { JwtSettings jwtSettings = new(); AppSettings.Bind("JwtSettings", jwtSettings); if (jwtSettings == null || jwtSettings.SecretKey.IsEmpty()) { throw new Exception("JwtSettings获取失败"); } var key = Encoding.ASCII.GetBytes(jwtSettings.SecretKey); 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对比 ClockSkew = TimeSpan.FromSeconds(30) //RequireExpirationTime = true,//过期时间 }; return tokenDescriptor; } /// /// 从令牌中获取数据声明 /// /// 令牌 /// public static JwtSecurityToken? ParseToken(string token) { var tokenHandler = new JwtSecurityTokenHandler(); var validateParameter = ValidParameters(); token = token.Replace("Bearer ", ""); try { tokenHandler.ValidateToken(token, validateParameter, out SecurityToken validatedToken); return tokenHandler.ReadJwtToken(token); } catch (Exception ex) { Console.WriteLine(ex.Message); // return null if validation fails return null; } } /// /// jwt token校验 /// /// /// public static LoginUser? ValidateJwtToken(JwtSecurityToken jwtSecurityToken) { try { if (jwtSecurityToken == null) return null; IEnumerable claims = jwtSecurityToken?.Claims; LoginUser loginUser = null; var userData = claims.FirstOrDefault(x => x.Type == ClaimTypes.UserData)?.Value; if (userData != null) { loginUser = JsonConvert.DeserializeObject(userData); loginUser.ExpireTime = jwtSecurityToken.ValidTo; } //var nowTime = DateTime.UtcNow; //TimeSpan ts = loginUser.ExpireTime - nowTime; //Console.WriteLine("jwt到期时间:" + loginUser.ExpireTime); //Console.WriteLine("nowTime" + nowTime + ",相隔" + ts.TotalSeconds); //if (loginUser != null && ts.TotalSeconds <= 30) //{ // var newToken = GenerateJwtToken(AddClaims(loginUser)); // var CK = "token_" + loginUser.UserId; // if (!CacheHelper.Exists(CK)) // { // CacheHelper.SetCache(CK, newToken); // } //} return loginUser; } catch (Exception ex) { Console.WriteLine(ex.Message); return null; } } /// ///组装Claims /// /// /// public static List AddClaims(LoginUser user) { var claims = new List() { new Claim(ClaimTypes.PrimarySid, user.UserId.ToString()), new Claim(ClaimTypes.Name, user.UserName), new Claim(ClaimTypes.UserData, JsonConvert.SerializeObject(user)) }; return claims; } } }