Spring Cloud Gateway 获取请求体消息

一、场景介绍

  在通常情况下,微服务的业务功能都需要通过网关来进行转发,记录业务请求日志通过网关来记录是最有利的方式。

二、演示代码
  • 测试版本

    组件名版本号
    Spring Boot2.2.5.RELEASE
    Spring CloudHoxton.SR3
    Spring Cloud Alibaba2.2.1.RELEASE
  • 测试代码

    import com.alibaba.fastjson.JSON;
    import lombok.extern.slf4j.Slf4j;
    import org.springframework.cloud.gateway.filter.GatewayFilter;
    import org.springframework.cloud.gateway.filter.GatewayFilterChain;
    import org.springframework.cloud.gateway.filter.GlobalFilter;
    import org.springframework.cloud.gateway.filter.factory.rewrite.ModifyRequestBodyGatewayFilterFactory;
    import org.springframework.cloud.gateway.filter.factory.rewrite.RewriteFunction;
    import org.springframework.core.Ordered;
    import org.springframework.http.HttpMethod;
    import org.springframework.http.MediaType;
    import org.springframework.http.server.reactive.ServerHttpRequest;
    import org.springframework.stereotype.Component;
    import org.springframework.web.server.ServerWebExchange;
    import reactor.core.publisher.Mono;
    
    import javax.annotation.PostConstruct;
    
    /**
     * @ClassName:   CustomGlobalLogFilter
     * @Description: 自定义全局日志过滤器(借用 ModifyRequestBodyGatewayFilterFactory, 该类的本身功能是为了修改请求体数据的, 这里也是比较巧妙的利用了他本身已经对请求体参数进行了封装, 在 RewriteFunction 中可以直接拿到请求参数的特点)
     * @Author:      Rambo
     * @CreateDate:  2020/7/5 21:43
     * @UpdateUser:  Rambo
     * @UpdateDate:  2020/7/5 21:43
     * @Version:     1.0.0
     */
    @Component
    @Slf4j
    public class CustomRequestGlobalLogFilter implements GlobalFilter, Ordered {
    
        private GatewayFilter delegate;
    
        @Override
        public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
            // 获取请求对象
            ServerHttpRequest request = exchange.getRequest();
    
            // 获取请求参数信息
            MediaType contentType = request.getHeaders().getContentType();
            log.info("------------------------> REQUEST URI = [{}]" , request.getURI());
            log.info("------------------------> REQUEST HEADER = [{}]" , request.getHeaders());
            log.info("------------------------> REQUEST METHOD = [{}]" , request.getMethodValue());
            if (HttpMethod.GET == request.getMethod()) {
                log.info("------------------------> REQUEST PARAMS = [{}]" , JSON.toJSONString(request.getQueryParams().toSingleValueMap()));
                // 正常放行的处理,进入下一个过滤链
                return chain.filter(exchange);
    
            // 只能读取到 contentType 为 application/json 的数据
            } else if (contentType.equals(MediaType.APPLICATION_JSON) || contentType.equals(MediaType.APPLICATION_JSON_UTF8)){
                return delegate.filter(exchange, chain);
            }
            // 正常放行的处理,进入下一个过滤链
            return chain.filter(exchange);
        }
    
        @PostConstruct
        public void init() {
            this.delegate = new ModifyRequestBodyGatewayFilterFactory().apply(this.getConfig());
        }
    
        private ModifyRequestBodyGatewayFilterFactory.Config getConfig() {
            ModifyRequestBodyGatewayFilterFactory.Config cf = new ModifyRequestBodyGatewayFilterFactory.Config();
            cf.setRewriteFunction(Object.class, Object.class, getRewriteFunction());
            return cf;
        }
    
        private RewriteFunction<Object, Object> getRewriteFunction() {
            return (serverWebExchange, body) -> {
                // 这里的 body 就是请求体参数, 类型是 LinkedHashMap, 可以根据需要转成JSON
                log.info("------------------------> REQUEST PARAMS = [{}]", JSON.toJSONString(body));
                return Mono.just(body);
            };
        }
    
        @Override
        public int getOrder() {
            // 定义过滤链的顺序
            return 0;
        }
    }
    
  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Cloud Gateway 是一个反向代理和路由器,它可以拦截进入的请求并将它们路由到不同的服务。Spring Cloud Gateway 可以通过以下步骤获取请求: 1. 在 Spring Cloud Gateway 的路由配置,定义一个过滤器: ```java @Bean public RouteLocator customRouteLocator(RouteLocatorBuilder builder) { return builder.routes() .route("path_route", r -> r.path("/foo/**") .filters(f -> f.filter(new RequestBodyFilter())) .uri("http://localhost:8080")) .build(); } ``` 这里定义了一个名为 `RequestBodyFilter` 的过滤器,用于获取请求。 2. 实现 `RequestBodyFilter` 过滤器: ```java public class RequestBodyFilter implements GatewayFilter { @Override public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) { return exchange.getRequest().getBody() .flatMap(body -> { // 处理请求 byte[] bytes = new byte[body.readableByteCount()]; body.read(bytes); String requestBody = new String(bytes); System.out.println(requestBody); // 重新设置请求 DataBuffer buffer = exchange.getResponse().bufferFactory().wrap(bytes); exchange.getRequest().mutate().body(buffer); return chain.filter(exchange); }); } } ``` 这里的 `filter` 方法会获取请求,并进行处理。处理完成后,需要重新设置请求,以便后续的过滤器或路由器可以正确地处理请求。 3. 在请求发送请求: ```bash curl -X POST http://localhost:8080/foo -d '{"name": "John"}' ``` 这里使用 `curl` 命令发送一个 POST 请求,并在请求包含 JSON 数据。Spring Cloud Gateway 会拦截这个请求,并使用定义的过滤器获取请求。 以上就是 Spring Cloud Gateway 获取请求的方法。需要注意的是,获取请求可能会影响性能,因此应该谨慎使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值