redis_exchange.go 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. package server
  2. import (
  3. "context"
  4. "encoding"
  5. "encoding/json"
  6. "fmt"
  7. "github.com/redis/go-redis/v9"
  8. "sikey.com/websocket/utils/zlog"
  9. )
  10. type Exchange struct {
  11. ctx context.Context
  12. rdb redis.UniversalClient
  13. serverId string
  14. Connect chan *EventConnect
  15. Disconnect chan *EventDisconnect
  16. Message chan *Message
  17. }
  18. func NewExchange(serverId string, rdb redis.UniversalClient) *Exchange {
  19. exc := &Exchange{
  20. ctx: context.Background(),
  21. rdb: rdb,
  22. serverId: serverId,
  23. Connect: make(chan *EventConnect),
  24. Disconnect: make(chan *EventDisconnect),
  25. Message: make(chan *Message),
  26. }
  27. go exc.remotelyEvent()
  28. return exc
  29. }
  30. var (
  31. channelEvent = "client.event.*"
  32. connectChannelEvent = "client.event.connect"
  33. disconnectChannelEvent = "client.event.disconnect"
  34. messageChannelEvent = "client.event.message"
  35. clients = "client.users.%s"
  36. )
  37. type (
  38. EventConnect struct {
  39. UserId string
  40. ServerId string
  41. }
  42. EventDisconnect struct {
  43. UserId string
  44. ServerId string
  45. }
  46. )
  47. var _ encoding.BinaryMarshaler = (*EventConnect)(nil)
  48. var _ encoding.BinaryUnmarshaler = (*EventConnect)(nil)
  49. var _ encoding.BinaryMarshaler = (*EventDisconnect)(nil)
  50. var _ encoding.BinaryUnmarshaler = (*EventDisconnect)(nil)
  51. func (e *EventDisconnect) UnmarshalBinary(data []byte) error {
  52. return json.Unmarshal(data, e)
  53. }
  54. func (e *EventDisconnect) MarshalBinary() (data []byte, err error) {
  55. return json.Marshal(e)
  56. }
  57. func (e *EventConnect) UnmarshalBinary(data []byte) error {
  58. return json.Unmarshal(data, e)
  59. }
  60. func (e *EventConnect) MarshalBinary() (data []byte, err error) {
  61. return json.Marshal(e)
  62. }
  63. func (exc *Exchange) OnPublishConnect(client *Client) error {
  64. // Create clients list
  65. var err error
  66. err = exc.rdb.Set(exc.ctx, fmt.Sprintf(clients, client.UserId), exc.serverId, 0).Err()
  67. if err != nil {
  68. zlog.Error(err)
  69. }
  70. event := &EventConnect{ServerId: exc.serverId, UserId: client.UserId}
  71. err = exc.rdb.Publish(exc.ctx, connectChannelEvent, event).Err()
  72. if err != nil {
  73. zlog.Error(err)
  74. }
  75. return err
  76. }
  77. func (exc *Exchange) OnPublishDisconnect(client *Client) error {
  78. exc.rdb.Del(exc.ctx, fmt.Sprintf(clients, client.UserId), exc.serverId)
  79. event := &EventDisconnect{ServerId: exc.serverId, UserId: client.UserId}
  80. return exc.rdb.Publish(exc.ctx, disconnectChannelEvent, event).Err()
  81. }
  82. func (exc *Exchange) OnPublishMessage(client *Client, message *Message) error {
  83. err := exc.rdb.Publish(exc.ctx, messageChannelEvent, message).Err()
  84. if err != nil {
  85. zlog.Error(err)
  86. return err
  87. }
  88. return nil
  89. }
  90. func (exc *Exchange) remotelyEvent() {
  91. ctx := context.TODO()
  92. pubsub := exc.rdb.PSubscribe(ctx, channelEvent)
  93. defer pubsub.Close()
  94. for {
  95. rMsg, err := pubsub.ReceiveMessage(ctx)
  96. if err != nil {
  97. zlog.Error(err)
  98. }
  99. switch rMsg.Channel {
  100. case connectChannelEvent:
  101. var event = deserializeEventConnect([]byte(rMsg.Payload))
  102. if event.ServerId != exc.serverId {
  103. exc.Connect <- &EventConnect{UserId: event.UserId}
  104. }
  105. case disconnectChannelEvent:
  106. var event = deserializeEventDisconnect([]byte(rMsg.Payload))
  107. if event.ServerId != exc.serverId {
  108. exc.Disconnect <- &EventDisconnect{UserId: event.UserId}
  109. }
  110. case messageChannelEvent:
  111. exc.Message <- deserializeMessage([]byte(rMsg.Payload))
  112. }
  113. }
  114. }
  115. func deserializeEventConnect(bytes []byte) *EventConnect {
  116. var event EventConnect
  117. _ = json.Unmarshal(bytes, &event)
  118. return &event
  119. }
  120. func deserializeEventDisconnect(bytes []byte) *EventDisconnect {
  121. var event EventDisconnect
  122. _ = json.Unmarshal(bytes, &event)
  123. return &event
  124. }