Slack.go improvements and increasing coverage (#193)

* Slack.go improvements and increasing converage

Slack.go Adding handling for trying to send responses when the websocket isn't connected
Slack.go Adding handling for an error response other than "Socket URL has expired"
Slack.go Making handleErrorResponse return an error
Slack.go Making use of s.Connected
Slack.go Making HandleMessage return an error
Slack.go Removing SendHTTPGetRequestSlack, code now calls SendHTTPGetRequest from common.go instead

More coverage for slack GetChannelsString test
More coverage for slack GetUsernameByID test
Improving slack GetIDByName test
Adding slack GetGroupIDByName test
Improving slack GetChannelIDByName test
More coverage for slack GetUsersInGroup test
Adding slack WebSocketConnect test
Adding slack handlePresenceChange test
Adding slack handleMessageResponse test
Adding slack handleErrorResponse test
Adding slack handleHelloResponse test
Adding slack handleReconnectResponse test
Adding slack WebsocketSend test
Adding slack HandleMessage test
Adding slack SendHTTPGetRequestSlack test

* Fixing race conditions and missing exclamation mark.
This commit is contained in:
cranktakular
2018-10-18 17:33:24 +11:00
committed by Adrian Gallagher
parent 0c92ccfa21
commit 415332b446
2 changed files with 359 additions and 70 deletions

View File

@@ -7,7 +7,6 @@ import (
"encoding/json"
"errors"
"fmt"
"io/ioutil"
"log"
"net/http"
"sync"
@@ -73,7 +72,6 @@ func (s *Slack) Connect() error {
if err := s.NewConnection(); err != nil {
return err
}
s.Connected = true
return nil
}
@@ -148,30 +146,33 @@ func (s *Slack) GetUsersInGroup(group string) []string {
// NewConnection connects the bot to a slack workgroup using a verification
// token and a channel
func (s *Slack) NewConnection() error {
err := s.SendHTTPGetRequestSlack(s.BuildURL(s.VerificationToken), true, &s.Details)
if err != nil {
return err
}
if !s.Connected {
err := common.SendHTTPGetRequest(s.BuildURL(s.VerificationToken), true, s.Verbose, &s.Details)
if err != nil {
return err
}
if !s.Details.Ok {
return errors.New(s.Details.Error)
}
if !s.Details.Ok {
return errors.New(s.Details.Error)
}
if s.Verbose {
log.Printf("%s [%s] connected to %s [%s] \nWebsocket URL: %s.\n",
s.Details.Self.Name,
s.Details.Self.ID,
s.Details.Team.Domain,
s.Details.Team.ID,
s.Details.URL)
log.Printf("Slack channels: %s", s.GetChannelsString())
}
if s.Verbose {
log.Printf("%s [%s] connected to %s [%s] \nWebsocket URL: %s.\n",
s.Details.Self.Name,
s.Details.Self.ID,
s.Details.Team.Domain,
s.Details.Team.ID,
s.Details.URL)
log.Printf("Slack channels: %s", s.GetChannelsString())
}
s.TargetChannelID, err = s.GetIDByName(s.TargetChannel)
if err != nil {
return err
s.TargetChannelID, err = s.GetIDByName(s.TargetChannel)
if err != nil {
return err
}
return s.WebsocketConnect()
}
return s.WebsocketConnect()
return errors.New("slack.go NewConnection() Already Connected")
}
// WebsocketConnect creates a websocket dialer amd initiates a websocket
@@ -213,7 +214,10 @@ func (s *Slack) WebsocketReader() {
switch data.Type {
case "error":
s.handleErrorResponse(data)
err = s.handleErrorResponse(data)
if err != nil {
continue
}
case "hello":
s.handleHelloResponse(data)
@@ -275,25 +279,32 @@ func (s *Slack) handleMessageResponse(resp []byte, data WebsocketResponse) error
msg.User, msg.Text)
}
if string(msg.Text[0]) == "!" {
s.HandleMessage(msg)
return s.HandleMessage(msg)
}
return nil
}
func (s *Slack) handleErrorResponse(data WebsocketResponse) {
func (s *Slack) handleErrorResponse(data WebsocketResponse) error {
if data.Error.Msg == "Socket URL has expired" {
if s.Verbose {
log.Println("Slack websocket URL has expired.. Reconnecting")
}
if s.WebsocketConn == nil {
return errors.New("Websocket connection is nil")
}
if err := s.WebsocketConn.Close(); err != nil {
log.Println(err)
}
s.ReconnectURL = ""
s.Connected = false
if err := s.NewConnection(); err != nil {
log.Fatal(err)
return err
}
return
return nil
}
return fmt.Errorf("Unknown error '%s'", data.Error.Msg)
}
func (s *Slack) handleHelloResponse(data WebsocketResponse) {
@@ -346,59 +357,35 @@ func (s *Slack) WebsocketSend(eventType, text string) error {
if err != nil {
return err
}
if s.WebsocketConn == nil {
return errors.New("Websocket not connected")
}
return s.WebsocketConn.WriteMessage(websocket.TextMessage, data)
}
// HandleMessage handles incoming messages and/or commands from slack
func (s *Slack) HandleMessage(msg Message) {
func (s *Slack) HandleMessage(msg Message) error {
msg.Text = common.StringToLower(msg.Text)
switch {
case common.StringContains(msg.Text, cmdStatus):
s.WebsocketSend("message", s.GetStatus())
return s.WebsocketSend("message", s.GetStatus())
case common.StringContains(msg.Text, cmdHelp):
s.WebsocketSend("message", getHelp)
return s.WebsocketSend("message", getHelp)
case common.StringContains(msg.Text, cmdTicker):
s.WebsocketSend("message", s.GetTicker("ANX"))
return s.WebsocketSend("message", s.GetTicker("ANX"))
case common.StringContains(msg.Text, cmdOrderbook):
s.WebsocketSend("message", s.GetOrderbook("ANX"))
return s.WebsocketSend("message", s.GetOrderbook("ANX"))
case common.StringContains(msg.Text, cmdSettings):
s.WebsocketSend("message", s.GetSettings())
return s.WebsocketSend("message", s.GetSettings())
case common.StringContains(msg.Text, cmdPortfolio):
s.WebsocketSend("message", s.GetPortfolio())
return s.WebsocketSend("message", s.GetPortfolio())
default:
s.WebsocketSend("message", "GoCryptoTrader SlackBot - Command Unknown!")
return s.WebsocketSend("message", "GoCryptoTrader SlackBot - Command Unknown!")
}
}
// SendHTTPGetRequestSlack sends a HTTP request
func (s *Slack) SendHTTPGetRequestSlack(url string, jsonDecode bool, result interface{}) error {
res, err := http.Get(url)
if err != nil {
return err
}
httpCode := res.StatusCode
if httpCode != 200 && httpCode != 400 {
log.Printf("HTTP status code: %d\n", httpCode)
return errors.New("status code was not 200")
}
contents, err := ioutil.ReadAll(res.Body)
if err != nil {
return err
}
defer res.Body.Close()
if jsonDecode {
return common.JSONDecode(contents, &result)
}
result = string(contents)
return nil
}

View File

@@ -3,6 +3,8 @@ package slack
import (
"testing"
"github.com/thrasher-/gocryptotrader/common"
"github.com/thrasher-/gocryptotrader/communications/base"
"github.com/thrasher-/gocryptotrader/config"
)
@@ -13,11 +15,37 @@ const (
var s Slack
type group struct {
ID string `json:"id"`
Name string `json:"name"`
IsGroup bool `json:"is_group"`
Created int64 `json:"created"`
Creator string `json:"creator"`
IsArchived bool `json:"is_archived"`
NameNormalised string `json:"name_normalised"`
IsMPIM bool `json:"is_mpim"`
HasPins bool `json:"has_pins"`
IsOpen bool `json:"is_open"`
LastRead string `json:"last_read"`
Members []string `json:"members"`
Topic struct {
Value string `json:"value"`
Creator string `json:"creator"`
LastSet int64 `json:"last_set"`
} `json:"topic"`
Purpose struct {
Value string `json:"value"`
Creator string `json:"creator"`
LastSet int64 `json:"last_set"`
} `json:"purpose"`
}
func TestSetup(t *testing.T) {
cfg := config.GetConfig()
cfg.LoadConfig(config.ConfigTestFile)
s.Setup(cfg.GetCommunicationsConfig())
s.Verbose = true
}
func TestConnect(t *testing.T) {
@@ -28,6 +56,7 @@ func TestConnect(t *testing.T) {
}
func TestPushEvent(t *testing.T) {
t.Parallel()
err := s.PushEvent(base.Event{})
if err == nil {
t.Error("test failed - slack PushEvent() error")
@@ -35,6 +64,7 @@ func TestPushEvent(t *testing.T) {
}
func TestBuildURL(t *testing.T) {
t.Parallel()
v := s.BuildURL("lol123")
if v != "https://slack.com/api/rtm.start?token=lol123" {
t.Error("test failed - slack BuildURL() error")
@@ -42,36 +72,143 @@ func TestBuildURL(t *testing.T) {
}
func TestGetChannelsString(t *testing.T) {
s.Details.Channels = append(s.Details.Channels, struct {
Created int `json:"created"`
Creator string `json:"creator"`
HasPins bool `json:"has_pins"`
ID string `json:"id"`
IsArchived bool `json:"is_archived"`
IsChannel bool `json:"is_channel"`
IsGeneral bool `json:"is_general"`
IsMember bool `json:"is_member"`
IsOrgShared bool `json:"is_org_shared"`
IsShared bool `json:"is_shared"`
Name string `json:"name"`
NameNormalized string `json:"name_normalized"`
PreviousNames []string `json:"previous_names"`
}{
NameNormalized: "General",
})
chans := s.GetChannelsString()
if len(chans) != 0 {
testpassed := false
for i := range chans {
if chans[i] == "General" {
testpassed = true
}
}
if testpassed == false {
t.Error("test failed - slack GetChannelsString() error")
}
}
func TestGetUsernameByID(t *testing.T) {
t.Parallel()
username := s.GetUsernameByID("1337")
if len(username) != 0 {
t.Error("test failed - slack GetUsernameByID() error")
}
s.Details.Users = append(s.Details.Users, struct {
Deleted bool `json:"deleted"`
ID string `json:"id"`
IsBot bool `json:"is_bot"`
Name string `json:"name"`
Presence string `json:"presence"`
Profile struct {
AvatarHash string `json:"avatar_hash"`
Email string `json:"email"`
Fields interface{} `json:"fields"`
FirstName string `json:"first_name"`
Image192 string `json:"image_192"`
Image24 string `json:"image_24"`
Image32 string `json:"image_32"`
Image48 string `json:"image_48"`
Image512 string `json:"image_512"`
Image72 string `json:"image_72"`
LastName string `json:"last_name"`
RealName string `json:"real_name"`
RealNameNormalized string `json:"real_name_normalized"`
} `json:"profile"`
TeamID string `json:"team_id"`
Updated int `json:"updated"`
}{
ID: "1337",
Name: "cranktakular",
})
username = s.GetUsernameByID("1337")
if username != "cranktakular" {
t.Error("test failed - slack GetUsernameByID() error")
}
}
func TestGetIDByName(t *testing.T) {
id, err := s.GetIDByName("batman")
if err == nil {
if err == nil || len(id) != 0 {
t.Error("test failed - slack GetIDByName() error")
}
if len(id) != 0 {
t.Error("test failed - slack GetIDByName() error")
s.Details.Groups = append(s.Details.Groups, group{
Name: "this is a group",
ID: "210314",
})
id, err = s.GetIDByName("this is a group")
if err != nil || id != "210314" {
t.Errorf("test failed - slack GetIDByName() Expected '210314' Actual '%s' Error: %s",
id, err)
}
}
func TestGetGroupIDByName(t *testing.T) {
id, err := s.GetGroupIDByName("batman")
if err == nil || len(id) != 0 {
t.Error("test failed - slack GetGroupIDByName() error")
}
s.Details.Groups = append(s.Details.Groups, group{
Name: "another group",
ID: "11223344",
})
id, err = s.GetGroupIDByName("another group")
if err != nil || id != "11223344" {
t.Errorf("test failed - slack GetGroupIDByName() Expected '11223344' Actual '%s' Error: %s",
id, err)
}
}
func TestGetChannelIDByName(t *testing.T) {
id, err := s.GetChannelIDByName("1337")
if err == nil {
if err == nil || len(id) != 0 {
t.Error("test failed - slack GetChannelIDByName() error")
}
if len(id) != 0 {
t.Error("test failed - slack GetChannelIDByName() error")
s.Details.Channels = append(s.Details.Channels, struct {
Created int `json:"created"`
Creator string `json:"creator"`
HasPins bool `json:"has_pins"`
ID string `json:"id"`
IsArchived bool `json:"is_archived"`
IsChannel bool `json:"is_channel"`
IsGeneral bool `json:"is_general"`
IsMember bool `json:"is_member"`
IsOrgShared bool `json:"is_org_shared"`
IsShared bool `json:"is_shared"`
Name string `json:"name"`
NameNormalized string `json:"name_normalized"`
PreviousNames []string `json:"previous_names"`
}{
ID: "2048",
Name: "Slack Test",
})
id, err = s.GetChannelIDByName("Slack Test")
if err != nil || id != "2048" {
t.Errorf("test failed - slack GetChannelIDByName() Expected '2048' Actual '%s' Error: %s",
id, err)
}
}
@@ -80,6 +217,18 @@ func TestGetUsersInGroup(t *testing.T) {
if len(username) != 0 {
t.Error("test failed - slack GetUsersInGroup() error")
}
s.Details.Groups = append(s.Details.Groups, group{
Name: "three guys",
ID: "3",
Members: []string{"Guy one", "Guy two", "Guy three"},
})
username = s.GetUsersInGroup("three guys")
if len(username) != 3 {
t.Errorf("test failed - slack GetUsersInGroup() Expected '3' Actual '%s'",
username)
}
}
func TestNewConnection(t *testing.T) {
@@ -88,3 +237,156 @@ func TestNewConnection(t *testing.T) {
t.Error("test failed - slack NewConnection() error")
}
}
func TestWebsocketConnect(t *testing.T) {
t.Parallel()
err := s.WebsocketConnect()
if err == nil {
t.Error("test failed - slack WebsocketConnect() error")
}
}
func TestHandlePresenceChange(t *testing.T) {
t.Parallel()
var pres PresenceChange
pres.User = "1337"
pres.Presence = "Present"
err := s.handlePresenceChange([]byte(`{"malformedjson}`))
if err == nil {
t.Error("test failed - slack handlePresenceChange(), unmarshalled malformed json")
}
data, _ := common.JSONEncode(pres)
err = s.handlePresenceChange(data)
if err != nil {
t.Errorf("test failed - slack handlePresenceChange() Error: %s", err)
}
}
func TestHandleMessageResponse(t *testing.T) {
t.Parallel()
var data WebsocketResponse
data.ReplyTo = 1
err := s.handleMessageResponse(nil, data)
if err.Error() != "ReplyTo != 0" {
t.Errorf("test failed - slack handleMessageResponse(), Incorrect Error: %s",
err)
}
data.ReplyTo = 0
err = s.handleMessageResponse([]byte(`{"malformedjson}`), data)
if err == nil {
t.Error("test failed - slack handleMessageResponse(), unmarshalled malformed json")
}
var msg Message
msg.User = "1337"
msg.Text = "Hello World!"
resp, _ := common.JSONEncode(msg)
err = s.handleMessageResponse(resp, data)
if err != nil {
t.Error("test failed - slack HandleMessage(), Sent message through nil websocket")
}
msg.Text = "!notacommand"
resp, _ = common.JSONEncode(msg)
err = s.handleMessageResponse(resp, data)
if err == nil {
t.Errorf("test failed - slack handleMessageResponse() Error: %s", err)
}
}
func TestHandleErrorResponse(t *testing.T) {
var data WebsocketResponse
err := s.handleErrorResponse(data)
if err == nil {
t.Error("test failed - slack handleErrorResponse() Ignored strange input")
}
data.Error.Msg = "Socket URL has expired"
err = s.handleErrorResponse(data)
if err == nil {
t.Error("test failed - slack handleErrorResponse() Didn't error on nil websocket")
}
}
func TestHandleHelloResponse(t *testing.T) {
t.Parallel()
var data WebsocketResponse
s.handleHelloResponse(data)
}
func TestHandleReconnectResponse(t *testing.T) {
t.Parallel()
err := s.handleReconnectResponse([]byte(`{"malformedjson}`))
if err == nil {
t.Error("test failed - slack handleReconnectResponse(), unmarshalled malformed json")
}
var testURL struct {
URL string `json:"url"`
}
testURL.URL = "https://www.thrasher.io"
data, _ := common.JSONEncode(testURL)
err = s.handleReconnectResponse(data)
if err != nil || s.ReconnectURL != "https://www.thrasher.io" {
t.Errorf("test failed - slack handleReconnectResponse() Expected 'https://www.thrasher.io' Actual '%s' Error: %s",
s.ReconnectURL, err)
}
}
func TestWebsocketSend(t *testing.T) {
t.Parallel()
err := s.WebsocketSend("test", "Hello World!")
if err == nil {
t.Error("test failed - slack WebsocketSend(), Sent message through nil websocket")
}
}
func TestHandleMessage(t *testing.T) {
t.Parallel()
var msg Message
err := s.HandleMessage(msg)
if err == nil {
t.Error("test failed - slack HandleMessage(), Sent message through nil websocket")
}
msg.Text = cmdStatus
err = s.HandleMessage(msg)
if err == nil {
t.Error("test failed - slack HandleMessage(), Sent message through nil websocket")
}
msg.Text = cmdHelp
err = s.HandleMessage(msg)
if err == nil {
t.Error("test failed - slack HandleMessage(), Sent message through nil websocket")
}
msg.Text = cmdTicker
err = s.HandleMessage(msg)
if err == nil {
t.Error("test failed - slack HandleMessage(), Sent message through nil websocket")
}
msg.Text = cmdOrderbook
err = s.HandleMessage(msg)
if err == nil {
t.Error("test failed - slack HandleMessage(), Sent message through nil websocket")
}
msg.Text = cmdSettings
err = s.HandleMessage(msg)
if err == nil {
t.Error("test failed - slack HandleMessage(), Sent message through nil websocket")
}
msg.Text = cmdPortfolio
err = s.HandleMessage(msg)
if err == nil {
t.Error("test failed - slack HandleMessage(), Sent message through nil websocket")
}
}