日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 前端技术 > javascript >内容正文

javascript

Shiro和SpringBoot简单集成

發布時間:2024/4/17 javascript 77 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Shiro和SpringBoot简单集成 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

Shiro是一種簡單的安全框架,可以用來處理系統的登錄和權限問題。
本篇記錄一下Spring Boot和Shiro集成,并使用Jwt Token進行無狀態登錄的簡單例子。
參考Demo地址,此Demo適合用于SpringBoot小型項目的快速開發。

環境

  • SpringBoot 版本 1.5.15.RELEASE
    不建議使用2.x版本的Springboot,與1.x相比很多地方代碼有所改動,很麻煩。
  • Shiro 版本 1.4.0
  • IntelliJ IDEA
  • jjwt 版本 0.9.0
  • lombok(可選)精簡代碼

思路

  • 使用Jwt Token實現無狀態登錄
    平時用戶登錄后,服務器將會把用戶信息存儲到Session里,在用戶數量很大的時候,服務器負擔會很大。而使用token方式登錄,服務器不存儲用戶信息,而是將其加密后生成token發送給請求方,請求方在請求需要權限的資源時,將token帶上,服務器解析token即可知道登錄用戶的信息。

  • 服務器自動刷新token
    token需要刷新。對于活躍的用戶,服務器自動完成刷新token;對于長期不活躍的用戶,服務器通過配置的 token有效期 來檢查,如果時間超過有效期的兩倍,則認為該用戶需要重新登錄。

  • 登錄流程

    • 用戶通過賬號密碼登錄
      用戶登錄成功后,服務器將用戶信息等集合起來做成Jwt Token(字符串),然后將其放入Response里的header,并發送請求成功的json給請求方。
      請求方接收到請求成功的json信息后,從header中拿出jwt token存儲起來。
    • 用戶請求需要驗證的資源
      請求方將token放入request的header,并發送請求。
      服務器收到請求,檢查request里的token,首先驗證token合法性,不合法返回token不合法的json給請求方。
      如果token合法,則檢查token是否過期:
      如果token簽發時間到現在,已經超過了有效期,卻沒有超過有效期的兩倍,則服務器自動生成新token,將其放入response的header,請求方接收到response后,可以檢查header里是否有token,有則更新一下token預備下次請求。
      如果token從簽發時間到現在,已經超過有效期的兩倍,則用戶需要重新登錄。
  • 集成步驟

    注意
    • @Slf4j(topic = "xxx")注解是lombok集成的日志模塊,可不使用,參考:日志處理方案
    數據庫建表

    思路:
    系統里有多個角色,每個角色對于多個權限。每個權限都是一個請求url,驗證權限時,后臺拿到用戶信息后即可知道該用戶的角色,而后去數據庫查詢該角色所擁有的權限集合,在其中查找是否存在當前請求url,存在說明用戶有訪問該url的權限,否則沒有權限

    -- Sql -- Mysql Version 5.7 -- author 1802226517@qq.comdrop database if exists `rb_demo`; CREATE DATABASE rb_demoDEFAULT CHARACTER SET utf8COLLATE utf8_general_ci; USE rb_demo;-- ------------------------------ 用戶部分 ------------------------------DROP TABLE IF EXISTS `user`; CREATE TABLE `user` (`id` BIGINT UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '主鍵',`account` VARCHAR(50) NOT NULL COMMENT '賬號,唯一',`password` VARCHAR(100) NOT NULL COMMENT '密碼',`name` VARCHAR(100) DEFAULT '默認用戶名' COMMENT '昵稱',`role_id` BIGINT UNSIGNED NOT NULL COMMENT '所屬角色id',`status` TINYINT UNSIGNED NOT NULL COMMENT '是否啟用',`is_deleted` TINYINT UNSIGNED NOT NULL COMMENT '是否刪除',`version` BIGINT UNSIGNED NOT NULL COMMENT '版本',`gmt_create` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP COMMENT '創建時間',`gmt_modified` TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP COMMENT '更新時間',PRIMARY KEY (`id`),KEY `idx_id` (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='用戶表';DROP TABLE IF EXISTS `role`; CREATE TABLE `role` (`id` BIGINT UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '主鍵',`name` VARCHAR(200) NOT NULL COMMENT '角色名稱',`version` BIGINT UNSIGNED NOT NULL COMMENT '版本',PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='角色表';DROP TABLE IF EXISTS `permission`; CREATE TABLE `permission` (`id` BIGINT UNSIGNED NOT NULL AUTO_INCREMENT COMMENT '主鍵',`role_id` BIGINT UNSIGNED NOT NULL COMMENT '所屬角色id',`name` VARCHAR(200) NOT NULL COMMENT '權限名稱',`url` VARCHAR(200) NOT NULL COMMENT '匹配url',`version` BIGINT UNSIGNED NOT NULL COMMENT '版本',PRIMARY KEY (`id`) ) ENGINE=InnoDB DEFAULT CHARSET=utf8 COMMENT='權限表';
    建立Springboot項目

    組件選擇 web、redis和lombok,Springboot版本選擇 1.5.15.RELEASE
    連接數據庫參考:Mybatis-Plus

    編寫Shiro配置類

    ShiroConfig.java 這個配置類主要配置了Shiro攔截器、自定義的Realm和禁用了Session。
    禁用Session方法參考代碼注釋。
    為什么要禁用?因為我們采用Jwt Token方式完成登錄驗證,不需要存用戶信息到Session。

    package com.spz.demo.security.shiro.config;import com.spz.demo.security.shiro.filter.ShiroLoginFilter; import com.spz.demo.security.shiro.matcher.PasswordCredentialsMatcher; import com.spz.demo.security.shiro.realm.UserRealm; import com.spz.demo.security.shiro.token.UserAuthenticationToken; import lombok.extern.slf4j.Slf4j; import org.apache.shiro.mgt.DefaultSessionStorageEvaluator; import org.apache.shiro.mgt.DefaultSubjectDAO; import org.apache.shiro.mgt.SecurityManager; import org.apache.shiro.realm.Realm; import org.apache.shiro.session.mgt.DefaultSessionManager; import org.apache.shiro.spring.web.ShiroFilterFactoryBean; import org.apache.shiro.web.mgt.DefaultWebSecurityManager; import org.apache.shiro.web.mgt.DefaultWebSubjectFactory; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import javax.servlet.Filter; import java.util.*;/*** Shiro 配置* 禁用 Shiro Session 步驟:* 1. SubjectContext 在創建的時候,需要關閉 session 的創建,這個由 DefaultWebSubjectFactory.createSubject 管理。* 參考自定義類:ASubjectFactory.java* 2. 禁用使用 Sessions 作為存儲策略的實現,這個由 securityManager 的 subjectDao.sessionStorageEvaluator 管理* 3. 禁用掉會話調度器,這個由 sessionManager 管理*/ @Slf4j(topic = "SYSTEM_LOG") @Configuration public class ShiroConfig {@Autowiredprivate UserRealm userRealm;/*** Shiro 安全管理器*/@Beanpublic DefaultWebSecurityManager securityManager() {DefaultWebSecurityManager manager = new DefaultWebSecurityManager();// 設置自定義的 SubjectFactorymanager.setSubjectFactory(subjectFactory());// 設置自定義的 SessionManagermanager.setSessionManager(sessionManager());// 禁用 Session((DefaultSessionStorageEvaluator)((DefaultSubjectDAO)manager.getSubjectDAO()).getSessionStorageEvaluator()).setSessionStorageEnabled(false);// 設置自定義的 Realmmanager.setRealms(getRealms());return manager;}/*** 設置過濾規則*/@Beanpublic ShiroFilterFactoryBean shiroFilter(SecurityManager securityManager) {ShiroFilterFactoryBean shiroFilterFactoryBean = new ShiroFilterFactoryBean();shiroFilterFactoryBean.setSecurityManager(securityManager);//自定義攔截器 參考 ShiroLoginFilter.javaMap<String, Filter> filtersMap = new LinkedHashMap<String, Filter>();filtersMap.put("shiroLoginFilter", new ShiroLoginFilter());//登錄驗證攔截器shiroFilterFactoryBean.setFilters(filtersMap);// 所有請求給這個攔截器處理Map<String,String> filterChainDefinitionMap = new LinkedHashMap<String,String>();filterChainDefinitionMap.put("/**", "shiroLoginFilter");shiroFilterFactoryBean.setFilterChainDefinitionMap(filterChainDefinitionMap);shiroFilterFactoryBean.setFilterChainDefinitionMap(filterChainDefinitionMap);return shiroFilterFactoryBean;}/*** 自定義的 subjectFactory* 禁用了 Session* @return*/@Beanpublic DefaultWebSubjectFactory subjectFactory(){ASubjectFactory mySubjectFactory = new ASubjectFactory();return mySubjectFactory;}/*** session管理器* 禁用了 Session* sessionManager通過sessionValidationSchedulerEnabled禁用掉會話調度器,* @return*/@Beanpublic DefaultSessionManager sessionManager(){DefaultSessionManager sessionManager = new DefaultSessionManager();sessionManager.setSessionValidationSchedulerEnabled(false);return sessionManager;}/*** 配置自定義的 Realm* @return*/@Beanpublic Collection<Realm> getRealms(){Collection<Realm> realms = new ArrayList<>();// 配置自定義 UserRealm// 由于UserRealm里使用了自動注入,所以這里需要注入Realm而不是new新建userRealm.setAuthenticationTokenClass(UserAuthenticationToken.class);userRealm.setCredentialsMatcher(new PasswordCredentialsMatcher());//使用自定義的密碼匹配器realms.add(userRealm);return realms;} }

    ASubjectFactory.java 和ShiroConfig配套使用,用于禁用Session。

    package com.spz.demo.security.shiro.config;import org.apache.shiro.authc.AuthenticationToken; import org.apache.shiro.mgt.DefaultSessionStorageEvaluator; import org.apache.shiro.subject.Subject; import org.apache.shiro.subject.SubjectContext; import org.apache.shiro.web.mgt.DefaultWebSubjectFactory;/*** 自定義的 SubjectFactory* 禁用Session* 對于無狀態的TOKEN不創建session 這里都不使用session*/ public class ASubjectFactory extends DefaultWebSubjectFactory {@Overridepublic Subject createSubject(SubjectContext context) {context.setSessionCreationEnabled(Boolean.FALSE);return super.createSubject(context);} }
    編寫自定義Shiro攔截器

    ShiroLoginFilter.java

    • Message類是包裝返回給請求方的類,需要將Message實例轉為json輸出到Response輸出流,參考:[SpringMVC] Web層返回值包裝JSON
    • WebUtil.isPublicRequest()方法判斷請求是否為公共請求
      建議將不需要驗證權限的請求設置一個前綴,比如/public/,這樣,isPublicRequest方法就可以檢查請求url里是否有/public,有則說明是公共請求,直接放行。
    • 所有請求(公共請求除外)都給* onAccessDenied*方法處理
      在onAccessDenied方法里,通過檢查請求url的方式來得知當前請求是什么類型的請求。
      如果是登錄請求,則直接放行,因為登錄邏輯放在了controller層方法。
      如果是其他請求,則需要驗證登錄和權限。
    • 檢查用戶是否具備權限
      將請求url和permission表里的url進行匹配,如果存在匹配,則說明有權限。
    package com.spz.demo.security.shiro.filter;import com.alibaba.fastjson.JSON; import com.alibaba.fastjson.JSONObject; import com.spz.demo.security.bean.Message; import com.spz.demo.security.common.MessageCode; import com.spz.demo.security.common.RequestMappingConst; import com.spz.demo.security.common.WebConst; import com.spz.demo.security.entity.Role; import com.spz.demo.security.exception.custom.RoleException; import com.spz.demo.security.util.CommonUtil; import com.spz.demo.security.util.JwtUtil; import com.spz.demo.security.util.WebUtil; import com.spz.demo.security.vo.JwtToken; import lombok.extern.slf4j.Slf4j; import org.apache.shiro.web.filter.AccessControlFilter; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.context.annotation.Lazy; import org.springframework.stereotype.Component;import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse;/*** 重寫shiro攔截器* 所有請求由此攔截器攔截*/ @Slf4j(topic = "USER_LOG") @Component public class ShiroLoginFilter extends AccessControlFilter {//由于項目啟動時,Shiro加載比其他bean快,所以這里需要加入Lazy注解,在使用時再加載。否則會出現jwtUtil為null的情況@Autowired@Lazyprivate JwtUtil jwtUtil;@Overrideprotected boolean isAccessAllowed(ServletRequest request,ServletResponse response, Object mappedValue) {// 判斷請求是否是公共請求,通過請求的url判斷if(WebUtil.isPublicRequest((HttpServletRequest) request)){return true;}return false;// 拒絕,統一交給 onAccessDenied 處理}@Overrideprotected boolean onAccessDenied(ServletRequest request, ServletResponse response) throws Exception {HttpServletRequest httpServletRequest = (HttpServletRequest)request;HttpServletResponse httpServletResponse = (HttpServletResponse) response;// ========== 判斷是否是登錄請求,是就放行,登錄處理放在了controller層 ==========if(WebUtil.isLoginRequest(httpServletRequest)){return true;}// ========== 其他請求,都需要驗證 ==========//驗證是否登錄(檢查json token)if(CommonUtil.isBlank(httpServletRequest.getHeader(WebConst.TOKEN))){// 返回JSON給請求方WebUtil.writeStringToResponse(httpServletResponse,JSON.toJSONString(new Message().setErrorMessage("[" + WebConst.TOKEN + "] 不能為空,請將token存入header")));return false;}String token = httpServletRequest.getHeader(WebConst.TOKEN);JwtToken jwtToken;try {jwtToken = jwtUtil.parseJwt(token);}catch (RoleException re){//出現異常,說明驗證失敗Message message = new Message();if(re.getMessage().equals(RoleException.MSG_TOKEN_ERROR)){//token錯誤異常message.setMessage(MessageCode.TOKEN_ERROR,RoleException.MSG_TOKEN_ERROR);}else{//token過期異常message.setMessage(MessageCode.TOKEN_OVERDUE,RoleException.MSG_TOKEN_OVERDUE);}WebUtil.writeStringToResponse((HttpServletResponse) response,JSON.toJSONString(message));//返回jsonreturn false;}if(jwtToken.getIsFlushed()){//需要刷新tokenhttpServletResponse.setHeader(WebConst.TOKEN,jwtToken.getToken());// 更新response}// 檢查用戶是否具備權限if(!jwtToken.hasUrl(((HttpServletRequest) request).getRequestURI())){WebUtil.writeStringToResponse((HttpServletResponse) response,JSON.toJSONString(new Message().setPermissionDeniedMessage("沒有權限")));return false;}else{//登錄驗證通過return true;}} }
    編寫自定義的 Realm 類
    • Realm類用來給shiro注入認證信息和授權信息,我們需要自定義。
    • @Value("${jwt.salt}")是從application.yml中讀取配置
    package com.spz.demo.security.shiro.realm;import com.spz.demo.security.common.DatabaseConst; import com.spz.demo.security.entity.User; import com.spz.demo.security.service.UserService; import com.spz.demo.security.shiro.token.UserAuthenticationToken; import lombok.extern.slf4j.Slf4j; import org.apache.shiro.authc.*; import org.apache.shiro.authz.AuthorizationInfo; import org.apache.shiro.realm.AuthorizingRealm; import org.apache.shiro.subject.PrincipalCollection; import org.apache.shiro.util.ByteSource; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Value; import org.springframework.context.annotation.Lazy; import org.springframework.stereotype.Component;@Slf4j(topic = "USER_LOG") @Component("userRealm") public class UserRealm extends AuthorizingRealm{@Autowiredprivate UserService userService;@Value("${jwt.salt}")private String jwtSalt;private static final String DEFAULT_JWT_SALT = "asdfh2738yWsdjDfha";//默認的鹽/*** 授權處理* 不使用* @param principals* @return*/@Overrideprotected AuthorizationInfo doGetAuthorizationInfo(PrincipalCollection principals) {return null;}/*** 身份認證*/@Overrideprotected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken authenticationToken) throws AuthenticationException {// 獲取用戶String account = (String) authenticationToken.getPrincipal();//這里的user里只有賬號和未加密的密碼User user = userService.getUserByAccount(account,DatabaseConst.STATUS_ENABLE,DatabaseConst.IS_DETETED_NO);if (user == null) {return null;}else{//這里這樣做是因為我需要在web層可以拿到userID((UserAuthenticationToken)authenticationToken).setUserId(user.getId());//賦值userId}return new SimpleAuthenticationInfo(user,user.getPassword().toCharArray(),ByteSource.Util.bytes((jwtSalt == null ? DEFAULT_JWT_SALT: jwtSalt)),//鹽getName());} }
    編寫自定義的 Matcher 類
    • AuthenticatingRealm使用CredentialsMatcher進行密碼匹配,我們需要自定義
    package com.spz.demo.security.shiro.matcher;import com.spz.demo.security.entity.User; import com.spz.demo.security.shiro.token.UserAuthenticationToken; import com.spz.demo.security.util.CommonUtil; import org.apache.shiro.authc.AuthenticationInfo; import org.apache.shiro.authc.AuthenticationToken; import org.apache.shiro.authc.credential.CredentialsMatcher;/*** 改寫原有的密碼匹配器* 用于賬號密碼登錄時的賬密匹配*/ public class PasswordCredentialsMatcher implements CredentialsMatcher {@Overridepublic boolean doCredentialsMatch(AuthenticationToken token, AuthenticationInfo info) {//賬號密碼登錄,則token應該是自定義的 AccountPasswordAuthenticationTokenif(token instanceof UserAuthenticationToken){//這里檢查賬號和密碼是否匹配//token是登錄接口那里獲取的,info是通過account獲取到數據里的信息//密碼需要進行md5處理,因為數據庫存儲的密碼為密文if(info.getPrincipals().getPrimaryPrincipal() instanceof User){User user = (User)info.getPrincipals().getPrimaryPrincipal();if(token.getPrincipal().equals(user.getAccount()) &&CommonUtil.md5((String) token.getCredentials()).equals(user.getPassword())){return true;}}}return false;} }
    編寫自定義的AuthenticationToken類
    package com.spz.demo.security.shiro.token;import com.spz.demo.security.entity.User; import lombok.Data; import org.apache.shiro.authc.AuthenticationToken;/*** 用于登錄* 登錄時給此類的account和password(明文)賦值* 然后在UserRealm里將查詢到的userId賦值給此類里的userId。controller層需要id*/ @Data public class UserAuthenticationToken implements AuthenticationToken {private Long userId;//用戶在數據庫中的idprivate String account;private String password;public UserAuthenticationToken(String account, String password){this.account = account;this.password = password;}/*** 返回 account* @return*/@Overridepublic Object getPrincipal() {return this.account;}/*** 返回 password* @return*/@Overridepublic Object getCredentials() {return this.password;} }
    編寫Jwt Token工具類
    package com.spz.demo.security.util;import com.alibaba.fastjson.JSONObject; import com.fasterxml.jackson.databind.ObjectMapper; import com.spz.demo.security.exception.custom.RoleException; import com.spz.demo.security.vo.JwtToken; import io.jsonwebtoken.*; import io.jsonwebtoken.impl.DefaultHeader; import io.jsonwebtoken.impl.DefaultJwsHeader; import io.jsonwebtoken.impl.TextCodec; import io.jsonwebtoken.impl.compression.DefaultCompressionCodecResolver; import io.jsonwebtoken.lang.Assert; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Value; import org.springframework.stereotype.Component; import org.springframework.util.CollectionUtils; import org.springframework.util.StringUtils; import sun.java2d.pipe.AlphaPaintPipe;import javax.swing.event.CaretListener; import javax.xml.bind.DatatypeConverter; import java.io.IOException; import java.util.*;/*** jwt 工具類** @author zp*/ @Slf4j(topic = "SYSTEM_LOG") @Component public class JwtUtil {@Value("${jwt.appKey}")private String appKey;//app key,用于加密@Value("${jwt.period}")private Long period;//token有效時間@Value(("${jwt.issuer}"))private String issuer;//jwt token 簽發人public static final long DEFAULT_PERIOD = 60*60*1000;//token默認有效時間,1小時public static final String DEFAULT_APPKEY = "defaultAppKey";//默認appkey,配置文件里讀不到appKey時用此值public static final String DEFAULT_ISSUER = "Server-System-2333";//默認簽發人private static final ObjectMapper MAPPER = new ObjectMapper();private static CompressionCodecResolver codecResolver = new DefaultCompressionCodecResolver();/*** 簽發 JWT Token Token* @param id 令牌ID* @param subject subject 用戶ID* @param issuer 簽發人,自定義* @param roles 角色* @param permissions 權限集合,建議傳入權限集合的json字符串* @param period 有效時間(ms)* 1. 在 當前時間-簽發時間>有效時間 時攜帶token訪問接口,會重新刷新token* 在 當前時間-簽發時間>有效時間*2 時,則需要重新登錄。* 2. 這樣可以分離長時間不活躍的用戶和活躍用戶* 活躍用戶感受不到token的刷新* 不活躍用戶需要登錄才可以重新獲取token* @param algorithm 加密算法* @return*/public String issueJWT(String id,String subject,String issuer,String roles,String permissions,Long period,SignatureAlgorithm algorithm) {// 需要讀取appKeyif(appKey == null || appKey.equals("")){log.error("appKey無法讀取:" + appKey);appKey = DEFAULT_APPKEY;}byte[] secreKeyBytes = DatatypeConverter.parseBase64Binary(appKey);// 秘鑰JwtBuilder jwtBuilder = Jwts.builder();if (!StringUtils.isEmpty(id)) {jwtBuilder.setId(id);}if (!StringUtils.isEmpty(subject)) {jwtBuilder.setSubject(subject);}if (!StringUtils.isEmpty(issuer)) {jwtBuilder.setIssuer(issuer);}// 設置簽發時間Date now = new Date();jwtBuilder.setIssuedAt(now);// 設置到期時間if (null != period) {jwtBuilder.setExpiration(new Date(now.getTime() + period + period)//簽發時間+有效期*2);}if (!StringUtils.isEmpty(roles)) {jwtBuilder.claim("roles",roles);}if (!StringUtils.isEmpty(permissions)) {jwtBuilder.claim("perms",permissions);}// 壓縮,可選GZIPjwtBuilder.compressWith(CompressionCodecs.DEFLATE);// 加密設置jwtBuilder.signWith(algorithm,secreKeyBytes);return jwtBuilder.compact();}/*** 驗簽JWT** @param jwt json web token* @return 如果驗證通過,且刷新了token,則設置 JwtToken.isFlushed 為true*/public JwtToken parseJwt(String jwt) throws RoleException {if(appKey == null || appKey.equals("")){log.error("appKey無法讀取:" + appKey);appKey = DEFAULT_APPKEY;}// 檢查 jwt token 合法性Claims claims;try{claims = Jwts.parser().setSigningKey(DatatypeConverter.parseBase64Binary(appKey)).parseClaimsJws(jwt).getBody();}catch (ExpiredJwtException ex){//token過期異常 token已經失效需要重新登錄throw new RoleException(RoleException.MSG_TOKEN_OVERDUE);}catch (SignatureException | UnsupportedJwtException | MalformedJwtException | IllegalArgumentException e){//不支持的tokenthrow new RoleException(RoleException.MSG_TOKEN_ERROR);}catch (Exception e){log.error("驗證token時出現未知錯誤: " + CommonUtil.getDetailExceptionMsg(e));throw new RoleException(RoleException.MSG_UNKNOWN_ERROR);}JwtToken jwtToken = new JwtToken();// 檢查是否需要刷新 jwt tokenlong time = claims.getIssuedAt().getTime();//token簽發時間long now = new Date().getTime();//當前時間period = (period == null ? JwtUtil.DEFAULT_PERIOD : period);if(time + period >= now){//還在有效期內,不需要刷新token // log.info("不需要刷新token");jwtToken.setToken(jwt);jwtToken.setIsFlushed(false);}else if(time + period < now &&//超過有效期,但未超過2倍有效期,此時應該刷新tokentime + period + period >= now){ // log.info("刷新token");jwtToken.setToken(issueJWT(// 制作JWT TokenCommonUtil.getRandomString(20),//令牌idclaims.getSubject(),//用戶id(issuer == null ? DEFAULT_ISSUER : issuer),//簽發人claims.get("roles", String.class),//訪問角色,設置為null,不使用claims.get("perms", String.class),//權限集合字符串,jsonperiod,//token有效時間*2SignatureAlgorithm.HS512));jwtToken.setIsFlushed(true);}else{log.error("未知錯誤 - Jwts.parser() 方法未對過期token拋出異常");}// 設置其他字段jwtToken.setId(claims.getSubject());//用戶idjwtToken.setPermissions(JSONObject.parseObject(claims.get("perms", String.class),List.class));//用戶權限集合,json轉為list集合return jwtToken;}/* ** @Description* @Param [val] 從json數據中讀取格式化map* @Return java.util.Map<java.lang.String,java.lang.Object>*/@SuppressWarnings("unchecked")public static Map<String, Object> readValue(String val) {try {return MAPPER.readValue(val, Map.class);} catch (IOException e) {throw new MalformedJwtException("Unable to read JSON value: " + val, e);}} }
    controller登錄驗證
    package com.spz.demo.security.controller;import com.alibaba.fastjson.JSONArray; import com.spz.demo.security.bean.Message; import com.spz.demo.security.common.MessageKeyConst; import com.spz.demo.security.common.RedisConst; import com.spz.demo.security.common.RequestMappingConst; import com.spz.demo.security.common.WebConst; import com.spz.demo.security.entity.Permission; import com.spz.demo.security.entity.User; import com.spz.demo.security.service.UserService; import com.spz.demo.security.shiro.token.UserAuthenticationToken; import com.spz.demo.security.util.CommonUtil; import com.spz.demo.security.util.JwtUtil; import com.spz.demo.security.util.RedisUtil; import com.spz.demo.security.util.WebUtil; import com.spz.demo.security.vo.JwtToken; import io.jsonwebtoken.SignatureAlgorithm; import lombok.extern.slf4j.Slf4j; import org.apache.shiro.SecurityUtils; import org.apache.shiro.subject.Subject; import org.hibernate.validator.constraints.NotEmpty; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.beans.factory.annotation.Value; import org.springframework.validation.annotation.Validated; import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RestController; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.util.ArrayList; import java.util.Date; import java.util.List;@Slf4j(topic = "USER_LOG") @RestController public class UserController {@Value("${jwt.period}")private Long period;//token有效時間(毫秒)@Value(("${jwt.issuer}"))private String issuer;//jwt token 簽發人@Autowiredprivate JwtUtil jwtUtil;@Autowiredprivate UserService userService;/*** 用戶登錄* 驗證碼校驗和請求參數校驗功能已去除,完整版參考Demo* @return*/@PostMapping(value = RequestMappingConst.LOGIN)public Message login(String account,String password,HttpServletRequest request,HttpServletResponse response)throws Exception{// 使用 Shiro 進行登錄Subject subject = SecurityUtils.getSubject();UserAuthenticationToken token = new UserAuthenticationToken(account,password);subject.login(token);// 登錄成功后,獲取userid,查詢該用戶擁有的權限List<String> permissions = userService.getUserPermissions(token.getUserId());// 制作JWT TokenString jwtToken = jwtUtil.issueJWT(CommonUtil.getRandomString(20),//令牌id,必須為整個系統唯一idtoken.getUserId() + "",//用戶id(issuer == null ? JwtUtil.DEFAULT_ISSUER : issuer),//簽發人,可隨便定義null,//訪問角色JSONArray.toJSONString(permissions),//用戶權限集合,json格式(period == null ? JwtUtil.DEFAULT_PERIOD : period),//token有效時間SignatureAlgorithm.HS512//簽名算法,我也不知道是啥來的);//token存入 response里的Headerresponse.setHeader(WebConst.TOKEN,jwtToken);// 返回Message的jsonMessage message = new Message().setSuccessMessage("登錄成功,token已存入header");message.getData().put("account",account);message.getData().put(MessageKeyConst.LOGIN_TIME,new Date().getTime());log.info("用戶登錄成功 ip=" + WebUtil.getIpAdrress(request));return message;} }
    POM文件參考
    <?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"><modelVersion>4.0.0</modelVersion><groupId>com.spz.demo</groupId><artifactId>security</artifactId><version>0.0.1-SNAPSHOT</version><packaging>jar</packaging><name>security</name><description>登錄和權限demo,適用于小項目</description><parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>1.5.15.RELEASE</version><relativePath/> <!-- lookup parent from repository --></parent><properties><project.build.sourceEncoding>UTF-8</project.build.sourceEncoding><project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding><java.version>1.8</java.version><fastjson.version>1.2.38</fastjson.version><mybatisplus.version>2.2.0</mybatisplus.version></properties><dependencies><!--json--><dependency><groupId>com.alibaba</groupId><artifactId>fastjson</artifactId><version>${fastjson.version}</version></dependency><!-- Mybatis Plus --><dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-boot-starter</artifactId><version>${mybatisplus.version}</version></dependency><!-- Mybatis 代碼生成器(模板引擎) --><dependency><groupId>org.apache.velocity</groupId><artifactId>velocity</artifactId><version>1.7</version></dependency><dependency><groupId>org.freemarker</groupId><artifactId>freemarker</artifactId><version>2.3.28</version></dependency><!-- redis --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artifactId></dependency><!-- Kaptcha驗證碼框架 --><dependency><groupId>com.github.axet</groupId><artifactId>kaptcha</artifactId><version>0.0.9</version></dependency><!-- apache --><dependency><groupId>commons-codec</groupId><artifactId>commons-codec</artifactId><version>1.11</version></dependency><!-- json 用于web層包裝請求返回--><dependency><groupId>com.fasterxml.jackson.core</groupId><artifactId>jackson-databind</artifactId><version>2.7.4</version></dependency><!-- lombok 精簡代碼用 --><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><version>1.18.0</version><scope>provided</scope></dependency><!-- Jwt --><dependency><groupId>io.jsonwebtoken</groupId><artifactId>jjwt</artifactId><version>0.9.0</version></dependency><!-- shiro --><dependency><groupId>org.apache.shiro</groupId><artifactId>shiro-spring-boot-starter</artifactId><version>1.4.0</version></dependency><!-- Mysql --><dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><version>5.1.47</version></dependency><!-- AOP --><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency></dependencies><build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId></plugin></plugins></build></project>
    application.yml參考
    spring:# AOP Configaop:auto: trueredis:host: 127.0.0.1password:port: 6379database: 0datasource:url: jdbc:mysql://xxx.xx.xx.xxx:3306/rb_demo?useUnicode=true&characterEncoding=UTF-8username: rootpassword:driver-class-name: com.mysql.jdbc.Driver# Jwt Token相關配置 jwt:appKey: ds[W&dsfa:dfhu12a%W@ // app秘鑰,隨便定義即可appId: 210293ajkw723o@7eh*db //appId,隨便定義即可period: 120000 # 有效期,單位msissuer: Server-System # 簽發者,用于制作 jwt tokensalt: salt-sdwbhx23i # 鹽,隨便定義即可, view UserRealm.doGetAuthenticationInfo()# Mybatis-Plus 配置,請參考官方文檔 mybatis-plus:mapper-locations: classpath:/mapper/*Mapper.xmltypeAliasesPackage: com.spz.demo.security.entityglobal-config:id-type: 2field-strategy: 0db-column-underline: truerefresh-mapper: trueconfiguration:map-underscore-to-camel-case: truecache-enabled: true
    工具類參考
    • 通用工具類
    package com.spz.demo.security.util;import lombok.extern.slf4j.Slf4j; import org.springframework.util.CollectionUtils; import org.springframework.util.StringUtils;import java.security.MessageDigest; import java.util.HashSet; import java.util.Random; import java.util.Set;/* ** @Author tomsun28* @Description 高頻方法工具類* @Date 14:08 2018/3/12*/ @Slf4j(topic = "SYSTEM_LOG") public class CommonUtil {/*** 獲取指定位數的隨機數* @param length* @return*/public static String getRandomString(int length) {String base = "abcdefghijklmnopqrstuvwxyz0123456789";Random random = new Random();StringBuilder sb = new StringBuilder();for (int i = 0; i < length; i++) {int number = random.nextInt(base.length());sb.append(base.charAt(number));}return sb.toString();}/*** MD5加密* @param content* @return*/public static String md5(String content) {// 用于加密的字符char[] md5String = {'0','1','2','3','4','5','6','7','8','9','A','B','C','D','E','F'};try {// 使用平臺默認的字符集將md5String編碼為byte序列,并將結果存儲到一個新的byte數組中byte[] byteInput = content.getBytes();// 信息摘要是安全的單向哈希函數,它接收任意大小的數據,并輸出固定長度的哈希值MessageDigest mdInst = MessageDigest.getInstance("MD5");// MessageDigest對象通過使用update方法處理數據,使用指定的byte數組更新摘要mdInst.update(byteInput);//摘要更新后通過調用digest() 執行哈希計算,獲得密文byte[] md = mdInst.digest();//把密文轉換成16進制的字符串形式int j = md.length;char[] str = new char[j*2];int k = 0;for (int i=0;i<j;i++) {byte byte0 = md[i];str[k++] = md5String[byte0 >>> 4 & 0xf];str[k++] = md5String[byte0 & 0xf];}// 返回加密后的字符串return new String(str);}catch (Exception e) {log.error("加密出現錯誤:" + e.toString());return null;}}/*** 分割字符串進SET*/@SuppressWarnings("unchecked")public static Set<String> split(String str) {Set<String> set = new HashSet<>();if (StringUtils.isEmpty(str))return set;set.addAll(CollectionUtils.arrayToList(str.split(",")));return set;}/*** 檢查字符串是否為空* @param str* @return*/public static boolean isBlank(String str){return (str == null || str.equals("") ? true : false);} }
    • Web請求工具類
    package com.spz.demo.security.util;import com.spz.demo.security.common.RedisConst; import com.spz.demo.security.common.RequestMappingConst; import org.apache.commons.lang.StringUtils;import javax.servlet.ServletRequest; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.io.PrintWriter;public class WebUtil {/*** 檢查url是否需要登錄驗證* @param url* @return false 不需要登錄即可訪問* true 需要登錄才可以訪問*/public static boolean needLogin(String url){if(url.indexOf(RequestMappingConst.V_CODE) >= 0 || //驗證碼url.indexOf(RequestMappingConst.LOGIN) >= 0){//登錄return false;}return true;}/*** 獲取Ip地址* @param request* @return*/public static String getIpAdrress(HttpServletRequest request) {String Xip = request.getHeader("X-Real-IP");String XFor = request.getHeader("X-Forwarded-For");if (StringUtils.isNotEmpty(XFor) && !"unKnown".equalsIgnoreCase(XFor)) {//多次反向代理后會有多個ip值,第一個ip才是真實ipint index = XFor.indexOf(",");if (index != -1) {return XFor.substring(0,index);} else {return XFor;}}XFor = Xip;if (StringUtils.isNotEmpty(XFor) && !"unKnown".equalsIgnoreCase(XFor)) {return XFor;}if (StringUtils.isBlank(XFor) || "unknown".equalsIgnoreCase(XFor)) {XFor = request.getHeader("Proxy-Client-IP");}if (StringUtils.isBlank(XFor) || "unknown".equalsIgnoreCase(XFor)) {XFor = request.getHeader("WL-Proxy-Client-IP");}if (StringUtils.isBlank(XFor) || "unknown".equalsIgnoreCase(XFor)) {XFor = request.getHeader("HTTP_CLIENT_IP");}if (StringUtils.isBlank(XFor) || "unknown".equalsIgnoreCase(XFor)) {XFor = request.getHeader("HTTP_X_FORWARDED_FOR");}if (StringUtils.isBlank(XFor) || "unknown".equalsIgnoreCase(XFor)) {XFor = request.getRemoteAddr();}return XFor;}/*** 檢查請求是否為登錄請求* @param request* @return*/public static boolean isLoginRequest(HttpServletRequest request) {if(request.getRequestURI().indexOf(RequestMappingConst.LOGIN) >= 0){return true;}return false;}/*** 檢查請求是否為注銷請求* @param request* @return*/public static boolean isLogoutRequest(HttpServletRequest request) {if(request.getRequestURI().indexOf(RequestMappingConst.LOGOUT) >= 0){return true;}return false;}/*** 檢查請求是否為公共請求* @param request* @return*/public static boolean isPublicRequest(HttpServletRequest request) {if(request.getRequestURI().indexOf(RequestMappingConst.BASIC_URL_PUBLIC) >= 0){return true;}return false;}/*** 輸出json字符串到 HttpServletResponse* @param response* @param str : 字符串*/public static void writeJSONToResponse(HttpServletResponse response, String str){PrintWriter jsonOut = null;response.setContentType("application/json;charset=UTF-8");try {jsonOut = response.getWriter();jsonOut.write(str);}catch (Exception e){e.printStackTrace();}finally{if(jsonOut != null){jsonOut.close();}}} }

    參考文章

    簽發的用戶認證token超時刷新策略
    shiro實現手機驗證碼登錄
    SpringBoot 集成無狀態的 Shiro

    總結

    以上是生活随笔為你收集整理的Shiro和SpringBoot简单集成的全部內容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

    久久久久在线 | 99久久精品免费看国产四区 | 国产亚洲欧美在线视频 | 国产成人精品在线播放 | 久久久久一区二区三区四区 | 香蕉视频在线观看免费 | 玖玖精品视频 | 国产一区二区三区在线 | 日本中文字幕在线免费观看 | 国产高清无线码2021 | 日韩视频中文字幕 | 久久九九久久九九 | 激情综合狠狠 | 婷婷久久五月天 | 最近免费中文视频 | 国产精品国产三级在线专区 | 国产乱码精品一区二区三区介绍 | 免费三级骚| 亚在线播放中文视频 | 国产精品18久久久久白浆 | 天天操天天干天天玩 | 久久久免费精品视频 | 欧美日韩国产在线观看 | 97精品国产91久久久久久久 | 亚洲精品视频在线观看免费视频 | 国产男女爽爽爽免费视频 | 色婷婷激情电影 | 91免费网| 一本一道久久a久久综合蜜桃 | 久久黄色成人 | av在线免费观看黄 | 中文字幕在线人 | 香蕉视频在线网站 | 成人一区二区三区在线观看 | 日韩一区在线播放 | 中文字幕在线专区 | 五月天久久久久久 | 成人免费在线观看av | 国产精品久久久久久一区二区三区 | 91精选| 国产精品久久久亚洲 | 国产精品18久久久久久久久 | 黄网站a| 国产精品一码二码三码在线 | 国产精品一区二区三区在线免费观看 | 免费看片色 | 97精品视频在线播放 | 天天干天天拍天天操 | 久草男人天堂 | 女人18片毛片90分钟 | 久久精品成人热国产成 | 亚洲精品久久久久中文字幕m男 | 999国产在线 | 欧美片网站yy| 久久免费视频5 | 欧美日韩国产一区二区在线观看 | 国产精品区一区 | 最新国产精品久久精品 | 中文字幕视频免费观看 | 手机成人在线 | 久久久亚洲网站 | 黄色片亚洲 | 久久99久久99精品免观看软件 | 欧美尹人| 久久综合中文字幕 | 精品视频久久 | 波多野结衣在线视频免费观看 | 亚洲黄色在线 | 日韩精品一区二区在线视频 | 国产福利91精品张津瑜 | 免费看的黄色的网站 | 亚洲午夜小视频 | 91成版人在线观看入口 | 欧美日韩三级 | 久久精品视频网 | 国产精品久久久久婷婷二区次 | 亚洲综合精品在线 | 日韩专区在线观看 | 在线黄色免费 | 九九九视频精品 | 天天综合网在线 | 91九色精品女同系列 | 国内精品久久影院 | 成人试看120秒 | 国产精品久久久久久久久久久杏吧 | 国产亚洲视频系列 | 久久免费视频观看 | 欧美精品视 | 亚洲欧美一区二区三区孕妇写真 | 久久婷婷综合激情 | 色网址99| 久久99精品久久久久久三级 | 99视频精品免费视频 | 国产女人18毛片水真多18精品 | 日韩午夜三级 | 国产精品1区2区3区 久久免费视频7 | 久久免费视频这里只有精品 | 成人黄色在线看 | 99久久99久久精品 | 一区二区三区观看 | 波多野结衣精品视频 | 深夜免费小视频 | 亚洲成av人片在线观看 | 日韩免费观看一区二区 | 亚洲欧美在线视频免费 | 国产精品久久久久久欧美 | 国产不卡精品 | 日韩一区在线播放 | 综合激情av | 亚洲精品久久激情国产片 | 国产亚洲一区二区三区 | 国产精品久久久久四虎 | 天天草夜夜 | 成人久久久久久久久久 | 亚洲,国产成人av | 中文字幕在线资源 | 日日夜夜精品视频天天综合网 | 中文在线a天堂 | 久久永久免费 | 久久网站最新地址 | 最新日韩视频在线观看 | 婷婷丁香七月 | 国产高清视频色在线www | 日韩色高清 | 亚洲三级在线免费观看 | 国产生活一级片 | 一区二区观看 | 天天操天天操天天操 | 日韩网| 99tvdz@gmail.com| 99久热在线精品视频 | 国产福利在线 | 国产首页| 日韩一级电影在线 | 中文字幕一区二区三区久久 | 国产日韩亚洲 | 国产又黄又爽又猛视频日本 | 亚洲精品乱码久久久一二三 | 欧美乱熟臀69xxxxxx | 日韩av成人 | 国产综合福利在线 | 天天天天天天天天操 | 黄色亚洲免费 | 婷婷久久五月天 | 精品国产成人av在线免 | 特级西西444www大胆高清无视频 | 在线一二三区 | 国产免费激情久久 | 日韩精品视频在线观看网址 | 国产精品 中文字幕 亚洲 欧美 | 精品在线免费视频 | 国产 中文 日韩 欧美 | 91免费试看 | 国产自在线观看 | 亚洲欧洲国产视频 | 五月婷婷在线视频观看 | 波多野结衣视频在线 | www免费网站在线观看 | 丁香六月欧美 | 色婷婷狠 | 人人爽人人爽人人片av免 | 国产一区二区三精品久久久无广告 | 天天色天天操综合网 | 麻豆视频国产精品 | 一本到在线 | 成人黄色在线观看视频 | 黄色看片 | 97超视频免费观看 | 97人人爽人人 | 国产免费小视频 | 欧美精品久久久久性色 | 97超碰精品 | 国产亚洲精品女人久久久久久 | 亚洲天堂网在线观看视频 | 97超碰在线人人 | 热精品| 国产福利小视频在线 | 视频国产一区二区三区 | 国产资源免费在线观看 | www色片| 97超在线视频 | 六月丁香综合 | 911久久香蕉国产线看观看 | 超级av在线 | 免费看一级特黄a大片 | 在线免费视频 你懂得 | 日韩在线观看视频中文字幕 | 日本中文字幕在线一区 | aav在线 | 亚洲精品国产精品国自产 | 麻豆视频在线观看免费 | 日韩视频在线播放 | 国产专区视频在线观看 | 日韩在线观看视频免费 | 免费色视频网址 | 成人动态视频 | 97在线精品视频 | 免费亚洲电影 | 在线观看久久久久久 | 国产韩国日本高清视频 | 天天弄天天操 | 精品国产一区二区三区男人吃奶 | a电影在线观看 | 国产视频日韩 | 日韩久久精品一区二区三区 | 91精品婷婷国产综合久久蝌蚪 | 视频一区在线免费观看 | 91成人午夜 | 深爱激情五月网 | 色在线最新 | 国产精品毛片一区二区三区 | 亚洲黄网址 | 久草精品视频 | 国产精品观看在线亚洲人成网 | 亚洲专区在线播放 | 国产视频一区二区在线播放 | 最近中文字幕免费视频 | 天天插日日插 | 特级黄色视频毛片 | 色欧美88888久久久久久影院 | 亚洲精品福利在线观看 | 91av蜜桃 | 久久久久久毛片 | 国产美女永久免费 | 成人免费看片98欧美 | 天堂在线视频免费观看 | 国产欧美精品一区二区三区 | 中文字幕资源在线 | 日韩久久精品一区二区三区 | 国产一二三区在线观看 | 人人澡澡人人 | 国产成人在线播放 | 午夜视频在线网站 | 久久久免费精品视频 | 在线免费色 | 在线观看91 | 国产不卡在线看 | 免费看的黄色的网站 | 久久国产露脸精品国产 | 日韩网站一区二区 | 国产中文字幕一区二区 | 天堂网av 在线 | 亚洲高清av在线 | 欧美a级一区二区 | wwxxxx日本| 久久久久久国产精品美女 | 久久免费毛片 | 亚洲激情视频 | 色网站免费在线看 | 西西444www大胆高清视频 | 国产日韩av在线 | 五月激情综合婷婷 | 日韩区欧美久久久无人区 | 欧洲视频一区 | 精品国产一区二区三区久久久蜜臀 | 国产一线天在线观看 | 亚洲综合欧美日韩狠狠色 | 亚洲精品在线免费观看视频 | 探花视频在线观看 | 国产黄色一级片在线 | 久久综合亚洲鲁鲁五月久久 | 国产视频丨精品|在线观看 国产精品久久久久久久久久久久午夜 | 黄在线| 久久超级碰视频 | 欧美日韩国产精品爽爽 | av在线等 | 在线观看免费日韩 | 日韩免费网址 | 日韩av一区二区三区在线观看 | 香蕉视频国产在线观看 | 日日夜夜操操操操 | 久久人人爽人人 | 黄色精品久久久 | 久久久99精品免费观看app | 国产主播大尺度精品福利免费 | 9久久精品 | 最近中文字幕免费视频 | 狠狠狠狠狠狠 | 成年人国产精品 | 国产精品久久网 | 亚洲精品视频免费看 | 中文字幕在线观看第三页 | 欧美 亚洲 另类 激情 另类 | www.香蕉视频| 玖玖视频 | 欧美精品天堂 | 国产精品18毛片一区二区 | 日韩高清在线不卡 | 国产精品热视频 | 久久精品一区二区国产 | 国产在线播放一区二区三区 | 色无五月| 天天操夜夜操 | 狠狠色婷婷丁香六月 | 美女久久久 | 精品国产不卡 | 成人av在线网址 | 欧美精品v国产精品 | 五月花激情 | 狠狠干 狠狠操 | 日韩四虎 | 麻豆国产精品va在线观看不卡 | 国产高清视频在线免费观看 | 国产精品嫩草在线 | 成人在线黄色电影 | 国产一区成人在线 | 99c视频高清免费观看 | 99精品电影 | 久草观看视频 | 天天干天天碰 | 亚洲专区视频在线观看 | 日韩sese | 亚洲午夜久久久影院 | 午夜av网站 | 偷拍精偷拍精品欧洲亚洲网站 | 免费日韩av电影 | 国产高清在线免费 | 亚洲高清视频在线 | 亚洲精品在线免费 | 99热手机在线观看 | 国产精华国产精品 | 国产一级在线看 | 日韩精品高清不卡 | www.五月婷婷 | av网站在线观看播放 | 天天色欧美 | 在线观看国产中文字幕 | 亚洲精品动漫久久久久 | 天堂av在线免费 | 97视频中文字幕 | 亚洲永久国产精品 | 久久精品xxx| 在线之家免费在线观看电影 | 日韩成人精品在线观看 | 国产一区二区视频在线 | 久久久麻豆视频 | 国产精品综合久久久 | 91精品国产91热久久久做人人 | 久久久久女人精品毛片九一 | 天天摸天天操天天舔 | 成人欧美亚洲 | 亚洲黄a| 手机看片1042| 亚洲一区二区精品视频 | 999精品在线| 久久久国产精品人人片99精片欧美一 | 欧美污网站 | 欧美性色网站 | 婷婷精品国产一区二区三区日韩 | 成人99免费视频 | 国产精品日韩久久久久 | 亚洲免费精品视频 | 亚洲国产精品va在线看 | 日本成人黄色片 | 奇米777777 | 精品久久免费 | 综合激情网 | 成人a在线观看高清电影 | 午夜在线免费视频 | 91精品久久久久久久久久久久久 | 国产成人亚洲精品自产在线 | 亚洲国产视频网站 | 一区二区精品在线观看 | 久草视频手机在线 | 亚一亚二国产专区 | 综合色婷婷 | 日韩久久电影 | 天天操网址| 久久久亚洲麻豆日韩精品一区三区 | 日本69hd| 精品国产一区二区三区久久影院 | 国产高清在线永久 | 国内视频1区 | 国产精品一区二区 91 | 国产中文字幕视频在线 | 日韩精品一区二区三区免费观看 | 91av中文字幕 | 欧美日韩成人 | 国产中文字幕一区二区 | 久草五月 | 91毛片在线观看 | 99久久精品国 | 欧美 亚洲 另类 激情 另类 | 国产精品丝袜久久久久久久不卡 | 片黄色毛片黄色毛片 | 国产精品s色| 日本性生活免费看 | 久久99深爱久久99精品 | 手机在线观看国产精品 | 日韩在线欧美在线 | 日日爱影视 | 在线小视频你懂的 | 97影视 | 91麻豆免费版 | 亚洲国产经典视频 | 久久精品牌麻豆国产大山 | 亚洲va韩国va欧美va精四季 | 99亚洲精品视频 | 中文字幕国语官网在线视频 | 夜夜看av | 黄色99视频| 精品亚洲免费 | 国产一区二区在线精品 | 日本aaaa级毛片在线看 | 久久久精品视频成人 | 成人免费视频观看 | 99久久99久久免费精品蜜臀 | 国产精品一区二区三区四区在线观看 | 国产中文字幕视频在线 | 亚洲精品国产视频 | 婷婷99 | 99九九99九九九视频精品 | 日韩美女久久 | 日韩两性视频 | 亚洲一区视频在线播放 | 国产毛片久久久 | 免费试看一区 | 久久久国产毛片 | 国产精品成人一区二区 | 97在线观看免费视频 | 午夜av大片 | 高清国产午夜精品久久久久久 | 色国产精品一区在线观看 | 久久综合婷婷综合 | 午夜精品久久久久久久99水蜜桃 | 中文字幕在线观看第一页 | 久久久五月婷婷 | 99热都是精品| 在线看片中文字幕 | 亚洲涩涩网 | 中文亚洲欧美日韩 | 韩国一区二区三区视频 | 黄色国产在线 | 欧美日韩3p| 女人18毛片90分钟 | 国产999久久久 | 色wwwww| 欧美国产精品久久久久久免费 | 欧美激情视频一二三区 | 国产一区二区在线观看免费 | 国产色婷婷在线 | 久久国产成人午夜av影院宅 | 五月婷婷激情六月 | 久久精品中文 | 久久99国产精品免费 | 激情五月婷婷综合 | 四虎影视8848aamm | 国产黄a三级三级三级三级三级 | 综合网色 | www.xxx.性狂虐 | 又黄又爽的视频在线观看网站 | 久久最新视频 | 色偷偷网站视频 | 狠狠色丁香久久婷婷综合五月 | 午夜视频在线观看一区 | 午夜久久福利视频 | 成人免费在线观看入口 | 国产视频在线观看一区 | 日本在线观看一区二区 | 波多野结衣小视频 | 日韩在线免费电影 | 夜夜夜影院| 成人一区二区在线 | 成人在线观看av | 久久久久久久久福利 | 免费网站看av片 | 免费看国产视频 | 日本中文一区二区 | 99re国产| 色久av | 一级片免费观看视频 | av电影中文| 中文字幕在线观看免费高清电影 | 久久精品一区二区 | 精品国产电影一区 | 五月婷婷天堂 | 天天操操操操操操 | 日本xxxx.com| 波多野结衣网址 | 在线免费观看国产 | 日韩在线免费看 | 久久久在线| 日日夜夜天天久久 | 亚洲国产资源 | 日本一区二区三区免费观看 | 免费a视频在线观看 | 狠狠地日 | 亚洲视频免费在线观看 | 久久久国际精品 | 亚洲国产经典视频 | av高清免费在线 | 韩国av在线播放 | 久久成人国产精品免费软件 | 欧美国产精品一区二区 | 国产一线二线三线在线观看 | 夜夜躁日日躁狠狠久久88av | 亚洲精品一区中文字幕乱码 | 亚洲精品中文字幕在线 | 操操操人人 | 五月天综合色 | 97精品在线 | 久久不卡国产精品一区二区 | 欧美在线一二 | 在线免费观看一区二区三区 | 亚洲第一区精品 | av在线播放中文字幕 | 国产麻豆精品一区二区 | 亚洲精选在线 | 毛片区 | 国产在线观看高清视频 | 精品爱爱 | 免费视频二区 | 干av在线 | 久久久久久久免费 | 97狠狠干| 色综合久久久久久久久五月 | 国产精品成人国产乱一区 | 亚洲综合欧美精品电影 | 黄色一区二区在线观看 | 99精品免费久久久久久久久日本 | 最近中文字幕免费观看 | 色com网| 97天堂网 | 国产91学生| 国产69精品久久99不卡的观看体验 | 观看免费av | 亚洲一级影院 | 国产在线播放不卡 | 亚洲在线视频免费观看 | 最新av电影网址 | 日韩在线看片 | 精品国产一区二区三区四 | 日本动漫做毛片一区二区 | 日本在线观看一区二区三区 | 爱爱av网 | 欧美一区二区在线免费观看 | www.一区二区三区 | 日韩黄在线观看 | 日日骑| 色吊丝在线永久观看最新版本 | 天天干.com| 亚洲精品久久久久久久不卡四虎 | 黄色成人影视 | 日本在线观看一区二区三区 | 国产成人一区二区啪在线观看 | 九色自拍视频 | 人人干网 | 69国产盗摄一区二区三区五区 | 亚洲伊人婷婷 | 99r精品视频在线观看 | 91丨九色丨高潮丰满 | 免费网站污 | 中文字幕二区在线观看 | 国产高清在线a视频大全 | 天天色天天综合 | a级片久久| 玖草在线观看 | 国产精品一区二区久久 | 婷婷六月丁 | 在线观看韩日电影免费 | 五月天久久婷 | 久久久久久久99精品免费观看 | 亚洲精品久久久蜜桃 | 99久久精品国产欧美主题曲 | 婷婷精品在线视频 | 97视频在线观看播放 | 天天操天天草 | 婷婷在线综合 | 国产高清无线码2021 | 天天射天天 | 日韩欧美高清不卡 | 日韩av图片 | 成年美女黄网站色大片免费看 | 国产成人一区二区三区电影 | 国产精品18毛片一区二区 | 激情视频久久 | 国产99久久精品一区二区300 | 在线视频免费观看 | 天天操夜夜拍 | 国产精品一区二区三区久久 | 日本中文字幕在线免费观看 | 99久久夜色精品国产亚洲96 | 日韩中文字幕国产精品 | 天天操天天添 | 久久系列 | 国产高清精 | 国产一性一爱一乱一交 | 波多野结衣电影一区二区 | a级片韩国 | 91高清视频在线 | 亚洲91视频 | 三级av免费 | 亚洲黄色小说网 | 国产最新在线观看 | 天天操天天射天天舔 | 成年人精品 | 九九色视频 | 一级免费看 | 一区二区三区免费在线播放 | 特级毛片爽www免费版 | 97视频免费在线看 | 美女搞黄国产视频网站 | 天天弄天天干 | 国产一区二区在线免费播放 | 福利视频 | 91一区啪爱嗯打偷拍欧美 | 午夜成人免费电影 | 欧美久久久久久久久久 | 你操综合 | 91中文字幕在线观看 | 免费又黄又爽视频 | 一区二区中文字幕在线 | 日韩精品欧美专区 | 婷婷综合五月天 | 91av视屏 | 婷婷国产在线观看 | 在线观看资源 | 9999精品| 干天天 | 日黄网站 | 在线视频观看国产 | 国产高清免费av | 麻豆视频国产精品 | 国产一区欧美二区 | 一区二区三区免费播放 | 国外成人在线视频网站 | 国产特级毛片aaaaaa高清 | 91亚洲精品视频 | 成人97视频一区二区 | 亚洲国产三级在线 | 欧美日韩中文国产一区发布 | 久久视频精品在线观看 | 色婷婷丁香 | 麻豆精品在线 | 91精品国产欧美一区二区 | 国产精品女人网站 | 免费h漫在线观看 | 婷婷六月丁香激情 | 99久久婷婷 | 欧美激情综合五月色丁香 | 奇米网777| 99色99| 国产精品人成电影在线观看 | 久久免费一级片 | 亚洲午夜精品久久久久久久久 | a级黄色片视频 | 国产精品1000 | 摸bbb搡bbb搡bbbb | 色99视频 | 国产成人一区二区三区久久精品 | 91丨九色丨蝌蚪丨对白 | 久草在线费播放视频 | 国产 日韩 中文字幕 | 韩日精品在线观看 | 国产一区网址 | 久久不卡国产精品一区二区 | 精品久久久久久久久久 | 在线看成人 | 黄色一区二区在线观看 | 亚洲aⅴ乱码精品成人区 | 在线观看aaa| 日韩有码专区 | 国产精品免费观看视频 | 国产精品白浆视频 | 国产国语在线 | 精品免费99久久 | 中文字幕日韩一区二区三区不卡 | 久久伊人色综合 | 国产黄色av影视 | 日韩在线第一区 | 国产1区2区3区精品美女 | 日韩亚洲国产中文字幕 | 黄色av一区二区三区 | 国产一区精品在线观看 | 伊人色**天天综合婷婷 | 亚洲激情在线 | 色在线网 | 国产在线不卡视频 | 免费亚洲黄色 | 日韩精品中文字幕一区二区 | 玖草在线观看 | 国产婷婷在线观看 | 麻豆一区在线观看 | www.99av| 免费视频久久久久 | 成人av资源在线 | 人人插人人艹 | 国产一级二级在线观看 | 日本中文字幕在线视频 | 在线精品视频免费播放 | 亚洲精品乱码久久久久久蜜桃动漫 | 中文字幕av全部资源www中文字幕在线观看 | 天天操夜夜操天天射 | 成人av资源网站 | 亚洲欧美成人网 | 中文字幕一区二区三区久久 | 成人免费在线看片 | 亚洲 欧美日韩 国产 中文 | 亚洲成人网av | 国产精品手机视频 | 91中文字幕在线播放 | 国产成人精品久久久久 | 亚洲狠狠操 | 亚洲综合一区二区精品导航 | 国产精品 中文字幕 亚洲 欧美 | 激情av网 | 国产精品一区二区三区四区在线观看 | 最新国产精品拍自在线播放 | 免费成人在线观看视频 | 91免费看片黄 | 干干夜夜 | 国产精品第一页在线观看 | 成人av电影网址 | 人人添人人澡人人澡人人人爽 | av黄色av| 高清av影院 | 久久久精品久久日韩一区综合 | 久久久久久久久久久精 | 蜜桃视频在线视频 | 91亚洲精品乱码久久久久久蜜桃 | 处女av在线 | 成人免费视频网站在线观看 | 有码视频在线观看 | 欧美福利久久 | 日韩精品免费 | av黄色在线播放 | 欧美一区免费在线观看 | 国产男女无遮挡猛进猛出在线观看 | 五月婷婷香蕉 | 91久久电影 | 久久成人精品视频 | 成年人黄色免费网站 | 中文字幕在线观看完整版 | 天天翘av| 国产日韩欧美视频 | 免费黄色特级片 | 国产精品精品久久久久久 | 国内精品在线一区 | 精品一区二区三区香蕉蜜桃 | 波多野结衣在线播放视频 | 五月婷婷狠狠 | 亚洲精品播放 | 99久国产 | 91亚洲狠狠婷婷综合久久久 | 午夜在线观看一区 | 四虎成人免费观看 | 亚洲成a人片在线www | 成人av片免费看 | 在线观看欧美成人 | 综合久久精品 | 久久久色 | 午夜色大片在线观看 | 天堂va欧美va亚洲va老司机 | 国产香蕉久久精品综合网 | 一区精品久久 | 亚洲综合导航 | 国产九色在线播放九色 | 亚洲第一伊人 | 国产精品网站一区二区三区 | 香蕉在线观看视频 | 亚洲免费av在线播放 | 天天做日日做天天爽视频免费 | 日韩黄色大片在线观看 | 亚洲狠狠操 | 91亚色在线观看 | 欧美最新大片在线看 | 国产精品自产拍在线观看网站 | 五月天综合色激情 | 日韩a在线播放 | 在线观看中文字幕一区 | 婷婷精品国产欧美精品亚洲人人爽 | 日韩精品一区二区三区水蜜桃 | 精品国产一二区 | 超碰免费久久 | 激情欧美xxxx | 日韩伦理一区二区三区av在线 | 午夜12点| 视频在线在亚洲 | 91精品国产乱码久久 | a黄色片在线观看 | 日韩高清一 | 国产高清无线码2021 | 一区二区三区韩国免费中文网站 | 国产免费区| 五月婷婷丁香在线观看 | 亚洲视频综合在线 | 91香蕉视频720p| 国产精品96久久久久久吹潮 | 91久久精 | 成年人电影免费在线观看 | 日韩视频在线播放 | 国产成人精品亚洲 | wwxxxx日本 | 91最新在线观看 | 国产盗摄精品一区二区 | 久久狠狠婷婷 | www91在线| 超碰日韩 | av电影 一区二区 | www.人人草 | 久久人人爽人人片av | 日韩电影中文 | 日日精品 | 久草在线免费看视频 | 在线综合 亚洲 欧美在线视频 | 看片网站黄 | 国产在线自 | 四虎国产精品成人免费影视 | 五月婷婷欧美视频 | 视频国产在线观看18 | 黄色av电影在线观看 | 不卡的av中文字幕 | 欧美aa一级 | 国产精品h在线观看 | 2021久久| 国产精品自在线 | 国产精品99久久久 | 国产97在线观看 | 精品免费久久久久久 | 久久精品国产一区二区电影 | 国产精品18久久久久白浆 | 又黄又刺激的视频 | 成人黄色大片 | 一级α片 | 国产不卡av在线播放 | 亚洲精品一区二区18漫画 | 97在线观看免费视频 | 国产成人久久精品一区二区三区 | 国产黄色一级片在线 | 免费视频黄 | 久久免费国产精品 | 国产精品日韩欧美一区二区 | 国产成人在线一区 | 免费黄色在线 | 日日操网 | .国产精品成人自产拍在线观看6 | 91在线影视| 日韩免费| 最近中文字幕国语免费高清6 | 久久久久| 欧美久久久一区二区三区 | www.久久色.com| 伊人超碰在线 | www国产亚洲 | 五月花激情 | 91精品福利在线 | 97超碰在线久草超碰在线观看 | 久久国产系列 | 婷婷视频在线 | 国产精品福利小视频 | 国产精品久久久久久久久久久免费看 | 伊人天天狠天天添日日拍 | 91精品久久久久久久99蜜桃 | 在线观看av小说 | 九九久久久久99精品 | 日韩三级免费 | 国产高h视频 | 色在线中文字幕 | 日日噜噜噜噜夜夜爽亚洲精品 | 中文乱码视频在线观看 | 四虎亚洲精品 | 国产精品黄色在线观看 | 国模精品在线 | 国产手机在线观看 | 中文字幕视频播放 | 中文字幕视频网 | 99色婷婷| 欧美极品在线播放 | 久久久久久片 | 久久99操 | 日本精品免费看 | 九九日韩 | 91看片淫黄大片一级在线观看 | 精品免费在线视频 | 99视频网址 | 国产精品久久久久久久婷婷 | 国产午夜麻豆影院在线观看 | 91精品国| 91av免费看 | 麻豆视频免费在线观看 | 国产精品免费麻豆入口 | 18岁免费看片 | 久久在线观看视频 | 最新日韩在线观看视频 | 不卡的av电影在线观看 | 日韩欧美视频免费看 | 欧美一级片在线 | 国产精品久久久久久久久岛 | 激情视频免费观看 | 97操碰| 天天拍天天操 | 91九色蝌蚪视频在线 | 国产一区久久久 | 日韩特级片| 不卡电影一区二区三区 | 国产直播av | 97超碰人 | 国产视频首页 | 911国产精品 | 成人av动漫在线观看 | 国产一级免费在线观看 | 欧美激情精品一区 | 狠狠五月婷婷 | 色操插 | 婷婷色网视频在线播放 | 一区二区三区在线观看免费 | 国产精品欧美久久久久久 | 激情综合啪啪 | 国产视频一区在线 | 天天精品视频 | 亚洲国产成人精品电影在线观看 | 手机看片国产日韩 | 成人午夜精品久久久久久久3d | 久久精品一区二区三区四区 | 我要看黄色一级片 | 99精品在线观看视频 | 国产精品原创在线 | 亚洲精品美女在线观看播放 | 超碰在线人人艹 | 91精品国产自产老师啪 | 日韩中文字幕亚洲一区二区va在线 | 久久综合久久伊人 | 91亚瑟视频| 欧美日韩精品二区第二页 | 91在线91| 综合色综合色 | 欧美一二三区播放 | www黄com| 色婷婷啪啪免费在线电影观看 | 中文字幕日韩免费视频 | 一区二区三区四区五区在线 | 91精品久久久久久久99蜜桃 | 欧美久久电影 | 夜夜澡人模人人添人人看 | 亚洲中字幕 | 欧美日韩3p | 超碰人人在| 国产精品久久久久久久久搜平片 | 超碰在线天天 | 色婷婷激情 | 日韩国产精品久久久久久亚洲 | 97国产在线| 久久午夜免费观看 | 国产午夜麻豆影院在线观看 | 人人玩人人爽 | 久久免费一 | 国产精品美 | 日韩在线观看高清 | 色综合色综合久久综合频道88 | 一色av| 亚洲国产成人精品在线观看 | 麻豆高清免费国产一区 | 99热99| 精品久久毛片 | 国产精品婷婷 | 久久久久久久久久久久久久电影 | 日本美女xx | 99精品国产视频 | av日韩不卡 | 亚洲成人av一区二区 | 欧美精品一区二区三区四区在线 | 日韩欧美视频在线观看免费 | 最近免费在线观看 | 青青河边草免费视频 | 在线看国产视频 | 精品国产一区二 | 色99色| av在线之家电影网站 | 日韩欧美精品在线 | 西西大胆免费视频 | 国产中文字幕在线 | 亚洲国产中文字幕在线视频综合 | 久久国语露脸国产精品电影 | 色综合久久久久综合 | 亚洲黄色片 | 丁香 婷婷 激情 | 国产97碰免费视频 | 日韩专区在线 | 久久久久久久免费看 | 中文字幕综合在线 | 在线观看日韩精品 | 日本视频不卡 | 欧美成人在线免费观看 | 不卡的av在线 | 麻豆久久久久 | 四虎最新域名 | 国产精品男女啪啪 | 一级黄毛片 | 99视频一区 | 亚洲永久在线 | 三级av片 | 超碰免费在线公开 | 精品国产大片 | 最新婷婷色 | 日韩精品免费一线在线观看 | 久久99影院 | 日日干夜夜干 | 欧美日韩免费在线视频 |