亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > Java > 正文

Spring-boot結合Shrio實現JWT的方法

2024-07-14 08:40:39
字體:
來源:轉載
供稿:網友

本文介紹了Spring-boot結合Shrio實現JWT的方法,分享給大家,具體如下:

關于驗證大致分為兩個方面:

  1. 用戶登錄時的驗證;
  2. 用戶登錄后每次訪問時的權限認證

主要解決方法:使用自定義的Shiro Filter

項目搭建:

這是一個spring-boot 的web項目,不了解spring-boot的項目搭建,請google。

pom.mx引入相關jar包

 <!-- shiro 權限管理 -->  <dependency>    <groupId>org.apache.shiro</groupId>    <artifactId>shiro-spring</artifactId>    <version>${shiro.version}</version>  </dependency>  <dependency>    <groupId>org.apache.shiro</groupId>    <artifactId>shiro-core</artifactId>    <version>${shiro.version}</version>  </dependency> <!-- JWT -->   <dependency>    <groupId>io.jsonwebtoken</groupId>    <artifactId>jjwt</artifactId>    <version>0.9.0</version>  </dependency>

Shrio 的相關配置

劃重點!!自定義了一個Filter

filterMap.put("JWTFilter", new JWTFilter());
@Configurationpublic class ShiroConfig {  @Bean  public ShiroFilterFactoryBean getShiroFilterFactoryBean(SecurityManager securityManager) {    ShiroFilterFactoryBean shiroFilterFactoryBean = new ShiroFilterFactoryBean();    shiroFilterFactoryBean.setSecurityManager(securityManager);    // 添加自己的過濾器并且取名為JWTFilter    Map<String, Filter> filterMap = new HashMap<>();    filterMap.put("JWTFilter", new JWTFilter());    shiroFilterFactoryBean.setFilters(filterMap);    /*     * 自定義url規則     * http://shiro.apache.org/web.html#urls-     */    Map<String, String> filterChainDefinitionMap = shiroFilterFactoryBean.getFilterChainDefinitionMap();    filterChainDefinitionMap.put("/**", "JWTFilter");    shiroFilterFactoryBean.setFilterChainDefinitionMap(filterChainDefinitionMap);    return shiroFilterFactoryBean;  }  /**   * securityManager 不用直接注入shiroDBRealm,可能會導致事務失效   * 解決方法見 handleContextRefresh   * http://www.debugrun.com/a/NKS9EJQ.html   */  @Bean("securityManager")  public DefaultWebSecurityManager securityManager(TokenRealm tokenRealm) {    DefaultWebSecurityManager manager = new DefaultWebSecurityManager();    manager.setRealm(tokenRealm);    /*     * 關閉shiro自帶的session,詳情見文檔     * http://shiro.apache.org/session-management.html#SessionManagement-StatelessApplications%28Sessionless%29     */    DefaultSubjectDAO subjectDAO = new DefaultSubjectDAO();    DefaultSessionStorageEvaluator defaultSessionStorageEvaluator = new DefaultSessionStorageEvaluator();    defaultSessionStorageEvaluator.setSessionStorageEnabled(false);    subjectDAO.setSessionStorageEvaluator(defaultSessionStorageEvaluator);    manager.setSubjectDAO(subjectDAO);    return manager;  }  @Bean  public LifecycleBeanPostProcessor lifecycleBeanPostProcessor() {    return new LifecycleBeanPostProcessor();  }  @Bean(name = "TokenRealm")  @DependsOn("lifecycleBeanPostProcessor")  public TokenRealm tokenRealm() {    return new TokenRealm();  }  @Bean  @DependsOn("lifecycleBeanPostProcessor")  public DefaultAdvisorAutoProxyCreator defaultAdvisorAutoProxyCreator() {    DefaultAdvisorAutoProxyCreator defaultAdvisorAutoProxyCreator = new DefaultAdvisorAutoProxyCreator();    // 強制使用cglib,防止重復代理和可能引起代理出錯的問題    // https://zhuanlan.zhihu.com/p/29161098    defaultAdvisorAutoProxyCreator.setProxyTargetClass(true);    return defaultAdvisorAutoProxyCreator;  }  @Bean  public AuthorizationAttributeSourceAdvisor getAuthorizationAttributeSourceAdvisor(SecurityManager securityManager) {    AuthorizationAttributeSourceAdvisor authorizationAttributeSourceAdvisor = new AuthorizationAttributeSourceAdvisor();    authorizationAttributeSourceAdvisor.setSecurityManager(securityManager);    return new AuthorizationAttributeSourceAdvisor();  }}

自定義Shrio filter

執行順序:preHandle -> doFilterInternal -> executeLogin -> onLoginSuccess

主要判斷是不是登錄請求的是 doFilterInternal

public class JWTFilter extends BasicHttpAuthenticationFilter {  /**   * 自定義執行登錄的方法   */  @Override  protected boolean executeLogin(ServletRequest request, ServletResponse response) throws IOException {    HttpServletRequest httpServletRequest = (HttpServletRequest) request;    UsernamePasswordToken usernamePasswordToken = JSON.parseObject(httpServletRequest.getInputStream(), UsernamePasswordToken.class);    // 提交給realm進行登入,如果錯誤他會拋出異常并被捕獲    Subject subject = this.getSubject(request, response);    subject.login(usernamePasswordToken);    return this.onLoginSuccess(usernamePasswordToken, subject, request, response);    //錯誤拋出異常  }  /**   * 最先執行的方法   */  @Override  protected boolean preHandle(ServletRequest request, ServletResponse response) throws Exception {    return super.preHandle(request, response);  }  /**   * 登錄成功后登錄的操作   * 加上jwt 的header   */  @Override  protected boolean onLoginSuccess(AuthenticationToken token, Subject subject, ServletRequest request, ServletResponse response) {    HttpServletResponse httpServletResponse = (HttpServletResponse) response;    String jwtToken = Jwts.builder()        .setId(token.getPrincipal().toString())        .setExpiration(DateTime.now().plusMinutes(30).toDate())        .signWith(SignatureAlgorithm.HS256, JWTCost.signatureKey)        .compact();    httpServletResponse.addHeader(AUTHORIZATION_HEADER, jwtToken);    return true;  }  /**   * 登錄以及校驗的主要流程   * 判斷是否是登錄,或者是登陸后普通的一次請求   */  @Override  public void doFilterInternal(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {    HttpServletRequest httpServletRequest = (HttpServletRequest) servletRequest;    HttpServletResponse httpServletResponse = (HttpServletResponse) servletResponse;    String servletPath = httpServletRequest.getServletPath();    if (StringUtils.equals(servletPath, "/login")) {      //執行登錄      this.executeLogin(servletRequest, servletResponse);    } else {      String authenticationHeader = httpServletRequest.getHeader(AUTHORIZATION_HEADER);      if (StringUtils.isNotEmpty(authenticationHeader)) {        Claims body = Jwts.parser()            .setSigningKey(JWTCost.signatureKey)            .parseClaimsJws(authenticationHeader)            .getBody();        if (body != null) {          //更新token          body.setExpiration(DateTime.now().plusMinutes(30).toDate());          String updateToken = Jwts.builder().setClaims(body).compact();          httpServletResponse.addHeader(AUTHORIZATION_HEADER, updateToken);          //添加用戶憑證          PrincipalCollection principals = new SimplePrincipalCollection(body.getId(), JWTCost.UserNamePasswordRealm);//拼裝shiro用戶信息          WebSubject.Builder builder = new WebSubject.Builder(servletRequest, servletResponse);          builder.principals(principals);          builder.authenticated(true);          builder.sessionCreationEnabled(false);          WebSubject subject = builder.buildWebSubject();          //塞入容器,統一調用          ThreadContext.bind(subject);          filterChain.doFilter(httpServletRequest, httpServletResponse);        }      } else {        httpServletResponse.setStatus(HttpStatus.FORBIDDEN.value());      }    }  }}

登錄失敗處理

處理Shrio異常

@RestControllerAdvicepublic class GlobalControllerExceptionHandler {  @ExceptionHandler(value = Exception.class)  public Object allExceptionHandler(HttpServletRequest request, HttpServletResponse response, Exception exception) {    String message = exception.getCause().getMessage();    LogUtil.error(message);    return new ResultInfo(exception.getClass().getName(), message);  }  /*=========== Shiro 異常攔截==============*/  @ExceptionHandler(value = IncorrectCredentialsException.class)  public String IncorrectCredentialsException(HttpServletRequest request, HttpServletResponse response, Exception exception) {    response.setStatus(HttpStatus.FORBIDDEN.value());    return "IncorrectCredentialsException";  }  @ExceptionHandler(value = UnknownAccountException.class)  public String UnknownAccountException(HttpServletRequest request, HttpServletResponse response, Exception exception) {    response.setStatus(HttpStatus.FORBIDDEN.value());    return "UnknownAccountException";  }  @ExceptionHandler(value = LockedAccountException.class)  public String LockedAccountException(HttpServletRequest request, HttpServletResponse response, Exception exception) {    response.setStatus(HttpStatus.FORBIDDEN.value());    return "LockedAccountException";  }  @ExceptionHandler(value = ExcessiveAttemptsException.class)  public String ExcessiveAttemptsException(HttpServletRequest request, HttpServletResponse response, Exception exception) {    response.setStatus(HttpStatus.FORBIDDEN.value());    return "ExcessiveAttemptsException";  }  @ExceptionHandler(value = AuthenticationException.class)  public String AuthenticationException(HttpServletRequest request, HttpServletResponse response, Exception exception) {    response.setStatus(HttpStatus.FORBIDDEN.value());    return "AuthenticationException";  }  @ExceptionHandler(value = UnauthorizedException.class)  public String UnauthorizedException(HttpServletRequest request, HttpServletResponse response, Exception exception) {    response.setStatus(HttpStatus.FORBIDDEN.value());    return "UnauthorizedException";  }}

處理JWT異常

這是個坑,因為是在filter內發生的異常,@ExceptionHandler是截獲不到的。

/** * 截獲spring boot Error頁面 */@RestControllerpublic class GlobalExceptionHandler implements ErrorController {  @Override  public String getErrorPath() {    return "/error";  }  @RequestMapping(value = "/error")  public Object error(HttpServletRequest request, HttpServletResponse response) throws Exception {    // 錯誤處理邏輯    Exception exception = (Exception) request.getAttribute("javax.servlet.error.exception");    Throwable cause = exception.getCause();    if (cause instanceof ExpiredJwtException) {      response.setStatus(HttpStatus.GATEWAY_TIMEOUT.value());      return new ResultInfo("ExpiredJwtException", cause.getMessage());    }    if (cause instanceof MalformedJwtException) {      response.setStatus(HttpStatus.FORBIDDEN.value());      return new ResultInfo("MalformedJwtException", cause.getMessage());    }    return new ResultInfo(cause.getCause().getMessage(), cause.getMessage());  }}

關于權限等授權信息,可以直接放到Redis中實現緩存。我認為也是不錯的。

源碼奉上:githup-shiro分支 :溫馨提示:平時測試代碼可能比較亂。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
青草青草久热精品视频在线网站| 欧美激情精品久久久久久久变态| 国产精品视频在线观看| 久久精品视频99| 色偷偷88888欧美精品久久久| 国产精品自拍视频| 九九久久久久99精品| 91精品国产综合久久香蕉最新版| 91国在线精品国内播放| 亚洲国产一区二区三区在线观看| 亚洲最大的免费| 欧美丰满少妇xxxx| 久久精品在线播放| 成人免费看吃奶视频网站| 日本亚洲欧美成人| 亚洲精品999| 亚洲精品狠狠操| 黑人精品xxx一区一二区| 综合av色偷偷网| 亚洲高清免费观看高清完整版| 亚洲第一男人av| 国产精品免费看久久久香蕉| 7777kkkk成人观看| 亚洲人午夜色婷婷| 国色天香2019中文字幕在线观看| 精品福利在线观看| 日韩欧美国产激情| 久久精品国产亚洲| 38少妇精品导航| 亚洲欧美激情四射在线日| 中文字幕亚洲欧美日韩高清| 亚洲美女免费精品视频在线观看| 久久免费成人精品视频| 欧美精品在线看| 精品亚洲va在线va天堂资源站| 欧美又大粗又爽又黄大片视频| 国模私拍视频一区| 亚洲大胆人体在线| 日韩电影免费观看中文字幕| 欧美视频在线视频| 欧美性精品220| 亚洲精品久久7777777| 国产成人精品久久二区二区| 97国产在线视频| 精品网站999www| 国产国语刺激对白av不卡| 91色琪琪电影亚洲精品久久| 91成人精品网站| 国产视频精品免费播放| 色偷偷av一区二区三区| 亚洲欧洲一区二区三区久久| 国产精品视频免费观看www| 成人久久久久久久| 日韩美女激情视频| 国产日韩欧美在线| 精品久久久久久久久国产字幕| 国语自产精品视频在线看一大j8| 久久久久一本一区二区青青蜜月| 日韩欧美a级成人黄色| 久久伊人精品视频| 九九久久久久99精品| 久久久视频精品| 久热99视频在线观看| 国产网站欧美日韩免费精品在线观看| 国产欧美久久一区二区| 亚洲欧洲黄色网| 亚洲成**性毛茸茸| 亚洲国产成人精品女人久久久| 日韩一区二区av| 日韩一区二区久久久| 久久精品亚洲国产| 国产亚洲欧洲高清| 久久99久久99精品免观看粉嫩| 日韩av有码在线| 亚洲欧美日韩爽爽影院| 色天天综合狠狠色| 亚洲精品一二区| 欧美日韩不卡合集视频| 一区二区三区国产在线观看| 色久欧美在线视频观看| 色妞欧美日韩在线| 欧美亚洲国产成人精品| 中文字幕一区电影| 国产精品一区专区欧美日韩| 日韩亚洲成人av在线| 中文字幕亚洲一区二区三区五十路| 日韩在线一区二区三区免费视频| 久久久精品2019中文字幕神马| 国产97在线|日韩| 欧美丝袜一区二区三区| 亚洲精品福利资源站| 91美女高潮出水| 久久久综合免费视频| 日韩www在线| 亚洲欧美综合区自拍另类| 国产欧美精品xxxx另类| 日韩在线视频播放| 国产三级精品网站| 亚洲在线观看视频| 欧美精品www在线观看| 欧美激情在线播放| 国产91热爆ts人妖在线| 日韩成人av在线| 国产成人一区二区| 国产精品久久久91| 国产精品男人的天堂| 久久精品小视频| 国产一区二区三区免费视频| 亚洲精品www久久久久久广东| 日韩av电影手机在线| 久久深夜福利免费观看| 国产精品爱啪在线线免费观看| 久久精品视频网站| 欧美午夜激情在线| 浅井舞香一区二区| 亚洲欧美日韩国产中文专区| 国产精品久久91| 国产精品扒开腿爽爽爽视频| 亚洲另类激情图| 成人精品久久久| 91亚洲精品久久久| 57pao成人国产永久免费| 亚洲成人久久久| 亚洲天堂av网| 亚洲欧美在线一区二区| 亚州精品天堂中文字幕| 欧美成人免费va影院高清| 精品成人在线视频| 久久av在线看| 午夜美女久久久久爽久久| 国产精品视频网址| 国产精品亚洲片夜色在线| 亚洲欧美一区二区三区四区| 国内精品久久久久影院 日本资源| 久久综合免费视频影院| 欧美中文在线观看| 成人欧美一区二区三区黑人| 色悠久久久久综合先锋影音下载| 在线观看国产精品日韩av| 欧美做受高潮电影o| 国产精品久久久精品| 久久69精品久久久久久国产越南| 欧美成人免费大片| 欧美激情免费看| 性金发美女69hd大尺寸| www国产亚洲精品久久网站| 国产精品精品国产| 91精品国产乱码久久久久久久久| 亚洲欧美制服中文字幕| 最近2019中文字幕mv免费看| 国产成人精品电影久久久| 51色欧美片视频在线观看| 国模吧一区二区| 91久久在线视频| 久久伊人免费视频| 久久av资源网站| 亚洲国产私拍精品国模在线观看| 欧美亚洲视频在线观看| 成人性生交大片免费看视频直播| www.国产一区| 亚洲欧美日韩中文视频| 亚洲一区二区久久| 91久久久久久国产精品| 日韩国产欧美精品一区二区三区|