using FileShare_EFCore.Models; using Microsoft.Extensions.Options; using Microsoft.IdentityModel.Tokens; using System.IdentityModel.Tokens.Jwt; using System.Security.Claims; using System.Text; namespace FileShare_API.Authentication { /// /// JWT Token 服务,负责创建包含用户声明和角色的 Access Token。 /// public sealed class JwtTokenService(IOptions options) { /// /// JWT 配置选项。 /// private readonly JwtOptions _options = options.Value; /// /// 创建包含用户声明和角色的 JWT Access Token。 /// /// 用户实体。 /// 角色集合。 /// 包含 Token 字符串和过期时间的元组。 public (string Token, DateTime ExpiresAt) CreateAccessToken(UserEntity user, IReadOnlyCollection roles) { var expiresAt = DateTime.UtcNow.AddMinutes(_options.AccessTokenMinutes); var claims = new List { new(JwtRegisteredClaimNames.Sub, user.Id.ToString()), new(ClaimTypes.NameIdentifier, user.Id.ToString()), new(ClaimTypes.Name, user.Name ?? user.Email ?? $"user-{user.Id}"), new("auth_type", "api-jwt"), }; foreach (var role in roles.Where(role => !string.IsNullOrWhiteSpace(role)).Distinct(StringComparer.OrdinalIgnoreCase)) { claims.Add(new Claim(ClaimTypes.Role, role)); } var signingKey = new SymmetricSecurityKey(Encoding.UTF8.GetBytes(_options.SigningKey)); var credentials = new SigningCredentials(signingKey, SecurityAlgorithms.HmacSha256); var jwt = new JwtSecurityToken( issuer: _options.Issuer, audience: _options.Audience, claims: claims, notBefore: DateTime.UtcNow, expires: expiresAt, signingCredentials: credentials); return (new JwtSecurityTokenHandler().WriteToken(jwt), expiresAt); } } }