解决SpringCloudGateway配置⾃定义路由404的坑
⽬录
问题背景
问题现象
解决过程
1 检查⽹关配置
2 跟源码,查可能的原因
3 异常原因分析
解决⽅法
⼼得
问题背景
将原有项⽬中的websocket模块迁移到基于SpringCloud Alibaba的微服务系统中,其中⽹关部分使⽤的是gateway。
问题现象
迁移后,我们在使⽤客户端连接websocket时报错:
dec.http.websocketx.WebSocketHandshakeException: Invalid subprotocol. Actual: null. Expected one of: protocol
...
同时,我们还有⼀个⽤py写的程序,⽤来模拟客户端连接,但是程序的websocket连接就是正常的。
解决过程
1 检查⽹关配置
先开始,我们以为是gateway的配置有问题。
但是在检查gateway的route配置后,发现并没有问题。很常见的那种。
...
gateway:
routes:
#表⽰websocket的转发
- id: user-service-websocket
uri: lb:ws://user-service
predicates:
- Path=/user-service/mq/**
filters:
- StripPrefix=1
其中,lb指负载均衡,ws指定websocket协议。
ps,如果,这⾥还有其他协议的相同路径的请求,也可以直接写成:
...
gateway:
微服务在哪里routes:
#表⽰websocket的转发
- id: user-service-websocket
uri: lb://user-service
predicates:
- Path=/user-service/mq/**
filters:
- StripPrefix=1
这样,其他协议的请求也可以通过这个规则进⾏转发了。
2 跟源码,查可能的原因
既然gate的配置没有问题,那我们就尝试从源码的⾓度,看看gateway是如何处理ws协议请求的。
⾸先,我们要对“gateway是如何⼯作的”有个⼤概的认识:
可见,在收到请求后,要先经过多个Filter才会到达Proxied Service。其中,要有⾃定义的Filter也有全局的Filter,全局的filter可以通过GET请求/actuator/gateway/globalfilters来查看
{
"org.springframework.cloud.gateway.filter.LoadBalancerClientFilter@77856cc5": 10100,
"org.springframework.cloud.gateway.filter.RouteToRequestUrlFilter@4f6fd101": 10000,
"org.springframework.cloud.gateway.filter.NettyWriteResponseFilter@32d22650": -1,
"org.springframework.cloud.gateway.filter.ForwardRoutingFilter@106459d9": 2147483647,
"org.springframework.cloud.gateway.filter.NettyRoutingFilter@1fbd5e0": 2147483647,
"org.springframework.cloud.gateway.filter.ForwardPathFilter@33a71d23": 0,
"org.springframework.cloud.gateway.filter.AdaptCachedBodyGlobalFilter@135064ea": 2147483637,
"org.springframework.cloud.gateway.filter.WebsocketRoutingFilter@23c05889": 2147483646
}
可以看到,其中的WebSocketRoutingFilter似乎与我们这⾥有关
public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) {
this.changeSchemeIfIsWebSocketUpgrade(exchange);
URI requestUrl = (RequiredAttribute(ServerWebExchangeUtils.GATEWAY_REQUEST_URL_ATTR);
String scheme = Scheme();
if (!ServerWebExchangeUtils.isAlreadyRouted(exchange) && ("ws".equals(scheme) || "wss".equals(scheme))) {
ServerWebExchangeUtils.setAlreadyRouted(exchange);
HttpHeaders headers = Request().getHeaders();
HttpHeaders filtered = HttpHeadersFilter.HeadersFilters(), exchange);
List<String> protocols = ("Sec-WebSocket-Protocol");
if (protocols != null) {
protocols = (("Sec-WebSocket-Protocol").stream().flatMap((header) -> {
return Arrays.stream(StringUtilsmaDelimitedListToStringArray(header));
}).map(String::trim).List());
}
return this.webSocketService.handleRequest(exchange, new WebsocketRoutingFilter.ProxyWebSocketHandler(requestUrl, this.webSocketClient, filtered, protocols)); } else {
return chain.filter(exchange);
}
}
以debug模式跟踪到这⾥后,可以看到,客户端请求中,⼦协议指定为“protocol”。
netty相关:
WebSocketClientHandshaker
WebSocketClientHandshakerFactory
这段烂尾了。。。
直接看结论吧
最后发现出错的原因是在netty的WebSocketClientHandshanker.finishHandshake
public final void finishHandshake(Channel channel, FullHttpResponse response) {
this.verify(response);
String receivedProtocol = response.headers().get(HttpHeaderNames.SEC_WEBSOCKET_PROTOCOL);
receivedProtocol = receivedProtocol != null ? im() : null;
String expectedProtocol = pectedSubprotocol != null ? pectedSubprotocol : "";
boolean protocolValid = false;
if (expectedProtocol.isEmpty() && receivedProtocol == null) {
protocolValid = true;
this.pectedSubprotocol);
} else if (!expectedProtocol.isEmpty() && receivedProtocol != null && !receivedProtocol.isEmpty()) {
String[] var6 = expectedProtocol.split(",");
int var7 = var6.length;
for(int var8 = 0; var8 < var7; ++var8) {
String protocol = var6[var8];
if (im().equals(receivedProtocol)) {
protocolValid = true;
this.setActualSubprotocol(receivedProtocol);
break;
}
}
}
if (!protocolValid) {
throw new WebSocketHandshakeException(String.format("Invalid subprotocol. Actual: %s. Expected one of: %s", receivedProtocol, pectedSubprotocol));
} else {
......
}
}
这⾥,当期望的⼦协议类型⾮空,⽽实际⼦协议不属于期望的⼦协议时,会抛出异常。也就是⽂章最初提到的那个。
3 异常原因分析
客户端在请求时,要求⼦协议为“protocol”,⽽我们的后台websocket组件中,并没有指定使⽤这个⼦协议,也就⽆法选出使⽤的哪个⼦协议。因此,在⾛到finishHandShaker时,netty在检查⼦协议是否匹配时抛出异常WebSocketHandshakeException。 py的模拟程序中,并没有指定⼦协议,也就不会出错。
⽽在springboot的版本中,由于我们是客户端直连(通过nginx转发)到websocket服务端的,因此也没有出错(猜测是客户端没有检查⼦协议是否合法)。。。
解决⽅法
在WebSocketServer类的注解@ServerEndpoint中,增加subprotocols={“protocol”}
@ServerEndpoint(value = "/ws/asset",subprotocols = {"protocol"})
随后由客户端发起websocket请求,请求连接成功,未抛出异常。
与客户端的开发⼈员交流后,其指出,他们的代码中,确实指定了⼦协议为“protocol”,当时随⼿写的…
⼼得
定位问题较慢,中间⾛了不少弯路。有优化的空间;
对gateway的模型有了更深刻的理解;
idea 可以⽤双击Shift键来查所有类,包括依赖包中的。
以上为个⼈经验,希望能给⼤家⼀个参考,也希望⼤家多多⽀持。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论