hub.go 2.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102
  1. package server
  2. import (
  3. "sync"
  4. "go.uber.org/zap"
  5. "sikey.com/websocket/config"
  6. )
  7. type Hub struct {
  8. serverId string // serverId 服务器ID
  9. clients map[string]*Client
  10. mutex sync.RWMutex
  11. Connect chan *Client
  12. Disconnect chan *Client
  13. Message chan *Message
  14. FirebaseMessage chan *FirebaseMessage
  15. FCM *FirebaseMessageServer
  16. Nats *Nats
  17. }
  18. func NewHub(serverId string) *Hub {
  19. hub := &Hub{
  20. serverId: serverId,
  21. clients: make(map[string]*Client),
  22. mutex: sync.RWMutex{},
  23. Connect: make(chan *Client, config.Websocket.ConnectSize),
  24. Disconnect: make(chan *Client, config.Websocket.ConnectSize),
  25. Message: make(chan *Message, config.Websocket.MessageSize),
  26. Nats: NewNats(config.Websocket.NatsUrl),
  27. FCM: NewFirebaseMessageServer(),
  28. }
  29. go hub.run()
  30. return hub
  31. }
  32. func (h *Hub) run() {
  33. for {
  34. select {
  35. case client := <-h.Connect:
  36. h.mutex.Lock()
  37. h.clients[client.UserId] = client
  38. h.mutex.Unlock()
  39. if h.Nats != nil {
  40. h.Nats.Subscribe <- &subscriber{client: client}
  41. }
  42. case client := <-h.Disconnect:
  43. h.mutex.Lock()
  44. close(client.Send)
  45. delete(h.clients, client.UserId)
  46. h.mutex.Unlock()
  47. if h.Nats != nil {
  48. h.Nats.Unsubscribe <- &subscriber{client: client}
  49. }
  50. case message := <-h.Message:
  51. h.mutex.RLock()
  52. if client, ok := h.clients[message.Receiver]; ok {
  53. zap.L().Info("[conn] message to client",
  54. zap.String("receiver", message.Receiver),
  55. zap.String("request_id", message.RequestId))
  56. client.Send <- message
  57. } else {
  58. h.Nats.Send <- &natsMessage{userId: message.Receiver, message: message}
  59. }
  60. h.mutex.RUnlock()
  61. // 检查用户如果不在线,并且有 FCM token,则发送 FCM 消息
  62. // case firebaseMessage := <-h.FirebaseMessage:
  63. // if err := h.FCM.Send(context.Background(), firebaseMessage.token, firebaseMessage.message); err != nil {
  64. // zap.L().Error("unable to send fcm message", zap.Error(err))
  65. // }
  66. }
  67. }
  68. }
  69. func (h *Hub) GetClients() []string {
  70. h.mutex.RLock()
  71. defer h.mutex.RUnlock()
  72. var clients = make([]string, 0)
  73. for _, c := range h.clients {
  74. clients = append(clients, c.UserId)
  75. }
  76. return clients
  77. }
  78. func (h *Hub) getClientByUserId(userId string) *Client {
  79. h.mutex.RLock()
  80. defer h.mutex.RUnlock()
  81. if client, ok := h.clients[userId]; ok {
  82. return client
  83. }
  84. return nil
  85. }