久久福利_99r_国产日韩在线视频_直接看av的网站_中文欧美日韩_久久一

您的位置:首頁技術文章
文章詳情頁

關于Spring Boot動態權限變更問題的實現方案

瀏覽:2日期:2023-07-05 09:24:02
1、前言

​  在Web項目中,權限管理即權限訪問控制為網站訪問安全提供了保障,并且很多項目使用了Session作為緩存,結合AOP技術進行token認證和權限控制。權限控制流程大致如下圖所示:

關于Spring Boot動態權限變更問題的實現方案

​  現在,如果管理員修改了用戶的角色,或修改了角色的權限,都會導致用戶權限發生變化,此時如何實現動態權限變更,使得前端能夠更新用戶的權限樹,后端訪問鑒權AOP模塊能夠知悉這種變更呢?

2、問題及解決方案

​​  現在的問題是,管理員沒法訪問用戶Session,因此沒法將變更通知此用戶。而用戶如果已經登錄,或直接關閉瀏覽器頁面而不是登出操作,Session沒有過期前,用戶訪問接口時,訪問鑒權AOP模塊仍然是根據之前緩存的Session信息進行處理,沒法做到動態權限變更。

​​  使用Security+WebSocket是一個方案,但沒法處理不在線用戶。

​​  ​解決方案的核心思想是利用ServletContext對象的共享特性,來實現用戶權限變更的信息傳遞。然后在AOP類中查詢用戶是否有變更通知記錄需要處理,如果權限發生變化,則修改response消息體,添加附加通知信息給前端。前端收到附加的通知信息,可更新功能權限樹,并進行相關處理。

​​​  這樣,利用的變更通知服務,不僅后端的用戶url訪問接口可第一時間獲悉變更,還可以通知到前端,從而實現了動態權限變更。

3、方案實現3.1、開發變更通知類

​​​  服務接口類ChangeNotifyService,代碼如下:

package com.abc.questInvest.service;/** * @className: ChangeNotifyService * @description: 變更通知服務 * @summary: * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/281.0.0sheng.zheng初版 * */public interface ChangeNotifyService {/** * * @methodName: getChangeNotifyInfo * @description: 獲取指定用戶ID的變更通知信息 * @param userId: 用戶ID * @return: 返回0表示無變更通知信息,其它值按照bitmap編碼。目前定義如下: * bit0:: 修改用戶的角色組合值,從而導致權限變更; * bit1:: 修改角色的功能項,從而導致權限變更; * bit2:: 用戶禁用,從而導致權限變更; * bit3:: 用戶調整部門,從而導致數據權限變更; * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/281.0.0sheng.zheng初版 * */public Integer getChangeNotifyInfo(Integer userId);/** * * @methodName: setChangeNotifyInfo * @description: 設置變更通知信息 * @param userId: 用戶ID * @param changeNotifyInfo: 變更通知值 * bit0:: 修改用戶的角色組合值,從而導致權限變更; * bit1:: 修改角色的功能項,從而導致權限變更; * bit2:: 用戶禁用,從而導致權限變更; * bit3:: 用戶調整部門,從而導致數據權限變更; * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/281.0.0sheng.zheng初版 * */public void setChangeNotifyInfo(Integer userId,Integer changeNotifyInfo); }

​​​  服務實現類ChangeNotifyServiceImpl,代碼如下:

package com.abc.questInvest.service.impl;import java.util.HashMap;import java.util.Map;import org.springframework.stereotype.Service;import com.abc.questInvest.service.ChangeNotifyService;/** * @className: ChangeNotifyServiceImpl * @description: ChangeNotifyService實現類 * @summary: * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/281.0.0sheng.zheng初版 * */@Servicepublic class ChangeNotifyServiceImpl implements ChangeNotifyService {//用戶ID與變更過通知信息映射表private Map<Integer,Integer> changeNotifyMap = new HashMap<Integer,Integer>();/** * * @methodName: getChangeNotifyInfo * @description: 獲取指定用戶ID的變更通知信息 * @param userId: 用戶ID * @return: 返回0表示無變更通知信息,其它值按照bitmap編碼。目前定義如下: * bit0:: 修改用戶的角色組合值,從而導致權限變更; * bit1:: 修改角色的功能項,從而導致權限變更; * bit2:: 用戶禁用,從而導致權限變更; * bit3:: 用戶調整部門,從而導致數據權限變更; * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/281.0.0sheng.zheng初版 * */@Overridepublic Integer getChangeNotifyInfo(Integer userId) {Integer changeNotifyInfo = 0;//檢查該用戶是否有變更通知信息if (changeNotifyMap.containsKey(userId)) {changeNotifyInfo = changeNotifyMap.get(userId);//移除數據,加鎖保護synchronized(changeNotifyMap) {changeNotifyMap.remove(userId);}}return changeNotifyInfo;}/** * * @methodName: setChangeNotifyInfo * @description: 設置變更通知信息,該功能一般由管理員觸發調用 * @param userId: 用戶ID * @param changeNotifyInfo: 變更通知值 * bit0:: 修改用戶的角色組合值,從而導致權限變更; * bit1:: 修改角色的功能項,從而導致權限變更; * bit2:: 用戶禁用,從而導致權限變更; * bit3:: 用戶調整部門,從而導致數據權限變更; * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/281.0.0sheng.zheng初版 * */@Overridepublic void setChangeNotifyInfo(Integer userId,Integer changeNotifyInfo) {//檢查該用戶是否有變更通知信息if (changeNotifyMap.containsKey(userId)) {//如果有,表示之前變更通知未處理//獲取之前的值Integer oldChangeNotifyInfo = changeNotifyMap.get(userId);//計算新值。bitmap編碼,或操作Integer newChangeNotifyInfo = oldChangeNotifyInfo | changeNotifyInfo;//設置數據,加鎖保護synchronized(changeNotifyMap) {changeNotifyMap.put(userId,newChangeNotifyInfo);}}else {//如果沒有,設置一條changeNotifyMap.put(userId,changeNotifyInfo);}}}

​​  此處,變更通知類型,與使用的demo項目有關,目前定義了4種變更通知類型。實際上,除了權限相關的變更,還有與Session緩存字段相關的變更,也需要通知,否則用戶還是在使用舊數據。

3.2、將變更通知類對象,納入全局配置服務對象中進行管理

​​​  全局配置服務類GlobalConfigService,負責管理全局的配置服務對象,服務接口類代碼如下:

package com.abc.questInvest.service;/** * @className: GlobalConfigService * @description: 全局變量管理類 * @summary: * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/021.0.0sheng.zheng初版 * */public interface GlobalConfigService {/** * * @methodName: loadData * @description: 加載數據 * @return: 成功返回true,否則返回false * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/021.0.0sheng.zheng初版 * */public boolean loadData();//獲取TableCodeConfigService對象public TableCodeConfigService getTableCodeConfigService();//獲取SysParameterService對象public SysParameterService getSysParameterService();//獲取FunctionTreeService對象public FunctionTreeService getFunctionTreeService();//獲取RoleFuncRightsService對象public RoleFuncRightsService getRoleFuncRightsService();//獲取ChangeNotifyService對象public ChangeNotifyService getChangeNotifyService();}

​​​  服務實現類GlobalConfigServiceImpl,代碼如下:

package com.abc.questInvest.service.impl;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.stereotype.Service;import com.abc.questInvest.service.ChangeNotifyService;import com.abc.questInvest.service.FunctionTreeService;import com.abc.questInvest.service.GlobalConfigService;import com.abc.questInvest.service.RoleFuncRightsService;import com.abc.questInvest.service.SysParameterService;import com.abc.questInvest.service.TableCodeConfigService;/** * @className: GlobalConfigServiceImpl * @description: GlobalConfigService實現類 * @summary: * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/021.0.0sheng.zheng初版 * */@Servicepublic class GlobalConfigServiceImpl implements GlobalConfigService{//ID編碼配置表數據服務@Autowiredprivate TableCodeConfigService tableCodeConfigService;//系統參數表數據服務@Autowiredprivate SysParameterService sysParameterService;//功能樹表數據服務@Autowiredprivate FunctionTreeService functionTreeService;//角色權限表數據服務@Autowiredprivate RoleFuncRightsService roleFuncRightsService;//變更通知服務@Autowiredprivate ChangeNotifyService changeNotifyService;/** * * @methodName: loadData * @description: 加載數據 * @return: 成功返回true,否則返回false * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/021.0.0sheng.zheng初版 * */@Overridepublic boolean loadData() {boolean bRet = false;//加載table_code_config表記錄bRet = tableCodeConfigService.loadData();if (!bRet) {return bRet;}//加載sys_parameters表記錄bRet = sysParameterService.loadData();if (!bRet) {return bRet;}//changeNotifyService目前沒有持久層,無需加載//如果服務重啟,信息丟失,也沒關系,因為此時Session也會失效//加載function_tree表記錄bRet = functionTreeService.loadData();if (!bRet) {return bRet;}//加載role_func_rights表記錄//先設置完整功能樹roleFuncRightsService.setFunctionTree(functionTreeService.getFunctionTree());//然后加載數據bRet = roleFuncRightsService.loadData();if (!bRet) {return bRet;}return bRet;}//獲取TableCodeConfigService對象@Overridepublic TableCodeConfigService getTableCodeConfigService() {return tableCodeConfigService;}//獲取SysParameterService對象@Overridepublic SysParameterService getSysParameterService() {return sysParameterService;}//獲取FunctionTreeService對象@Overridepublic FunctionTreeService getFunctionTreeService() {return functionTreeService;}//獲取RoleFuncRightsService對象@Overridepublic RoleFuncRightsService getRoleFuncRightsService() {return roleFuncRightsService;}//獲取ChangeNotifyService對象@Overridepublic ChangeNotifyService getChangeNotifyService() {return changeNotifyService;}}

​​  GlobalConfigServiceImpl類,管理了很多配置服務類,此處主要關注ChangeNotifyService類對象。

3.3、使用ServletContext,管理全局配置服務類對象

​​​  全局配置服務類在應用啟動時加載到Spring容器中,這樣可實現共享,減少對數據庫的訪問壓力。

​​​  實現一個ApplicationListener類,代碼如下:

package com.abc.questInvest;import javax.servlet.ServletContext;import org.springframework.context.ApplicationListener;import org.springframework.context.event.ContextRefreshedEvent;import org.springframework.stereotype.Component;import org.springframework.web.context.WebApplicationContext;import com.abc.questInvest.service.GlobalConfigService;/** * @className: ApplicationStartup * @description: 應用偵聽器 * */@Componentpublic class ApplicationStartup implements ApplicationListener<ContextRefreshedEvent>{ //全局變量管理對象,此處不能自動注入 private GlobalConfigService globalConfigService = null;@Override public void onApplicationEvent(ContextRefreshedEvent contextRefreshedEvent) {try {if(contextRefreshedEvent.getApplicationContext().getParent() == null){ //root application context 沒有parent.System.out.println('========定義全局變量==================');// 將 ApplicationContext 轉化為 WebApplicationContext WebApplicationContext webApplicationContext = (WebApplicationContext)contextRefreshedEvent.getApplicationContext(); // 從 webApplicationContext 中獲取 servletContext ServletContext servletContext = webApplicationContext.getServletContext();//加載全局變量管理對象 globalConfigService = (GlobalConfigService)webApplicationContext.getBean(GlobalConfigService.class); //加載數據 boolean bRet = globalConfigService.loadData(); if (false == bRet) { System.out.println('加載全局變量失敗'); return; } //====================================================================== // servletContext設置值 servletContext.setAttribute('GLOBAL_CONFIG_SERVICE', globalConfigService); } } catch (Exception e) {e.printStackTrace(); } }}

​​​  在啟動類中,加入該應用偵聽器ApplicationStartup。

public static void main(String[] args) { SpringApplication springApplication = new SpringApplication(QuestInvestApplication.class);springApplication.addListeners(new ApplicationStartup());springApplication.run(args); }

​​  現在,有了一個GlobalConfigService類型的全局變量globalConfigService。

3.4、發出變更通知

​​​  此處舉2個例子,說明發出變更通知的例子,這兩個例子,都在用戶管理模塊,UserManServiceImpl類中。

​​​  1)管理員修改用戶信息,可能導致權限相關項發生變動,2)禁用用戶,發出變更過通知。

​​​  發出通知的相關代碼如下:

/** * * @methodName: editUser * @description: 修改用戶信息 * @param userInfo: 用戶信息對象 * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/081.0.0sheng.zheng初版 * 2021/06/281.0.1sheng.zheng增加變更通知的處理 * */@Overridepublic void editUser(HttpServletRequest request,UserInfo userInfo) {//輸入參數校驗checkValidForParams('editUser',userInfo);//獲取操作人賬號String operatorName = (String) request.getSession().getAttribute('username');userInfo.setOperatorName(operatorName);//登錄名和密碼不修改userInfo.setLoginName(null);userInfo.setSalt(null);userInfo.setPasswd(null);//獲取修改之前的用戶信息Integer userId = userInfo.getUserId();UserInfo oldUserInfo = userManDao.selectUserByKey(userId);//修改用戶記錄try {userManDao.updateSelective(userInfo);}catch(Exception e) {e.printStackTrace();log.error(e.getMessage());throw new BaseException(ExceptionCodes.USERS_EDIT_USER_FAILED);}//檢查是否有需要通知的變更Integer changeFlag = 0;if (userInfo.getRoles() != null) {if(oldUserInfo.getRoles() != userInfo.getRoles()) {//角色組合有變化,bit0changeFlag |= 0x01;}}if (userInfo.getDeptId() != null) {if (oldUserInfo.getDeptId() != userInfo.getDeptId()) {//部門ID有變化,bit3changeFlag |= 0x08;}}if (changeFlag > 0) {//如果有變更過通知項//獲取全局變量ServletContext servletContext = request.getServletContext();GlobalConfigService globalConfigService = (GlobalConfigService)servletContext.getAttribute('GLOBAL_CONFIG_SERVICE');globalConfigService.getChangeNotifyService().setChangeNotifyInfo(userId, changeFlag);}}/** * * @methodName: disableUser * @description: 禁用用戶 * @param params: map對象,形式如下: * { * 'userId': 1 * } * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/081.0.0sheng.zheng初版 * 2021/06/281.0.1sheng.zheng增加變更通知的處理 * */@Overridepublic void disableUser(HttpServletRequest request,Map<String,Object> params) {//輸入參數校驗checkValidForParams('disableUser',params);UserInfo userInfo = new UserInfo();//獲取操作人賬號String operatorName = (String) request.getSession().getAttribute('username');//設置userInfo信息Integer userId = (Integer)params.get('userId');userInfo.setUserId(userId);userInfo.setOperatorName(operatorName);//設置禁用標記userInfo.setDeleteFlag((byte)1);//修改密碼try {userManDao.updateEnable(userInfo);}catch(Exception e) {e.printStackTrace();log.error(e.getMessage());throw new BaseException(ExceptionCodes.USERS_EDIT_USER_FAILED);}//禁用用戶,發出變更通知//獲取全局變量ServletContext servletContext = request.getServletContext();GlobalConfigService globalConfigService = (GlobalConfigService)servletContext.getAttribute('GLOBAL_CONFIG_SERVICE');//禁用用戶:bit2globalConfigService.getChangeNotifyService().setChangeNotifyInfo(userId, 0x04);}

​​  本demo項目的角色相對較少,沒有使用用戶角色關系表,而是使用了bitmap編碼,角色ID取值為2^n,用戶角色組合roles字段為一個Integer值。如roles=7,表示角色ID組合=[1,2,4]。​​  另外,如果修改了角色的功能權限集合,則需要查詢受影響的用戶ID列表,依次發出通知,可類似處理。

3.5、修改Response響應消息體

​​​  Response響應消息體,為BaseResponse,代碼如下:

package com.abc.questInvest.vo.common;import lombok.Data;/** * @className: BaseResponse * @description: 基本響應消息體對象 * @summary: * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/05/311.0.0sheng.zheng初版 * 2021/06/281.0.1sheng.zheng增加變更通知的附加信息 * */@Datapublic class BaseResponse<T> { //響應碼 private int code; //響應消息 private String message; //響應實體信息 private T data; //分頁信息 private Page page; //附加通知信息 private Additional additional;}

​​  BaseResponse類增加了Additional類型的additional屬性字段,用于輸出附加信息。

​​  Additional類的定義如下:

package com.abc.questInvest.vo.common;import lombok.Data;/** * @className: Additional * @description: 附加信息 * @summary: * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/281.0.0sheng.zheng初版 * */@Datapublic class Additional { //通知碼,附加信息 private int notifycode; //通知碼對應的消息 private String notification;//更新的token private String token;//更新的功能權限樹 private String rights;}

​​  附加信息類Additional中,各屬性字段的說明:

notifycode,為通知碼,即可對應通知消息的類型,目前只有一種,可擴展。 notification,為通知碼對應的消息。

​​  通知碼,在ExceptionCodes枚舉文件中定義:

//變更通知信息 USER_RIGHTS_CHANGED(51, 'message.USER_RIGHTS_CHANGED', '用戶權限發生變更'),; //end enum ExceptionCodes(int code, String messageId, String message) {this.code = code;this.messageId = messageId;this.message = message; } token,用于要求前端更新token。更新token的目的是確認前端已經收到權限變更通知。因為下次url請求將使用新的token,如果前端未收到或未處理,仍然用舊的token訪問,就要跳到登錄頁了。 rights,功能樹的字符串輸出,是樹型結構的JSON字符串。

3.6、AOP鑒權處理

​​​  AuthorizationAspect為鑒權認證的切面類,代碼如下:

package com.abc.questInvest.aop;import java.util.List;import javax.servlet.ServletContext;import javax.servlet.http.HttpServletRequest;import org.aspectj.lang.annotation.AfterReturning;import org.aspectj.lang.annotation.Aspect;import org.aspectj.lang.annotation.Before;import org.aspectj.lang.annotation.Pointcut;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.core.annotation.Order;import org.springframework.stereotype.Component;import org.springframework.web.context.request.RequestContextHolder;import org.springframework.web.context.request.ServletRequestAttributes;import com.abc.questInvest.common.constants.Constants;import com.abc.questInvest.common.utils.Utility;import com.abc.questInvest.dao.UserManDao;import com.abc.questInvest.entity.FunctionInfo;import com.abc.questInvest.entity.UserInfo;import com.abc.questInvest.exception.BaseException;import com.abc.questInvest.exception.ExceptionCodes;import com.abc.questInvest.service.GlobalConfigService;import com.abc.questInvest.service.LoginService;import com.abc.questInvest.vo.TreeNode;import com.abc.questInvest.vo.common.Additional;import com.abc.questInvest.vo.common.BaseResponse;/** * @className: AuthorizationAspect * @description: 接口訪問鑒權切面類 * @summary: 使用AOP,進行token認證以及用戶對接口的訪問權限鑒權 * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks * ------------------------------------------------------------------------------ * 2021/06/061.0.0sheng.zheng初版 * 2021/06/281.0.1sheng.zheng增加變更通知的處理,增加了afterReturning增強 * */@Aspect@Component@Order(2)public class AuthorizationAspect {@Autowired private UserManDao userManDao;//設置切點 @Pointcut('execution(public * com.abc.questInvest.controller..*.*(..))' + '&& !execution(public * com.abc.questInvest.controller.LoginController.*(..))' + '&& !execution(public * com.abc.questInvest.controller.QuestInvestController.*(..))')public void verify(){}@Before('verify()') public void doVerify(){ ServletRequestAttributes attributes=(ServletRequestAttributes) RequestContextHolder.getRequestAttributes();HttpServletRequest request=attributes.getRequest(); // ================================================================================// token認證//從header中獲取token值String token = request.getHeader('Authorization');if (null == token || token.equals('')){ //return;throw new BaseException(ExceptionCodes.TOKEN_IS_NULL); } //從session中獲取token和過期時間String sessionToken = (String)request.getSession().getAttribute('token');//判斷session中是否有信息,可能是非登錄用戶if (null == sessionToken || sessionToken.equals('')) {throw new BaseException(ExceptionCodes.TOKEN_WRONG);} //比較tokenif(!token.equals(sessionToken)) {//如果請求頭中的token與存在session中token兩者不一致throw new BaseException(ExceptionCodes.TOKEN_WRONG);}long expireTime = (long)request.getSession().getAttribute('expireTime');//檢查過期時間long time = System.currentTimeMillis();if (time > expireTime) {//如果token過期throw new BaseException(ExceptionCodes.TOKEN_EXPIRED);}else {//token未過期,更新過期時間long newExpiredTime = time + Constants.TOKEN_EXPIRE_TIME * 1000;request.getSession().setAttribute('expireTime', newExpiredTime);}// ============================================================================// 接口調用權限//獲取用戶IDInteger userId = (Integer)request.getSession().getAttribute('userId'); //獲取全局變量ServletContext servletContext = request.getServletContext();GlobalConfigService globalConfigService = (GlobalConfigService)servletContext.getAttribute('GLOBAL_CONFIG_SERVICE');//===================變更通知處理開始==============================================//檢查有無變更通知信息Integer changeNotifyInfo = globalConfigService.getChangeNotifyService().getChangeNotifyInfo(userId);//通知前端權限變更的標記boolean rightsChangedFlag = false;if (changeNotifyInfo > 0) {//有通知信息if ((changeNotifyInfo & 0x09) > 0) {//bit0:修改用戶的角色組合值,從而導致權限變更//bit3:用戶調整部門,從而導致數據權限變更//mask 0b1001 = 0x09 //都需要查詢用戶表,并更新信息;合在一起查詢。UserInfo userInfo = userManDao.selectUserByKey(userId);//更新Session request.getSession().setAttribute('roles', userInfo.getRoles()); request.getSession().setAttribute('deptId', userInfo.getDeptId()); if ((changeNotifyInfo & 0x01) > 0) { //權限變更標志置位 rightsChangedFlag = true; }}else if((changeNotifyInfo & 0x02) > 0) {//bit1:修改角色的功能值,從而導致權限變更 //權限變更標志置位 rightsChangedFlag = true;}else if((changeNotifyInfo & 0x04) > 0) {//bit2:用戶禁用,從而導致權限變更//設置無效token,可阻止該用戶訪問系統request.getSession().setAttribute('token', '');//直接拋出異常,由前端顯示:Forbidden頁面throw new BaseException(ExceptionCodes.ACCESS_FORBIDDEN);}if (rightsChangedFlag == true) {//寫Session,用于將信息傳遞到afterReturning方法中request.getSession().setAttribute('rightsChanged', 1);}}//===================變更通知處理結束==============================================//從session中獲取用戶權限值Integer roles = (Integer)request.getSession().getAttribute('roles');//獲取當前接口url值String servletPath = request.getServletPath();//獲取該角色對url的訪問權限Integer rights = globalConfigService.getRoleFuncRightsService().getRoleUrlRights(Utility.parseRoles(roles), servletPath);if (rights == 0) {//如果無權限訪問此接口,拋出異常,由前端顯示:Forbidden頁面throw new BaseException(ExceptionCodes.ACCESS_FORBIDDEN);} } @AfterReturning(value='verify()' ,returning='result') public void afterReturning(BaseResponse result) { //限制必須是BaseResponse類型,其它類型的返回值忽略 //獲取SessionServletRequestAttributes sra = (ServletRequestAttributes)RequestContextHolder.getRequestAttributes();HttpServletRequest request = sra.getRequest(); Integer rightsChanged = (Integer)request.getSession().getAttribute('rightsChanged'); if (rightsChanged != null && rightsChanged == 1) { //如果有用戶權限變更,通知前端來刷新該用戶的功能權限樹 //構造附加信息 Additional additional = new Additional(); additional.setNotifycode(ExceptionCodes.USER_RIGHTS_CHANGED.getCode()); additional.setNotification(ExceptionCodes.USER_RIGHTS_CHANGED.getMessage()); //更新token String loginName = (String)request.getSession().getAttribute('username'); String token = LoginService.generateToken(loginName); additional.setToken(token); //更新token,要求下次url訪問使用新的token request.getSession().setAttribute('token', token); //獲取用戶的功能權限樹 Integer roles = (Integer)request.getSession().getAttribute('roles'); ServletContext servletContext = request.getServletContext(); GlobalConfigService globalConfigService = (GlobalConfigService)servletContext.getAttribute('GLOBAL_CONFIG_SERVICE');//獲取用戶權限的角色功能樹 List<Integer> roleList = Utility.parseRoles(roles);TreeNode<FunctionInfo> rolesFunctionTree = globalConfigService.getRoleFuncRightsService().getRoleRights(roleList);additional.setRights(rolesFunctionTree.toString()); //修改response信息result.setAdditional(additional); //移除Session的rightsChanged項 request.getSession().removeAttribute('rightsChanged'); } }}

​​  AuthorizationAspect類定義了切點verify(),@Before增強用于鑒權驗證,增加了對變更通知信息的處理。并利用Session,用rightsChanged屬性字段記錄需要通知前端的標志,在@AfterReturning后置增強中根據該屬性字段的值,進行一步的處理。

​​  @Before增強的doVerify方法中,如果發現角色組合有改變,但仍有訪問此url權限時,會繼續后續處理,這樣不會中斷業務;如果沒有訪問此url權限,則返回訪問受限異常信息,由前端顯示訪問受限頁碼(類似403 Forbidden 頁碼)。

​​  在后置增強@AfterReturning中,限定了返回值類型,如果該請求響應的類型是BaseResponse類型,則修改reponse消息體,附加通知信息;如果不是,則不處理,會等待下一個url請求,直到返回類型是BaseResponse類型。也可以采用自定義response的header的方式,這樣,就無需等待了。

​​  generateToken方法,是LoginService類的靜態方法,用于生成用戶token。

​​  至于Utility的parseRoles方法,是將bitmap編碼的roles解析為角色ID的列表,代碼如下:

//========================= 權限組合值解析 ======================================/** * * @methodName: parseRoles * @description: 解析角色組合值 * @param roles: 按位設置的角色組合值 * @return: 角色ID列表 * @history: * ------------------------------------------------------------------------------ * dateversionmodifierremarks* ------------------------------------------------------------------------------ * 2021/06/241.0.0sheng.zheng初版 * */ public static List<Integer> parseRoles(int roles){ List<Integer> roleList = new ArrayList<Integer>(); int newRoles = roles; int bit0 = 0; int roleId = 0; for (int i = 0; i < 32; i++) { //如果組合值的余位都為0,則跳出 if (newRoles == 0) { break; }//取得最后一位 bit0 = newRoles & 0x01; if (bit0 == 1) { //如果該位為1,左移i位 roleId = 1 << i; roleList.add(roleId); }//右移一位 newRoles = newRoles >> 1; } return roleList; }

​​  getRoleRights方法,是角色功能權限服務類RoleFuncRightsService的方法,它提供了根據List類型的角色ID列表,快速獲取功能權限樹的功能。​​  關于功能權限樹TreeNode類型,請參閱:《Java通用樹結構數據管理》。

到此這篇關于Spring Boot動態權限變更實現的整體方案的文章就介紹到這了,更多相關Spring Boot動態權限內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Spring
相關文章:
主站蜘蛛池模板: 欧美日韩精品一二区 | 日韩欧美精品一区 | 日韩综合在线 | 91亚洲精品乱码久久久久久蜜桃 | 成人黄色在线观看 | 国产成人精品久久二区二区 | 亚洲一区在线免费观看 | 黄色国产区 | 日韩福利一区二区 | 日韩精品免费视频 | 久久久久久久久免费视频 | 国产午夜精品一区二区三区视频 | 欧美一区二区视频 | 国产男人天堂 | 亚洲啊v | 久久精品一区二区三区四区 | 韩日一区 | 97人人草 | 最新超碰 | 国产伦精品一区二区三区四区视频 | 国产在线日韩 | 免费视频99 | 91久久久久久 | 亚洲伦理一区 | 亚洲欧美电影 | 欧洲美女性开放视频 | 精品1区2区| 日本中文字幕一区二区 | 久久中文字幕一区二区三区 | 日韩中文一区二区三区 | 日本黄色大片免费 | 日韩艹逼视频 | 久久亚洲精品视频 | 五月婷婷色 | 亚洲免费观看视频 | 欧美国产日本一区 | 国产在线专区 | 99精品久久精品一区二区爱城 | 国产福利在线观看视频 | 一区二区三区四区在线播放 | www..99热 | 成人免费在线看片 | 亚洲九九九 | 一级黄色大片在线 | 日韩欧美在线播放 | 久久久国产精品视频 | 日韩成人一区 | 精品国产乱码久久久久久1区2区 | a一级毛片 | 亚洲综合在线视频 | 精品久久中文字幕 | 久久久久国产一级毛片高清版小说 | 国产视频9999 | 成人午夜视频在线观看 | 六月丁香啪啪 | 99色综合 | 久久久久亚洲精品 | 午夜爱视频 | 免费看黄视频网站 | 在线观看国精产品二区1819 | 日本免费三片免费观看 | 亚洲高清视频一区二区三区 | 国产欧美精品一区 | 欧美天天 | 99精品国产高清一区二区麻豆 | 久久久久久久国产 | 午夜视频一区二区 | 免费av片 | 欧美日韩国产高清 | 韩国一区二区视频 | 久久色视频 | 天堂久久一区 | 久久99精品一区二区三区三区 | 天天干夜夜骑 | 国产免费一区二区三区 | 国产精选视频 | 毛片天堂| www.久久久久久久久久久久 | 精品国产乱码久久久久久1区2区 | 亚洲视频一区 | 日本不卡一区二区三区在线观看 | a级在线免费| 久久久久亚洲视频 | 超碰在线人人 | 99久久久无码国产精品 | 亚洲综合国产 | 人人草在线观看视频 | 成人精品一区二区三区中文字幕 | 欧美午夜电影 | 国产福利在线视频 | 自拍一区视频 | 男女深夜网站 | 91精品国产91久久久久久最新 | 精品久久精品 | 亚洲成人二区 | 欧美一级二级三级视频 | 天天干视频 | 在线中文 | 国产午夜小视频 | 精品中文字幕在线观看 | av久久| 国产一区二区三区久久久 | 欧美黑人一级爽快片淫片高清 | 日韩一区二区在线观看视频 | 91久久久久久久久 | 日韩免费一区二区 | 色综合久久网 | 一级免费黄色免费片 | 成人h漫在线观看 | 欧美午夜精品久久久久久浪潮 | 色婷婷av一区二区三区软件 | 日韩视频在线观看不卡 | 欧美亚洲国产一区 | 国产一区免费在线观看 | 国产97人人超碰caoprom | 在线观看成人国产 | 狠狠草视频 | 欧美人成在线视频 | 国产亚洲欧美在线 | 日韩在线资源 | 国产美女av | 成人av网址在线观看 | 亚洲国产日韩一区 | 国产欧美专区 | 干干人人| 色黄网站| 亚洲综合网站 | 91最新视频 | 综合久久久 | 欧美肉体xxxx肉交高潮 | 老司机深夜福利视频 | 亚洲综合电影 | 久久噜噜噜精品国产亚洲综合 | 韩国av一区二区 | 欧美一级小视频 | 四虎影音| 亚洲黄色高清视频 | 亚洲欧美一区二区三区在线 | 自拍偷拍专区 | 国产女人爽到高潮免费视频 | 2018天天操夜夜操 | 成人精品视频 | 国产一区二区在线免费 | 国产资源视频在线观看 | 久久久久久国产视频 | 91在线视频免费观看 | 这里只有精品视频 | 中文字幕成人免费视频 | 欧美xxxxxx视频 | 在线播放国产精品 | 天天干天天草 | 一区二区免费 | 亚洲 自拍 另类 欧美 丝袜 | 91久久久久久 | 午夜亚洲| 亚洲欧美中文日韩v在线观看 | 色婷婷亚洲一区二区三区 | 99精品国自产在线 | 夜夜艹| 国产精品久久二区 | 9久久精品 | 日韩高清在线 | www.日韩在线视频 | 欧美夜夜骑 | 国产成人精品一区二区三区视频 | 伊人网在线视频 | 爱爱视频免费 | 美女视频一区二区三区 | 国产精品视频免费播放 | 免费欧美一级 | 欧美成人性生活 | 亚洲免费视频网 | 亚洲欧美在线一区 | 成人国产一区 | 一级片视频在线观看 | 日韩免费 | 国产亚洲久久 | 天天操妹子 | 午夜成人免费视频 | 色黄视频在线 | 成人精品网站在线观看 | 91国内精品久久 | 国产精品视频一区二区三区 | www.国产高清 | 犬夜叉在线观看 | 国产拍拍拍拍拍拍拍拍拍拍拍拍拍 | 国产精品夜间视频香蕉 | 国产精品对白一区二区三区 | 亚洲国产精品人人爽夜夜爽 | 欧美一级在线视频 | 天天操天天色天天 | 久久一区二区三区四区五区 | 毛片在线看片 | 成人av免费观看 | 免费看国产片在线观看 | 欧美激情在线播放 | 97视频久久| 一区二区三区四区视频 | 久久精品二区 | 国产精品日产欧美久久久久 | 免费一级在线观看 | 欧美精品久久久 | 久久国产视频一区二区 | 二区视频 | 一级毛片黄 | 日韩中文字幕在线 | 国产综合精品视频 | 国产一区二区三区免费视频 | 精品久久久久久久 | 亚洲欧洲一区二区三区 | 欧美a在线 | 久久一区| 免费一二三区 | 中文字幕在线观看的电影 | 欧美另类亚洲 | 蜜桃精品久久久久久久免费影院 | 欧美性猛片aaaaaaa做受 | 国产精品成人免费视频 | 一区二区视频 | 久久久久亚洲一区二区三区 | 欧美一区二区三区在线观看视频 | 成人小视频在线观看 | 国产精品国色综合久久 | 91国内视频在线观看 | 中文一区 | 国产精品一区二区三区在线 | 国产欧美久久一区二区三区 | 国产精品视频一区二区三区四蜜臂 | 日韩高清在线一区 | 亚洲三级在线播放 | 日韩一区二区三区在线观看 | 欧美激情一区二区三区蜜桃视频 | 91丁香婷婷综合久久欧美 | 国产精品久久久久久吹潮 | 国产毛片在线 | 成人深夜福利视频 | 亚洲精品国产第一综合99久久 | 国产成人精品午夜 | 伊人久操 | 人人澡人人草 | 中文字幕一区二区在线观看 | av影片在线 | 国产成人av一区二区 | 日本一区二区高清视频 | 日韩午夜免费视频 | 久久人体| 91av国产视频| 中文字幕av网 | 精品免费国产一区二区三区 | 欧美日韩在线精品 | 欧美伊人影院 | 2018天天操夜夜操 | 午夜午夜精品一区二区三区文 | 午夜免费小视频 | 亚洲最大的黄色网 | 黄色一级网站视频 | 成人免费视频网站在线看 | www.国产91| 午夜av电影院 | 综合二区 | 男女视频一区二区 | 欧美久久一级特黄毛片 | 在线91 | 欧美黄色片免费观看 | 中文字幕亚洲在线 | 久色 | 黄色激情网站 | 在线观看国产 | 美女午夜影院 | 超碰人操 | 81精品国产乱码久久久久久 | 成年人在线观看 | 午夜视频免费网站 | 中文字幕一区在线观看 | 久久精品无码一区二区日韩av | 欧美国产日韩一区 | 日本黄色片免费看 | 成人精品在线 | 婷婷av在线| 成人久久久 | 欧美日韩一区在线 | 欧美一级二级视频 | 9久久婷婷国产综合精品性色 | 欧美一区二区三区黄色 | 中文字幕精品一区久久久久 | 免费黄色特级片 | 国产精品久久久久久久久费观看 | 自拍偷拍专区 | 日韩城人网站 | √天堂在线 | 91在线中文字幕 | 日韩五月 | 日韩精品一区在线 | 亚洲精品二区三区 | 久久亚洲精品中文字幕 | 亚洲性视频网站 | 亚洲成av人影片在线观看 | 水卜樱一区二区av | 蜜桃臀一区二区三区 | 黄色大片在线播放 | 国产午夜视频 | 久草在线高清 | 性色av网 | 在线观看av网站永久 | 精品久久久久久国产 | 91国在线高清视频 | 男女精品视频 | 欧美日韩视频在线第一区 | 国产精品毛片一区二区三区 | 国产精品成人免费视频 | 欧美精品一区二区三区四区 | 超碰激情 | 在线日韩| 羞羞视频在线观看入口 | 亚洲欧洲综合av | 日韩色在线 | 久久久成人精品 | 国产精品高潮呻吟av久久4虎 | 91精品久久久久久久久久 | 成人免费在线电影 | 日韩艹逼视频 | 精品一区二区三区视频 | 亚洲精品成人无限看 | 亚洲精品一区二区网址 | 在线观看理论电影 | 日本在线看 | 黑人巨大精品欧美一区免费视频 | 中文字幕一区二区三区在线视频 | 91视频一区二区三区 | 日韩欧美视频 | 欧美日韩精品久久久 | 久草免费在线 | 亚洲成人三级 | 亚洲免费视频一区二区 | www婷婷av久久久影片 | 三级成人在线 | 色婷婷久久久久swag精品 | 免费久久99精品国产婷婷六月 | 国产成人免费视频网站视频社区 | 久久免费视频9 | 欧美国产一区二区 | 午夜精品久久久久久久久久久久久 | 日韩小视频 | 亚洲第一av | 精品久久久久久久 | 日韩激情网 | 国产日韩一级片 | 国产精品毛片一区二区 | 人人草天天草 | 国内福利视频 | 亚洲日韩中文字幕一区 | 国产美女网站 | 亚洲精品一区二区三区中文字幕 | 成人久久18 | 男人的天堂在线视频 | 亚洲精品一区在线观看 | 色视频www在线播放国产人成 | 国产精品视频播放 | 精品国产91亚洲一区二区三区www | 日韩精品中文字幕一区二区三区 | 久久国产精品免费一区二区三区 | 在线日韩欧美 | 午夜男人的天堂 | 免费看的毛片 | 桃花久久| 超级黄色一级片 | 手机久久看片 | 国产精品毛片久久久久久 | 国产精品高清在线观看 | 日韩一区二区三区精品 | 天天操夜夜操免费视频 | 日韩欧美一区二区在线 | 毛片毛片毛片毛片毛片毛片 | 婷婷色在线 | 日韩精品在线一区 | 一区不卡 | 91在线影院| 精品免费 | 日日操夜夜操天天操 | 久久天堂 | 久久精品久久久久久久久久久久久 | 午夜精品久久久久久 | 亚洲成人在线网站 | 麻豆一区一区三区四区 | 青青草久久网 | 国产一区二区三区久久久 | 99re国产| 另类sb东北妇女av | 精品国产乱码久久久久久1区2区 | 欧美欧美欧美 | 麻豆一区一区三区四区 | 国产亚洲网站 | 一级毛片网 | 超碰免费在线观看 | 亚洲综合视频一区 | 久久久久久久一区 | 一区二区精品在线 | 天天干一干 | 欧美激情在线播放 | 国产精品一区二区三区在线播放 | 日韩三级电影免费观看 | 毛片搜索 | 免费黄色小视频 | 天天干,夜夜操 | 美女黄网| 国产一区二区久久 | 精品久久久久av | 99精品电影| 2019亚洲日韩新视频 | 99热精品在线 | 欧美黑人狂躁日本寡妇 | 色婷婷综合久色 | 久久精品一级 | 日韩精品一区二区三区视频播放 | 久热热| 欧美一a一片一级一片 | 午夜精品一区二区三区在线 | 国产精品成人一区二区三区夜夜夜 | 亚洲精品一区二区三区蜜桃久 | 国产成人精品一区二区三区四区 | 99riav国产一区二区三区 | 国产精品高清在线 | 一二三区字幕免费观看av | 永久91嫩草亚洲精品人人 | 天天综合网网欲色 | 欧洲美女7788成人免费视频 | 一级免费黄色 | 手机久久看片 | 激情亚洲 | 成人tv| 色综合色综合网色综合 | 日本久久久久久久久久久久 | 国产亚洲精品精品国产亚洲综合 | 成人免费一区二区三区 | 亚洲一区二区三区视频免费观看 | 欧美精品在线免费观看 | 国产日韩欧美精品一区二区三区 | 成人一级 | 国产精品国产成人国产三级 | 日本在线视频中文字幕 | av在线大全| 日韩成人中文字幕 | 日韩电影专区 | h视频免费看 | 一区二区三区视频在线观看 | 91在线免费视频 | 久操视频在线观看 | 一区二区三区四区在线 | 欧美久久精品 | 小川阿佐美88av在线播放 | 簧片毛片| 精品国产三级a在线观看 | 成人精品一区二区 | 视频精品一区二区 | 欧美在线 | 亚洲 | av成人在线观看 | 久久久久久久久久国产精品 | 欧美日韩精品 | 国产美女一区二区 | 亚洲精品日韩在线 | 精品视频在线观看一区二区 | 久久综合一区 | 亚洲电影一区二区 | 超碰在线人 | 国产精品一卡二卡 | 精品1区 | www.日韩大片 | 亚洲国产成人在线 | 亚洲视频免费观看 | 91视频在线网址 | sese综合 | 搡女人真爽免费午夜网站 | 成人国产一区二区 | h片在线 | 九色91九色porny永久 | 日本 欧美 三级 高清 视频 | 欧美精品亚洲 | 亚洲欧洲在线观看 | 99精品国产在热久久 | 亚洲精品专区 | 欧美亚洲免费 | 精品久久久久久久久久久 | 狠狠影院 | 国产欧美日韩在线观看 | 永久在线观看 | 美女久久久久 | 色网站在线观看 | 久久久久久国产精品 | 成人精品视频 | 男人天堂网av | 色噜噜视频 | 成人欧美 | 久久精品亚洲精品国产欧美kt∨ | 久久久网| 欧美福利影院 | 男女做爰高清无遮挡免费视频 | 日韩一区精品视频 | 91精品国产91久久久久久不卡 | 中文在线一区 | 国产一级在线观看 | 日本精品中文字幕 | 亚洲久久一区 | 成人av网站在线观看 | 免费在线看a| 日韩大片一区 | 一级在线播放 | 成人av网站免费观看 | 奇米色欧美一区二区三区 | 日本一区二区电影 | 日韩一区二区在线视频 | 国产精品日本一区二区不卡视频 | 欧洲一区在线 | 亚洲福利一区 | 国产一级片 | 91无吗| 午夜在线观看免费 | 成人欧美一区二区三区在线播放 | 日韩精品一区二 | 亚洲乱码国产乱码精品精 | 探花在线观看 | 久久久国产精品 | 欧美成年黄网站色视频 | 亚洲天堂中文字幕 | 91伊人网 | 欧美一区二区三区 | 在线干| www.99热这里只有精品 | 伊人春色成人 | 日韩视频三区 | 欧美淫视频 | 国产九九在线观看 | 在线观看免费黄色小视频 | 欧美精品tv | 一级在线免费视频 | 久久久精品免费观看 | 天天看天天干 | 久久久xxx | 日韩在线观看一区 | 成人免费视频观看视频 | 国产精品久久久久久久久久久久久久 | 日韩国产欧美一区 | 中文字幕亚洲综合 | aaa在线观看 | 日韩视频在线观看 | 精品久久一区二区 | 午夜欧美一区二区三区在线播放 | 久久99久久98精品免观看软件 | 欧美最猛性xxxxx亚洲精品 | 中文字幕二区 | 先锋av资源网 | 国产亚洲精品美女久久久久久久久久 | 国产一区二区三区免费 | 九九免费观看全部免费视频 | 国产高清美女一级a毛片久久 | 日韩视频在线观看 | 四虎动漫| 日韩午夜 | 国产精品69久久久久水密桃 | 成人影院在线 | 亚洲精品一区二区三区精华液 | 亚洲精品乱码久久久久久金桔影视 | 日韩在线视频观看 | 亚洲综合日韩 | 人人爱超碰 | 久久99精品国产麻豆婷婷洗澡 | 久久久精品网站 | 一区不卡 | 欧美一a一片一级一片 | 亚洲福利社区 | 日韩a视频 | 天堂av2020 | 少妇精品视频在线观看 | 一区二区在线电影 | 国产免费一区二区 | 午夜国产精品成人 | av免费在线播放 | 欧美2区 | 视频网站免费观看 | 欧美日韩亚洲一区二区 | 国产亚洲精品v | 欧美精品在线看 | 欧美日韩在线免费观看 | 国产免费一区二区 | 老司机午夜免费精品视频 | 国产成人免费视频网站高清观看视频 | 日韩成人免费 | 香蕉91| 中文在线a在线 | 夜本色| 欧美精品网站 | 亚洲精品福利在线 | 国产一区二区免费电影 | 欧美11一13sex性hd | 久久9999久久 | 伊人网在线视频免费观看 | 一区二区免费在线观看 | 国产视频三区 | 特级做a爰片毛片免费看108 | 成人在线欧美 | 亚洲国产精品成人 | 亚洲成av人片在线观看无码 | 9999国产精品 | 国产美女一区二区 | 精品一区在线 | 日韩成人在线观看 | 免费三片在线观看网站 | 日韩中文字幕一区二区高清99 | 精品日韩在线观看 | 特级淫片女子高清视频在线观看 | 一本久久a久久精品亚洲 | 国产一级特黄 | 精品成人一区 | 国产乱码精品一区二区三区忘忧草 | 欧美精品一二三 | 久久久久久亚洲精品视频 |