Spring Cloud Gateway 打印 Request/Response Body

1 篇文章 0 订阅

一、需求

要求对Spring Cloud Gateway(后面统一使用SCG表述)代理的后端服务进行日志记录。记录的内容包括请求信息(请求方式、请求地址、请求头、RequestBody等)、响应信息(响应头、响应码及ResponseBody等);

二、环境及版本

        Spring Boot  2.3.5.RELEASE

        Spring Cloud Gateway  2.2.7.RELEASE

三、问题描述

因为SCG 是基于 Spring 5.x 版本的 WebFlux 实现的,源码中很多地方也都是异步实现的;且我本人对 WebFlux 并没有太多开发经验。结果在获取请求与响应的 Body 内容时遇到了很多坑....

先说第一个坑: 

java.lang.IllegalStateException: Only one connection receive subscriber allowed.

 这个问题是由于在 Filter 中提前消费了 RequestBody 导致后面的 Filter 在获取 RequestBody时返回 Null 就会提示这个异常;网上对这个问题的解决文章有很多,基本都是 2018年左右的文章内容,当你尝试按照这些文章去解决问题的时候你会发现这并无卵用;再此处贴出一段代码来为大家讲解究其何因:

位置 1 创建了一个  AtomicReference 的实例对象用来缓存 Body数据(只要知道它是一个容器就可以了).

位置 2 是一个异步流订阅的骚操作,在这个骚操作中将得到的 Body 数据转换为 String 类型 然后释放内存空间;接着在将这个 String 放倒 AtomicReference 对象容器中;

位置 3 将缓存在 AtomicReference 对象容器中的 Body 取出;

执行完上面三个步骤后你会发现拿到的 Body 是空的!!!! 

原因是 位置2的操作 是一个异步订阅操;在你当前线程执行的时候 位置2地方的代码可能并没有执行,所以导致你在位置3 获取数据是一个 Null 对象。这段代码应该在 2018 年发布的 SCG 版本中应该是可以获得对象的(盲目猜测当时的SCG 并非是真正的异步执行),但是现在肯定不行了。

 四、解决方案

         注: 你现在看到的解决方式可能会在以后的SCG 版本中失效,所以你在测试的时候如果发现没有用一定要认真看一下自己当前的SCG 版本是否和我的一致;

经过几天时间的源码阅读发现了两个可以借鉴的类:ModifyRequestBodyGatewayFilterFactory.java 

ModifyResponseBodyGatewayFilterFactory.java

通过对上面这两个类的理解我把它整理成为了我需要的 FilterFactory :

打印 RequestBody 代码:

@Override
        public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {

            if (isTargetLogMediaType(exchange.getRequest())) {
                Class inClass = String.class;
                Class outClass = String.class;
                String requestId = exchange.getAttribute(AppConfig.GATEWAY_REQUEST_ID_KEY);
                ServerRequest serverRequest = ServerRequest.create(exchange, messageReaders);
                Mono<?> modifiedBody = serverRequest.bodyToMono(inClass).flatMap(originalBody -> {
                    MDC.put("type", "REQUEST-BODY");
                    log.info(String.format("[%s] %s", requestId, originalBody));
                    return Mono.just(originalBody);
                });

                BodyInserter bodyInserter = BodyInserters.fromPublisher(modifiedBody, outClass);
                HttpHeaders headers = new HttpHeaders();
                headers.putAll(exchange.getRequest().getHeaders());
                headers.remove(HttpHeaders.CONTENT_LENGTH);

                CachedBodyOutputMessage outputMessage = new CachedBodyOutputMessage(exchange, headers);
                return bodyInserter.insert(outputMessage, new BodyInserterContext()).then(Mono.defer(() ->
                        chain.filter(exchange.mutate().request(decorate(exchange, headers, outputMessage)).build())
                )).onErrorResume((Function<Throwable, Mono<Void>>) throwable -> Mono.error(throwable));
            }

            return chain.filter(exchange);
        }

private ServerHttpRequestDecorator decorate(ServerWebExchange exchange, HttpHeaders headers, CachedBodyOutputMessage outputMessage) {
        return new ServerHttpRequestDecorator(exchange.getRequest()) {
            @Override
            public HttpHeaders getHeaders() {
                long contentLength = headers.getContentLength();
                HttpHeaders httpHeaders = new HttpHeaders();
                httpHeaders.putAll(headers);
                if (contentLength > 0) {
                    httpHeaders.setContentLength(contentLength);
                } else {
                    httpHeaders.set(HttpHeaders.TRANSFER_ENCODING, "chunked");
                }
                return httpHeaders;
            }

            @Override
            public Flux<DataBuffer> getBody() {
                return outputMessage.getBody();
            }
        };
    }

打印 ResponseBody 代码:

@Override
public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
     return chain.filter(exchange.mutate().response(decorate(exchange)).build());
}


private ServerHttpResponse decorate(ServerWebExchange exchange) {
      return new ServerHttpResponseDecorator(exchange.getResponse()) {

      @Override
      public Mono<Void> writeWith(Publisher<? extends DataBuffer> body) { // Mono<NettyDataBuffer>
                    if (isTargetLogMediaType(getDelegate())) {
                        Class inClass = String.class;
                        Class outClass = String.class;

                        String requestId = exchange.getAttribute(AppConfig.GATEWAY_REQUEST_ID_KEY);
                        String originalResponseContentType = exchange.getAttribute(ORIGINAL_RESPONSE_CONTENT_TYPE_ATTR);
                        HttpHeaders httpHeaders = new HttpHeaders();
                        httpHeaders.add(HttpHeaders.CONTENT_TYPE, originalResponseContentType);

                        ClientResponse clientResponse = ClientResponse
                                .create(exchange.getResponse().getStatusCode())
                                .headers(headers -> headers.putAll(httpHeaders))
                                .body(Flux.from(body)).build();

                        Mono<?> modifiedBody = clientResponse.bodyToMono(inClass).flatMap(originalBody -> {
                            MDC.put("type", "RESPONSE-BODY");
                            log.info(String.format("[%s] %s", requestId, originalBody));
                            return Mono.just(originalBody);
                        });

                        BodyInserter bodyInserter = BodyInserters.fromPublisher(modifiedBody, outClass);
                        CachedBodyOutputMessage outputMessage = new CachedBodyOutputMessage(exchange, exchange.getResponse().getHeaders());
                        return bodyInserter.insert(outputMessage, new BodyInserterContext()).then(Mono.defer(() -> {
                            Flux<DataBuffer> messageBody = outputMessage.getBody();
                            HttpHeaders headers = getDelegate().getHeaders();
                            if (!headers.containsKey(HttpHeaders.TRANSFER_ENCODING)) {
                                messageBody = messageBody.doOnNext(data -> headers.setContentLength(data.readableByteCount()));
                            }
                            return getDelegate().writeWith(messageBody);
                        }));
                    }
                    return getDelegate().writeWith(body);
                }

                @Override
                public Mono<Void> writeAndFlushWith(Publisher<? extends Publisher<? extends DataBuffer>> body) {
                    return writeWith(Flux.from(body).flatMapSequential(p -> p));
                }
            };
        }

 

 

  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: Spring Cloud Gateway 可以通过自定义过滤器来获取请求体(body)。具体步骤如下: 1. 创建一个自定义过滤器类,实现 GatewayFilter 接口。 2. 在过滤器类中重写 filter 方法,在该方法中获取请求体。 3. 在 Spring Cloud Gateway 配置文件中配置该过滤器。 示例代码如下: ```java @Component public class MyFilter implements GatewayFilter { @Override public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) { // 获取请求体 ServerHttpRequest request = exchange.getRequest(); Flux<DataBuffer> body = request.getBody(); // 处理请求体 // ... // 调用下一个过滤器 return chain.filter(exchange); } } ``` 在 Spring Cloud Gateway 配置文件中配置该过滤器: ```yaml spring: cloud: gateway: routes: - id: my_route uri: http://localhost:8080 predicates: - Path=/my_path/** filters: - MyFilter ``` 其中,MyFilter 是自定义过滤器类的名称。在 filters 配置中指定该过滤器即可。 ### 回答2: Spring Cloud Gateway是一个基于Spring Boot的API网关,它允许开发者以统一的方式管理和路由HTTP请求到多个微服务。在实际开发中,有时需要获取HTTP请求的body,在Spring Cloud Gateway中获取HTTP请求的body需要注意以下几点: 1. 所有的Route Predicate都需要配置读取HTTP请求体,否则在路由到下游服务时,请求体会丢失。 2. 如果请求体是将JSON字符串作为参数传递,则需要使用JSON库将字符串转成JSON对象。Spring Cloud Gateway中推荐使用与Spring Framework组件集成的Jackson JSON库。 3. HTTP请求的body只能读取一次,所以需要配置路由过滤器来实现将读取过的请求体保存在请求上下文中,以便后续的路由过滤器和路由处理器获取请求体。 在Spring Cloud Gateway中获取HTTP请求的body,可以通过自定义GatewayFilter来实现。下面给出获取HTTP请求体的代码示例: ```java public class BodyGatewayFilterFactory extends AbstractGatewayFilterFactory<BodyGatewayFilterFactory.Config> { public BodyGatewayFilterFactory() { super(Config.class); } @Override public GatewayFilter apply(Config config) { return (exchange, chain) -> { return DataBufferUtils.join(exchange.getRequest().getBody()) .flatMap(dataBuffer -> { byte[] bytes = new byte[dataBuffer.readableByteCount()]; dataBuffer.read(bytes); DataBufferUtils.release(dataBuffer); String requestBody = new String(bytes, Charset.forName("UTF-8")); exchange.getAttributes().put("requestBody", requestBody); return chain.filter(exchange); }); }; } public static class Config { } } ``` 在上面的代码中,使用DataBufferUtils.join()函数将请求体存储在字节数组中,并通过exchange的setAttribute()方法存储到请求上下文中。这样,在后续的路由过滤器和路由处理器中就可以通过读取exchange.getAttributes().get("requestBody")来获取HTTP请求的body,而无需重新读取请求体。 ### 回答3: Spring Cloud Gateway是一个基于Spring Boot网关。它可以在微服务架构中起到路由、负载均衡、API管理等多种作用。 在Spring Cloud Gateway中,获取请求体有两种方式:获取单个请求体和获取多个请求体。 获取单个请求体: 在Spring Cloud Gateway中,获取单个请求体可以使用Exchange对象的getBody()方法。这个方法会返回一个Mono对象,需要使用subscribe()方法来订阅结果。 例如: ```java public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) { Mono<String> requestBody = exchange.getRequest().getBodyToMono(String.class); requestBody.subscribe(content -> { // 对请求体进行操作 }); return chain.filter(exchange); } ``` 上面代码中,我们使用getBodyToMono()获取请求体,然后使用subscribe()方法来订阅请求体的内容。订阅成功后,我们可以对请求体进行操作。 获取多个请求体: 在Spring Cloud Gateway中,获取多个请求体可以使用GlobalFilter。GlobalFilter是一种全局过滤器,可以对所有的请求进行处理。 我们可以创建一个自定义的GlobalFilter,然后在filter()方法中获取请求体。 例如: ```java @Component public class MyGlobalFilter implements GlobalFilter { @Override public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) { MediaType mediaType = exchange.getRequest().getHeaders().getContentType(); Flux<DataBuffer> body = exchange.getRequest().getBody(); return chain.filter(exchange.mutate().request( exchange.getRequest().mutate().body(Flux.just(body)).build()) .build()); } } ``` 上面代码中,我们创建了一个MyGlobalFilter类,并实现了GlobalFilter接口。在filter()方法中,我们使用getBody()获取请求体。获取请求体后,我们更改了请求体的数据,然后使用build()方法创建了一个新的Exchange对象,并返回chain.filter()。 总结: Spring Cloud Gateway可以通过Exchange对象来获取请求体。可以使用getBody()方法获取单个请求体,也可以使用GlobalFilter获取多个请求体。 注意:在Spring Cloud Gateway中,请求体是一个Flux对象。如果需要将请求体转换成其他类型,请使用getBodyToMono()方法。由于Flux对象可能包含多个元素,因此在订阅操作时需要注意。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值