Spring и веб-сокет: выполните аутентификацию и закройте соединение при смене / закрытии страницы - PullRequest
0 голосов
/ 31 октября 2018

впервые я использую веб-сокет внутри своего весеннего проекта. Это конфигурация:

@Configuration
@EnableWebSocketMessageBroker
public class WebSocketConfig implements WebSocketMessageBrokerConfigurer {

    @Autowired 
    private Environment env;
    public static final String TOPICS_ROOT_PATH = "ws.topic";
    public static final String APPLICATION_PREFIX_PATH = "ws.prefix_path";
    public static final String SOCKET_PATH = "ws.socket_path";

    @Override
    public void configureMessageBroker(MessageBrokerRegistry config) {
        config.enableSimpleBroker(env.getRequiredProperty(TOPICS_ROOT_PATH));
        config.setApplicationDestinationPrefixes(env.getRequiredProperty(APPLICATION_PREFIX_PATH));
    }

    @Override
    public void registerStompEndpoints(StompEndpointRegistry registry) {
        registry.addEndpoint(env.getRequiredProperty(SOCKET_PATH));
        registry.addEndpoint(env.getRequiredProperty(SOCKET_PATH)).withSockJS();
    }
}

Затем я настроил фабрику в angularjs для управления веб-сокетом:

angular.module('myApp').factory(
        'wsService',
        ['$rootScope', '$timeout', function ($rootScope, $timeout) {
                const RETRY_TIMEOUT = 10000;
                const WEBSOCKET_PATH = './wsendpoint';
                const APP_PATH = '/app';

                var wsClient = {
                        wsClient: this,
                        stompClient: null,
                        socket: null,
                        subscriptions: new Map(),

                        connect: function () {
                            var url = WEBSOCKET_PATH;
                            wsClient.socket = new SockJS(url);
                            wsClient.stompClient = Stomp.over(wsClient.socket);
                            wsClient.stompClient.debug = null;

                            const connectCallback = function (frame) {
                                $rootScope.$broadcast('STOMP_CONNECTED', {});
                            };

                            const errorCallback = function () {
                                $timeout( function(){
                                    wsClient.connect();
                                }, RETRY_TIMEOUT);
                            };

                            wsClient.stompClient.connect({}, connectCallback, errorCallback);
                        },

                        isConnected: function () {
                            return wsClient.stompClient.connected;
                        },

                        send: function (destination, headers, payload) {
                            if (wsClient.isConnected()) {
                                console.log("Sending message to " + destination);
                                wsClient.stompClient.send(destination, headers, payload);
                            } else {
                                $rootScope.$on('STOMP_CONNECTED', function () {
                                    console.log("Sending message to " + destination);
                                    wsClient.stompClient.send(destination, headers, payload);
                                });
                            }
                        },

                        unsubscribe: function (topic) {
                            if (wsClient.subscriptions.get(topic)) {
                                wsClient.subscriptions.get(topic).unsubscribe();
                                wsClient.subscriptions.delete(topic);
                            }
                        },

                        subscribe: function (topicName, callback) {
                            subscribeWhenConnected(topicName, callback);
                        }        
                };

                function subscribeWhenConnected(topicName, callback) {
                    console.log("Subscribing to topic: " + topicName);
                    if (wsClient.isConnected()) {
                        subscribeAndRunJob(topicName, callback);
                    } else {
                        $rootScope.$on('STOMP_CONNECTED', function () {
                            subscribeAndRunJob(topicName, callback);
                        });
                    }
                };

                function subscribeAndRunJob(topicName, callback) {        
                    const sub = wsClient.stompClient.subscribe(topicName, callback);
                    wsClient.subscriptions.set(topicName, sub);        
                } 

                wsClient.connect();
                return wsClient;

        }]
)
.run(function () {
    console.debug("Web Socket is connected");
});

Для отправки с сервера на клиент я использую SimpMessagingTemplate с методом convertAndSend. Теперь у меня два вопроса:

  1. Эти веб-сокеты доступны только зарегистрированному пользователю, поэтому они аутентифицированы?
  2. Когда соединение обрывается или пользователь меняет или закрывает страницу с открытым сокетом, как я могу остановить сокет с сервером? На данный момент я использую:

    $ scope. $ On ('$ locationChangeStart', функция (событие, следующее, текущее) { wsService.unsubscribe (wsTopic); });

Спасибо

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...