没有 SockJS 的 Spring 启动 Websocket

Man*_*n84 7 websocket rxjs sockjs spring-boot spring-websocket

我已经为此挣扎了至少两个星期。我对 websockets 很陌生。我对休息端点有很好的经验。

我的用例很简单。客户端启动一个 websocket 连接,向服务器发送一些信息,服务器使用该信息,并以固定的时间间隔(例如每 5 秒)向客户端发送一些信息。我按照这里的教程 - https://spring.io/guides/gs/messaging-stomp-websocket/ 它按照解释完美运行。根据上面的教程,客户端发起一个 http 请求,该请求被升级为 websocket。

就我而言,前端是一个 angular 10 应用程序,前端开发人员更喜欢使用rxjs/websocket,不想使用SockJS client,因为他确信我们不必支持任何旧版浏览器,这就是我感到震惊的地方. 显然rxjs/websocketws://协议中需要 url 。

从下面的片段中,我认为我的等效 ws 协议是 ws://localhost:8080/test但是,它似乎不起作用。我不确定出了什么问题。任何帮助是极大的赞赏!

@Configuration
@EnableWebSocketMessageBroker
public class WebSocketConfig implements WebSocketMessageBrokerConfigurer
{

    @Override
    public void configureMessageBroker(MessageBrokerRegistry config)
    {
        config.enableSimpleBroker("/topic");
        config.setApplicationDestinationPrefixes("/ws/");
    

    @Override
    public void registerStompEndpoints(StompEndpointRegistry registry)
    {
        registry.addEndpoint("/test");
    }

}
Run Code Online (Sandbox Code Playgroud)

从教程中,我更改了app.js,如下以测试这一点。

function connect() {
    // var socket = new SockJS('http://localhost:8080/test'); This works perfectly
    // stompClient = Stomp.over(socket);
    ws = new WebSocket('ws://localhost:8080/test');
    stompClient = Stomp.client(ws);

    stompClient.connect({}, function (frame) {
        setConnected(true);
        console.log('Connected: ' + frame);
        stompClient.subscribe('/topic/' + $("#site").val(), function (message) {
            showMessageSentFromServer(JSON.stringify(message.body));
        });
    });
}
Run Code Online (Sandbox Code Playgroud)

当我打开 chrome 的开发者工具并检查时,我看到 websocket 连接已经建立和升级,或者我看到的就是这样。但是,在控制台中,我看到如下错误日志。我不确定出了什么问题。

网络截图:

在此处输入图片说明

控制台故障日志:

stomp.min.js:8 Uncaught DOMException: Failed to construct 'WebSocket': The URL '[object WebSocket]' is invalid.
    at Object.client (http://localhost:8080/webjars/stomp-websocket/stomp.min.js:8:7229)
    at connect (http://localhost:8080/app.js:18:25)
    at HTMLButtonElement.<anonymous> (http://localhost:8080/app.js:54:9)
    at HTMLButtonElement.dispatch (http://localhost:8080/webjars/jquery/jquery.min.js:3:10315)
    at HTMLButtonElement.q.handle (http://localhost:8080/webjars/jquery/jquery.min.js:3:8342)
Run Code Online (Sandbox Code Playgroud)

现在长话短说,我设法在服务器端禁用了 SockJs,通过删除withSockJS(). 那么我的等效ws协议 URL 是什么?

此外,除此之外,我面临的另一个挑战是,如何根据客户端的输入设置一个计划进程,该进程可以将消息发送到客户端订阅的 websocket 主题。我知道使用@Scheduled注释设置预定流程很容易。但在我的情况下,我想要来自客户端的一些输入,这些输入在预定流程中是必需的。

另外,请分享您拥有的任何资源或示例,这些资源或示例解释了如何使用 Websocket stomp 客户端订阅主题 rxjs

Man*_*n84 20

我通过进行两个简单的更改就解决了这个问题。

  1. 我通过添加两个端点来使我的后端能够支持wshttp协议,如下WebSocketConfig所示 - 一个端点有,另一个端点没有,sockjs如下所示,这使得我的后端在支持这两种协议建立websocket连接方面更加灵活。我不知道为什么 Spring 文档或其他地方没有提到这一点。也许,人们认为这是暗示吧!
@Configuration
@EnableWebSocketMessageBroker
public class WebSocketConfig implements WebSocketMessageBrokerConfigurer
{

        @Override
        public void configureMessageBroker(MessageBrokerRegistry config)
        {
                config.enableSimpleBroker("/topic");
                config.setApplicationDestinationPrefixes("/ws/");
        }

        @Override
        public void registerStompEndpoints(StompEndpointRegistry registry)
        {
                registry.addEndpoint("/test"); // This will allow you to use ws://localhost:8080/test to establish websocket connection
                registry.addEndpoint("/test").withSockJS(); // This will allow you to use http://localhost:8080/test to establish websocket connection
        }

}
Run Code Online (Sandbox Code Playgroud)
  1. 正如前面的评论所指出的@arturgrzesiak,传递给 I 的参数存在错误,Stomp.client(url)我很愚蠢地传递了wsobject 而不是 plain url
stompClient = Stomp.client('ws://localhost:8080/test');
Run Code Online (Sandbox Code Playgroud)

最后,

如果有人想使用SockJS客户端连接,他们可以使用

var socket = new SockJS('http://localhost:8080/test');
stompClient = Stomp.over(socket);

Run Code Online (Sandbox Code Playgroud)

如果有人只想使用普通Websocket对象进行连接,请使用以下内容。

stompClient = Stomp.client('ws://localhost:8080/test');
Run Code Online (Sandbox Code Playgroud)

我发布这个解决方案是因为这对于有类似痛苦经历的人来说很有用,而且他们可能会发现它很有用。