websocket 增加多分组 fork https://github.com/olahol/melody
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

125 lines
2.3 KiB

10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
10 years ago
  1. package melody
  2. import (
  3. "errors"
  4. "github.com/gorilla/websocket"
  5. "net/http"
  6. "time"
  7. )
  8. // A melody session.
  9. type Session struct {
  10. Request *http.Request
  11. conn *websocket.Conn
  12. output chan *envelope
  13. melody *Melody
  14. }
  15. func (s *Session) writeMessage(message *envelope) {
  16. select {
  17. case s.output <- message:
  18. default:
  19. s.melody.errorHandler(s, errors.New("Message buffer full"))
  20. }
  21. }
  22. func (s *Session) writeRaw(message *envelope) error {
  23. s.conn.SetWriteDeadline(time.Now().Add(s.melody.Config.WriteWait))
  24. err := s.conn.WriteMessage(message.t, message.msg)
  25. if err != nil {
  26. return err
  27. }
  28. if message.t == websocket.CloseMessage {
  29. err := s.conn.Close()
  30. if err != nil {
  31. return err
  32. }
  33. }
  34. return nil
  35. }
  36. func (s *Session) close() {
  37. s.writeRaw(&envelope{t: websocket.CloseMessage, msg: []byte{}})
  38. }
  39. func (s *Session) ping() {
  40. s.writeRaw(&envelope{t: websocket.PingMessage, msg: []byte{}})
  41. }
  42. func (s *Session) writePump() {
  43. defer s.conn.Close()
  44. ticker := time.NewTicker(s.melody.Config.PingPeriod)
  45. defer ticker.Stop()
  46. loop:
  47. for {
  48. select {
  49. case msg, ok := <-s.output:
  50. if !ok {
  51. s.close()
  52. break loop
  53. }
  54. if err := s.writeRaw(msg); err != nil {
  55. s.melody.errorHandler(s, err)
  56. break loop
  57. }
  58. case <-ticker.C:
  59. s.ping()
  60. }
  61. }
  62. }
  63. func (s *Session) SetPongHandler(f func() error) {
  64. s.conn.SetPongHandler(func(string) error {
  65. s.conn.SetReadDeadline(time.Now().Add(s.melody.Config.PongWait))
  66. return f()
  67. })
  68. }
  69. func (s *Session) readPump() {
  70. defer s.conn.Close()
  71. s.conn.SetReadLimit(s.melody.Config.MaxMessageSize)
  72. s.conn.SetReadDeadline(time.Now().Add(s.melody.Config.PongWait))
  73. s.SetPongHandler(func() error {
  74. return nil
  75. })
  76. for {
  77. t, message, err := s.conn.ReadMessage()
  78. if err != nil {
  79. s.melody.errorHandler(s, err)
  80. break
  81. }
  82. if t == websocket.TextMessage {
  83. s.melody.messageHandler(s, message)
  84. }
  85. if t == websocket.BinaryMessage {
  86. s.melody.messageHandlerBinary(s, message)
  87. }
  88. }
  89. }
  90. // Write message to session.
  91. func (s *Session) Write(msg []byte) {
  92. s.writeMessage(&envelope{t: websocket.TextMessage, msg: msg})
  93. }
  94. // Write binary message to session.
  95. func (s *Session) WriteBinary(msg []byte) {
  96. s.writeMessage(&envelope{t: websocket.BinaryMessage, msg: msg})
  97. }
  98. // Close session.
  99. func (s *Session) Close() {
  100. s.writeMessage(&envelope{t: websocket.CloseMessage, msg: []byte{}})
  101. }