package melody import ( "github.com/gorilla/websocket" "time" ) // A melody session. type Session struct { conn *websocket.Conn output chan *envelope config *Config } func newSession(config *Config, conn *websocket.Conn) *Session { return &Session{ conn: conn, output: make(chan *envelope, config.MessageBufferSize), config: config, } } func (s *Session) writeMessage(message *envelope) { s.output <- message } func (s *Session) writeRaw(message *envelope) error { s.conn.SetWriteDeadline(time.Now().Add(s.config.WriteWait)) return s.conn.WriteMessage(message.t, message.msg) } func (s *Session) close() { s.writeRaw(&envelope{t: websocket.CloseMessage, msg: []byte{}}) } func (s *Session) ping() { s.writeMessage(&envelope{t: websocket.PingMessage, msg: []byte{}}) } func (s *Session) writePump(errorHandler handleErrorFunc) { defer s.conn.Close() ticker := time.NewTicker(s.config.PingPeriod) defer ticker.Stop() for { select { case msg, ok := <-s.output: if !ok { s.close() return } if err := s.writeRaw(msg); err != nil { go errorHandler(s, err) return } case <-ticker.C: s.ping() } } } func (s *Session) readPump(messageHandler handleMessageFunc, errorHandler handleErrorFunc) { defer s.conn.Close() s.conn.SetReadLimit(s.config.MaxMessageSize) s.conn.SetReadDeadline(time.Now().Add(s.config.PongWait)) s.conn.SetPongHandler(func(string) error { s.conn.SetReadDeadline(time.Now().Add(s.config.PongWait)) return nil }) for { _, message, err := s.conn.ReadMessage() if err != nil { go errorHandler(s, err) break } go messageHandler(s, message) } } // Write a message to session. func (s *Session) Write(msg []byte) { s.writeMessage(&envelope{t: websocket.TextMessage, msg: msg}) } // Close a session. func (s *Session) Close() { s.writeMessage(&envelope{t: websocket.CloseMessage, msg: []byte{}}) }