134 lines
4.2 KiB
Go
Raw Normal View History

2017-08-26 10:53:28 +02:00
package slack
import (
2017-12-01 23:52:25 +01:00
"context"
"encoding/json"
2017-08-26 10:53:28 +02:00
"net/url"
2018-07-21 13:21:22 +02:00
"sync"
2017-12-01 23:52:25 +01:00
"time"
2018-07-21 13:21:22 +02:00
"github.com/gorilla/websocket"
2017-08-26 10:53:28 +02:00
)
const (
websocketDefaultTimeout = 10 * time.Second
2018-08-26 14:12:23 +02:00
defaultPingInterval = 30 * time.Second
)
const (
rtmEventTypeAck = ""
rtmEventTypeHello = "hello"
rtmEventTypeGoodbye = "goodbye"
rtmEventTypePong = "pong"
rtmEventTypeDesktopNotification = "desktop_notification"
)
2017-12-01 23:52:25 +01:00
// StartRTM calls the "rtm.start" endpoint and returns the provided URL and the full Info block.
2017-08-26 10:53:28 +02:00
//
2017-12-01 23:52:25 +01:00
// To have a fully managed Websocket connection, use `NewRTM`, and call `ManageConnection()` on it.
2017-08-26 10:53:28 +02:00
func (api *Client) StartRTM() (info *Info, websocketURL string, err error) {
ctx, cancel := context.WithTimeout(context.Background(), websocketDefaultTimeout)
defer cancel()
return api.StartRTMContext(ctx)
2017-12-01 23:52:25 +01:00
}
// StartRTMContext calls the "rtm.start" endpoint and returns the provided URL and the full Info block with a custom context.
//
// To have a fully managed Websocket connection, use `NewRTM`, and call `ManageConnection()` on it.
func (api *Client) StartRTMContext(ctx context.Context) (info *Info, websocketURL string, err error) {
2017-08-26 10:53:28 +02:00
response := &infoResponseFull{}
err = api.postMethod(ctx, "rtm.start", url.Values{"token": {api.token}}, response)
2017-12-01 23:52:25 +01:00
if err != nil {
2018-07-21 13:21:22 +02:00
return nil, "", err
2017-12-01 23:52:25 +01:00
}
2018-07-21 13:21:22 +02:00
2017-12-01 23:52:25 +01:00
api.Debugln("Using URL:", response.Info.URL)
2018-07-21 13:21:22 +02:00
return &response.Info, response.Info.URL, response.Err()
2017-12-01 23:52:25 +01:00
}
// ConnectRTM calls the "rtm.connect" endpoint and returns the provided URL and the compact Info block.
//
// To have a fully managed Websocket connection, use `NewRTM`, and call `ManageConnection()` on it.
func (api *Client) ConnectRTM() (info *Info, websocketURL string, err error) {
ctx, cancel := context.WithTimeout(context.Background(), websocketDefaultTimeout)
defer cancel()
return api.ConnectRTMContext(ctx)
2017-12-01 23:52:25 +01:00
}
2018-07-21 13:21:22 +02:00
// ConnectRTMContext calls the "rtm.connect" endpoint and returns the
// provided URL and the compact Info block with a custom context.
2017-12-01 23:52:25 +01:00
//
// To have a fully managed Websocket connection, use `NewRTM`, and call `ManageConnection()` on it.
func (api *Client) ConnectRTMContext(ctx context.Context) (info *Info, websocketURL string, err error) {
response := &infoResponseFull{}
err = api.postMethod(ctx, "rtm.connect", url.Values{"token": {api.token}}, response)
2017-08-26 10:53:28 +02:00
if err != nil {
api.Debugf("Failed to connect to RTM: %s", err)
2018-07-21 13:21:22 +02:00
return nil, "", err
2017-08-26 10:53:28 +02:00
}
2018-07-21 13:21:22 +02:00
2017-08-26 10:53:28 +02:00
api.Debugln("Using URL:", response.Info.URL)
2018-07-21 13:21:22 +02:00
return &response.Info, response.Info.URL, response.Err()
2017-08-26 10:53:28 +02:00
}
2018-07-21 13:21:22 +02:00
// RTMOption options for the managed RTM.
type RTMOption func(*RTM)
// RTMOptionUseStart as of 11th July 2017 you should prefer setting this to false, see:
// https://api.slack.com/changelog/2017-04-start-using-rtm-connect-and-stop-using-rtm-start
func RTMOptionUseStart(b bool) RTMOption {
return func(rtm *RTM) {
rtm.useRTMStart = b
}
}
// RTMOptionDialer takes a gorilla websocket Dialer and uses it as the
// Dialer when opening the websocket for the RTM connection.
func RTMOptionDialer(d *websocket.Dialer) RTMOption {
return func(rtm *RTM) {
rtm.dialer = d
}
2017-12-01 23:52:25 +01:00
}
2018-08-26 14:12:23 +02:00
// RTMOptionPingInterval determines how often to deliver a ping message to slack.
func RTMOptionPingInterval(d time.Duration) RTMOption {
return func(rtm *RTM) {
rtm.pingInterval = d
rtm.resetDeadman()
}
}
// RTMOptionConnParams installs parameters to embed into the connection URL.
func RTMOptionConnParams(connParams url.Values) RTMOption {
return func(rtm *RTM) {
rtm.connParams = connParams
}
}
2018-07-21 13:21:22 +02:00
// NewRTM returns a RTM, which provides a fully managed connection to
2017-12-01 23:52:25 +01:00
// Slack's websocket-based Real-Time Messaging protocol.
2018-07-21 13:21:22 +02:00
func (api *Client) NewRTM(options ...RTMOption) *RTM {
2017-12-01 23:52:25 +01:00
result := &RTM{
Client: *api,
IncomingEvents: make(chan RTMEvent, 50),
outgoingMessages: make(chan OutgoingMessage, 20),
2018-08-26 14:12:23 +02:00
pingInterval: defaultPingInterval,
pingDeadman: time.NewTimer(deadmanDuration(defaultPingInterval)),
2017-12-01 23:52:25 +01:00
killChannel: make(chan bool),
disconnected: make(chan struct{}),
disconnectedm: &sync.Once{},
2017-12-01 23:52:25 +01:00
forcePing: make(chan bool),
rawEvents: make(chan json.RawMessage),
idGen: NewSafeID(1),
2018-07-21 13:21:22 +02:00
mu: &sync.Mutex{},
2017-12-01 23:52:25 +01:00
}
2018-07-21 13:21:22 +02:00
for _, opt := range options {
opt(result)
2017-12-01 23:52:25 +01:00
}
return result
2017-08-26 10:53:28 +02:00
}