webflux切面拦截权限,webflux整合aop,webflux获取request

本文介绍了在Spring WebFlux与Netty结合的异步环境中,如何通过ReactiveHttpContextHolder模拟RequestContextHolder来获取ServerHttpRequest和ServerHttpResponse。并展示了如何编写一个WebFilter设置请求到上下文,以及一个切面实现基于token的权限校验,确保在AOP中能正确获取filter中设置的请求。测试结果显示该方法在WebFlux+Netty场景下有效。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

背景

在springboot+tomcat应用中获取request对象可以使用RequestContextHolder.getRequestAttributes()的方式来获取,此种方式的核心在于request所在容器被放在threadlocal中,但是webflux结合netty项目却不能这么使用,因为webflux是异步响应式的,下面介绍下异步服务webflux+netty如何便捷获取request。

编写基于webflux的RequestContextHolder实现

  1. 编写ReactiveHttpContextHolder类用来模拟RequestContextHolder
public class ReactiveHttpContextHolder {
    //获取当前请求对象
    public static Mono<ServerHttpRequest> getRequest() {
        return Mono.subscriberContext()
                .map(context -> context.get(Info.CONTEXT_KEY).getRequest());
    }

    //获取当前response
    public static Mono<ServerHttpResponse> getResponse(){
        return Mono.subscriberContext()
                .map(context -> context.get(Info.CONTEXT_KEY).getResponse());
    }

    public static final class Info{
        public static final Class<ServerWebExchange> CONTEXT_KEY = ServerWebExchange.class;
    }
}
  1. 编写过滤器类用于设置request到容器中
@Component
@ConditionalOnWebApplication(type = ConditionalOnWebApplication.Type.REACTIVE)
public class AppFilter implements WebFilter {

    @Override
    public Mono<Void> filter(ServerWebExchange exchange, WebFilterChain chain) {
        //设置当前请求
         return chain.filter(exchange)
                .subscriberContext(context -> context.put(ReactiveHttpContextHolder.Info.CONTEXT_KEY, exchange));
    }
}

编写切面实现token拦截,校验权限

  1. 切面逻辑代码
@Component
@Aspect
public class AuthAspect extends BaseController {

    @Pointcut("execution(public * xx.xx.xxController.*(..))")
    private void classRule(){}
    //带有AuthIgnore注解的不校验
    @Pointcut("@annotation(xx.AuthIgnore)")
    private void ignoreRule(){}

    @Around("classRule() && !ignoreRule()")
    public Mono<Object> aroundInvoke(ProceedingJoinPoint joinPoint){
        //获取token
        Mono<ServerHttpRequest> requestMono = ReactiveHttpContextHolder.getRequest();
        return requestMono.flatMap(request -> {
            //检查token
            String token = getHeader("token", request);
            if (token == null || TokenCacheContainer.getTokenCache().get(token) == null){
                BusException ex = new BusException("01300001");
                return buildErrMsg(ex);
            } else {
                try {
                    return (Mono<Object>) joinPoint.proceed();
                } catch (Throwable throwable) {
                    LogFactory.getLog(AuthAspect.class).error("", throwable);
                    BusException ex = new BusException("01300000");
                    return buildErrMsg(ex);
                }
            }
        });
    }

    private Mono<String> buildErrMsg(BusException ex){
        ApiJsonResult errResult = ApiJsonResult.createErrResult(ex.getErrorCode(), ex.getReason(), ex.getResolve());
        return Mono.just( errResult.toString() );
    }
}
  1. 说明
    过滤器会在切面之前执行,因此aop中能获取到在filter中设置的request
  2. 测试结果
    在这里插入图片描述
### 如何在 Spring AOP 切面获取请求的指定参数 为了实现在 AOP 切面获取请求的特定参数,可以利用 `ProceedingJoinPoint` 对象来访问方法签名以及传递给该方法的实际参数。通过解析这些信息并结合目标对象的方法反射机制,能够定位到具体的参数名称及其对应的值。 当处理基于注解的方式时,还可以借助于 `@AspectJ` 风格的支持来进行更加灵活的操作。下面是一个简单的例子展示如何创建一个切面用于捕获控制器中的入参: #### 定义切入点表达式匹配所有带有@RequestMapping注解的方法 ```java @Pointcut("@annotation(org.springframework.web.bind.annotation.RequestMapping)") public void requestMappingAnnotatedMethods() {} ``` #### 实现环绕通知,在其中提取请求的具体参数 ```java @Around("requestMappingAnnotatedMethods()") public Object logRequestParameters(ProceedingJoinPoint joinPoint) throws Throwable { MethodSignature signature = (MethodSignature) joinPoint.getSignature(); // 获取方法形参列表 String[] paramNames = signature.getParameterNames(); // 获取实际传入的参数值数组 Object[] args = joinPoint.getArgs(); // 将参数名与其对应值组合成键值对形式输出 StringBuilder paramsBuilder = new StringBuilder("Request Parameters: "); for(int i=0; i<paramNames.length && i<args.length ;i++){ paramsBuilder.append(paramNames[i]).append("=").append(args[i]); if(i != paramNames.length - 1){ paramsBuilder.append(", "); } } logger.info(paramsBuilder.toString()); try{ // 执行被代理的目标方法 return joinPoint.proceed(); } catch(Throwable t){ throw t; } } ``` 上述代码片段展示了如何在一个环绕通知(`@Around`)内工作,它会拦截任何标注有 `@RequestMapping` 注解的方法调用,并记录下每次调用所携带的所有参数的信息[^2]。 对于更复杂的情况,比如想要只针对某些特定类型的参数进行操作,则可以在编写切入点规则的时候进一步细化条件;或者是采用其他方式如使用 AspectJ 的 ITD(Inter-Type Declaration)特性等高级功能来增强灵活性和可维护性。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值