gateway 获取响应数据缺失问题

11 篇文章 0 订阅
5 篇文章 0 订阅
本文介绍了在使用Spring Cloud Gateway时遇到的响应数据丢失问题,特别是在浏览器访问接口时。通过代码分析发现,问题出在数据转换过程中,原始代码在处理Flux时没有正确合并数据缓冲区。为了解决这个问题,将数据缓冲区一次性join完成,避免了数据不完整的问题。最终,通过修改`writeWith`方法中对Flux数据的处理逻辑,成功修复了浏览器访问接口时的错误。
摘要由CSDN通过智能技术生成

在网上找到了一篇关于gateway获取响应数据的代码,自己加了了一下代码 如下`

import org.reactivestreams.Publisher;
import org.springframework.cloud.gateway.filter.GatewayFilterChain;
import org.springframework.cloud.gateway.filter.GlobalFilter;
import org.springframework.core.Ordered;
import org.springframework.core.io.buffer.DataBuffer;
import org.springframework.core.io.buffer.DataBufferFactory;
import org.springframework.core.io.buffer.DataBufferUtils;
import org.springframework.http.server.reactive.ServerHttpResponse;
import org.springframework.http.server.reactive.ServerHttpResponseDecorator;
import org.springframework.stereotype.Component;
import org.springframework.web.server.ServerWebExchange;
import reactor.core.publisher.Flux;
import reactor.core.publisher.Mono;
 
import java.nio.charset.Charset;
 

@Component
public class WrapperResponseGlobalFilter implements GlobalFilter, Ordered {
    @Override
    public int getOrder() {
        return -2;
    }
 
    @Override
    public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
        ServerHttpResponse originalResponse = exchange.getResponse();
        DataBufferFactory bufferFactory = originalResponse.bufferFactory();
        ServerHttpResponseDecorator decoratedResponse = new ServerHttpResponseDecorator(originalResponse) {
            @Override
            public Mono<Void> writeWith(Publisher<? extends DataBuffer> body) {
                if (body instanceof Flux) {
                    Flux<? extends DataBuffer> fluxBody = (Flux<? extends DataBuffer>) body;
                    return super.writeWith(fluxBody.map(dataBuffer -> {
                        // probably should reuse buffers
                        byte[] content = new byte[dataBuffer.readableByteCount()];
                        dataBuffer.read(content);
                        //释放掉内存
                        DataBufferUtils.release(dataBuffer);
                        String s = new String(content, Charset.forName("UTF-8"));
                        JSONObject respondseObject = JSONUtil.parseObj(s);
                        String status = String.valueOf(respondseObject.get("status"));
                        logger.info("接口状态: {}", status);
                        //TODO,s就是response的值,想修改、查看就随意而为了
                        byte[] uppedContent = new String(content, Charset.forName("UTF-8")).getBytes();
                        return bufferFactory.wrap(uppedContent);
                    }));
                }
                // if body is not a flux. never got there.
                return super.writeWith(body);
            }
        };
        // replace response with decorator
        return chain.filter(exchange.mutate().response(decoratedResponse).build());
    }
}

用postman测试接口,返回的数据是正常的,但是浏览器访问接口时却报错了,debug后问题就出在 String s = new String(content, Charset.forName(“UTF-8”)); 因为postman请求是转换的数据是完整的,但是浏览器访问接口时,转换出的数据缺失了,以至于 JSONObject respondseObject = JSONUtil.parseObj(s); 报错cn.hutool.json.JSONException: Unterminated string at 604 [character 605 line 1]

经过上网查阅终于解决了

修改上述代码 如下

ServerHttpResponseDecorator decoratedResponse = new ServerHttpResponseDecorator(originalResponse) {
            @Override
            public Mono<Void> writeWith(Publisher<? extends DataBuffer> body) {
                if (body instanceof Flux) {
                    Flux<? extends DataBuffer> fluxBody = (Flux<? extends DataBuffer>) body;
                    return super.writeWith(fluxBody.buffer().map(dataBuffer -> {
//                            HttpStatus statusCode = exchange.getResponse().getStatusCode();
                        // probably should reuse buffers
                        DataBufferFactory dataBufferFactory = new DefaultDataBufferFactory();
                        DataBuffer buffer= dataBufferFactory.join(dataBuffer);

                        byte[] content = new byte[buffer.readableByteCount()];
                        buffer.read(content);
                        //释放掉内存
                        DataBufferUtils.release(buffer);
                        String s = new String(content, Charset.forName("UTF-8"));

                        JSONObject respondseObject = JSONUtil.parseObj(s);
                        String status = String.valueOf(respondseObject.get("status"));
                        logger.info("接口状态: {}", status);
                        //TODO,s就是response的值,想修改、查看就随意而为了
                        byte[] uppedContent = s.getBytes();
                        return bufferFactory.wrap(uppedContent);
                    }));
                }
                // if body is not a flux. never got there.
                return super.writeWith(body);
            }
        };

DataBufferFactory dataBufferFactory = new DefaultDataBufferFactory();
DataBufferFactory可以一次性join完所有数据, 所以就不会出现数据缺失问题

问题解决!

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
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 获取请求体的方法。需要注意的是,获取请求体可能会影响性能,因此应该谨慎使用。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值