go游戏后端开发21:处理nats消息

处理 NATS 订阅的消息

在 WebSocket 的管理模块中,我们之前已经处理了一些消息。这些消息通过 NATS 订阅过来,我们需要对这些消息进行进一步的处理。一旦消息到达,我们需要执行相应的操作,并将结果发送回去,包括之前的操作。

理论上,所有消息都应该能够到达这里进行处理。目前,我们还没有对这部分进行处理,接下来需要完善这部分逻辑。

3. 解析消息并处理

当收到消息后,我们需要解析消息内容。解析完成后,我们需要根据消息的类型(type)进行不同的处理。

  • 特殊类型处理

    • 如果消息类型是"三审"(假设的类型),我们需要进行特殊处理。因为这种类型的消息可能存储在 WebSocket 的连接中,而不是直接推送给客户端。

    • 如果消息类型是"replace"或"response",我们需要将消息类型改为"response",以便正确地将消息发送回客户端。

  • 普通类型处理

    • 如果消息类型是"pose",我们需要将消息放入一个专门的通道(Channel)中进行处理。这样可以提高系统的承载能力,避免消息积压。
4. 消息推送逻辑

在处理完消息后,我们需要将消息推送给客户端。具体步骤如下:

  1. 获取客户端 ID

    • 从消息中获取当前客户端的 ID。如果客户端不存在,我们需要记录日志并提示客户端已下线。
  2. 编码消息

    • 对消息进行编码处理,确保消息格式正确。
  3. 发送消息

    • 将编码后的消息发送给客户端。如果消息类型是"pose",我们需要循环处理所有相关客户端,并将消息推送给每个客户端。
5. 代码实现

以下是优化后的代码实现:

复制代码
package websocket

import (
	"encoding/json"
	"fmt"
	"log"
	"sync"

	"github.com/nats-io/nats.go"
	"github.com/gorilla/websocket"
)

type WebSocketManager struct {
	sync.RWMutex
	clients    map[*websocket.Conn]string
	messageCh  chan *Message
	pushCh     chan *Message
	natsConn   *nats.Conn
}

type Message struct {
	Type    string          `json:"type"`
	Data    json.RawMessage `json:"data"`
	ClientID string          `json:"clientID"`
}

func NewWebSocketManager(natsConn *nats.Conn) *WebSocketManager {
	return &WebSocketManager{
		clients:   make(map[*websocket.Conn]string),
		messageCh: make(chan *Message, 1024),
		pushCh:    make(chan *Message, 1024),
		natsConn:  natsConn,
	}
}

func (wm *WebSocketManager) handleMessage(msg *Message) {
	log.Printf("Handling message: %+v", msg)

	// 获取客户端连接
	clientID := msg.ClientID
	clientConn, exists := wm.clients[clientID]
	if !exists {
		log.Printf("Client not found: %s", clientID)
		return
	}

	// 根据消息类型处理
	switch msg.Type {
	case "response":
		// 处理响应消息
		wm.sendToClient(clientConn, msg)
	case "pose":
		// 处理推送消息
		wm.pushCh <- msg
	default:
		log.Printf("Unknown message type: %s", msg.Type)
	}
}

func (wm *WebSocketManager) sendToClient(clientConn *websocket.Conn, msg *Message) {
	encodedMsg, err := json.Marshal(msg)
	if err != nil {
		log.Printf("Failed to encode message: %v", err)
		return
	}

	err = clientConn.WriteMessage(websocket.TextMessage, encodedMsg)
	if err != nil {
		log.Printf("Failed to send message to client: %v", err)
	}
}

func (wm *WebSocketManager) processPushMessages() {
	for msg := range wm.pushCh {
		for clientConn := range wm.clients {
			wm.sendToClient(clientConn, msg)
		}
	}
}

func (wm *WebSocketManager) start() {
	go wm.processPushMessages()

	// NATS 订阅消息
	nc := wm.natsConn
	nc.Subscribe("ws.messages", func(msg *nats.Msg) {
		var wsMsg Message
		err := json.Unmarshal(msg.Data, &wsMsg)
		if err != nil {
			log.Printf("Failed to unmarshal NATS message: %v", err)
			return
		}
		wm.handleMessage(&wsMsg)
	})
}
相关推荐
Sammyyyyy5 分钟前
macOS是开发的终极进化版吗?
开发语言·macos·开发工具
青草地溪水旁16 分钟前
23 种设计模式
开发语言·c++·设计模式
草履虫建模19 分钟前
在 RuoYi 中接入 3D「园区驾驶舱」:Vue2 + Three.js + Nginx
运维·开发语言·javascript·spring boot·nginx·spring cloud·微服务
编码浪子19 分钟前
趣味学RUST基础篇(函数式编程闭包)
开发语言·算法·rust
wanhengidc20 分钟前
高性价比云手机挑选指南
运维·网络·安全·游戏·智能手机
MC皮蛋侠客29 分钟前
使用python test测试http接口
开发语言·python·http
胡耀超1 小时前
5、Python-NumPy科学计算基础
开发语言·人工智能·python·深度学习·numpy
点灯小铭1 小时前
基于MATLAB的车牌识别系统
开发语言·单片机·数码相机·matlab·毕业设计·课程设计
十八旬1 小时前
苍穹外卖项目实战(day7-2)-购物车操作功能完善-记录实战教程、问题的解决方法以及完整代码
java·开发语言·windows·spring boot·mysql
BIGSHU09232 小时前
java多线程场景3-并发处理和异步请求
java·开发语言·python