1、前言

​ 在Web项目中,权限管理即权限访问控制为网站访问安全提供了保障,并且很多项目使用了Session作为缓存,结合AOP技术进行token认证和权限控制。权限控制流程大致如下图所示:

​ 现在,如果管理员修改了用户的角色,或修改了角色的权限,都会导致用户权限发生变化,此时如何实现动态权限变更,使得前端能够更新用户的权限树,后端访问鉴权AOP模块能够知悉这种变更呢?

2、问题及解决方案

​ 现在的问题是,管理员没法访问用户Session,因此没法将变更通知此用户。而用户如果已经登录,或直接关闭浏览器页面而不是登出操作,Session没有过期前,用户访问接口时,访问鉴权AOP模块仍然是根据之前缓存的Session信息进行处理,没法做到动态权限变更。

​ 使用WebSocket是一个方案,但没法处理不在线用户。

​ 解决方案的核心思想是利用ServletContext对象的共享特性,来实现用户权限变更的信息传递。然后在AOP类中查询用户是否有变更通知记录需要处理,如果权限发生变化,则修改response消息体,添加附加通知信息给前端。前端收到附加的通知信息,可更新功能权限树,并进行相关处理。

​ 这样,利用的变更通知服务,不仅后端的用户url访问接口可第一时间获悉变更,还可以通知到前端,从而实现了动态权限变更。

3、方案实现

3.1、开发变更通知类

​ 服务接口类ChangeNotifyService,代码如下:

package com.abc.questInvest.service;

/**
* @className : ChangeNotifyService
* @description : 变更通知服务
* @summary :
* @history :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/28 1.0.0 sheng.zheng 初版
*
*/
public interface ChangeNotifyService { /**
*
* @methodName : getChangeNotifyInfo
* @description : 获取指定用户ID的变更通知信息
* @param userId : 用户ID
* @return : 返回0表示无变更通知信息,其它值按照bitmap编码。目前定义如下:
* bit0: : 修改用户的角色组合值,从而导致权限变更;
* bit1: : 修改角色的功能项,从而导致权限变更;
* bit2: : 用户禁用,从而导致权限变更;
* bit3: : 用户调整部门,从而导致数据权限变更;
* @history :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/28 1.0.0 sheng.zheng 初版
*
*/
public Integer getChangeNotifyInfo(Integer userId); /**
*
* @methodName : setChangeNotifyInfo
* @description : 设置变更通知信息
* @param userId : 用户ID
* @param changeNotifyInfo : 变更通知值
* bit0: : 修改用户的角色组合值,从而导致权限变更;
* bit1: : 修改角色的功能项,从而导致权限变更;
* bit2: : 用户禁用,从而导致权限变更;
* bit3: : 用户调整部门,从而导致数据权限变更;
* @history :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/28 1.0.0 sheng.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 :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/28 1.0.0 sheng.zheng 初版
*
*/
@Service
public 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 :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/28 1.0.0 sheng.zheng 初版
*
*/
@Override
public 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 :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/28 1.0.0 sheng.zheng 初版
*
*/
@Override
public 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 :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/02 1.0.0 sheng.zheng 初版
*
*/
public interface GlobalConfigService { /**
*
* @methodName : loadData
* @description : 加载数据
* @return : 成功返回true,否则返回false
* @history :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/02 1.0.0 sheng.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 :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/02 1.0.0 sheng.zheng 初版
*
*/
@Service
public class GlobalConfigServiceImpl implements GlobalConfigService{ //ID编码配置表数据服务
@Autowired
private TableCodeConfigService tableCodeConfigService; //系统参数表数据服务
@Autowired
private SysParameterService sysParameterService; //功能树表数据服务
@Autowired
private FunctionTreeService functionTreeService; //角色权限表数据服务
@Autowired
private RoleFuncRightsService roleFuncRightsService; //变更通知服务
@Autowired
private ChangeNotifyService changeNotifyService; /**
*
* @methodName : loadData
* @description : 加载数据
* @return : 成功返回true,否则返回false
* @history :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/02 1.0.0 sheng.zheng 初版
*
*/
@Override
public 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对象
@Override
public TableCodeConfigService getTableCodeConfigService() {
return tableCodeConfigService;
} //获取SysParameterService对象
@Override
public SysParameterService getSysParameterService() {
return sysParameterService;
} //获取FunctionTreeService对象
@Override
public FunctionTreeService getFunctionTreeService() {
return functionTreeService;
} //获取RoleFuncRightsService对象
@Override
public RoleFuncRightsService getRoleFuncRightsService() {
return roleFuncRightsService;
} //获取ChangeNotifyService对象
@Override
public 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 : 应用侦听器
*
*/
@Component
public 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 :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/08 1.0.0 sheng.zheng 初版
* 2021/06/28 1.0.1 sheng.zheng 增加变更通知的处理
*
*/
@Override
public 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()) {
//角色组合有变化,bit0
changeFlag |= 0x01;
}
}
if (userInfo.getDeptId() != null) {
if (oldUserInfo.getDeptId() != userInfo.getDeptId()) {
//部门ID有变化,bit3
changeFlag |= 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 :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/08 1.0.0 sheng.zheng 初版
* 2021/06/28 1.0.1 sheng.zheng 增加变更通知的处理
*
*/
@Override
public 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");
//禁用用户:bit2
globalConfigService.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 :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/05/31 1.0.0 sheng.zheng 初版
* 2021/06/28 1.0.1 sheng.zheng 增加变更通知的附加信息
*
*/
@Data
public 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 :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/28 1.0.0 sheng.zheng 初版
*
*/
@Data
public 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 :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/06 1.0.0 sheng.zheng 初版
* 2021/06/28 1.0.1 sheng.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);
} //比较token
if(!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);
} // ============================================================================
// 接口调用权限
//获取用户ID
Integer userId = (Integer)request.getSession().getAttribute("userId");
//获取全局变量
ServletContext servletContext = request.getServletContext();
GlobalConfigService globalConfigService = (GlobalConfigService)servletContext.getAttribute("GLOBAL_CONFIG_SERVICE"); //===================变更通知处理开始==============================================
//检查有无变更通知信息
Integer changeNotifyInfo = globalConfigService.getChangeNotifyService().getChangeNotifyInfo(userId);
//设置成员属性为false
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类型,其它类型的返回值忽略
//获取Session
ServletRequestAttributes 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 :
* ------------------------------------------------------------------------------
* date version modifier remarks
* ------------------------------------------------------------------------------
* 2021/06/24 1.0.0 sheng.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<Integer>类型的角色ID列表,快速获取权限权限树的功能。

Spring Boot动态权限变更实现的整体方案的更多相关文章

  1. 利用ServletContext,实现Session动态权限变更

    1.前言 很多Spring Boot应用使用了Session作为缓存,一般会在用户登录后保存用户的关键信息,如: 用户ID. 用户名. 用户token. 权限角色集合. 等等... 在管理员修改了用户 ...

  2. (39.3) Spring Boot Shiro权限管理【从零开始学Spring Boot】

    在学习此小节之前您可能还需要学习: (39.1) Spring Boot Shiro权限管理[从零开始学Spring Boot] http://412887952-qq-com.iteye.com/b ...

  3. (39.2). Spring Boot Shiro权限管理【从零开始学Spring Boot】

    (本节提供源代码,在最下面可以下载) (4). 集成Shiro 进行用户授权 在看此小节前,您可能需要先看: http://412887952-qq-com.iteye.com/blog/229973 ...

  4. (39.1) Spring Boot Shiro权限管理【从零开始学Spring Boot】

    (本节提供源代码,在最下面可以下载)距上一个章节过了二个星期了,最近时间也是比较紧,一直没有时间可以写博客,今天难得有点时间,就说说Spring Boot如何集成Shiro吧.这个章节会比较复杂,牵涉 ...

  5. (39.4) Spring Boot Shiro权限管理【从零开始学Spring Boot】

    在读此文章之前您还可能需要先了解: (39.1) Spring Boot Shiro权限管理[从零开始学Spring Boot] http://412887952-qq-com.iteye.com/b ...

  6. Spring Boot Shiro 权限管理

    Spring Boot Shiro 权限管理 标签: springshiro 2016-01-14 23:44 94587人阅读 评论(60) 收藏 举报 .embody{ padding:10px ...

  7. Spring Boot动态注入删除bean

    Spring Boot动态注入删除bean 概述 因为如果采用配置文件或者注解,我们要加入对象的话,还要重启服务,如果我们想要避免这一情况就得采用动态处理bean,包括:动态注入,动态删除. 动态注入 ...

  8. Spring Boot 动态数据源(多数据源自己主动切换)

    本文实现案例场景: 某系统除了须要从自己的主要数据库上读取和管理数据外.另一部分业务涉及到其它多个数据库,要求能够在不论什么方法上能够灵活指定详细要操作的数据库. 为了在开发中以最简单的方法使用,本文 ...

  9. Spring Boot 动态数据源(多数据源自动切换)

    本文实现案例场景: 某系统除了需要从自己的主要数据库上读取和管理数据外,还有一部分业务涉及到其他多个数据库,要求可以在任何方法上可以灵活指定具体要操作的数据库. 为了在开发中以最简单的方法使用,本文基 ...

随机推荐

  1. [BUAA2021软工助教]结对项目-第二阶段小结

    一.作业链接 结对项目-第二阶段 二.优秀作业推荐 本次博客作业虽然是简单总结,但是以下作业中都不乏有思考.有亮点的精彩内容,推荐给同学们阅读学习. 磨练,结对编程!(中) zzx 和 zzy 同学实 ...

  2. OO随笔之魔鬼的第一单元——多项式求导

    OO是个借助Java交我们面向对象的课,可是萌新们总是喜欢带着面向过程的脑子去写求导,然后就是各种一面(main)到底.各种方法杂糅,然后就是被hack的很惨. 第一次作业:萌新入门面向对象 题目分析 ...

  3. Java安全之Filter权限绕过

    Java安全之Filter权限绕过 0x00 前言 在一些需要挖掘一些无条件RCE中,大部分类似于一些系统大部分地方都做了权限控制的,而这时候想要利用权限绕过就显得格外重要.在此来学习一波权限绕过的思 ...

  4. ES系列(五):获取单条数据get处理过程实现

    前面讲的都是些比较大的东西,即框架层面的东西.今天咱们来个轻松点的,只讲一个点:如题,get单条记录的es查询实现. 1. get语义说明 get是用于搜索单条es的数据,是根据主键id查询数据方式. ...

  5. ActiveMQ FileServer漏洞(详细)

    半个月前,巡检时发现服务器出现不明进程,对其进行了处理,由于当时没有做详细记录,在这里把大致过程描述一下. 症状: ps命令发现出现几个不明进程, 1.于/tmp下运行的,名称随机的进程.占用CPU高 ...

  6. 克隆并编译otter

    源码编译: git clone 项目到本地,用IDEA打开,等待Maven下载完jar包,打开命令行,进入当前项目的lib目录 执行install.bat命令,该批处理文件会将缺失的jar包安装到你本 ...

  7. 8.1 fdisk:磁盘分区工具

    fdisk 是Linux下常用的磁盘分区工具.受mbr分区表的限制,fdisk工具只能给小于2TB的磁盘划分分区.如果使用fdisk对大于2TB的磁盘进行分区,虽然可以分区,但其仅识别2TB的空间,所 ...

  8. 【ArcGIS遇上Python】ArcGIS Python批处理入门到精通实用教程目录

    目录 1. 专栏简介 2. 专栏地址 3. 专栏目录 1. 专栏简介 Python语言是目前很火热的语言,极大的促进了人工智能发展.你知道在ArcGIS中也会有python的身影吗?事实上,在ArcG ...

  9. 安装Keras出现的问题

    先是pip install tensorflow  给装好了,但是pip install  keras出现如下的问题: 只好搜帖子,参考如下的帖子,我直接 conda install keras wi ...

  10. 工作流中的数据持久化详解!Activiti框架中JPA的使用分析

    Activiti中JPA简介 可以使用JPA实体作为流程变量, 并进行操作: 基于流程变量更新已有的JPA实体,可以在用户任务的表单中填写或者由服务任务生成 重用已有的领域模型,不需要编写显示的服务获 ...