spring-cloud-gateway使用websocket出现Max frame length of 65536 has been exceeded

前言:

在 Spring Cloud Gateway 3.1.4 版本中配置 WebSocket 通信时,遇到 Max frame length of 65536 has been exceeded 错误。经过源码分析发现,网上流传的通过 spring.cloud.gateway.httpclient.websocket.max-frame-payload-length 配置参数修改最大帧长度的方法无效。本文通过源码解析问题根源,并提供有效解决方案。

问题现象

WebSocket 配置伪代码

typescript 复制代码
@Autowired
@Bean
public HandlerMapping webSocketMapping(final EchoHandler echoHandler) {
    Map<String, WebSocketHandler> map = new HashMap<>();
    map.put("/echo", echoHandler);
    SimpleUrlHandlerMapping mapping = new SimpleUrlHandlerMapping();
    mapping.setOrder(Ordered.HIGHEST_PRECEDENCE);
    mapping.setUrlMap(map);
    return mapping;
}

@Bean
public WebSocketHandlerAdapter handlerAdapter() {
    return new WebSocketHandlerAdapter(); // 问题关键在这里
}

即使配置了:

yaml 复制代码
spring:
  cloud:
    gateway:
      httpclient:
        websocket:
          max-frame-payload-length: xxxxxx

仍然会触发默认最大帧长度 65536 的限制错误。

为什么配置了 max-frame-payload-length 却无效?

这要从 WebSocketHandlerAdapter 的源码开始分析。

csharp 复制代码
public WebSocketHandlerAdapter() {
    this(new HandshakeWebSocketService()); // 使用默认HandshakeWebSocketService
}

HandshakeWebSocketService 的默认构造器如下:

csharp 复制代码
// gateway 使用的是reactor netty 所以最后初始化出来的是ReactorNettyRequestUpgradeStrategy
public HandshakeWebSocketService() {
    this(initUpgradeStrategy()); // 初始化 ReactorNettyRequestUpgradeStrategy
}

默认会使用 ReactorNettyRequestUpgradeStrategy

csharp 复制代码
public ReactorNettyRequestUpgradeStrategy() {
    this(WebsocketServerSpec::builder); // 使用无参 builder
}

进一步查看 WebsocketServerSpec.Builder 的实现:

scala 复制代码
public static class Builder<SPEC extends Builder<SPEC>> implements Supplier<SPEC> {
    int maxFramePayloadLength = 65536; // 默认值
    ...
}

结论

Spring Cloud Gateway 的 spring.cloud.gateway.httpclient.websocket.max-frame-payload-length 配置项实际上无法修改底层 WebsocketServerSpecmaxFramePayloadLength 值,因为默认构造方式不会读取该配置。

有效解决方案

我们可以自己构建一个带参数的 ReactorNettyRequestUpgradeStrategy,明确设置 maxFramePayloadLength

通过显式配置 ReactorNettyRequestUpgradeStrategy 并自定义 WebsocketServerSpec

java 复制代码
@Bean
public WebSocketHandlerAdapter handlerAdapter() {
    int frameSizeLimit = 655360; // 设置为你所需的帧长度限制
    WebsocketServerSpec.Builder builder = WebsocketServerSpec.builder().maxFramePayloadLength(frameSizeLimit);
    ReactorNettyRequestUpgradeStrategy upgradeStrategy = new ReactorNettyRequestUpgradeStrategy(builder);
    return new WebSocketHandlerAdapter(new HandshakeWebSocketService(upgradeStrategy));
}

补充说明

若需要配置更多 WebSocket 参数(如启用压缩、调整 Ping/Pong 处理策略),可在 WebsocketServerSpec.Builder 中链式调用:

scss 复制代码
WebsocketServerSpec.Builder builder = WebsocketServerSpec.builder()
        .maxFramePayloadLength(655360)
        .compress(true)  // 启用压缩
        .handlePing(false); // 禁止自动处理Ping帧

总结:

  • spring.cloud.gateway.httpclient.websocket.max-frame-payload-length针对 gateway 客户端通信(即作为 WebSocket 客户端)配置的;

  • 如果你是通过自定义 WebSocketHandlerAdapter 来处理 WebSocket 请求(即作为 WebSocket 服务端),则必须手动指定 maxFramePayloadLength

  • 默认构造路径中的帧大小限制是写死在源码中的,必须通过构造器覆盖。

相关推荐
古城小栈2 小时前
从 cargo-whero 库中,找到提升 rust 的契机
开发语言·后端·rust
keep one's resolveY2 小时前
SpringBoot实现重试机制的四种方案
java·spring boot·后端
阿丰资源4 小时前
基于Spring Boot的电影城管理系统(直接运行)
java·spring boot·后端
IT_陈寒4 小时前
SpringBoot自动配置的坑差点让我加班到天亮
前端·人工智能·后端
消失的旧时光-19435 小时前
Spring Boot 工程化进阶:统一返回 + 全局异常 + AOP 通用工具包
java·spring boot·后端·aop·自定义注解
追风筝的人er6 小时前
SpringBoot+Vue3 企业考勤如何处理法定假期?节假日方案、调休补班与工作日判断链路拆解
前端·vue.js·后端
金銀銅鐵6 小时前
[git] 如何丢弃对一个文件的改动?
git·后端
橘子海全栈攻城狮7 小时前
【最新源码】养老院系统管理A013
java·spring boot·后端·web安全·微信小程序
smallyoung7 小时前
具有反思能力的 Agentic RAG 实战:用 LangChain4j 实现 CRAG 纠错检索
人工智能·后端
EthanYuan7 小时前
💡RAG实践:从云知识库迁移到PostgreSQL ,并使用PGVector实现向量存储
后端