JWTUtils

发布时间 2023-12-06 18:56:42作者: Aotle
@Service
public class JwtUtils {
    /**
     * 加密盐值
     */
    @Value("${application.security.jwt.secret-key}")
    private String secretKey;

    /**
     * Token失效时间
     */
    @Value("${application.security.jwt.expiration}")
    private long jwtExpiration;

    /**
     * Token刷新时间
     */
    @Value("${application.security.jwt.refresh-token.expiration}")
    private long refreshExpiration;

    /**
     * 从Token中获取Username
     * @param token Token
     * @return String
     */
    public String extractUsername(String token) {
        return extractClaim(token, Claims::getSubject);
    }

    /**
     * 从Token中回去数据,根据传入不同的Function返回不同的数据
     * eg: String extractUsername(String token)
     */
    public <T> T extractClaim(String token, Function<Claims, T> claimsResolver) {
        final Claims claims = extractAllClaims(token);
        return claimsResolver.apply(claims);
    }

    /**
     * 生成Token无额外信息
     */
    public String generateToken(User usersDetails) {
        return generateToken(new HashMap<>(), usersDetails);
    }

    /**
     * 生成Token,有额外信息
     * @param extraClaims 额外的数据
     * @param users 用户信息
     * @return String
     */
    public String generateToken(
            Map<String, Object> extraClaims,
            User users
    ) {
        return buildToken(extraClaims, users, jwtExpiration);
    }

    /**
     * 生成刷新用的Token
     * @param users 用户信息
     * @return String
     */
    public String generateRefreshToken(
            User users
    ) {
        return buildToken(new HashMap<>(), users, refreshExpiration);
    }

    /**
     * 构建Token方法
     * @param extraClaims 额外信息
     * @param users //用户信息
     * @param expiration //失效时间
     * @return String
     */
    private String buildToken(
            Map<String, Object> extraClaims,
            User users,
            long expiration
    ) {
        return Jwts
                .builder()
                .setClaims(extraClaims) //body
                .setSubject(users.getUsername()) //主题数据
                .setIssuedAt(new Date(System.currentTimeMillis())) //设置发布时间
                .setExpiration(new Date(System.currentTimeMillis() + expiration)) //设置过期时间
                .signWith(getSignInKey())//设置摘要算法
                .compact();
    }

    /**
     * 验证Token是否有效
     * @param token Token
     * @param users 用户信息
     * @return boolean
     */
    public boolean isTokenValid(String token, Users users) {
        final String username = extractUsername(token);
        return (username.equals(users.getName())) && !isTokenExpired(token);
    }

    public boolean isTokenValid(String token, String userName) {
        final String username = extractUsername(token);
        return (username.equals(userName) && !isTokenExpired(token));
    }

    /**
     * 判断Token是否过去
     */
    private boolean isTokenExpired(String token) {
        return extractExpiration(token).before(new Date());
    }

    /**
     * 从Token中获取失效时间
     */
    private Date extractExpiration(String token) {
        //通用方法,传入一个Function,返回一个T
        return extractClaim(token, Claims::getExpiration);
    }

    /**
     * 从Token中获取所有数据
     */
    private Claims extractAllClaims(String token) {
        return Jwts
                .parserBuilder()
                .setSigningKey(getSignInKey())
                .build()
                .parseClaimsJws(token)
                .getBody();
    }

    /**
     * 获取签名Key
     * Token 加密解密使用
     */
    private Key getSignInKey() {
        byte[] keyBytes = Decoders.BASE64.decode(secretKey);
        return Keys.hmacShaKeyFor(keyBytes);
    }

}