免費注冊163成都seo招聘信息
目錄
- 一、序言
- 二、開啟RabbitMQ外部消息代理
- 三、代碼示例
- 1、Maven依賴項
- 2、相關實體
- 3、自定義用戶認證攔截器
- 4、Websocket外部消息代理配置
- 5、ChatController
- 6、前端頁面chat.html
- 四、測試示例
- 1、群聊、私聊、后臺定時推送測試
- 2、登錄RabbitMQ控制臺查看隊列信息
- 五、結(jié)語
一、序言
上節(jié)我們在 WebSocket的那些事(4-Spring中的STOMP支持詳解) 中詳細說明了通過Spring內(nèi)置消息代理
結(jié)合STOMP子協(xié)議進行Websocket通信,以及相關注解的使用及原理。
但是Spring內(nèi)置消息代理會有一些限制,比如只支持STOMP協(xié)議的一部分命令,像acks
、receipts
命令都是不支持的,還有由于內(nèi)置消息代理把消息存儲在內(nèi)存,當應用不可用時,客戶端也就訂閱不到到后臺推送的消息。
這節(jié)我們將會使用支持STOMP協(xié)議的外部消息代理(RabbitMQ
)進行Websocket通信。
二、開啟RabbitMQ外部消息代理
服務端路由發(fā)送消息以及客戶端訂閱消息都要通過STOMP協(xié)議與RabbitMQ進行交互,由于RabbitMQ默認沒有啟動STOMP插件,因此我們需要先啟用該插件。
rabbitmq-plugins enable rabbitmq_stomp
啟動該插件后,RabbitMQ中STOMP適配器
默認會監(jiān)聽61613
端口,如果是云服務器,需要把該端口在安全組中放開。
關于該插件說明請參考:RabbitMQ中STOMP插件說明。
三、代碼示例
我們在 WebSocket的那些事(4-Spring中的STOMP支持詳解)中寫了一個簡單的聊天Demo示例,下面我們對該聊天Demo示例進行改造,將Spring內(nèi)置消息代理替換成RabbitMQ
外部消息代理。
1、Maven依賴項
服務端和客戶端與外部消息代理都是通過TCP進行通信,Spring底層默認使用的是Netty
和Reactor
,因此需要引入相關依賴項。
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-websocket</artifactId>
</dependency>
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-reactor-netty</artifactId>
</dependency>
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>
2、相關實體
(1) 請求消息參數(shù)
@Data
public class WebSocketMsgDTO {private String name;private String content;
}
(2) 響應消息內(nèi)容
@Data
@Builder
@AllArgsConstructor
@NoArgsConstructor
public class WebSocketMsgVO {private String content;
}
(3) 自定義認證用戶信息
@Data
@AllArgsConstructor
@NoArgsConstructor
public class StompAuthenticatedUser implements Principal {/*** 用戶唯一ID*/private String userId;/*** 用戶昵稱*/private String nickName;/*** 用于指定用戶消息推送的標識* @return*/@Overridepublic String getName() {return this.userId;}}
3、自定義用戶認證攔截器
@Slf4j
public class UserAuthenticationChannelInterceptor implements ChannelInterceptor {private static final String USER_ID = "User-ID";private static final String USER_NAME = "User-Name";@Overridepublic Message<?> preSend(Message<?> message, MessageChannel channel) {StompHeaderAccessor accessor = MessageHeaderAccessor.getAccessor(message, StompHeaderAccessor.class);// 如果是連接請求,記錄userIdif (StompCommand.CONNECT.equals(accessor.getCommand())) {String userID = accessor.getFirstNativeHeader(USER_ID);String username = accessor.getFirstNativeHeader(USER_NAME);log.info("Stomp User-Related headers found, userID: {}, username:{}", userID, username);accessor.setUser(new StompAuthenticatedUser(userID, username));}return message;}}
4、Websocket外部消息代理配置
Spring中與外部消息代理通信的中間方被稱之為Broker Relay,它會維護一個系統(tǒng)共享的單一TCP連接和外部消息代理進行通信,該TCP連接僅僅適用于服務端,用來發(fā)送消息,而不是接收消息,通過Broker Relay的systemLogin
和systemPasscode
屬性可以設置該連接的認證信息。
Broker Relay也會為每個連接的Websocket客戶端創(chuàng)建一個TCP連接,該連接用來接收消息,通過clientLogin
和clientPasscode
屬性可以設置連接的認證信息。
/*** Websocket連接外部消息代理配置* @author Nick Liu* @date 2023/9/6*/
@Configuration
@EnableWebSocketMessageBroker
public class WebsocketExternalMessageBrokerConfig implements WebSocketMessageBrokerConfigurer {@Overridepublic void configureClientInboundChannel(ChannelRegistration registration) {// 攔截器配置registration.interceptors(new UserAuthenticationChannelInterceptor());}@Overridepublic void registerStompEndpoints(StompEndpointRegistry registry) {registry.addEndpoint("/websocket") // WebSocket握手端口.addInterceptors(new HttpSessionHandshakeInterceptor()).setAllowedOriginPatterns("*") // 設置跨域.withSockJS(); // 開啟SockJS回退機制}@Overridepublic void configureMessageBroker(MessageBrokerRegistry registry) {registry.setApplicationDestinationPrefixes("/app") // 發(fā)送到服務端目的地前綴.enableStompBrokerRelay("/topic") // 開啟外部消息代理,指定消息訂閱前綴.setRelayHost("localhost") // 外部消息代理Host.setRelayPort(61613) // 外部消息代理STOMP端口.setSystemLogin("admin") // 共享系統(tǒng)連接用戶名,該連接主要用來發(fā)送消息.setSystemPasscode("admin") // 共享系統(tǒng)連接密碼,該連接主要用來發(fā)送消息.setClientLogin("admin") // 客戶端連接用戶名,該連接主要用來接收消息.setClientPasscode("admin") // 客戶端連接密碼,該連接主要用來接收消息.setVirtualHost("/stomp"); // RabbitMQ虛擬主機}
}
備注:我們可以為服務端與客戶端的連接設置不同的用戶,針對客戶端連接用戶進行權(quán)限管控,保證系統(tǒng)的安全性,在這里為了方便測試我們統(tǒng)一用一個用戶。
5、ChatController
STOMP協(xié)議并沒有規(guī)定消息代理必須支持哪種類型的Destinations(目的地)
,但是RabbitMQ STOMP適配器只支持一些指定的目的地類型,如下圖:
/exchange
:指定交換機和路由key,發(fā)送和訂閱來自隊列的消息。/queue
:發(fā)送和訂閱受STOMP網(wǎng)關管理的隊列的消息,最多只有一個訂閱者能到消息。/amq/queue
:發(fā)送和訂閱不受STOMP網(wǎng)關管理的隊列的消息。/topic
:發(fā)送和訂閱來自臨時或者持久Topic的消息,多個訂閱者都能接收到消息。/temp-queue/
:發(fā)送和訂閱來自臨時隊列的消息。
參考文檔見:RabbitMQ中STOMP插件說明。
在下面的示例中,我們選用了/topic
的開頭的消息發(fā)送和訂閱前綴,目的地格式只能為/topic/{routing-key}
,routing-key不能有斜杠,否則會報錯。
@Slf4j
@Controller
@RequiredArgsConstructor
public class ChatController {private final SimpUserRegistry simpUserRegistry;private final SimpMessagingTemplate simpMessagingTemplate;/*** 模板引擎為Thymeleaf,需要加上spring-boot-starter-thymeleaf依賴,* @return*/@GetMapping("/page/chat")public ModelAndView turnToChatPage() {return new ModelAndView("chat");}/*** 群聊消息處理* 這里我們通過@SendTo注解指定消息目的地為"/topic/chat/group",如果不加該注解則會自動發(fā)送到"/topic" + "/chat/group"* @param webSocketMsgDTO 請求參數(shù),消息處理器會自動將JSON字符串轉(zhuǎn)換為對象* @return 消息內(nèi)容,方法返回值將會廣播給所有訂閱"/topic/chat/group"的客戶端*/@MessageMapping("/chat/group")@SendTo("/topic/chat-group")public WebSocketMsgVO groupChat(WebSocketMsgDTO webSocketMsgDTO) {log.info("Group chat message received: {}", FastJsonUtils.toJsonString(webSocketMsgDTO));String content = String.format("來自[%s]的群聊消息: %s", webSocketMsgDTO.getName(), webSocketMsgDTO.getContent());return WebSocketMsgVO.builder().content(content).build();}/*** 私聊消息處理* 這里我們通過@SendToUser注解指定消息目的地為"/topic/chat/private",發(fā)送目的地默認會拼接上"/user/"前綴* 實際發(fā)送目的地為"/user/topic/chat/private"* @param webSocketMsgDTO 請求參數(shù),消息處理器會自動將JSON字符串轉(zhuǎn)換為對象* @return 消息內(nèi)容,方法返回值將會基于SessionID單播給指定用戶*/@MessageMapping("/chat/private")@SendToUser("/topic/chat-private")public WebSocketMsgVO privateChat(WebSocketMsgDTO webSocketMsgDTO) {log.info("Private chat message received: {}", FastJsonUtils.toJsonString(webSocketMsgDTO));String content = "私聊消息回復:" + webSocketMsgDTO.getContent();return WebSocketMsgVO.builder().content(content).build();}/*** 定時消息推送,這里我們會列舉所有在線的用戶,然后單播給指定用戶。* 通過SimpMessagingTemplate實例可以在任何地方推送消息。*/@Scheduled(fixedRate = 10 * 1000)public void pushMessageAtFixedRate() {log.info("當前在線人數(shù): {}", simpUserRegistry.getUserCount());if (simpUserRegistry.getUserCount() <= 0) {return;}// 這里的Principal為StompAuthenticatedUser實例Set<StompAuthenticatedUser> users = simpUserRegistry.getUsers().stream().map(simpUser -> StompAuthenticatedUser.class.cast(simpUser.getPrincipal())).collect(Collectors.toSet());users.forEach(authenticatedUser -> {String userId = authenticatedUser.getUserId();String nickName = authenticatedUser.getNickName();WebSocketMsgVO webSocketMsgVO = new WebSocketMsgVO();webSocketMsgVO.setContent(String.format("定時推送的私聊消息, 接收人: %s, 時間: %s", nickName, LocalDateTime.now()));log.info("開始推送消息給指定用戶, userId: {}, 消息內(nèi)容:{}", userId, FastJsonUtils.toJsonString(webSocketMsgVO));simpMessagingTemplate.convertAndSendToUser(userId, "/topic/chat-push", webSocketMsgVO);});}}
6、前端頁面chat.html
<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head><meta charset="UTF-8"><title>chat</title><script src="https://cdnjs.cloudflare.com/ajax/libs/sockjs-client/1.6.1/sockjs.min.js"></script><script src="https://cdnjs.cloudflare.com/ajax/libs/stomp.js/2.3.3/stomp.min.js"></script><script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.6.4/jquery.min.js"></script><style>#mainWrapper {width: 600px;margin: auto;}</style>
</head>
<body>
<div id="mainWrapper"><div><label for="username" style="margin-right: 5px">姓名:</label><input id="username" type="text"/></div><div id="msgWrapper"><p style="vertical-align: top">發(fā)送的消息:</p><textarea id="msgSent" style="width: 600px;height: 100px"></textarea><p style="vertical-align: top">收到的群聊消息:</p><textarea id="groupMsgReceived" style="width: 600px;height: 100px"></textarea><p style="vertical-align: top">收到的私聊消息:</p><textarea id="privateMsgReceived" style="width: 600px;height: 200px"></textarea></div><div style="margin-top: 5px;"><button onclick="connect()">連接</button><button onclick="sendGroupMessage()">發(fā)送群聊消息</button><button onclick="sendPrivateMessage()">發(fā)送私聊消息</button><button onclick="disconnect()">斷開連接</button></div>
</div>
<script type="text/javascript">$(() => {$('#msgSent').val('');$("#groupMsgReceived").val('');$("#privateMsgReceived").val('');});let stompClient = null;// 連接服務器const connect = () => {const header = {"User-ID": new Date().getTime().toString(), "User-Name": $('#username').val()};const ws = new SockJS('http://localhost:8080/websocket');stompClient = Stomp.over(ws);stompClient.connect(header, () => subscribeTopic());}// 訂閱主題const subscribeTopic = () => {alert("連接成功!");// 訂閱廣播消息stompClient.subscribe('/topic/chat-group', function (message) {console.log(`Group message received : ${message.body}`);const resp = JSON.parse(message.body);const previousMsg = $("#groupMsgReceived").val();$("#groupMsgReceived").val(`${previousMsg}${resp.content}\n`);});// 訂閱單播消息stompClient.subscribe('/user/topic/chat-private', message => {console.log(`Private message received : ${message.body}`);const resp = JSON.parse(message.body);const previousMsg = $("#privateMsgReceived").val();$("#privateMsgReceived").val(`${previousMsg}${resp.content}\n`);});// 訂閱定時推送的單播消息stompClient.subscribe(`/user/topic/chat-push`, message => {console.log(`Private message received : ${message.body}`);const resp = JSON.parse(message.body);const previousMsg = $("#privateMsgReceived").val();$("#privateMsgReceived").val(`${previousMsg}${resp.content}\n`);});};// 斷連const disconnect = () => {stompClient.disconnect(() => {$("#msgReceived").val('Disconnected from WebSocket server');});}// 發(fā)送群聊消息const sendGroupMessage = () => {const msg = {name: $('#username').val(), content: $('#msgSent').val()};stompClient.send('/app/chat/group', {}, JSON.stringify(msg));}// 發(fā)送私聊消息const sendPrivateMessage = () => {const msg = {name: $('#username').val(), content: $('#msgSent').val()};stompClient.send('/app/chat/private', {}, JSON.stringify(msg));}
</script>
</body>
</html>
四、測試示例
1、群聊、私聊、后臺定時推送測試
啟動應用程序,日志打印顯示系統(tǒng)連接建立成功,如下:
打開瀏覽器訪問http://localhost:8080/page/chat
可進入聊天頁,同時打開兩個窗口訪問。
2、登錄RabbitMQ控制臺查看隊列信息
可以看到所有消息都發(fā)送到了amq.topic
交換機上(Topic類型), RabbitMQ會為每個連接的客戶端創(chuàng)建3個隊列。
因為我們在ChatController
中定義了三個目的地,Routing Key分別是/topic/chat-group
、/topic/chat-private
、/topic/chat-push
。群聊消息目的地/topic/chat-group
綁定了兩個隊列,用于實現(xiàn)廣播訂閱,其它兩個Routing Key分別綁定到了不同的隊列上,實現(xiàn)唯一訂閱。
五、結(jié)語
下一節(jié)我們將會詳細說明RabbitMQ STOMP適配器支持的各種消息目的地類型的區(qū)別以及適用場景。