赞
踩
例如一个学校图书馆的管理系统,如果是普通学生登录就能看到借书还书相关的功能,不可能让他看到并且去使用添加书籍信息,删除书籍信息等功能。但是如果是一个图书馆管理员的账号登录了,应该就能看到并使用添加书籍信息,删除书籍信息等功能。
总结起来就是不同的用户可以使用不同的功能。这就是权限系统要去实现的效果。
我们不能只依赖前端去判断用户的权限来选择显示哪些菜单哪些按钮。因为如果只是这样,如果有人知道了对应功能的接口地址就可以不通过前端,直接去发送请求来实现相关功能操作。
所以我们还需要在后台进行用户权限的判断,判断当前用户是否有相应的权限,必须具有所需权限才能进行相应的操作。
在SpringSecurity中,会使用默认的FilterSecurityInterceptor来进行权限校验。在FilterSecurityInterceptor中会从SecurityContextHolder获取其中的Authentication,然后获取其中的权限信息。当前用户是否拥有访问当前资源所需的权限。
所以我们在项目中只需要把当前登录用户的权限信息也存入Authentication。
然后设置我们的资源所需要的权限即可。
在SecurityConfig上添加注解@EnableGlobalMethodSecurity(prePostEnabled = true)开启注解功能
之后在Controller层需要添加权限的方法上添加注解
定义了UserDetails的实现类LoginUser,想要让其能封装权限信息就要对其进行修改
- @Data
- @NoArgsConstructor //无参构造
-
- public class LoginUser implements UserDetails {
-
- private User user;
-
- //存储权限信息
- private List<String> permissions;
-
- //权限集合的转换
- @Override
- public Collection<? extends GrantedAuthority> getAuthorities() {
- if(authorities!=null){
- return authorities;
- }
-
- //把permission中String类型的权限信息封装成SimpleGrantedAuthority对象
- authorities = new ArrayList<>();
- for (String permission : permissions) {
- SimpleGrantedAuthority authority = new SimpleGrantedAuthority(permission);
- authorities.add(authority);
- }
- return authorities;
- }

(在此操作之前需编写mapper方法从数据库内查询到权限信息)
LoginUser修改完后我们就可以在UserDetailsServiceImpl中去把权限信息封装到LoginUser中了
在过滤器中加入权限即可
/** * 过滤器 */ @Configuration public class JwtAuthenticationTokenFilter extends OncePerRequestFilter { @Autowired private RedisCache redisCache; @Override protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException { //获取token String token = request.getHeader("token"); if(!StringUtils.hasLength(token)){ //放行 filterChain.doFilter(request,response); return; } //解析token String userid; try { Claims claims = JwtUtil.parseJWT(token); //获取用户id userid = claims.getSubject(); } catch (Exception e) { e.printStackTrace(); throw new RuntimeException("token非法"); } //从redis中获取用户信息 String redisKey="login:"+userid; //通过key从redis获取数据 LoginUser loginUser = redisCache.getCacheObject(redisKey); if(Objects.isNull(loginUser)){ throw new RuntimeException("用户未登录"); } //存入SecurityContextHolder //TODO 获取权限信息封装到Authentication中 UsernamePasswordAuthenticationToken authenticationToken = new UsernamePasswordAuthenticationToken(loginUser,null,loginUser.getAuthorities()); SecurityContextHolder.getContext().setAuthentication(authenticationToken); //放行 filterChain.doFilter(request,response); } }
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。