AppTokenService.java 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. package com.ruoyi.app.base;
  2. import com.auth0.jwt.JWT;
  3. import com.auth0.jwt.algorithms.Algorithm;
  4. import com.ruoyi.app.domain.TbAppUser;
  5. import com.ruoyi.common.constant.Constants;
  6. import com.ruoyi.common.core.domain.model.LoginUser;
  7. import com.ruoyi.common.core.redis.RedisCache;
  8. import com.ruoyi.common.utils.ServletUtils;
  9. import com.ruoyi.common.utils.StringUtils;
  10. import com.ruoyi.common.utils.ip.AddressUtils;
  11. import com.ruoyi.common.utils.ip.IpUtils;
  12. import com.ruoyi.common.utils.uuid.IdUtils;
  13. import eu.bitwalker.useragentutils.UserAgent;
  14. import io.jsonwebtoken.Claims;
  15. import io.jsonwebtoken.Jwts;
  16. import io.jsonwebtoken.SignatureAlgorithm;
  17. import org.springframework.beans.factory.annotation.Autowired;
  18. import org.springframework.beans.factory.annotation.Value;
  19. import org.springframework.stereotype.Service;
  20. import javax.servlet.http.HttpServletRequest;
  21. import java.util.HashMap;
  22. import java.util.Map;
  23. import java.util.concurrent.TimeUnit;
  24. /**
  25. * @author Alex
  26. */
  27. @Service("AppTokenService")
  28. public class AppTokenService {
  29. /**令牌自定义标识*/
  30. @Value("${token.header}")
  31. private String header;
  32. /**令牌秘钥*/
  33. @Value("${token.secret}")
  34. private String secret;
  35. /**令牌有效期(默认7天)*/
  36. @Value("${token.expireTime}")
  37. private int expireTime;
  38. protected static final long MILLIS_SECOND = 1000;
  39. protected static final long MILLIS_MINUTE = 60 * MILLIS_SECOND;
  40. private static final Long MILLIS_MINUTE_TEN = 20 * 60 * 1000L;
  41. @Autowired
  42. private RedisCache redisCache;
  43. /**
  44. * 获取用户身份信息
  45. *
  46. * @return 用户信息
  47. */
  48. public AppLoginUser getLoginUser() {
  49. HttpServletRequest request = ServletUtils.getRequest();
  50. // 获取请求携带的令牌
  51. String token = getToken(request);
  52. if (StringUtils.isNotEmpty(token)) {
  53. Claims claims = parseToken(token);
  54. // 解析对应的用户信息
  55. String mobile = (String) claims.get(Constants.APP_LOGIN_USER_KEY);
  56. String salt = (String) claims.get("salt");
  57. String userKey = getTokenKey(mobile);
  58. AppLoginUser user = redisCache.getCacheObject(userKey);
  59. if (!salt.equals(user.getSalt())){
  60. return null;
  61. }
  62. return user;
  63. }
  64. return null;
  65. }
  66. /**
  67. * 获取用户身份信息
  68. *
  69. * @return 用户信息
  70. */
  71. public AppLoginUser getLoginUser(String token) {
  72. if (StringUtils.isNotEmpty(token)) {
  73. Claims claims = parseToken(token);
  74. // 解析对应的用户信息
  75. String mobile = (String) claims.get(Constants.APP_LOGIN_USER_KEY);
  76. String salt = (String) claims.get("salt");
  77. String userKey = getTokenKey(mobile);
  78. AppLoginUser user = redisCache.getCacheObject(userKey);
  79. if (!salt.equals(user.getSalt())){
  80. return null;
  81. }
  82. return user;
  83. }
  84. return null;
  85. }
  86. /**
  87. * 设置用户身份信息
  88. */
  89. public void setLoginUser(AppLoginUser loginUser) {
  90. if (StringUtils.isNotNull(loginUser) && StringUtils.isNotEmpty(loginUser.getUser().getMobile())) {
  91. refreshToken(loginUser);
  92. }
  93. }
  94. /**
  95. * 删除用户身份信息
  96. */
  97. public void delLoginUser(String mobile) {
  98. if (StringUtils.isNotEmpty(mobile)) {
  99. String userKey = getTokenKey(mobile);
  100. redisCache.deleteObject(userKey);
  101. }
  102. }
  103. /**
  104. * 创建令牌
  105. *
  106. * @param loginUser 用户信息
  107. * @return 令牌
  108. */
  109. public String createToken(AppLoginUser loginUser) {
  110. String salt = IdUtils.fastUUID();
  111. loginUser.setSalt(salt);
  112. setUserAgent(loginUser);
  113. refreshToken(loginUser);
  114. Map<String, Object> claims = new HashMap<>(1);
  115. claims.put(Constants.APP_LOGIN_USER_KEY, loginUser.getUser().getMobile());
  116. claims.put("salt", salt);
  117. return createToken(claims);
  118. }
  119. /**
  120. * 验证令牌有效期,相差不足20分钟,自动刷新缓存
  121. *
  122. * @param loginUser
  123. * @return 令牌
  124. */
  125. public void verifyToken(AppLoginUser loginUser) {
  126. long expireTime = loginUser.getExpireTime();
  127. long currentTime = System.currentTimeMillis();
  128. if (expireTime - currentTime <= MILLIS_MINUTE_TEN) {
  129. refreshToken(loginUser);
  130. }
  131. }
  132. /**
  133. * 刷新令牌有效期
  134. *
  135. * @param loginUser 登录信息
  136. */
  137. public void refreshToken(AppLoginUser loginUser)
  138. {
  139. loginUser.setLoginTime(System.currentTimeMillis());
  140. loginUser.setExpireTime(loginUser.getLoginTime() + expireTime * MILLIS_MINUTE);
  141. // 根据mobile将loginUser缓存
  142. String userKey = getTokenKey(loginUser.getUser().getMobile());
  143. redisCache.setCacheObject(userKey, loginUser, expireTime, TimeUnit.MINUTES);
  144. }
  145. /**
  146. * 设置用户代理信息
  147. *
  148. * @param loginUser 登录信息
  149. */
  150. public void setUserAgent(AppLoginUser loginUser) {
  151. UserAgent userAgent = UserAgent.parseUserAgentString(ServletUtils.getRequest().getHeader("User-Agent"));
  152. String ip = IpUtils.getIpAddr(ServletUtils.getRequest());
  153. loginUser.setIpaddr(ip);
  154. loginUser.setLoginLocation(AddressUtils.getRealAddressByIP(ip));
  155. loginUser.setBrowser(userAgent.getBrowser().getName());
  156. loginUser.setOs(userAgent.getOperatingSystem().getName());
  157. }
  158. /**
  159. * 从数据声明生成令牌
  160. *
  161. * @param claims 数据声明
  162. * @return 令牌
  163. */
  164. private String createToken(Map<String, Object> claims) {
  165. String token = Jwts.builder()
  166. .setClaims(claims)
  167. .signWith(SignatureAlgorithm.HS512, secret).compact();
  168. return token;
  169. }
  170. /**
  171. * 从令牌中获取数据声明
  172. *
  173. * @param token 令牌
  174. * @return 数据声明
  175. */
  176. private Claims parseToken(String token) {
  177. return Jwts.parser()
  178. .setSigningKey(secret)
  179. .parseClaimsJws(token)
  180. .getBody();
  181. }
  182. /**
  183. * 从令牌中获取用户名
  184. *
  185. * @param token 令牌
  186. * @return 用户名
  187. */
  188. public String getUsernameFromToken(String token) {
  189. Claims claims = parseToken(token);
  190. return claims.getSubject();
  191. }
  192. /**
  193. * 获取请求token
  194. *
  195. * @param request
  196. * @return token
  197. */
  198. private String getToken(HttpServletRequest request) {
  199. String token = request.getHeader(header);
  200. if (StringUtils.isNotEmpty(token) && token.startsWith(Constants.TOKEN_PREFIX))
  201. {
  202. token = token.replace(Constants.TOKEN_PREFIX, "");
  203. }
  204. return token;
  205. }
  206. private String getTokenKey(String mobile) {
  207. return Constants.APP_LOGIN_TOKEN_KEY + mobile;
  208. }
  209. }