9

我正在尝试在简单的 Java App(使用java.net.http.WebSocket类)和远程google-chrome运行之间创建通信google-chrome --remote-debugging-port=9222 --user-data-dir=.

发送和接收小消息按预期工作,但对于较大的消息(16kb)存在问题。

这是java源代码的一部分:


var uri = new URI("ws://127.0.0.1:9222/devtools/page/C0D7B4DBC53FB39F7A4BE51DA79E96BB");

/// create websocket client
WebSocket ws = HttpClient
    .newHttpClient()
    .newWebSocketBuilder()
    .connectTimeout(Duration.ofSeconds(30))
    .buildAsync(uri, simpleListener)
    .join();

// session Id attached to chrome tab
String sessionId = "...";

// send message
String message = "{\"id\":1,\"method\":\"Runtime.evaluate\",\"params\":{\"expression\":\"document.body.style.backgroundColor = 'blue';\",\"returnByValue\":true,\"awaitPromise\":true,\"userGesture\":true},\"sessionId\":\"" + sessionId + "\"}";

// this works
ws.send(message, true);

// generate big string contains over 18k chars for testing purpose
String bigMessage = "{\"id\":2,\"method\":\"Runtime.evaluate\",\"params\":{\"expression\":\"[" + ("1,".repeat(9000)) + "1]\",\"returnByValue\":true,\"awaitPromise\":true,\"userGesture\":true},\"sessionId\":\"" + sessionId + "\"}";

// this doesn't work
ws.send(bigMessage, true);

这是堆栈:

java.net.SocketException: Connection reset
    at java.base/sun.nio.ch.SocketChannelImpl.throwConnectionReset(SocketChannelImpl.java:345)
    at java.base/sun.nio.ch.SocketChannelImpl.read(SocketChannelImpl.java:376)
    at java.net.http/jdk.internal.net.http.SocketTube.readAvailable(SocketTube.java:1153)
    at java.net.http/jdk.internal.net.http.SocketTube$InternalReadPublisher$InternalReadSubscription.read(SocketTube.java:821)
    at java.net.http/jdk.internal.net.http.SocketTube$SocketFlowTask.run(SocketTube.java:175)
    at java.net.http/jdk.internal.net.http.common.SequentialScheduler$SchedulableTask.run(SequentialScheduler.java:198)
...

我已经通过使用puppeteer(nodejs library)尝试了基本相同的方法,它按预期工作。

我在网上找不到有关此问题的任何资源。我的例子中有什么我遗漏的吗?


这是简单示例的网址: https ://github.com/zeljic/websocket-devtools-protocol

4

2 回答 2

2

根据我目前所见,我最好的猜测是 Chrome 开发工具不会在该暴露webSocketDebuggerUrl端点上处理碎片化的文本消息。是否可以将 Chrome 开发工具配置为这样做是另一个问题。但是,我必须指出,RFC 6455(WebSocket 协议)要求它:

客户端和服务器必须支持接收分片和非分片消息。

我可以在这里看到一种解决方法。请记住,这是不受支持的,并且将来可能会意外更改。运行客户端时,在命令行上指定以下系统属性-Djdk.httpclient.websocket.intermediateBufferSize=1048576(或选择任何其他合适的大小)。只要您继续发送true作为boolean last参数传递给send*方法的消息,java.net.http.WebSocket就会在单个 WebSocket 帧中发送未分段的消息。

于 2020-05-11T09:49:58.860 回答
0

好吧,通过在 java 中使用带有 tomcat 服务器的 web-sockets 发送大字符串时,我遇到了类似的问题。在 websocket 服务器中发送或接收可能存在有效负载限制。


在tomcat 的文档中检查 org.apache.tomcat.websocket.textBufferSize 。默认情况下它是8192字节尝试增加大小。

于 2020-05-13T21:14:31.593 回答