Spring AOP实现权限检查的功能
作者:溪源的奇思妙想 发布时间:2023-08-10 06:51:14
前言
最近开发了一个接口,完成后准备自测时,却被 * 拦截了,提示:(AUTH-NO)未能获得有效的请求参数!
怎么会这样呢?
于是我全局搜了这个提示语,结果发现它被出现在一个Aspect类当中了,并且把一个 @interface 作为了一个切点,原来这里利用了Spring AOP面向切面的方式进行权限控制。
正文
Spring AOP 即面向切面,是对OOP面向对象的一种延伸。
AOP机制可以让开发者把业务流程中的通用功能抽取出来,单独编写功能代码。在业务流程执行过程中,Spring框架会根据业务流程要求,自动把独立编写的功能代码切入到流程的合适位置。
我们通过AOP机制可以实现:Authentication 权限检查、Caching 缓存、Context passing 内容传递、Error handling 错误处理等功能,这里我们讲一下怎么用Spring AOP来实现权限检查。
Spring AOP实现权限检查
引入依赖
<!--lombok-->
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.18.2</version>
<optional>true</optional>
</dependency>
<!--Spring AOP-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aspects</artifactId>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.2</version>
</dependency>
<dependency>
<groupId>aopalliance</groupId>
<artifactId>aopalliance</artifactId>
<version>1.0</version>
</dependency>
MyPermissionTag.class自定义注解
@Retention: 用来修饰注解,是注解的注解,称为元注解。
@Target:用来说明对象的作用范围
/**
* 用户请求权限校验
*/
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface MyPermissionTag {
String value() default "";
String name() default "";
}
这里特别讲一下@Retention
,按生命周期来划分可分为3类:
RetentionPolicy.SOURCE
:注解只保留在源文件,当Java文件编译成class文件的时候,注解被遗弃(运行时去动态获取注解信息);RetentionPolicy.CLASS
:注解被保留到class文件,但jvm加载class文件时候被遗弃,这是默认的生命周期(在编译时进行一些预处理操作);RetentionPolicy.RUNTIME
:注解不仅被保存到class文件中,jvm加载class文件之后,仍然存在(做一些检查性的操作);
这3个生命周期分别对应于:Java源文件(.java文件) —> .class文件 —> 内存中的字节码。
AuthInterceptor 权限检查的切面
这里简单介绍一下,切面的执行方法和其执行顺序:
@Around
通知方法将目标方法封装起来@Before
通知方法会在目标方法调用之前执行@After
通知方法会在目标方法返回或者异常后执行@AfterReturning
通知方法会在目标方法返回时执行@Afterthrowing
通知方法会在目标方法抛出异常时执行
这里以一个返回正常的情况为例:(异常替换最后一步即可)
AuthInterceptor.class
注意要在启动类扫描这个class,并且添加 @EnableAspectJAutoProxy(proxyTargetClass =
true)
@Slf4j
@Aspect
@Component
public class AuthInterceptor {
/**
* 参数处理
*
* @param point
*/
@Before("@annotation(com.luo.common.tag.MyPermissionTag)")
public void beforeProReq(JoinPoint point) {
log.info("前置拦截-开始");
Request req = getOperationRequest(point.getArgs());
if (req != null) {
//解密帐号
log.info("前置拦截-开始解密ACCOUNT:{}", req.getAccount());
log.info("前置拦截-结束解密ACCOUNT:{}", req.getAccount());
}
log.info("前置拦截-结束");
}
@Around("@annotation(com.luo.common.tag.MyPermissionTag)")
public Object authCheck(ProceedingJoinPoint pjp) throws Throwable {
log.info("权限拦截-开始");
//请求方法
ReqMethod reqMethod = getPermissionTag(pjp);
MyPermissionTag myPermissionTag =reqMethod.perTag;
log.info(myPermissionTag.value()); //获取配置的值
log.info("权限拦截-开始-拦截到方法:{}", reqMethod.getMethodName());
if("true".equals(myPermissionTag.value().toString())){
//错误返回
Response notGoRes = new Response();
Request req = getOperationRequest(pjp.getArgs());
// 校验请求对象
if (req == null) {
notGoRes.setErrorMsg("(AUTH)未能获得有效的请求参数!");
log.info("(AUTH-NO)未能获得有效的请求参数!");
return notGoRes;
}else {//可以在这里根据请求参数对请求做进一步校验
log.info("完成请求校验:"+req);
}
}else {
log.info("未开启权限校验");
}
return pjp.proceed();
}
/**
* 获取 request 接口中的请求参数
* @param args
* @return
*/
private Request getOperationRequest(Object[] args) {
if (args == null || args.length <= 0) {
log.error("AUTH权限验证:拦截方法的请求参数为空!");
return null;
}
Object obj = args[0];
if (obj instanceof Request) {
log.info("AUTH权限验证:请求对象为正确的OperationRequest对象");
return (Request) obj;
}
return null;
}
/**
* 获取拦截的资源标签
* 这里可以获取方法名+注解信息(包括 key+value 等)
* @param pjp
* @return
* @throws SecurityException
* @throws NoSuchMethodException
*/
private ReqMethod getPermissionTag(ProceedingJoinPoint pjp) throws NoSuchMethodException, SecurityException {
Signature signature = pjp.getSignature();
MethodSignature methodSignature = (MethodSignature) signature;
Method targetMethod = methodSignature.getMethod();
Method realMethod = pjp.getTarget().getClass().getDeclaredMethod(signature.getName(), targetMethod.getParameterTypes());
MyPermissionTag permissionTag = realMethod.getAnnotation(MyPermissionTag.class);
return new ReqMethod(permissionTag, realMethod.getName());
}
@Setter
@Getter
class ReqMethod {
private MyPermissionTag perTag;
private String methodName;
public ReqMethod(MyPermissionTag perTag, String methodName) {
this.perTag = perTag;
this.methodName = methodName;
}
}
}
验证
测试接口
@PostMapping("/helloluo")
@MyPermissionTag(value = "true")
public String helloluo(UserPojoReq userPojoReq){
return "Hello World";
}
发送请求
验证
来源:https://blog.csdn.net/weixin_40990818/article/details/108269875


猜你喜欢
- using System.Drawing;using System.Drawing.Drawing2D;using System.Drawi
- SelectMany操作符提供了将多个from子句组合起来的功能,相当于数据库中的多表连接查询,它将每个对象的结果合并成单个序列。示例:st
- 本文实例讲述了C#获取汉字字符串拼音首字母的方法。分享给大家供大家参考。具体如下:这个C#类经常能够用到,将提取汉字的拼音首字母,方便用户查
- 在java里, 我们可以使用Executors.newFixedThreadPool 来创建线程池, 然后就可以不停的创建新任务,并用线程池
- 前言:一个游戏里的一个人物会存在多种状态,那么就需要有一个专门管理这些状态的类。不然会显得杂乱无章,不易于后面状态的增加或者减少。思路:既然
- 在 fluro 中,定义路由处理器 Handler 时可以指定该页面的默认转场形式,或者在使用 navigateTo 方法是可以设置页面跳转
- 这几天面试中有遇到关于main数组中的args数组传值的问题,一般是从命令提示符中传值,也可以直接在java代码中赋值。而且这个数组的长度是
- Android 自定义gradle property在Android studio上运行项目,gradle的配置是必不可少的,但是随着项目的
- 在网上很多关于dubbo异常统一处理的博文,90%都是抄来抄去。大多都是先上一段dubbo中对于异常的统一处理的原码,然后说一堆的(甚至有1
- 悬浮窗在安卓中实现起来还是比较容易的,这几天在网上温习了相关资料,运行在我安卓6.0手机上才发现,原来在6.0手机上不是行的。第一反应肯定是
- 从接触springboot开始,便深深的被它的简洁性深深的折服了,精简的配置,方便的集成,使我再也不想用传统的ssm框架来搭建项目,一大堆的
- 前言字符串分割函数strtok,大家可能都知道他怎么使用,一旦要用的时候就会心生疑惑,不知道它的内部的实现,废话不多说,本篇就来带大家看看s
- Spring SecuritySpring Security是能够为J2EE项目提供综合性的安全访问控制解决方案的安全框架。它依赖于Serv
- 方法一:实现Comparator接口,并重写compare方法实体类代码:import java.util.Comparator;/** *
- 我实现的思路:1.继承ImageView类2.重写onTouchEvent方法,在ACTION_MOVE(即移动时),记录下所经过的点坐标,
- 使用ehcache-spring-annotations使得在工程中简单配置即可使用缓存下载地址:http://code.google.co
- Java是一门天然的面向对象的语言。而所有我们手动创造出来的类,都继承于同一个类,即Object类。可以看一下Object类的结构nativ
- 写在前面:接下来很长一段时间的文章主要会记录一些项目中实际遇到的问题及对应的解决方案,在相应代码分析时会直指问题所在,不会将无关的流程代码贴
- java文件输出流是一种用于处理原始二进制数据的字节流类。为了将数据写入到文件中,必须将数据转换为字节,并保存到文件。package com
- 基于 springboot+vue 的测试平台(练手项目)开发继续更新。在接口编辑页中点击发送接口请求,除了显示响应体外,还可以显示响应头等