Browse Source

binary messages and tests

master
Ola Holmström 10 years ago
parent
commit
0c379737b4
  1. 5
      .gitignore
  2. 7
      .travis.yml
  3. 4
      CHANGELOG.md
  4. 2
      examples/chat/main.go
  5. 2
      examples/filewatch/main.go
  6. 2
      examples/multichat/main.go
  7. 16
      melody.go
  8. 183
      melody_test.go
  9. 31
      session.go

5
.gitignore

@ -0,0 +1,5 @@
.DS_Store
benchmark
*.swp
coverage.out
Makefile

7
.travis.yml

@ -1,6 +1,11 @@
language: go language: go
go: go:
- 1.4 - 1.4
before_install:
- go get github.com/axw/gocov/gocov
- go get github.com/mattn/goveralls
- if ! go get code.google.com/p/go.tools/cmd/cover; then go get golang.org/x/tools/cmd/cover; fi
install: install:
- go get github.com/gorilla/websocket - go get github.com/gorilla/websocket
script: go test -v
script:
- $HOME/gopath/bin/goveralls -service=travis-ci

4
CHANGELOG.md

@ -0,0 +1,4 @@
### 2015-06-10
* Support for binary messages.
* BroadcastOthers method.

2
examples/chat/main.go

@ -1,8 +1,8 @@
package main package main
import ( import (
"../../"
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
"github.com/olahol/melody"
"net/http" "net/http"
) )

2
examples/filewatch/main.go

@ -1,9 +1,9 @@
package main package main
import ( import (
"../../"
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
"github.com/go-fsnotify/fsnotify" "github.com/go-fsnotify/fsnotify"
"github.com/olahol/melody"
"io/ioutil" "io/ioutil"
"net/http" "net/http"
) )

2
examples/multichat/main.go

@ -1,8 +1,8 @@
package main package main
import ( import (
"../../"
"github.com/gin-gonic/gin" "github.com/gin-gonic/gin"
"github.com/olahol/melody"
"net/http" "net/http"
) )

16
melody.go

@ -14,6 +14,7 @@ type Melody struct {
Config *Config Config *Config
Upgrader *websocket.Upgrader Upgrader *websocket.Upgrader
messageHandler handleMessageFunc messageHandler handleMessageFunc
messageHandlerBinary handleMessageFunc
errorHandler handleErrorFunc errorHandler handleErrorFunc
connectHandler handleSessionFunc connectHandler handleSessionFunc
disconnectHandler handleSessionFunc disconnectHandler handleSessionFunc
@ -35,6 +36,7 @@ func New() *Melody {
Config: newConfig(), Config: newConfig(),
Upgrader: upgrader, Upgrader: upgrader,
messageHandler: func(*Session, []byte) {}, messageHandler: func(*Session, []byte) {},
messageHandlerBinary: func(*Session, []byte) {},
errorHandler: func(*Session, error) {}, errorHandler: func(*Session, error) {},
connectHandler: func(*Session) {}, connectHandler: func(*Session) {},
disconnectHandler: func(*Session) {}, disconnectHandler: func(*Session) {},
@ -57,6 +59,11 @@ func (m *Melody) HandleMessage(fn func(*Session, []byte)) {
m.messageHandler = fn m.messageHandler = fn
} }
// Callback when a binary message comes in.
func (m *Melody) HandleMessageBinary(fn func(*Session, []byte)) {
m.messageHandlerBinary = fn
}
// Fires when a session has an error. // Fires when a session has an error.
func (m *Melody) HandleError(fn func(*Session, error)) { func (m *Melody) HandleError(fn func(*Session, error)) {
m.errorHandler = fn m.errorHandler = fn
@ -79,7 +86,7 @@ func (m *Melody) HandleRequest(w http.ResponseWriter, r *http.Request) {
go session.writePump(m.errorHandler) go session.writePump(m.errorHandler)
session.readPump(m.messageHandler, m.errorHandler)
session.readPump(m.messageHandler, m.messageHandlerBinary, m.errorHandler)
m.hub.unregister <- session m.hub.unregister <- session
@ -97,3 +104,10 @@ func (m *Melody) BroadcastFilter(msg []byte, fn func(*Session) bool) {
message := &envelope{t: websocket.TextMessage, msg: msg, filter: fn} message := &envelope{t: websocket.TextMessage, msg: msg, filter: fn}
m.hub.broadcast <- message m.hub.broadcast <- message
} }
// Broadcasts a message to all sessions except session `s`.
func (m *Melody) BroadcastOthers(msg []byte, s *Session) {
m.BroadcastFilter(msg, func(q *Session) bool {
return s != q
})
}

183
melody_test.go

@ -77,6 +77,69 @@ func TestEcho(t *testing.T) {
} }
} }
func TestEchoBinary(t *testing.T) {
echo := NewTestServer()
echo.m.HandleMessageBinary(func(session *Session, msg []byte) {
session.WriteBinary(msg)
})
server := httptest.NewServer(echo)
defer server.Close()
fn := func(msg string) bool {
conn, err := NewDialer(server.URL)
defer conn.Close()
if err != nil {
t.Error(err)
return false
}
conn.WriteMessage(websocket.BinaryMessage, []byte(msg))
_, ret, err := conn.ReadMessage()
if err != nil {
t.Error(err)
return false
}
if msg != string(ret) {
t.Errorf("%s should equal %s", msg, string(ret))
return false
}
return true
}
if err := quick.Check(fn, nil); err != nil {
t.Error(err)
}
}
func TestHandlers(t *testing.T) {
echo := NewTestServer()
echo.m.HandleMessage(func(session *Session, msg []byte) {
session.Write(msg)
})
server := httptest.NewServer(echo)
defer server.Close()
var q *Session
echo.m.HandleConnect(func(session *Session) {
q = session
session.Close()
})
echo.m.HandleDisconnect(func(session *Session) {
if q != session {
t.Error("disconnecting session should be the same as connecting")
}
})
NewDialer(server.URL)
}
func TestUpgrader(t *testing.T) { func TestUpgrader(t *testing.T) {
broadcast := NewTestServer() broadcast := NewTestServer()
broadcast.m.HandleMessage(func(session *Session, msg []byte) { broadcast.m.HandleMessage(func(session *Session, msg []byte) {
@ -141,6 +204,65 @@ func TestBroadcast(t *testing.T) {
} }
} }
_, ret, err := conn.ReadMessage()
if err != nil {
t.Error(err)
return false
}
if msg != string(ret) {
t.Errorf("%s should equal %s", msg, string(ret))
return false
}
return true
}
if err := quick.Check(fn, nil); err != nil {
t.Error(err)
}
}
func TestBroadcastOthers(t *testing.T) {
broadcast := NewTestServer()
broadcast.m.HandleMessage(func(session *Session, msg []byte) {
broadcast.m.BroadcastOthers(msg, session)
})
broadcast.m.Config.PongWait = time.Second
broadcast.m.Config.PingPeriod = time.Second * 9 / 10
server := httptest.NewServer(broadcast)
defer server.Close()
n := 10
fn := func(msg string) bool {
conn, _ := NewDialer(server.URL)
defer conn.Close()
listeners := make([]*websocket.Conn, n)
for i := 0; i < n; i++ {
listener, _ := NewDialer(server.URL)
listeners[i] = listener
defer listeners[i].Close()
}
conn.WriteMessage(websocket.TextMessage, []byte(msg))
for i := 0; i < n; i++ {
_, ret, err := listeners[i].ReadMessage()
if err != nil {
t.Error(err)
return false
}
if msg != string(ret) {
t.Errorf("%s should equal %s", msg, string(ret))
return false
}
}
return true return true
} }
@ -175,23 +297,18 @@ func TestPingPong(t *testing.T) {
} }
} }
/*
func TestBroadcastFilter(t *testing.T) { func TestBroadcastFilter(t *testing.T) {
echo := NewTestServer()
echo.m.HandleMessage(func(session *Session, msg []byte) {
echo.m.BroadcastFilter(func(s *Session) bool {
//return s == session
return false
}, msg)
broadcast := NewTestServer()
broadcast.m.HandleMessage(func(session *Session, msg []byte) {
broadcast.m.BroadcastFilter(msg, func(q *Session) bool {
return session == q
}) })
server := httptest.NewServer(echo)
})
server := httptest.NewServer(broadcast)
defer server.Close() defer server.Close()
fn := func(msg string) bool { fn := func(msg string) bool {
conn, err := NewDialer(server.URL) conn, err := NewDialer(server.URL)
conn.SetPingHandler(func(string) error {
return nil
})
defer conn.Close() defer conn.Close()
if err != nil { if err != nil {
@ -220,47 +337,3 @@ func TestBroadcastFilter(t *testing.T) {
t.Error(err) t.Error(err)
} }
} }
*/
func BenchmarkEcho(b *testing.B) {
echo := NewTestServerHandler(func(session *Session, msg []byte) {
session.Write(msg)
})
server := httptest.NewServer(echo)
defer server.Close()
conn, _ := NewDialer(server.URL)
defer conn.Close()
for i := 0; i < b.N; i++ {
conn.WriteMessage(websocket.TextMessage, []byte("test"))
conn.ReadMessage()
}
}
func BenchmarkBroadcast(b *testing.B) {
broadcast := NewTestServer()
broadcast.m.HandleMessage(func(session *Session, msg []byte) {
broadcast.m.Broadcast(msg)
})
server := httptest.NewServer(broadcast)
defer server.Close()
conn, _ := NewDialer(server.URL)
defer conn.Close()
n := 10
listeners := make([]*websocket.Conn, n)
for i := 0; i < n; i++ {
listener, _ := NewDialer(server.URL)
listeners[i] = listener
defer listeners[i].Close()
}
for i := 0; i < b.N; i++ {
conn.WriteMessage(websocket.TextMessage, []byte("test"))
for i := 0; i < n; i++ {
listeners[i].ReadMessage()
}
}
}

31
session.go

@ -29,7 +29,21 @@ func (s *Session) writeMessage(message *envelope) {
func (s *Session) writeRaw(message *envelope) error { func (s *Session) writeRaw(message *envelope) error {
s.conn.SetWriteDeadline(time.Now().Add(s.config.WriteWait)) s.conn.SetWriteDeadline(time.Now().Add(s.config.WriteWait))
return s.conn.WriteMessage(message.t, message.msg)
err := s.conn.WriteMessage(message.t, message.msg)
if err != nil {
return err
}
if message.t == websocket.CloseMessage {
err := s.conn.Close()
if err != nil {
return err
}
}
return nil
} }
func (s *Session) close() { func (s *Session) close() {
@ -63,7 +77,7 @@ func (s *Session) writePump(errorHandler handleErrorFunc) {
} }
} }
func (s *Session) readPump(messageHandler handleMessageFunc, errorHandler handleErrorFunc) {
func (s *Session) readPump(messageHandler handleMessageFunc, messageHandlerBinary handleMessageFunc, errorHandler handleErrorFunc) {
defer s.conn.Close() defer s.conn.Close()
s.conn.SetReadLimit(s.config.MaxMessageSize) s.conn.SetReadLimit(s.config.MaxMessageSize)
@ -75,15 +89,26 @@ func (s *Session) readPump(messageHandler handleMessageFunc, errorHandler handle
}) })
for { for {
_, message, err := s.conn.ReadMessage()
t, message, err := s.conn.ReadMessage()
if err != nil { if err != nil {
go errorHandler(s, err) go errorHandler(s, err)
break break
} }
if t == websocket.TextMessage {
go messageHandler(s, message) go messageHandler(s, message)
} }
if t == websocket.BinaryMessage {
go messageHandlerBinary(s, message)
}
}
}
// Write message to session.
func (s *Session) WriteBinary(msg []byte) {
s.writeMessage(&envelope{t: websocket.BinaryMessage, msg: msg})
} }
// Write message to session. // Write message to session.

Loading…
Cancel
Save