fayec/transport/websocket/websocket.go
2018-09-05 11:59:20 +02:00

175 lines
3.7 KiB
Go

package websocket
import (
"encoding/json"
"github.com/gorilla/websocket"
"github.com/thesyncim/faye/message"
"github.com/thesyncim/faye/transport"
"log"
"strconv"
"strings"
"sync/atomic"
)
const transportName = "websocket"
func init() {
transport.RegisterTransport(&Websocket{})
}
var Debug = true
func debugJson(v interface{}) string {
b, _ := json.MarshalIndent(v, "", " ")
return string(b)
}
type Websocket struct {
TransportOpts *transport.Options
conn *websocket.Conn
clientID string
msgID *uint64
subs map[string]chan struct{}
}
var _ transport.Transport = (*Websocket)(nil)
func (w *Websocket) Init(options *transport.Options) error {
var (
err error
msgID uint64
)
w.TransportOpts = options
w.msgID = &msgID
w.subs = map[string]chan struct{}{}
w.conn, _, err = websocket.DefaultDialer.Dial(options.Url, nil)
if err != nil {
return err
}
return nil
}
func (w *Websocket) Name() string {
return transportName
}
func (w *Websocket) nextMsgID() string {
return strconv.Itoa(int(atomic.AddUint64(w.msgID, 1)))
}
func (w *Websocket) Options() *transport.Options {
return w.TransportOpts
}
func (w *Websocket) Handshake() (err error) {
var payload []message.Message
payload = append(payload, message.Message{
Channel: string(transport.Handshake),
Version: "1.0", //todo const
SupportedConnectionTypes: []string{transportName},
})
if Debug {
log.Println("handshake request", debugJson(payload))
}
if err = w.conn.WriteJSON(payload); err != nil {
return err
}
var hsResps []message.Message
if err = w.conn.ReadJSON(&hsResps); err != nil {
return err
}
if Debug {
log.Println("handshake response", debugJson(hsResps))
}
resp := hsResps[0]
if resp.GetError() != nil {
return err
}
w.clientID = resp.ClientId
return nil
}
func (w *Websocket) Connect() error {
var payload []message.Message
payload = append(payload, message.Message{
Channel: string(transport.Connect),
ClientId: w.clientID,
ConnectionType: transportName,
Id: w.nextMsgID(),
})
if Debug {
log.Println("connect request", debugJson(payload))
}
//todo verify if extensions are applied on connect,verify if hs is complete
return w.conn.WriteJSON(payload)
}
func (w *Websocket) Subscribe(subscription string, onMessage func(message *message.Message)) error {
m := &message.Message{
Channel: string(transport.Subscribe),
ClientId: w.clientID,
Subscription: subscription,
Id: w.nextMsgID(),
}
if w.TransportOpts.OutExt != nil {
w.TransportOpts.OutExt(m)
}
var payload []message.Message
payload = append(payload, *m)
if Debug {
log.Println("subscribe request", debugJson(payload))
}
err := w.conn.WriteJSON(payload)
if err != nil {
return err
}
var hsResps []message.Message
if err = w.conn.ReadJSON(&hsResps); err != nil {
return err
}
if Debug {
log.Println("subscribe response", debugJson(hsResps))
}
subResp := hsResps[0]
if subResp.GetError() != nil {
return err
}
if !subResp.Successful {
//report err just for sanity
}
unsubsCh := make(chan struct{}, 0)
w.subs[subscription] = unsubsCh
for {
select {
case <-unsubsCh:
return nil
default:
}
//todo guard unsusribe
var payload []message.Message
err := w.conn.ReadJSON(&payload)
if err != nil {
return err
}
//hack
msg := payload[0]
if strings.HasPrefix(msg.Channel, "/meta") {
continue //todo update introspect message and update state
}
onMessage(&msg)
}
return nil
}
func (w *Websocket) Unsubscribe(subscription string) error {
panic("not implemented")
}
func (w *Websocket) Publish(subscription string, message *message.Message) error {
panic("not implemented")
}