Spring Boot AOP 自定义注解实现权限控制

╰半橙微兮° 2022-04-03 07:12 766阅读 0赞

写在前面

在网上搜了一天,结果居然没有发现什么太详细的spring boot中使用的aop教程,很多照着做了,结果都没有什么效果实现。。无奈,只能去查谷歌,翻官方文档了

附:Spring中aop的官方文档地址:https://docs.spring.io/spring/docs/5.0.12.BUILD-SNAPSHOT/spring-framework-reference/core.html#aop

Aop的说明

在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。

环境说明

基于spring boot 2.0.4 jdk1.8 实现

接口测试工具:insomnia

事先声明:为了方便流程的演示,本文不涉及token的相关操作
若请求头带入携带id,则视为已登录。
角色信息,用head头的形式提交。

准备阶段

第一步,导入spring boot对spring aop的支持组件

  1. <dependency>
  2. <groupId>org.springframework.boot</groupId>
  3. <artifactId>spring-boot-starter-aop</artifactId>
  4. </dependency>

此组件是基于spring boot自动配置的spring-boot-starter,可以无需配置文件,与spring boot集成

此外,有的人说直接导入aspectj包的,在这里说明一下,这个包是包含aspectj的,而且spring boot官方推荐使用此依赖来在spring boot中使用aop

第二步:构建Controller,提供入口点

  1. @RestController
  2. public class MyController {
  3. @GetMapping("hello")
  4. public String hello(Integer id, String name, Integer age) {
  5. System.out.println("hello方法执行:id==>" + id + ",name==>" + name + ",age==>" + age);
  6. return "hi~ 我不需要用户权限";
  7. }
  8. @GetMapping("user")
  9. public String user(Integer id, String name, Integer age) {
  10. System.out.println("user方法执行:id==>" + id + ",name==>" + name + ",age==>" + age);
  11. return "hi~ 我需要登陆后才可以访问";
  12. }
  13. @GetMapping("admin")
  14. public String admin(Integer id, String name, Integer age) {
  15. System.out.println("admin方法执行:id==>" + id + ",name==>" + name + ",age==>" + age);
  16. return "hi~ 我需要管理员身份才可以访问";
  17. }
  18. }

controller中为了提供了三个方法,分别是不需要具有权限、登陆即可访问、需要管理员角色才可以访问。

同时为了模拟实际操作,传入参数并执行方法。

测试访问:

20181220221559180

准备阶段完毕。开始构建权限操作。

权限逻辑操作

第一步:构建注解类

  1. @Target({ ElementType.METHOD, ElementType.TYPE })
  2. @Retention(RetentionPolicy.RUNTIME)
  3. @Documented
  4. public @interface AuthToken {
  5. /**
  6. * 访问所需的身份,默认为空,为登录即可访问,可以多个定义
  7. *
  8. * @return
  9. * @data 2018年12月19日
  10. * @version v1.0.0.0
  11. */
  12. String[] role_name() default "";
  13. }

此注解定义一个数组属性,role_name,用于标识访问被此注解修饰的方法需要访问的用户具有什么身份。

注:为了方便流程的演示,本文不涉及token的相关操作,权限角色,用head头的形式提交。且若请求头带入携带id,则视为已登录。

第二步:为Controller提供的方法加入此注解。

  1. @RestController
  2. public class MyController {
  3. /**
  4. * 无需校验,不加注解
  5. */
  6. @GetMapping("hello")
  7. public String hello(Integer id, String name, Integer age) {
  8. System.out.println("hello方法执行:id==>" + id + ",name==>" + name + ",age==>" + age);
  9. return "hi~ 我不需要用户权限";
  10. }
  11. /**
  12. * 需要登录校验,加上注解,但不传所需角色
  13. */
  14. @GetMapping("user")
  15. @AuthToken
  16. public String user(Integer id, String name, Integer age) {
  17. System.out.println("user方法执行:id==>" + id + ",name==>" + name + ",age==>" + age);
  18. return "hi~ 我需要登陆后才可以访问";
  19. }
  20. /**
  21. * 需要角色校验,加上注解,并且写入两个角色,本文演示两个角色有一个即可访问,当然写一个可以。
  22. * 注:若想两个角色同时具有,修改后文的逻辑判断即可。
  23. * 若需要更复杂的逻辑操作,推荐使用Spring Security框架。
  24. */
  25. @GetMapping("admin")
  26. @AuthToken(role_name = { "admin", "Administrator" })
  27. public String admin(Integer id, String name, Integer age) {
  28. System.out.println("admin方法执行:id==>" + id + ",name==>" + name + ",age==>" + age);
  29. return "hi~ 我需要管理员身份才可以访问";
  30. }
  31. }

方法解释请看代码中的注释。

第三步(重头戏):写Spring Aop切面,增强方法

先搭出来基本的架子,新建AuthTokenAspect类,引入@Aspect和@Component两个注解

@Aspect标识这个类是一个spring 切面类

@Component 将这个类交给spring处理

二者缺一不可

  1. @Aspect
  2. @Component
  3. public class AuthTokenAspect {
  4. /**
  5. * Spring中使用@Pointcut注解来定义方法切入点
  6. * @Pointcut 用来定义切点,针对方法 @Aspect 用来定义切面,针对类
  7. * 后面的增强均是围绕此切入点来完成的
  8. * 此处仅配置被我们刚才定义的注解:AuthToken修饰的方法即可
  9. *
  10. */
  11. @Pointcut("@annotation(authToken)")
  12. public void doAuthToken(AuthToken authToken) {
  13. }
  14. /**
  15. * 此处我使用环绕增强,在方法执行之前或者执行之后均会执行。
  16. */
  17. @Around("doAuthToken(authToken)")
  18. public Object deBefore(ProceedingJoinPoint pjp, AuthToken authToken) throws Throwable {
  19. System.out.println("---------方法执行之前-------------");
  20. // 执行原方法,并记录返回值。
  21. Object proceed = pjp.proceed();
  22. System.out.println("---------方法执行之后-------------");
  23. return proceed;
  24. }
  25. }

想了解@Pointcut注解中的语法的使用,可以去看看这位博主的博客:https://www.jianshu.com/p/0b78f1156642

这位博主写了一套Spring Boot Aop的学习教程 目录地址:https://www.jianshu.com/p/9093e6ca3378

十分感谢!

先测试一下(此时是没有加权限或者登陆验证的)

针对于上文的Controller 运行user方法来测试

20181220221559199

方法按照预期,执行前和执行后都输出了我们的语句。然而,此处方法的返回值是需要我们注意的。

20181220221559217

此处的返回值不是由原方法返回的。还记得上面写切面时的那句“ 执行原方法,并记录返回值。”吗?
对,上文的方法被我们执行后,将返回值记录了下来。用于保存!注意是保存!
记录完之后,我们的切点还没有走完呢,它还要继续往下走,输出一条“方法执行之后”的语句,才执行的return语句。
不信的话,回来看一下代码。

20181220221559233

对吧,此时接口测试工具的返回值,是我们事先记录下来的返回值(既然是记录下来的,那么如果我们不记录,或者说不执行方法直接自己定义一个返回值呢??),正确返回值只能说是我们制造的一种错觉,让人感觉这个方法正常的执行,并按照我们的预期执行了正确的返回值。

相信到这里,聪明的童鞋就可以悟出来了,这个方法在切点里就像是一个小姑娘,我们想让她执行她就执行,不想执行就不让她执行(不执行pjp.proceed()方法就行了),想返回什么数据,我们自己定义就好了!(但是要与原方法的返回值类型相同,比如我原方法的返回值是String,那么我这里也只能返回String类型的,不能说本来我要的是苹果,你给我一个榴莲就不好了~~)

为了验证我们的推断,我们修改代码如下:

  1. @Around("doAuthToken(authToken)")
  2. public Object deBefore(ProceedingJoinPoint pjp, AuthToken authToken) throws Throwable {
  3. System.out.println("---------方法执行之前-------------");
  4. // 执行原方法,并记录返回值。
  5. Object proceed = pjp.proceed();
  6. System.out.println("---------方法执行之后-------------");
  7. return "哈哈哈哈,方法执行了也没有你要的返回值!";
  8. }

20181220221559249

方法如我们的预期被执行了。那返回值呢?

20181220221559266

好的,到这里为止,我们的大概思路应该就出来了,我们首先拦截带有@AuthToken注解的方法执行,然后去判断@AuthToken注解是否带有所需的角色名(权限),接着去判断request请求中的请求头是否带有id和我们对应所需的角色。(事先就声明了,本文不涉及token,若请求头中id不为空,则为登录,若role_name值对应上我们注解中所携带的注解,则为有权限)

若判断成功,则方法正常执行,并如预期让这个小姑娘有正确的返回值。

那若判断失败,不好意思,方法不执行,且返回值就是我们自定义错误信息。

获得request可以用以下代码:

  1. ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
  2. HttpServletRequest request = attributes.getRequest();

此处涉及到的Spring AOP原理可以去官网自行翻阅文档,用谷歌翻译看,不算太吃力。

上文用到的ProceedingJoinPoint是方法的连接点对象,它是无所不能的,详细信息,同样可以去翻阅本文首提供的Spring官方文档,或者翻阅源码,本文不再赘述。

第四步:完善切面类

完善代码如下:

  1. @Aspect
  2. @Component
  3. public class AuthTokenAspect {
  4. /**
  5. * Spring中使用@Pointcut注解来定义方法切入点
  6. *
  7. * @Pointcut 用来定义切点,针对方法
  8. * @Aspect 用来定义切面,针对类 后面的增强均是围绕此切入点来完成的
  9. * 此处仅配置被我们刚才定义的注解:AuthToken修饰的方法即可
  10. *
  11. */
  12. @Pointcut("@annotation(authToken)")
  13. public void doAuthToken(AuthToken authToken) {
  14. }
  15. /**
  16. * 此处我使用环绕增强,在方法执行之前或者执行之后均会执行。
  17. */
  18. @Around("doAuthToken(authToken)")
  19. public Object deBefore(ProceedingJoinPoint pjp, AuthToken authToken) throws Throwable {
  20. ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
  21. HttpServletRequest request = attributes.getRequest();
  22. // 获取访问该方法所需的role_name信息
  23. String[] role_name = authToken.role_name();
  24. if (role_name == null || role_name.length == 0) {
  25. // 只需登录,验证是否具有id即可。
  26. String id = request.getHeader("id");
  27. /**
  28. * 此处使用短路与,若id==null直接会执行if体,不会继续判断
  29. * 若不等于null,则去验证后面的条件,但是也不会出现因为id为null而出现的空指针异常
  30. * 所以这样写也是安全的。
  31. */
  32. if (id != null && !id.equals("")) {
  33. // 已登录,执行原方法并返回即可。
  34. return pjp.proceed();
  35. }
  36. // 未登录,不执行方法,直接返回错误信息
  37. return "请登陆后再试!";
  38. } else {
  39. // 需要验证身份
  40. String role = request.getHeader("role");
  41. for (String str : role_name) {
  42. /**
  43. * 此处str由于是用role_name中取值,则str必定不为空
  44. * 而从请求头中获取的role有可能为空,则此处调用str的equals方法
  45. * 当然可以直接在获得请求头后进行验证是是否不为空。
  46. */
  47. if (str.equals(role)) {
  48. // 身份匹配成功
  49. return pjp.proceed();
  50. }
  51. }
  52. return "权限校验失败,不具有指定的身份";
  53. }
  54. }
  55. }

方法说明见注释。

测试方法(统一不传id和role):

hello方法:

20181220221559281

user方法:

20181220221559297

admin方法(因为通常逻辑下,具有身份的前提下肯定是登陆过的。head传入id):

20181220221559315

20181220221559442

测试方法(传id和role):

user方法:

20181220221559460

20181220221559559

admin方法:

20181220221559576

20181220221559594

就测试到这吧。。。

这个小Demo的GitHub地址:https://github.com/zichun0507/SpringBoot_AOP

本人也是菜鸟,有何不对的地方,请多多担待。谢谢。

另外,在实际项目开发中,通常是用响应类来封装Controller的返回值的,所以小弟的方法,修改返回值的方法,也是可以用的。。。

响应类代码

  1. package com.lzc.demo.response;
  2. import java.io.Serializable;
  3. import java.util.Date;
  4. import java.util.HashMap;
  5. import java.util.Map;
  6. import com.fasterxml.jackson.annotation.JsonFormat;
  7. public class ResponseResult<T> implements Serializable {
  8. private static final long serialVersionUID = 2719931935414658118L;
  9. private int status;
  10. private String message;
  11. @JsonFormat(timezone = "GMT+8", pattern = "yyyy-MM-dd HH:mm:ss")
  12. private final Date timestamp;
  13. private T data;
  14. private String[] exceptions;
  15. public ResponseResult(int status, String message) {
  16. this.status = status;
  17. this.message = message;
  18. this.timestamp = new Date();
  19. }
  20. public ResponseResult(int status, String message, T data) {
  21. this(status, message);
  22. this.data = data;
  23. }
  24. public ResponseResult(T data) {
  25. this(200, "请求成功!");
  26. this.data = data;
  27. }
  28. public ResponseResult(int status, String message, String[] exceptions) {
  29. this(status, message);
  30. this.exceptions = exceptions;
  31. }
  32. public ResponseResult(ResponseMessage responseMessage) {
  33. this(responseMessage.getStatus(), responseMessage.getMessage());
  34. }
  35. public ResponseResult(ResponseMessage responseMessage, T data) {
  36. this(responseMessage);
  37. this.data = data;
  38. }
  39. public Map<String, Object> toMap() {
  40. Map<String, Object> map = new HashMap<>();
  41. map.put("status", status);
  42. if (message != null) {
  43. map.put("message", message);
  44. }
  45. if (timestamp != null) {
  46. map.put("timestamp", timestamp);
  47. }
  48. if (data != null) {
  49. map.put("data", data);
  50. }
  51. if (exceptions != null) {
  52. map.put("exceptions", exceptions);
  53. }
  54. return map;
  55. }
  56. public int getStatus() {
  57. return status;
  58. }
  59. public void setStatus(int status) {
  60. this.status = status;
  61. }
  62. public String getMessage() {
  63. return message;
  64. }
  65. public void setMessage(String message) {
  66. this.message = message;
  67. }
  68. public Date getTimestamp() {
  69. return timestamp;
  70. }
  71. public T getData() {
  72. return data;
  73. }
  74. public void setData(T data) {
  75. this.data = data;
  76. }
  77. public String[] getExceptions() {
  78. return exceptions;
  79. }
  80. public void setExceptions(String[] exceptions) {
  81. this.exceptions = exceptions;
  82. }
  83. }

发表评论

表情:
评论列表 (有 0 条评论,766人围观)

还没有评论,来说两句吧...

相关阅读

    相关 aop实现定义注解

    注解简单知识 关键字 自定义注解的关键字是@interface 参数类型 自定义注解的参数类型:八大基础类型、String、枚举、注解,还可以是以上类型对应