впервые я использую веб-сокет внутри своего весеннего проекта.
Это конфигурация:
@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
.
Теперь у меня два вопроса:
- Эти веб-сокеты доступны только зарегистрированному пользователю, поэтому они аутентифицированы?
Когда соединение обрывается или пользователь меняет или закрывает страницу с открытым сокетом, как я могу остановить сокет с сервером? На данный момент я использую:
$ scope. $ On ('$ locationChangeStart', функция (событие, следующее, текущее) {
wsService.unsubscribe (wsTopic);
});
Спасибо