分享

springboot+拦截器+注解实现自定义权限控制

 jackeyqing 2020-06-08

最近用到一种前端模板技术:vue,在权限控制上没有用springSecurity。因此用拦截器和注解结合实现了权限控制。

实现如下:

1.1 定义权限常量 PermissionConstants.java

  1. /**
  2. * @author blueriver
  3. * @description 权限常量
  4. * @date 2017/11/17
  5. * @since 1.0
  6. */
  7. public class PermissionConstants {
  8. /**
  9. * 管理员-产品列表查询
  10. */
  11. public static final String ADMIN_PRODUCT_LIST = "admin_product_list";
  12. /**
  13. * 管理员-产品详情
  14. */
  15. public static final String ADMIN_PRODUCT_DETAIL = "admin_product_detail";
  16. }
  • 权限也可以不定义为常量,看项目情况


1.2 定义权限的注解 RequiredPermission.java

  1. /**
  2. * @author blueriver
  3. * @description 与拦截器结合使用 验证权限
  4. * @date 2017/11/17
  5. * @since 1.0
  6. */
  7. @Target({ElementType.TYPE, ElementType.METHOD})
  8. @Retention(RetentionPolicy.RUNTIME)
  9. @Inherited
  10. @Documented
  11. public @interface RequiredPermission {
  12. String value();
  13. }
  • ElementType.TYPE,ElementType.METHOD表示注解可以标记类和方法


1.3 权限拦截器 SecurityInterceptor.java

  1. /**
  2. * @author blueriver
  3. * @description 权限拦截器
  4. * @date 2017/11/17
  5. * @since 1.0
  6. */
  7. public class SecurityInterceptor implements HandlerInterceptor {
  8. @Autowired
  9. private AdminUserService adminUserService;
  10. @Override
  11. public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
  12. // 验证权限
  13. if (this.hasPermission(handler)) {
  14. return true;
  15. }
  16. // null == request.getHeader("x-requested-with") TODO 暂时用这个来判断是否为ajax请求
  17. // 如果没有权限 则抛403异常 springboot会处理,跳转到 /error/403 页面
  18. response.sendError(HttpStatus.FORBIDDEN.value(), "无权限");
  19. return false;
  20. }
  21. /**
  22. * 是否有权限
  23. *
  24. * @param handler
  25. * @return
  26. */
  27. private boolean hasPermission(Object handler) {
  28. if (handler instanceof HandlerMethod) {
  29. HandlerMethod handlerMethod = (HandlerMethod) handler;
  30. // 获取方法上的注解
  31. RequiredPermission requiredPermission = handlerMethod.getMethod().getAnnotation(RequiredPermission.class);
  32. // 如果方法上的注解为空 则获取类的注解
  33. if (requiredPermission == null) {
  34. requiredPermission = handlerMethod.getMethod().getDeclaringClass().getAnnotation(RequiredPermission.class);
  35. }
  36. // 如果标记了注解,则判断权限
  37. if (requiredPermission != null && StringUtils.isNotBlank(requiredPermission.value())) {
  38. // redis或数据库 中获取该用户的权限信息 并判断是否有权限
  39. Set<String> permissionSet = adminUserService.getPermissionSet();
  40. if (CollectionUtils.isEmpty(permissionSet) ){
  41. return false;
  42. }
  43. return permissionSet.contains(requiredPermission.value());
  44. }
  45. }
  46. return true;
  47. }
  48. @Override
  49. public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
  50. // TODO
  51. }
  52. @Override
  53. public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
  54. // TODO
  55. }
  56. }

1.4 拦截器注入的配置 MVCConfig.java

  1. @Configuration
  2. public class MVCConfig extends WebMvcConfigurerAdapter {
  3. @Bean
  4. public SecurityInterceptor securityInterceptor() {
  5. return new SecurityInterceptor();
  6. }
  7. @Override
  8. public void addInterceptors(InterceptorRegistry registry) {
  9. registry.addInterceptor(securityInterceptor()).excludePathPatterns("/static/*")
  10. .excludePathPatterns("/error").addPathPatterns("/**");
  11. }
  12. }
  • springboot中注入拦截器

  • 这里说一下多拦截器的配置方式


  • 最后强调一点:只有经过DispatcherServlet 的请求,才会走拦截器链,我们自定义的Servlet 请求是不会被拦截的,比如我们自定义的Servlet地址 http://localhost:8080/xs/myservlet 是不会被拦截器拦截的。并且不管是属于哪个Servlet 只要复合过滤器的过滤规则,过滤器都会拦截。

    最后说明下,我们上面用到的 WebMvcConfigurerAdapter 并非只是注册添加拦截器使用,其顾名思义是做Web配置用的,它还可以有很多其他作用,通过下面截图便可以大概了解,具体每个方法都是干什么用的,留给大家自己研究(其实都大同小异也很简单)。



1.5 ProductController.java

  1. /**
  2. * @author blueriver
  3. * @description 产品管理
  4. * @date 2017/10/25
  5. * @since 1.0
  6. */
  7. @Controller
  8. @RequestMapping("/product")
  9. // @PermissionConstants.ADMIN_PRODUCT_MANAGEMENT
  10. public class ProductController {
  11. /**
  12. * 产品列表
  13. *
  14. * @return
  15. */
  16. @RequestMapping("/list")
  17. @RequiredPermission(PermissionConstants.ADMIN_PRODUCT_LIST) // 权限注解
  18. public String list() {
  19. // 省略产品列表查询逻辑
  20. return "/product/list";
  21. }
  22. /**
  23. * 产品详情
  24. *
  25. * @return
  26. */
  27. @RequestMapping("/detail")
  28. @RequiredPermission(PermissionConstants.ADMIN_PRODUCT_DETAIL) // 权限注解
  29. public String detail() {
  30. // 省略查询产品详情的逻辑
  31. return "/product/edit";
  32. }
  33. /**
  34. * 删除产品
  35. *
  36. * @return
  37. */
  38. @RequestMapping("/delete")
  39. public String delete() {
  40. // 省略删除产品的逻辑
  41. return "/product/list";
  42. }
  43. }
  44. 如果没有标记权限注解,则不会验证该请求的权限,如/product/delete 请求

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多