当前位置:   article > 正文

全网超细--Java实现Token登录验证步骤实现_java token验证

java token验证

一、JWT说明

token进行用户身份验证的流程:

1、客户端使用用户名和密码请求登录

2、服务端收到请求,验证用户名和密码

3、验证成功后,服务端会签发一个token,再把这个token返回给客户端

4、客户端收到token后可以把它存储起来,比如放到cookie中

5、客户端每次向服务端请求资源时需要携带服务端签发的token,可以在cookie或者header中携带

6、服务端收到请求,然后去验证客户端请求里面带着的token,如果验证成功,就向客户端返回请求数据

这种基于token的认证方式相比传统的session认证方式更节约服务器资源,并且对移动端和分布式更加友好。其优点如下:

支持跨域访问:cookie是无法跨域的,而token由于没有用到cookie(前提是将token放到请求头中),所以跨域后不会存在信息丢失问题
无状态:token机制在服务端不需要存储session信息,因为token自身包含了所有登录用户的信息,所以可以减轻服务端压力
更适用CDN:可以通过内容分发网络请求服务端的所有资料
更适用于移动端:当客户端是非浏览器平台时,cookie是不被支持的,此时采用token认证方式会简单很多
无需考虑CSRF:由于不再依赖cookie,所以采用token认证方式不会发生CSRF,所以也就无需考虑CSRF的防御

JWT就是上述流程当中token的一种具体实现方式,其全称是JSON Web Token,官网地址:https://jwt.io/

通俗地说,JWT的本质就是一个字符串,它是将用户信息保存到一个Json字符串中,然后进行编码后得到一个JWT token,并且这个JWT token带有签名信息,接收后可以校验是否被篡改,所以可以用于在各方之间安全地将信息作为Json对象传输。JWT的认证流程如下:

1、首先,前端通过Web表单将自己的用户名和密码发送到后端的接口,这个过程一般是一个POST请求。建议的方式是通过SSL加密的传输(HTTPS),从而避免敏感信息被嗅探

2、后端核对用户名和密码成功后,将包含用户信息的数据作为JWT的Payload,将其与JWT Header分别进行Base64编码拼接后签名,形成一个JWT Token,形成的JWT Token就是一个如同lll.zzz.xxx的字符串
3、后端将JWT Token字符串作为登录成功的结果返回给前端。前端可以将返回的结果保存在浏览器中,退出登录时删除保存的JWT Token即可

4、前端在每次请求时将JWT Token放入HTTP请求头中的Authorization属性中(解决XSS和XSRF问题)

5、后端检查前端传过来的JWT Token,验证其有效性,比如检查签名是否正确、是否过期、token的接收方是否是自己等等

6、验证通过后,后端解析出JWT Token中包含的用户信息,进行其他逻辑操作(一般是根据用户信息得到权限等),返回结果

二、使用步骤

JwtUtil类(生成和解析JWT)

  1. import io.jsonwebtoken.Claims;
  2. import io.jsonwebtoken.JwtBuilder;
  3. import io.jsonwebtoken.Jwts;
  4. import io.jsonwebtoken.SignatureAlgorithm;
  5. import java.nio.charset.StandardCharsets;
  6. import java.util.Date;
  7. import java.util.Map;
  8. public class JwtUtil {
  9. /**
  10. * 生成jwt
  11. * 使用Hs256算法, 私匙使用固定秘钥
  12. *
  13. * @param secretKey jwt秘钥
  14. * @param ttlMillis jwt过期时间(毫秒)
  15. * @param claims 设置的信息
  16. * @return
  17. */
  18. public static String createJWT(String secretKey, long ttlMillis, Map<String, Object> claims) {
  19. // 指定签名的时候使用的签名算法,也就是header那部分
  20. SignatureAlgorithm signatureAlgorithm = SignatureAlgorithm.HS256;
  21. // 生成JWT的时间
  22. long expMillis = System.currentTimeMillis() + ttlMillis;
  23. Date exp = new Date(expMillis);
  24. // 设置jwt的body
  25. JwtBuilder builder = Jwts.builder()
  26. // 如果有私有声明,一定要先设置这个自己创建的私有的声明,这个是给builder的claim赋值,一旦写在标准的声明赋值之后,就是覆盖了那些标准的声明的
  27. .setClaims(claims)
  28. // 设置签名使用的签名算法和签名使用的秘钥
  29. .signWith(signatureAlgorithm, secretKey.getBytes(StandardCharsets.UTF_8))
  30. // 设置过期时间
  31. .setExpiration(exp);
  32. return builder.compact();
  33. }
  34. /**
  35. * Token解密
  36. *
  37. * @param secretKey jwt秘钥 此秘钥一定要保留好在服务端, 不能暴露出去, 否则sign就可以被伪造, 如果对接多个客户端建议改造成多个
  38. * @param token 加密后的token
  39. * @return
  40. */
  41. public static Claims parseJWT(String secretKey, String token) {
  42. // 得到DefaultJwtParser
  43. Claims claims = Jwts.parser()
  44. // 设置签名的秘钥
  45. .setSigningKey(secretKey.getBytes(StandardCharsets.UTF_8))
  46. // 设置需要解析的jwt
  47. .parseClaimsJws(token).getBody();
  48. return claims;
  49. }
  50. }
JwtProperties类(JWT属性)
  1. @Component //bean的注入
  2. @ConfigurationProperties(prefix = "sky.jwt")//配置属性类,读取配置文件,在application。yml中配置值
  3. @Data
  4. public class JwtProperties {
  5. /**
  6. * 管理端员工生成jwt令牌相关配置
  7. */
  8. private String adminSecretKey;
  9. private long adminTtl;
  10. private String adminTokenName;
  11. /**
  12. * 用户端微信用户生成jwt令牌相关配置
  13. */
  14. private String userSecretKey;
  15. private long userTtl;
  16. private String userTokenName;
  17. }

定义拦截器(继承 HandlerInterceptor,这里我分为管理员和用户)

  1. //管理员
  2. import com.sky.constant.JwtClaimsConstant;
  3. import com.sky.context.BaseContext;
  4. import com.sky.properties.JwtProperties;
  5. import com.sky.utils.JwtUtil;
  6. import io.jsonwebtoken.Claims;
  7. import lombok.extern.slf4j.Slf4j;
  8. import org.springframework.beans.factory.annotation.Autowired;
  9. import org.springframework.stereotype.Component;
  10. import org.springframework.web.method.HandlerMethod;
  11. import org.springframework.web.servlet.HandlerInterceptor;
  12. import javax.servlet.http.HttpServletRequest;
  13. import javax.servlet.http.HttpServletResponse;
  14. /**
  15. * jwt令牌校验的拦截器
  16. */
  17. @Component //生成Bean
  18. @Slf4j
  19. public class JwtTokenAdminInterceptor implements HandlerInterceptor {
  20. @Autowired
  21. private JwtProperties jwtProperties;
  22. /**
  23. * 校验jwt
  24. *
  25. * @param request
  26. * @param response
  27. * @param handler
  28. * @return
  29. * @throws Exception
  30. */
  31. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
  32. //判断当前拦截到的是Controller的方法还是其他资源
  33. if (!(handler instanceof HandlerMethod)) {
  34. //当前拦截到的不是动态方法,直接放行
  35. return true;
  36. }
  37. //1、从请求头中获取令牌
  38. String token = request.getHeader(jwtProperties.getAdminTokenName());
  39. //2、校验令牌
  40. try {
  41. log.info("jwt校验:{}", token);
  42. Claims claims = JwtUtil.parseJWT(jwtProperties.getAdminSecretKey(), token);
  43. Long empId = Long.valueOf(claims.get(JwtClaimsConstant.EMP_ID).toString());
  44. log.info("当前员工id:{}", empId);
  45. //前端发一次请求,是一次线程(拦截器,controller,service,mapper共有同一线程) (threadLocal)
  46. BaseContext.setCurrentId(empId);
  47. //3、通过,放行
  48. return true;
  49. } catch (Exception ex) {
  50. //4、不通过,响应401状态码
  51. response.setStatus(401);
  52. return false;
  53. }
  54. }
  55. }
  1. //用户
  2. import com.sky.constant.JwtClaimsConstant;
  3. import com.sky.context.BaseContext;
  4. import com.sky.properties.JwtProperties;
  5. import com.sky.utils.JwtUtil;
  6. import io.jsonwebtoken.Claims;
  7. import lombok.extern.slf4j.Slf4j;
  8. import org.springframework.beans.factory.annotation.Autowired;
  9. import org.springframework.stereotype.Component;
  10. import org.springframework.web.method.HandlerMethod;
  11. import org.springframework.web.servlet.HandlerInterceptor;
  12. import javax.servlet.http.HttpServletRequest;
  13. import javax.servlet.http.HttpServletResponse;
  14. /**
  15. * jwt令牌校验的拦截器 拦截器校验令牌,注册是拦截路径
  16. */
  17. @Component
  18. @Slf4j
  19. public class JwtTokenUserInterceptor implements HandlerInterceptor {
  20. @Autowired
  21. private JwtProperties jwtProperties;
  22. /**
  23. * 校验jwt
  24. *
  25. * @param request
  26. * @param response
  27. * @param handler
  28. * @return
  29. * @throws Exception
  30. */
  31. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
  32. //判断当前拦截到的是Controller的方法还是其他资源
  33. if (!(handler instanceof HandlerMethod)) {
  34. //当前拦截到的不是动态方法,直接放行
  35. return true;
  36. }
  37. //1、从请求头中获取令牌
  38. String token = request.getHeader(jwtProperties.getUserTokenName());
  39. //2、校验令牌
  40. try {
  41. log.info("jwt校验:{}", token);
  42. Claims claims = JwtUtil.parseJWT(jwtProperties.getUserSecretKey(), token);
  43. Long userId = Long.valueOf(claims.get(JwtClaimsConstant.USER_ID).toString());
  44. log.info("当前用户id:{}", userId);
  45. //前端发一次请求,是一次线程(拦截器,controller,service,mapper共有同一线程) (threadLocal)
  46. BaseContext.setCurrentId(userId);
  47. //3、通过,放行
  48. return true;
  49. } catch (Exception ex) {
  50. //4、不通过,响应401状态码
  51. response.setStatus(401);
  52. return false;
  53. }
  54. }
  55. }

定义拦截器后,要注册才能实现拦截作用(addInterceptors方法中添加拦截器)

  1. import com.sky.interceptor.JwtTokenAdminInterceptor;
  2. import com.sky.interceptor.JwtTokenUserInterceptor;
  3. import com.sky.json.JacksonObjectMapper;
  4. import lombok.extern.slf4j.Slf4j;
  5. import org.springframework.beans.factory.annotation.Autowired;
  6. import org.springframework.context.annotation.Bean;
  7. import org.springframework.context.annotation.Configuration;
  8. import org.springframework.http.converter.HttpMessageConverter;
  9. import org.springframework.http.converter.cbor.MappingJackson2CborHttpMessageConverter;
  10. import org.springframework.http.converter.json.MappingJackson2HttpMessageConverter;
  11. import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
  12. import org.springframework.web.servlet.config.annotation.ResourceHandlerRegistry;
  13. import org.springframework.web.servlet.config.annotation.WebMvcConfigurationSupport;
  14. import springfox.documentation.builders.ApiInfoBuilder;
  15. import springfox.documentation.builders.PathSelectors;
  16. import springfox.documentation.builders.RequestHandlerSelectors;
  17. import springfox.documentation.service.ApiInfo;
  18. import springfox.documentation.spi.DocumentationType;
  19. import springfox.documentation.spring.web.plugins.Docket;
  20. import java.util.List;
  21. /**
  22. * 配置类,注册web层相关组件
  23. */
  24. @Configuration
  25. @Slf4j
  26. public class WebMvcConfiguration extends WebMvcConfigurationSupport {
  27. @Autowired
  28. private JwtTokenAdminInterceptor jwtTokenAdminInterceptor;
  29. @Autowired
  30. private JwtTokenUserInterceptor jwtTokenUserInterceptor;
  31. /**
  32. * 注册自定义拦截器 配置路径
  33. *
  34. * @param registry
  35. */
  36. protected void addInterceptors(InterceptorRegistry registry) {
  37. log.info("开始注册自定义拦截器...");
  38. registry.addInterceptor(jwtTokenAdminInterceptor)
  39. .addPathPatterns("/admin/**")
  40. .excludePathPatterns("/admin/employee/login");
  41. registry.addInterceptor(jwtTokenUserInterceptor)
  42. .addPathPatterns("/user/**")//拦截路径
  43. .excludePathPatterns("/user/user/login")//排除拦截路径
  44. .excludePathPatterns("/user/shop/status");
  45. }
  46. /**
  47. * 通过knife4j生成接口文档
  48. * @return
  49. */
  50. @Bean
  51. public Docket docket1() {
  52. log.info("准备生成接口文档...");
  53. ApiInfo apiInfo = new ApiInfoBuilder()
  54. .title("苍穹外卖项目接口文档")
  55. .version("2.0")
  56. .description("苍穹外卖项目接口文档")
  57. .build();
  58. Docket docket = new Docket(DocumentationType.SWAGGER_2)
  59. .groupName("管理端接口")
  60. .apiInfo(apiInfo)
  61. .select()
  62. .apis(RequestHandlerSelectors.basePackage("com.sky.controller.admin"))
  63. .paths(PathSelectors.any())
  64. .build();
  65. return docket;
  66. }
  67. @Bean
  68. public Docket docket2() {
  69. log.info("准备生成接口文档...");
  70. ApiInfo apiInfo = new ApiInfoBuilder()
  71. .title("苍穹外卖项目接口文档")
  72. .version("2.0")
  73. .description("苍穹外卖项目接口文档")
  74. .build();
  75. Docket docket = new Docket(DocumentationType.SWAGGER_2)
  76. .groupName("用户端接口")
  77. .apiInfo(apiInfo)
  78. .select()
  79. .apis(RequestHandlerSelectors.basePackage("com.sky.controller.user"))
  80. .paths(PathSelectors.any())
  81. .build();
  82. return docket;
  83. }
  84. /**
  85. * 设置静态资源映射
  86. * @param registry
  87. */
  88. protected void addResourceHandlers(ResourceHandlerRegistry registry) {
  89. log.info("开始设置静态资源映射...");
  90. registry.addResourceHandler("/doc.html").addResourceLocations("classpath:/META-INF/resources/");
  91. registry.addResourceHandler("/webjars/**").addResourceLocations("classpath:/META-INF/resources/webjars/");
  92. }
  93. //扩展springmvc框架的消息转化器
  94. @Override
  95. protected void extendMessageConverters(List<HttpMessageConverter<?>> converters) {
  96. log.info("扩展消息转化器...");
  97. //创建一个消息转化器对象
  98. MappingJackson2HttpMessageConverter converter = new MappingJackson2HttpMessageConverter();
  99. //需要为消息转化器设置一个对象转换器,对象转换器可以将Java对象序列化为json数据
  100. converter.setObjectMapper(new JacksonObjectMapper());
  101. //将自己的消息转化器加入容器中
  102. converters.add(0,converter);
  103. }
  104. }
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/空白诗007/article/detail/931616
推荐阅读
相关标签
  

闽ICP备14008679号