2017-08-16 23:37:37 +02:00
|
|
|
// Copyright (c) 2016-present Mattermost, Inc. All Rights Reserved.
|
2016-08-15 18:47:31 +02:00
|
|
|
// See License.txt for license information.
|
|
|
|
|
|
|
|
package model
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
2018-02-09 00:11:04 +01:00
|
|
|
"net/http"
|
2018-11-18 17:55:05 +00:00
|
|
|
"time"
|
2018-02-09 00:11:04 +01:00
|
|
|
|
2016-08-15 18:47:31 +02:00
|
|
|
"github.com/gorilla/websocket"
|
|
|
|
)
|
|
|
|
|
2017-03-25 21:04:10 +01:00
|
|
|
const (
|
2018-11-18 17:55:05 +00:00
|
|
|
SOCKET_MAX_MESSAGE_SIZE_KB = 8 * 1024 // 8KB
|
|
|
|
PING_TIMEOUT_BUFFER_SECONDS = 5
|
2017-03-25 21:04:10 +01:00
|
|
|
)
|
|
|
|
|
2016-08-15 18:47:31 +02:00
|
|
|
type WebSocketClient struct {
|
2018-11-18 17:55:05 +00:00
|
|
|
Url string // The location of the server like "ws://localhost:8065"
|
|
|
|
ApiUrl string // The api location of the server like "ws://localhost:8065/api/v3"
|
|
|
|
ConnectUrl string // The websocket URL to connect to like "ws://localhost:8065/api/v3/path/to/websocket"
|
|
|
|
Conn *websocket.Conn // The WebSocket connection
|
|
|
|
AuthToken string // The token used to open the WebSocket
|
|
|
|
Sequence int64 // The ever-incrementing sequence attached to each WebSocket action
|
|
|
|
PingTimeoutChannel chan bool // The channel used to signal ping timeouts
|
|
|
|
EventChannel chan *WebSocketEvent
|
|
|
|
ResponseChannel chan *WebSocketResponse
|
|
|
|
ListenError *AppError
|
|
|
|
pingTimeoutTimer *time.Timer
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewWebSocketClient constructs a new WebSocket client with convenience
|
2016-08-15 18:47:31 +02:00
|
|
|
// methods for talking to the server.
|
|
|
|
func NewWebSocketClient(url, authToken string) (*WebSocketClient, *AppError) {
|
2018-11-18 17:55:05 +00:00
|
|
|
return NewWebSocketClientWithDialer(websocket.DefaultDialer, url, authToken)
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewWebSocketClientWithDialer constructs a new WebSocket client with convenience
|
|
|
|
// methods for talking to the server using a custom dialer.
|
|
|
|
func NewWebSocketClientWithDialer(dialer *websocket.Dialer, url, authToken string) (*WebSocketClient, *AppError) {
|
|
|
|
conn, _, err := dialer.Dial(url+API_URL_SUFFIX+"/websocket", nil)
|
2016-08-15 18:47:31 +02:00
|
|
|
if err != nil {
|
2018-02-09 00:11:04 +01:00
|
|
|
return nil, NewAppError("NewWebSocketClient", "model.websocket_client.connect_fail.app_error", nil, err.Error(), http.StatusInternalServerError)
|
2016-08-15 18:47:31 +02:00
|
|
|
}
|
|
|
|
|
2016-11-12 22:00:53 +01:00
|
|
|
client := &WebSocketClient{
|
2016-08-15 18:47:31 +02:00
|
|
|
url,
|
2018-11-18 17:55:05 +00:00
|
|
|
url + API_URL_SUFFIX,
|
|
|
|
url + API_URL_SUFFIX + "/websocket",
|
2017-08-16 23:37:37 +02:00
|
|
|
conn,
|
|
|
|
authToken,
|
|
|
|
1,
|
2018-11-18 17:55:05 +00:00
|
|
|
make(chan bool, 1),
|
2017-08-16 23:37:37 +02:00
|
|
|
make(chan *WebSocketEvent, 100),
|
|
|
|
make(chan *WebSocketResponse, 100),
|
|
|
|
nil,
|
2018-11-18 17:55:05 +00:00
|
|
|
nil,
|
2017-08-16 23:37:37 +02:00
|
|
|
}
|
|
|
|
|
2018-11-18 17:55:05 +00:00
|
|
|
client.configurePingHandling()
|
|
|
|
|
2017-08-16 23:37:37 +02:00
|
|
|
client.SendMessage(WEBSOCKET_AUTHENTICATION_CHALLENGE, map[string]interface{}{"token": authToken})
|
|
|
|
|
|
|
|
return client, nil
|
|
|
|
}
|
|
|
|
|
2018-11-18 17:55:05 +00:00
|
|
|
// NewWebSocketClient4 constructs a new WebSocket client with convenience
|
2017-08-16 23:37:37 +02:00
|
|
|
// methods for talking to the server. Uses the v4 endpoint.
|
|
|
|
func NewWebSocketClient4(url, authToken string) (*WebSocketClient, *AppError) {
|
2018-11-18 17:55:05 +00:00
|
|
|
return NewWebSocketClient4WithDialer(websocket.DefaultDialer, url, authToken)
|
|
|
|
}
|
2016-11-12 22:00:53 +01:00
|
|
|
|
2018-11-18 17:55:05 +00:00
|
|
|
// NewWebSocketClient4WithDialer constructs a new WebSocket client with convenience
|
|
|
|
// methods for talking to the server using a custom dialer. Uses the v4 endpoint.
|
|
|
|
func NewWebSocketClient4WithDialer(dialer *websocket.Dialer, url, authToken string) (*WebSocketClient, *AppError) {
|
|
|
|
return NewWebSocketClientWithDialer(dialer, url, authToken)
|
2016-08-15 18:47:31 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (wsc *WebSocketClient) Connect() *AppError {
|
2018-11-18 17:55:05 +00:00
|
|
|
return wsc.ConnectWithDialer(websocket.DefaultDialer)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (wsc *WebSocketClient) ConnectWithDialer(dialer *websocket.Dialer) *AppError {
|
2016-08-15 18:47:31 +02:00
|
|
|
var err error
|
2018-11-18 17:55:05 +00:00
|
|
|
wsc.Conn, _, err = dialer.Dial(wsc.ConnectUrl, nil)
|
2016-08-15 18:47:31 +02:00
|
|
|
if err != nil {
|
2018-02-09 00:11:04 +01:00
|
|
|
return NewAppError("Connect", "model.websocket_client.connect_fail.app_error", nil, err.Error(), http.StatusInternalServerError)
|
2016-08-15 18:47:31 +02:00
|
|
|
}
|
|
|
|
|
2018-11-18 17:55:05 +00:00
|
|
|
wsc.configurePingHandling()
|
|
|
|
|
2016-11-12 22:00:53 +01:00
|
|
|
wsc.EventChannel = make(chan *WebSocketEvent, 100)
|
|
|
|
wsc.ResponseChannel = make(chan *WebSocketResponse, 100)
|
|
|
|
|
|
|
|
wsc.SendMessage(WEBSOCKET_AUTHENTICATION_CHALLENGE, map[string]interface{}{"token": wsc.AuthToken})
|
|
|
|
|
2016-08-15 18:47:31 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (wsc *WebSocketClient) Close() {
|
|
|
|
wsc.Conn.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (wsc *WebSocketClient) Listen() {
|
|
|
|
go func() {
|
2016-11-12 22:00:53 +01:00
|
|
|
defer func() {
|
|
|
|
wsc.Conn.Close()
|
|
|
|
close(wsc.EventChannel)
|
|
|
|
close(wsc.ResponseChannel)
|
|
|
|
}()
|
|
|
|
|
2016-08-15 18:47:31 +02:00
|
|
|
for {
|
|
|
|
var rawMsg json.RawMessage
|
|
|
|
var err error
|
|
|
|
if _, rawMsg, err = wsc.Conn.ReadMessage(); err != nil {
|
2016-11-12 22:00:53 +01:00
|
|
|
if !websocket.IsCloseError(err, websocket.CloseNormalClosure, websocket.CloseNoStatusReceived) {
|
2018-02-09 00:11:04 +01:00
|
|
|
wsc.ListenError = NewAppError("NewWebSocketClient", "model.websocket_client.connect_fail.app_error", nil, err.Error(), http.StatusInternalServerError)
|
2016-11-12 22:00:53 +01:00
|
|
|
}
|
|
|
|
|
2016-08-15 18:47:31 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
var event WebSocketEvent
|
|
|
|
if err := json.Unmarshal(rawMsg, &event); err == nil && event.IsValid() {
|
|
|
|
wsc.EventChannel <- &event
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
var response WebSocketResponse
|
|
|
|
if err := json.Unmarshal(rawMsg, &response); err == nil && response.IsValid() {
|
|
|
|
wsc.ResponseChannel <- &response
|
|
|
|
continue
|
|
|
|
}
|
2016-11-12 22:00:53 +01:00
|
|
|
|
2016-08-15 18:47:31 +02:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (wsc *WebSocketClient) SendMessage(action string, data map[string]interface{}) {
|
|
|
|
req := &WebSocketRequest{}
|
|
|
|
req.Seq = wsc.Sequence
|
|
|
|
req.Action = action
|
|
|
|
req.Data = data
|
|
|
|
|
|
|
|
wsc.Sequence++
|
|
|
|
|
|
|
|
wsc.Conn.WriteJSON(req)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UserTyping will push a user_typing event out to all connected users
|
|
|
|
// who are in the specified channel
|
|
|
|
func (wsc *WebSocketClient) UserTyping(channelId, parentId string) {
|
|
|
|
data := map[string]interface{}{
|
|
|
|
"channel_id": channelId,
|
|
|
|
"parent_id": parentId,
|
|
|
|
}
|
|
|
|
|
|
|
|
wsc.SendMessage("user_typing", data)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetStatuses will return a map of string statuses using user id as the key
|
|
|
|
func (wsc *WebSocketClient) GetStatuses() {
|
|
|
|
wsc.SendMessage("get_statuses", nil)
|
|
|
|
}
|
2016-11-12 22:00:53 +01:00
|
|
|
|
|
|
|
// GetStatusesByIds will fetch certain user statuses based on ids and return
|
|
|
|
// a map of string statuses using user id as the key
|
|
|
|
func (wsc *WebSocketClient) GetStatusesByIds(userIds []string) {
|
|
|
|
data := map[string]interface{}{
|
|
|
|
"user_ids": userIds,
|
|
|
|
}
|
|
|
|
wsc.SendMessage("get_statuses_by_ids", data)
|
|
|
|
}
|
2018-11-18 17:55:05 +00:00
|
|
|
|
|
|
|
func (wsc *WebSocketClient) configurePingHandling() {
|
|
|
|
wsc.Conn.SetPingHandler(wsc.pingHandler)
|
|
|
|
wsc.pingTimeoutTimer = time.NewTimer(time.Second * (60 + PING_TIMEOUT_BUFFER_SECONDS))
|
|
|
|
go wsc.pingWatchdog()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (wsc *WebSocketClient) pingHandler(appData string) error {
|
|
|
|
if !wsc.pingTimeoutTimer.Stop() {
|
|
|
|
<-wsc.pingTimeoutTimer.C
|
|
|
|
}
|
|
|
|
|
|
|
|
wsc.pingTimeoutTimer.Reset(time.Second * (60 + PING_TIMEOUT_BUFFER_SECONDS))
|
|
|
|
wsc.Conn.WriteMessage(websocket.PongMessage, []byte{})
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (wsc *WebSocketClient) pingWatchdog() {
|
|
|
|
<-wsc.pingTimeoutTimer.C
|
|
|
|
wsc.PingTimeoutChannel <- true
|
|
|
|
}
|