From 34e2404e8bb68d281be77a174c6f7df3393d03b8 Mon Sep 17 00:00:00 2001 From: Scott Kiesel Date: Tue, 31 Oct 2017 06:22:36 -0400 Subject: [PATCH 01/28] hacked poc for edits as an animation mechanism --- bot/bot.go | 6 +- bot/handlers.go | 6 ++ bot/interfaces.go | 3 + bot/mock.go | 2 + irc/irc.go | 4 ++ main.go | 2 + plugins/rpgORdie/rpgORdie.go | 61 +++++++++++++++++++ plugins/rpgORdie/rpgORdie_test.go | 4 ++ slack/slack.go | 98 +++++++++++++++++++++++++++---- 9 files changed, 175 insertions(+), 11 deletions(-) create mode 100644 plugins/rpgORdie/rpgORdie.go create mode 100644 plugins/rpgORdie/rpgORdie_test.go diff --git a/bot/bot.go b/bot/bot.go index 873d361..2507586 100644 --- a/bot/bot.go +++ b/bot/bot.go @@ -110,6 +110,10 @@ func (b *bot) DB() *sqlx.DB { return b.db } +func (b *bot) Conn() Connector { + return b.conn +} + // Create any tables if necessary based on version of DB // Plugins should create their own tables, these are only for official bot stuff // Note: This does not return an error. Database issues are all fatal at this stage. @@ -145,7 +149,7 @@ func (b *bot) migrateDB() { // Adds a constructed handler to the bots handlers list func (b *bot) AddHandler(name string, h Handler) { - b.plugins[strings.ToLower(name)] = h + b.plugins[name] = h b.pluginOrdering = append(b.pluginOrdering, name) if entry := h.RegisterWeb(); entry != nil { b.httpEndPoints[name] = *entry diff --git a/bot/handlers.go b/bot/handlers.go index e28dd65..9df5896 100644 --- a/bot/handlers.go +++ b/bot/handlers.go @@ -23,6 +23,8 @@ func (b *bot) MsgReceived(msg msg.Message) { // msg := b.buildMessage(client, inMsg) // do need to look up user and fix it + log.Println(msg.User.Name) + if strings.HasPrefix(msg.Body, "help ") && msg.Command { parts := strings.Fields(strings.ToLower(msg.Body)) b.checkHelp(msg.Channel, parts) @@ -65,6 +67,10 @@ func (b *bot) React(channel, reaction string, message msg.Message) { b.conn.React(channel, reaction, message) } +func (b *bot) Edit(channel, newMessage, identifier string) { + b.conn.Edit(channel, newMessage, identifier) +} + func (b *bot) GetEmojiList() map[string]string { return b.conn.GetEmojiList() } diff --git a/bot/interfaces.go b/bot/interfaces.go index 23e1185..d476642 100644 --- a/bot/interfaces.go +++ b/bot/interfaces.go @@ -13,11 +13,13 @@ type Bot interface { Config() *config.Config DBVersion() int64 DB() *sqlx.DB + Conn() Connector Who(string) []user.User AddHandler(string, Handler) SendMessage(string, string) SendAction(string, string) React(string, string, msg.Message) + Edit(string, string, string) MsgReceived(msg.Message) EventReceived(msg.Message) Filter(msg.Message, string) string @@ -34,6 +36,7 @@ type Connector interface { SendMessage(channel, message string) SendAction(channel, message string) React(string, string, msg.Message) + Edit(string, string, string) GetEmojiList() map[string]string Serve() error diff --git a/bot/mock.go b/bot/mock.go index 6b809f0..997fe5f 100644 --- a/bot/mock.go +++ b/bot/mock.go @@ -25,6 +25,7 @@ type MockBot struct { func (mb *MockBot) Config() *config.Config { return &mb.Cfg } func (mb *MockBot) DBVersion() int64 { return 1 } func (mb *MockBot) DB() *sqlx.DB { return mb.db } +func (mb *MockBot) Conn() Connector { return nil } func (mb *MockBot) Who(string) []user.User { return []user.User{} } func (mb *MockBot) AddHandler(name string, f Handler) {} func (mb *MockBot) SendMessage(ch string, msg string) { @@ -40,6 +41,7 @@ func (mb *MockBot) LastMessage(ch string) (msg.Message, error) { return msg.Mess func (mb *MockBot) CheckAdmin(nick string) bool { return false } func (mb *MockBot) React(channel, reaction string, message msg.Message) {} +func (mb *MockBot) Edit(channel, newMessage, identifier string) {} func (mb *MockBot) GetEmojiList() map[string]string { return make(map[string]string) } func (mb *MockBot) RegisterFilter(s string, f func(string) string) {} diff --git a/irc/irc.go b/irc/irc.go index 5ff2b07..da2a806 100644 --- a/irc/irc.go +++ b/irc/irc.go @@ -103,6 +103,10 @@ func (i *Irc) React(channel, reaction string, message msg.Message) { //we're not goign to do anything because it's IRC } +func (i *Irc) Edit(channel, newMessage, identifier string) { + //we're not goign to do anything because it's IRC +} + func (i *Irc) GetEmojiList() map[string]string { //we're not goign to do anything because it's IRC return make(map[string]string) diff --git a/main.go b/main.go index f576aca..069a27f 100644 --- a/main.go +++ b/main.go @@ -21,6 +21,7 @@ import ( "github.com/velour/catbase/plugins/leftpad" "github.com/velour/catbase/plugins/reaction" "github.com/velour/catbase/plugins/reminder" + "github.com/velour/catbase/plugins/rpgORdie" "github.com/velour/catbase/plugins/rss" "github.com/velour/catbase/plugins/stats" "github.com/velour/catbase/plugins/talker" @@ -69,6 +70,7 @@ func main() { b.AddHandler("emojifyme", emojifyme.New(b)) b.AddHandler("twitch", twitch.New(b)) b.AddHandler("inventory", inventory.New(b)) + b.AddHandler("rpgORdie", rpgORdie.New(b)) // catches anything left, will always return true b.AddHandler("factoid", fact.New(b)) diff --git a/plugins/rpgORdie/rpgORdie.go b/plugins/rpgORdie/rpgORdie.go new file mode 100644 index 0000000..7201c1b --- /dev/null +++ b/plugins/rpgORdie/rpgORdie.go @@ -0,0 +1,61 @@ +package rpgORdie + +import ( + "strings" + // "log" + "time" + + "github.com/velour/catbase/bot" + "github.com/velour/catbase/bot/msg" + "github.com/velour/catbase/slack" +) + +type RPGPlugin struct { + Bot bot.Bot + Slack *slack.Slack //nasty +} + +func New(b bot.Bot) *RPGPlugin { + return &RPGPlugin{ + Bot: b, + Slack: b.Conn().(*slack.Slack), //oh boy, this is just filthy + } +} + +func (p *RPGPlugin) Message(message msg.Message) bool { + if strings.ToLower(message.Body) == "start rpg" { + p.Bot.SendMessage(message.Channel, "I'll edit this.") + ts := p.Slack.GetLastMessageId() + + time.Sleep(2 * time.Second) + + edited := "" + for i := 0; i <= 5; i++ { + p.Bot.Edit(message.Channel, edited, ts) + edited += ":fire:" + time.Sleep(2 * time.Second) + } + p.Bot.Edit(message.Channel, "HECK YES", ts) + } + return false +} + +func (p *RPGPlugin) LoadData() { + +} + +func (p *RPGPlugin) Help(channel string, parts []string) { + p.Bot.SendMessage(channel, "Go find a walkthrough or something.") +} + +func (p *RPGPlugin) Event(kind string, message msg.Message) bool { + return false +} + +func (p *RPGPlugin) BotMessage(message msg.Message) bool { + return false +} + +func (p *RPGPlugin) RegisterWeb() *string { + return nil +} diff --git a/plugins/rpgORdie/rpgORdie_test.go b/plugins/rpgORdie/rpgORdie_test.go new file mode 100644 index 0000000..42db0f4 --- /dev/null +++ b/plugins/rpgORdie/rpgORdie_test.go @@ -0,0 +1,4 @@ +package rpgORdie + +import ( +) diff --git a/slack/slack.go b/slack/slack.go index 5f885bb..22c5c3a 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -15,7 +15,7 @@ import ( "regexp" "strconv" "strings" - "sync/atomic" + // "sync/atomic" "time" "github.com/velour/catbase/bot" @@ -38,6 +38,8 @@ type Slack struct { emoji map[string]string + lastMessageId string + eventReceived func(msg.Message) messageReceived func(msg.Message) } @@ -160,6 +162,7 @@ func New(c *config.Config) *Slack { lastRecieved: time.Now(), users: make(map[string]string), emoji: make(map[string]string), + lastMessageId: "", } } @@ -172,17 +175,47 @@ func (s *Slack) RegisterMessageReceived(f func(msg.Message)) { } func (s *Slack) SendMessageType(channel, messageType, subType, message string) error { - m := slackMessage{ - ID: atomic.AddUint64(&idCounter, 1), - Type: messageType, - SubType: subType, - Channel: channel, - Text: message, - } - err := websocket.JSON.Send(s.ws, m) + resp, err := http.PostForm("https://slack.com/api/chat.postMessage", + url.Values{"token": {s.config.Slack.Token}, + "channel": {channel}, + "text": {message}, + }) + if err != nil { - log.Printf("Error sending Slack message: %s", err) + log.Printf("Error sending Slack reaction: %s", err) } + + body, err := ioutil.ReadAll(resp.Body) + resp.Body.Close() + if err != nil { + log.Fatalf("Error reading Slack API body: %s", err) + } + + log.Println(string(body)) + + type MessageResponse struct { + OK bool `json:ok` + Channel string `json:channel` + Timestamp string `json:ts` + } + + var mr MessageResponse + err = json.Unmarshal(body, &mr) + if err != nil { + log.Fatalf("Error parsing message response: %s", err) + } + + bodyAsString := string(body) + + //what the fuck + if strings.Contains(bodyAsString, "\"ts\":\"") { + mr.Timestamp = strings.Split(strings.Split(bodyAsString, "\"ts\":\"")[1], "\"")[0] + } + + s.lastMessageId = mr.Timestamp + + log.Println(mr) + return err } @@ -209,6 +242,46 @@ func (s *Slack) React(channel, reaction string, message msg.Message) { log.Print(resp) } +func (s* Slack) GetLastMessageId() string { + return s.lastMessageId +} + +func (s* Slack) PrintHistory(channel string, howMany int) { + resp, err := http.PostForm("https://slack.com/api/channels.history", + url.Values{"token": {s.config.Slack.Token}, + "channel": {channel}, + "count": {fmt.Sprintf("%d", howMany)}}) + if err != nil { + log.Printf("Error getting slack history: %s", err) + } + body, err := ioutil.ReadAll(resp.Body) + resp.Body.Close() + if err != nil { + log.Fatalf("Error reading Slack API body: %s", err) + } + + log.Println(string(body)) +} + +func (s *Slack) Edit(channel, newMessage, identifier string) { + log.Printf("Editing in (%s) %s: %s", identifier, channel, newMessage) + resp, err := http.PostForm("https://slack.com/api/chat.update", + url.Values{"token": {s.config.Slack.Token}, + "channel": {channel}, + "text": {newMessage}, + "ts": {identifier}}) + if err != nil { + log.Printf("Error sending Slack reaction: %s", err) + } + body, err := ioutil.ReadAll(resp.Body) + resp.Body.Close() + if err != nil { + log.Fatalf("Error reading Slack API body: %s", err) + } + + log.Println(string(body)) +} + func (s *Slack) GetEmojiList() map[string]string { return s.emoji } @@ -275,6 +348,11 @@ func (s *Slack) Serve() error { case "message": if !msg.Hidden { m := s.buildMessage(msg) + + log.Println() + log.Println(m) + log.Println() + if m.Time.Before(s.lastRecieved) { log.Printf("Ignoring message: %+v\nlastRecieved: %v msg: %v", msg.ID, s.lastRecieved, m.Time) } else { From d6c35b94ecf16e8f151a66d391b761dfadb4e810 Mon Sep 17 00:00:00 2001 From: skkiesel Date: Tue, 31 Oct 2017 09:40:03 -0400 Subject: [PATCH 02/28] cleanup! --- bot/bot.go | 4 -- bot/handlers.go | 16 ++--- bot/interfaces.go | 17 +++--- bot/mock.go | 47 ++++++++++++--- irc/irc.go | 12 ++-- plugins/rpgORdie/rpgORdie.go | 10 +--- plugins/rpgORdie/rpgORdie_test.go | 3 +- slack/slack.go | 99 ++++++++++++++----------------- 8 files changed, 113 insertions(+), 95 deletions(-) diff --git a/bot/bot.go b/bot/bot.go index 2507586..eac6f28 100644 --- a/bot/bot.go +++ b/bot/bot.go @@ -110,10 +110,6 @@ func (b *bot) DB() *sqlx.DB { return b.db } -func (b *bot) Conn() Connector { - return b.conn -} - // Create any tables if necessary based on version of DB // Plugins should create their own tables, these are only for official bot stuff // Note: This does not return an error. Database issues are all fatal at this stage. diff --git a/bot/handlers.go b/bot/handlers.go index 9df5896..bb4c340 100644 --- a/bot/handlers.go +++ b/bot/handlers.go @@ -55,20 +55,20 @@ func (b *bot) EventReceived(msg msg.Message) { } } -func (b *bot) SendMessage(channel, message string) { - b.conn.SendMessage(channel, message) +func (b *bot) SendMessage(channel, message string) string { + return b.conn.SendMessage(channel, message) } -func (b *bot) SendAction(channel, message string) { - b.conn.SendAction(channel, message) +func (b *bot) SendAction(channel, message string) string { + return b.conn.SendAction(channel, message) } -func (b *bot) React(channel, reaction string, message msg.Message) { - b.conn.React(channel, reaction, message) +func (b *bot) React(channel, reaction string, message msg.Message) bool { + return b.conn.React(channel, reaction, message) } -func (b *bot) Edit(channel, newMessage, identifier string) { - b.conn.Edit(channel, newMessage, identifier) +func (b *bot) Edit(channel, newMessage, identifier string) bool { + return b.conn.Edit(channel, newMessage, identifier) } func (b *bot) GetEmojiList() map[string]string { diff --git a/bot/interfaces.go b/bot/interfaces.go index d476642..2fc783d 100644 --- a/bot/interfaces.go +++ b/bot/interfaces.go @@ -13,13 +13,12 @@ type Bot interface { Config() *config.Config DBVersion() int64 DB() *sqlx.DB - Conn() Connector Who(string) []user.User AddHandler(string, Handler) - SendMessage(string, string) - SendAction(string, string) - React(string, string, msg.Message) - Edit(string, string, string) + SendMessage(string, string) string + SendAction(string, string) string + React(string, string, msg.Message) bool + Edit(string, string, string) bool MsgReceived(msg.Message) EventReceived(msg.Message) Filter(msg.Message, string) string @@ -33,10 +32,10 @@ type Connector interface { RegisterEventReceived(func(message msg.Message)) RegisterMessageReceived(func(message msg.Message)) - SendMessage(channel, message string) - SendAction(channel, message string) - React(string, string, msg.Message) - Edit(string, string, string) + SendMessage(channel, message string) string + SendAction(channel, message string) string + React(string, string, msg.Message) bool + Edit(string, string, string) bool GetEmojiList() map[string]string Serve() error diff --git a/bot/mock.go b/bot/mock.go index 997fe5f..1e51bb9 100644 --- a/bot/mock.go +++ b/bot/mock.go @@ -3,7 +3,10 @@ package bot import ( + "fmt" "log" + "strconv" + "strings" "github.com/jmoiron/sqlx" "github.com/stretchr/testify/mock" @@ -25,14 +28,16 @@ type MockBot struct { func (mb *MockBot) Config() *config.Config { return &mb.Cfg } func (mb *MockBot) DBVersion() int64 { return 1 } func (mb *MockBot) DB() *sqlx.DB { return mb.db } -func (mb *MockBot) Conn() Connector { return nil } +func (mb *MockBot) Conn() Connector { return nil } func (mb *MockBot) Who(string) []user.User { return []user.User{} } func (mb *MockBot) AddHandler(name string, f Handler) {} -func (mb *MockBot) SendMessage(ch string, msg string) { +func (mb *MockBot) SendMessage(ch string, msg string) string { mb.Messages = append(mb.Messages, msg) + return fmt.Sprintf("m-%d", len(mb.Actions)-1) } -func (mb *MockBot) SendAction(ch string, msg string) { +func (mb *MockBot) SendAction(ch string, msg string) string { mb.Actions = append(mb.Actions, msg) + return fmt.Sprintf("a-%d", len(mb.Actions)-1) } func (mb *MockBot) MsgReceived(msg msg.Message) {} func (mb *MockBot) EventReceived(msg msg.Message) {} @@ -40,10 +45,38 @@ func (mb *MockBot) Filter(msg msg.Message, s string) string { return "" } func (mb *MockBot) LastMessage(ch string) (msg.Message, error) { return msg.Message{}, nil } func (mb *MockBot) CheckAdmin(nick string) bool { return false } -func (mb *MockBot) React(channel, reaction string, message msg.Message) {} -func (mb *MockBot) Edit(channel, newMessage, identifier string) {} -func (mb *MockBot) GetEmojiList() map[string]string { return make(map[string]string) } -func (mb *MockBot) RegisterFilter(s string, f func(string) string) {} +func (mb *MockBot) React(channel, reaction string, message msg.Message) bool { return false } + +func (mb *MockBot) Edit(channel, newMessage, identifier string) bool { + isMessage := identifier[0] == 'm' + if !isMessage && identifier[0] != 'a' { + log.Printf("failed to parse identifier: %s", identifier) + return false + } + + index, err := strconv.Atoi(strings.Split(identifier, "-")[1]) + if err != nil { + log.Printf("failed to parse identifier: %s", identifier) + return false + } + + if isMessage { + if index < len(mb.Messages) { + mb.Messages[index] = newMessage + } else { + return false + } + } else { + if index < len(mb.Actions) { + mb.Actions[index] = newMessage + } else { + return false + } + } + return true +} +func (mb *MockBot) GetEmojiList() map[string]string { return make(map[string]string) } +func (mb *MockBot) RegisterFilter(s string, f func(string) string) {} func NewMockBot() *MockBot { db, err := sqlx.Open("sqlite3_custom", ":memory:") diff --git a/irc/irc.go b/irc/irc.go index da2a806..65de1a7 100644 --- a/irc/irc.go +++ b/irc/irc.go @@ -66,7 +66,7 @@ func (i *Irc) JoinChannel(channel string) { i.Client.Out <- irc.Msg{Cmd: irc.JOIN, Args: []string{channel}} } -func (i *Irc) SendMessage(channel, message string) { +func (i *Irc) SendMessage(channel, message string) string { for len(message) > 0 { m := irc.Msg{ Cmd: "PRIVMSG", @@ -90,21 +90,25 @@ func (i *Irc) SendMessage(channel, message string) { i.Client.Out <- m } + return "NO_IRC_IDENTIFIERS" } // Sends action to channel -func (i *Irc) SendAction(channel, message string) { +func (i *Irc) SendAction(channel, message string) string { message = actionPrefix + " " + message + "\x01" i.SendMessage(channel, message) + return "NO_IRC_IDENTIFIERS" } -func (i *Irc) React(channel, reaction string, message msg.Message) { +func (i *Irc) React(channel, reaction string, message msg.Message) bool { //we're not goign to do anything because it's IRC + return false } -func (i *Irc) Edit(channel, newMessage, identifier string) { +func (i *Irc) Edit(channel, newMessage, identifier string) bool { //we're not goign to do anything because it's IRC + return false } func (i *Irc) GetEmojiList() map[string]string { diff --git a/plugins/rpgORdie/rpgORdie.go b/plugins/rpgORdie/rpgORdie.go index 7201c1b..17418c8 100644 --- a/plugins/rpgORdie/rpgORdie.go +++ b/plugins/rpgORdie/rpgORdie.go @@ -7,25 +7,21 @@ import ( "github.com/velour/catbase/bot" "github.com/velour/catbase/bot/msg" - "github.com/velour/catbase/slack" ) type RPGPlugin struct { - Bot bot.Bot - Slack *slack.Slack //nasty + Bot bot.Bot } func New(b bot.Bot) *RPGPlugin { return &RPGPlugin{ - Bot: b, - Slack: b.Conn().(*slack.Slack), //oh boy, this is just filthy + Bot: b, } } func (p *RPGPlugin) Message(message msg.Message) bool { if strings.ToLower(message.Body) == "start rpg" { - p.Bot.SendMessage(message.Channel, "I'll edit this.") - ts := p.Slack.GetLastMessageId() + ts := p.Bot.SendMessage(message.Channel, "I'll edit this.") time.Sleep(2 * time.Second) diff --git a/plugins/rpgORdie/rpgORdie_test.go b/plugins/rpgORdie/rpgORdie_test.go index 42db0f4..ddcd924 100644 --- a/plugins/rpgORdie/rpgORdie_test.go +++ b/plugins/rpgORdie/rpgORdie_test.go @@ -1,4 +1,3 @@ package rpgORdie -import ( -) +import () diff --git a/slack/slack.go b/slack/slack.go index 22c5c3a..3e7e2ed 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -38,8 +38,6 @@ type Slack struct { emoji map[string]string - lastMessageId string - eventReceived func(msg.Message) messageReceived func(msg.Message) } @@ -162,10 +160,30 @@ func New(c *config.Config) *Slack { lastRecieved: time.Now(), users: make(map[string]string), emoji: make(map[string]string), - lastMessageId: "", } } +func checkReturnStatus(response *http.Response) bool { + type Response struct { + OK bool `json:"ok"` + } + + body, err := ioutil.ReadAll(response.Body) + response.Body.Close() + if err != nil { + log.Printf("Error reading Slack API body: %s", err) + return false + } + + var resp Response + err = json.Unmarshal(body, &resp) + if err != nil { + log.Printf("Error parsing message response: %s", err) + return false + } + return resp.OK +} + func (s *Slack) RegisterEventReceived(f func(msg.Message)) { s.eventReceived = f } @@ -174,11 +192,11 @@ func (s *Slack) RegisterMessageReceived(f func(msg.Message)) { s.messageReceived = f } -func (s *Slack) SendMessageType(channel, messageType, subType, message string) error { +func (s *Slack) SendMessageType(channel, messageType, subType, message string) (string, error) { resp, err := http.PostForm("https://slack.com/api/chat.postMessage", url.Values{"token": {s.config.Slack.Token}, - "channel": {channel}, - "text": {message}, + "channel": {channel}, + "text": {message}, }) if err != nil { @@ -194,9 +212,9 @@ func (s *Slack) SendMessageType(channel, messageType, subType, message string) e log.Println(string(body)) type MessageResponse struct { - OK bool `json:ok` - Channel string `json:channel` - Timestamp string `json:ts` + OK bool `json:"ok"` + Channel string `json:"channel"` + Timestamp string `json:"ts"` } var mr MessageResponse @@ -212,24 +230,22 @@ func (s *Slack) SendMessageType(channel, messageType, subType, message string) e mr.Timestamp = strings.Split(strings.Split(bodyAsString, "\"ts\":\"")[1], "\"")[0] } - s.lastMessageId = mr.Timestamp - - log.Println(mr) - - return err + return mr.Timestamp, err } -func (s *Slack) SendMessage(channel, message string) { +func (s *Slack) SendMessage(channel, message string) string { log.Printf("Sending message to %s: %s", channel, message) - s.SendMessageType(channel, "message", "", message) + identifier, _ := s.SendMessageType(channel, "message", "", message) + return identifier } -func (s *Slack) SendAction(channel, message string) { +func (s *Slack) SendAction(channel, message string) string { log.Printf("Sending action to %s: %s", channel, message) - s.SendMessageType(channel, "message", "me_message", "_"+message+"_") + identifier, _ := s.SendMessageType(channel, "message", "me_message", "_"+message+"_") + return identifier } -func (s *Slack) React(channel, reaction string, message msg.Message) { +func (s *Slack) React(channel, reaction string, message msg.Message) bool { log.Printf("Reacting in %s: %s", channel, reaction) resp, err := http.PostForm("https://slack.com/api/reactions.add", url.Values{"token": {s.config.Slack.Token}, @@ -237,49 +253,24 @@ func (s *Slack) React(channel, reaction string, message msg.Message) { "channel": {channel}, "timestamp": {message.AdditionalData["RAW_SLACK_TIMESTAMP"]}}) if err != nil { - log.Printf("Error sending Slack reaction: %s", err) + log.Println("reaction failed: %s", err) + return false } - log.Print(resp) + return checkReturnStatus(resp) } -func (s* Slack) GetLastMessageId() string { - return s.lastMessageId -} - -func (s* Slack) PrintHistory(channel string, howMany int) { - resp, err := http.PostForm("https://slack.com/api/channels.history", - url.Values{"token": {s.config.Slack.Token}, - "channel": {channel}, - "count": {fmt.Sprintf("%d", howMany)}}) - if err != nil { - log.Printf("Error getting slack history: %s", err) - } - body, err := ioutil.ReadAll(resp.Body) - resp.Body.Close() - if err != nil { - log.Fatalf("Error reading Slack API body: %s", err) - } - - log.Println(string(body)) -} - -func (s *Slack) Edit(channel, newMessage, identifier string) { +func (s *Slack) Edit(channel, newMessage, identifier string) bool { log.Printf("Editing in (%s) %s: %s", identifier, channel, newMessage) resp, err := http.PostForm("https://slack.com/api/chat.update", url.Values{"token": {s.config.Slack.Token}, - "channel": {channel}, - "text": {newMessage}, - "ts": {identifier}}) + "channel": {channel}, + "text": {newMessage}, + "ts": {identifier}}) if err != nil { - log.Printf("Error sending Slack reaction: %s", err) + log.Println("edit failed: %s", err) + return false } - body, err := ioutil.ReadAll(resp.Body) - resp.Body.Close() - if err != nil { - log.Fatalf("Error reading Slack API body: %s", err) - } - - log.Println(string(body)) + return checkReturnStatus(resp) } func (s *Slack) GetEmojiList() map[string]string { From 42b1356eb20d772729fec42191e1eec621088ce9 Mon Sep 17 00:00:00 2001 From: skkiesel Date: Tue, 31 Oct 2017 09:48:35 -0400 Subject: [PATCH 03/28] remove some unneccessary code when parsing sent message response --- slack/slack.go | 7 ------- 1 file changed, 7 deletions(-) diff --git a/slack/slack.go b/slack/slack.go index 3e7e2ed..9808e2c 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -223,13 +223,6 @@ func (s *Slack) SendMessageType(channel, messageType, subType, message string) ( log.Fatalf("Error parsing message response: %s", err) } - bodyAsString := string(body) - - //what the fuck - if strings.Contains(bodyAsString, "\"ts\":\"") { - mr.Timestamp = strings.Split(strings.Split(bodyAsString, "\"ts\":\"")[1], "\"")[0] - } - return mr.Timestamp, err } From 3009a646e60fb042877c12285f636640f1cc05f5 Mon Sep 17 00:00:00 2001 From: skkiesel Date: Tue, 31 Oct 2017 10:07:20 -0400 Subject: [PATCH 04/28] support replies --- bot/handlers.go | 4 +++ bot/interfaces.go | 2 ++ bot/mock.go | 1 + irc/irc.go | 4 +++ plugins/rpgORdie/rpgORdie.go | 2 ++ slack/slack.go | 62 ++++++++++++++++++++++++++++++++---- 6 files changed, 69 insertions(+), 6 deletions(-) diff --git a/bot/handlers.go b/bot/handlers.go index bb4c340..81e9154 100644 --- a/bot/handlers.go +++ b/bot/handlers.go @@ -63,6 +63,10 @@ func (b *bot) SendAction(channel, message string) string { return b.conn.SendAction(channel, message) } +func (b *bot) ReplyToMessage(channel, message, identifier string) (string, bool) { + return b.conn.ReplyToMessage(channel, message, identifier) +} + func (b *bot) React(channel, reaction string, message msg.Message) bool { return b.conn.React(channel, reaction, message) } diff --git a/bot/interfaces.go b/bot/interfaces.go index 2fc783d..d1e374a 100644 --- a/bot/interfaces.go +++ b/bot/interfaces.go @@ -17,6 +17,7 @@ type Bot interface { AddHandler(string, Handler) SendMessage(string, string) string SendAction(string, string) string + ReplyToMessage(channel, message, identifier string) (string, bool) React(string, string, msg.Message) bool Edit(string, string, string) bool MsgReceived(msg.Message) @@ -34,6 +35,7 @@ type Connector interface { SendMessage(channel, message string) string SendAction(channel, message string) string + ReplyToMessage(channel, message, identifier string) (string, bool) React(string, string, msg.Message) bool Edit(string, string, string) bool GetEmojiList() map[string]string diff --git a/bot/mock.go b/bot/mock.go index 1e51bb9..fdb5178 100644 --- a/bot/mock.go +++ b/bot/mock.go @@ -39,6 +39,7 @@ func (mb *MockBot) SendAction(ch string, msg string) string { mb.Actions = append(mb.Actions, msg) return fmt.Sprintf("a-%d", len(mb.Actions)-1) } +func (mb *MockBot) ReplyToMessage(channel, message, identifier string) (string, bool) { return "", false } func (mb *MockBot) MsgReceived(msg msg.Message) {} func (mb *MockBot) EventReceived(msg msg.Message) {} func (mb *MockBot) Filter(msg msg.Message, s string) string { return "" } diff --git a/irc/irc.go b/irc/irc.go index 65de1a7..4d7b3bf 100644 --- a/irc/irc.go +++ b/irc/irc.go @@ -101,6 +101,10 @@ func (i *Irc) SendAction(channel, message string) string { return "NO_IRC_IDENTIFIERS" } +func (i *Irc) ReplyToMessage(channel, message, identifier string) (string, bool) { + return "NO_IRC_IDENTIFIERS", false +} + func (i *Irc) React(channel, reaction string, message msg.Message) bool { //we're not goign to do anything because it's IRC return false diff --git a/plugins/rpgORdie/rpgORdie.go b/plugins/rpgORdie/rpgORdie.go index 17418c8..5c5b619 100644 --- a/plugins/rpgORdie/rpgORdie.go +++ b/plugins/rpgORdie/rpgORdie.go @@ -32,6 +32,8 @@ func (p *RPGPlugin) Message(message msg.Message) bool { time.Sleep(2 * time.Second) } p.Bot.Edit(message.Channel, "HECK YES", ts) + + p.Bot.ReplyToMessage(message.Channel, "How's this reply?", ts) } return false } diff --git a/slack/slack.go b/slack/slack.go index 9808e2c..29e5ef9 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -5,6 +5,7 @@ package slack import ( "encoding/json" + "errors" "fmt" "html" "io" @@ -192,15 +193,20 @@ func (s *Slack) RegisterMessageReceived(f func(msg.Message)) { s.messageReceived = f } -func (s *Slack) SendMessageType(channel, messageType, subType, message string) (string, error) { - resp, err := http.PostForm("https://slack.com/api/chat.postMessage", +func (s *Slack) SendMessageType(channel, message string, meMessage bool) (string, error) { + postUrl := "https://slack.com/api/chat.postMessage" + if meMessage { + postUrl = "https://slack.com/api/chat.meMessage" + } + + resp, err := http.PostForm(postUrl, url.Values{"token": {s.config.Slack.Token}, "channel": {channel}, "text": {message}, }) if err != nil { - log.Printf("Error sending Slack reaction: %s", err) + log.Printf("Error sending Slack message: %s", err) } body, err := ioutil.ReadAll(resp.Body) @@ -213,7 +219,6 @@ func (s *Slack) SendMessageType(channel, messageType, subType, message string) ( type MessageResponse struct { OK bool `json:"ok"` - Channel string `json:"channel"` Timestamp string `json:"ts"` } @@ -223,21 +228,66 @@ func (s *Slack) SendMessageType(channel, messageType, subType, message string) ( log.Fatalf("Error parsing message response: %s", err) } + if !mr.OK { + return "", errors.New("failure response received") + } + return mr.Timestamp, err } func (s *Slack) SendMessage(channel, message string) string { log.Printf("Sending message to %s: %s", channel, message) - identifier, _ := s.SendMessageType(channel, "message", "", message) + identifier, _ := s.SendMessageType(channel, message, false) return identifier } func (s *Slack) SendAction(channel, message string) string { log.Printf("Sending action to %s: %s", channel, message) - identifier, _ := s.SendMessageType(channel, "message", "me_message", "_"+message+"_") + identifier, _ := s.SendMessageType(channel, "_"+message+"_", true) return identifier } +func (s *Slack) ReplyToMessage(channel, message, identifier string) (string, bool) { + resp, err := http.PostForm("https://slack.com/api/chat.postMessage", + url.Values{"token": {s.config.Slack.Token}, + "channel": {channel}, + "text": {message}, + "thread_ts": {identifier}, + }) + + if err != nil { + log.Printf("Error sending Slack reply: %s", err) + return "", false + } + + body, err := ioutil.ReadAll(resp.Body) + resp.Body.Close() + if err != nil { + log.Printf("Error reading Slack API body: %s", err) + return "", false + } + + log.Println(string(body)) + + type MessageResponse struct { + OK bool `json:"ok"` + Timestamp string `json:"ts"` + } + + var mr MessageResponse + err = json.Unmarshal(body, &mr) + if err != nil { + log.Printf("Error parsing message response: %s", err) + return "", false + } + + if !mr.OK { + return "", false + } + + return mr.Timestamp, err == nil +} + func (s *Slack) React(channel, reaction string, message msg.Message) bool { log.Printf("Reacting in %s: %s", channel, reaction) resp, err := http.PostForm("https://slack.com/api/reactions.add", From d9bb7ec3c0aead5e4852e90870e38b6090cbcdae Mon Sep 17 00:00:00 2001 From: skkiesel Date: Tue, 31 Oct 2017 10:16:41 -0400 Subject: [PATCH 05/28] sometimes you have an id, sometimes a message, we should support relpying to both --- bot/handlers.go | 8 ++++++-- bot/interfaces.go | 6 ++++-- bot/mock.go | 3 ++- irc/irc.go | 6 +++++- plugins/rpgORdie/rpgORdie.go | 2 +- slack/slack.go | 6 +++++- 6 files changed, 23 insertions(+), 8 deletions(-) diff --git a/bot/handlers.go b/bot/handlers.go index 81e9154..234e85e 100644 --- a/bot/handlers.go +++ b/bot/handlers.go @@ -63,8 +63,12 @@ func (b *bot) SendAction(channel, message string) string { return b.conn.SendAction(channel, message) } -func (b *bot) ReplyToMessage(channel, message, identifier string) (string, bool) { - return b.conn.ReplyToMessage(channel, message, identifier) +func (b *bot) ReplyToMessageIdentifier(channel, message, identifier string) (string, bool) { + return b.conn.ReplyToMessageIdentifier(channel, message, identifier) +} + +func (b *bot) ReplyToMessage(channel, message string, replyTo msg.Message) (string, bool) { + return b.conn.ReplyToMessage(channel, message, replyTo) } func (b *bot) React(channel, reaction string, message msg.Message) bool { diff --git a/bot/interfaces.go b/bot/interfaces.go index d1e374a..2e76e93 100644 --- a/bot/interfaces.go +++ b/bot/interfaces.go @@ -17,7 +17,8 @@ type Bot interface { AddHandler(string, Handler) SendMessage(string, string) string SendAction(string, string) string - ReplyToMessage(channel, message, identifier string) (string, bool) + ReplyToMessageIdentifier(string, string, string) (string, bool) + ReplyToMessage(string, string, msg.Message) (string, bool) React(string, string, msg.Message) bool Edit(string, string, string) bool MsgReceived(msg.Message) @@ -35,7 +36,8 @@ type Connector interface { SendMessage(channel, message string) string SendAction(channel, message string) string - ReplyToMessage(channel, message, identifier string) (string, bool) + ReplyToMessageIdentifier(string, string, string) (string, bool) + ReplyToMessage(string, string, msg.Message) (string, bool) React(string, string, msg.Message) bool Edit(string, string, string) bool GetEmojiList() map[string]string diff --git a/bot/mock.go b/bot/mock.go index fdb5178..1286e3f 100644 --- a/bot/mock.go +++ b/bot/mock.go @@ -39,7 +39,8 @@ func (mb *MockBot) SendAction(ch string, msg string) string { mb.Actions = append(mb.Actions, msg) return fmt.Sprintf("a-%d", len(mb.Actions)-1) } -func (mb *MockBot) ReplyToMessage(channel, message, identifier string) (string, bool) { return "", false } +func (mb *MockBot) ReplyToMessageIdentifier(channel, message, identifier string) (string, bool) { return "", false } +func (mb *MockBot) ReplyToMessage(channel, message string, replyTo msg.Message) (string, bool) { return "", false } func (mb *MockBot) MsgReceived(msg msg.Message) {} func (mb *MockBot) EventReceived(msg msg.Message) {} func (mb *MockBot) Filter(msg msg.Message, s string) string { return "" } diff --git a/irc/irc.go b/irc/irc.go index 4d7b3bf..d348610 100644 --- a/irc/irc.go +++ b/irc/irc.go @@ -101,7 +101,11 @@ func (i *Irc) SendAction(channel, message string) string { return "NO_IRC_IDENTIFIERS" } -func (i *Irc) ReplyToMessage(channel, message, identifier string) (string, bool) { +func (i *Irc) ReplyToMessageIdentifier(channel, message, identifier string) (string, bool) { + return "NO_IRC_IDENTIFIERS", false +} + +func (i *Irc) ReplyToMessage(channel, message string, replyTo msg.Message) (string, bool) { return "NO_IRC_IDENTIFIERS", false } diff --git a/plugins/rpgORdie/rpgORdie.go b/plugins/rpgORdie/rpgORdie.go index 5c5b619..2733d7b 100644 --- a/plugins/rpgORdie/rpgORdie.go +++ b/plugins/rpgORdie/rpgORdie.go @@ -33,7 +33,7 @@ func (p *RPGPlugin) Message(message msg.Message) bool { } p.Bot.Edit(message.Channel, "HECK YES", ts) - p.Bot.ReplyToMessage(message.Channel, "How's this reply?", ts) + p.Bot.ReplyToMessageIdentifier(message.Channel, "How's this reply?", ts) } return false } diff --git a/slack/slack.go b/slack/slack.go index 29e5ef9..b00ed3e 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -247,7 +247,7 @@ func (s *Slack) SendAction(channel, message string) string { return identifier } -func (s *Slack) ReplyToMessage(channel, message, identifier string) (string, bool) { +func (s *Slack) ReplyToMessageIdentifier(channel, message, identifier string) (string, bool) { resp, err := http.PostForm("https://slack.com/api/chat.postMessage", url.Values{"token": {s.config.Slack.Token}, "channel": {channel}, @@ -288,6 +288,10 @@ func (s *Slack) ReplyToMessage(channel, message, identifier string) (string, boo return mr.Timestamp, err == nil } +func (s *Slack) ReplyToMessage(channel, message string, replyTo msg.Message) (string, bool) { + return s.ReplyToMessageIdentifier(channel, message, replyTo.AdditionalData["RAW_SLACK_TIMESTAMP"]) +} + func (s *Slack) React(channel, reaction string, message msg.Message) bool { log.Printf("Reacting in %s: %s", channel, reaction) resp, err := http.PostForm("https://slack.com/api/reactions.add", From 54b83cd44433df483732a961ca11dbf93cf26cc9 Mon Sep 17 00:00:00 2001 From: skkiesel Date: Tue, 31 Oct 2017 14:14:45 -0400 Subject: [PATCH 06/28] listening to thread responses to rpg threads only --- bot/bot.go | 1 + bot/handlers.go | 15 ++++++++-- bot/interfaces.go | 3 ++ irc/irc.go | 5 ++++ plugins/admin/admin.go | 2 ++ plugins/babbler/babbler.go | 2 ++ plugins/beers/beers.go | 2 ++ plugins/counter/counter.go | 2 ++ plugins/dice/dice.go | 2 ++ plugins/downtime/downtime.go | 2 ++ plugins/emojifyme/emojifyme.go | 2 ++ plugins/fact/factoid.go | 2 ++ plugins/fact/remember.go | 2 ++ plugins/first/first.go | 2 ++ plugins/inventory/inventory.go | 2 ++ plugins/leftpad/leftpad.go | 2 ++ plugins/reaction/reaction.go | 2 ++ plugins/reminder/reminder.go | 2 ++ plugins/rpgORdie/rpgORdie.go | 18 ++++++++++-- plugins/rss/rss.go | 2 ++ plugins/stats/stats.go | 2 ++ plugins/talker/talker.go | 2 ++ plugins/twitch/twitch.go | 2 ++ plugins/your/your.go | 2 ++ plugins/zork/zork.go | 2 ++ slack/slack.go | 52 +++++++++++++++++++++++++++++----- 26 files changed, 122 insertions(+), 12 deletions(-) diff --git a/bot/bot.go b/bot/bot.go index eac6f28..777229c 100644 --- a/bot/bot.go +++ b/bot/bot.go @@ -93,6 +93,7 @@ func New(config *config.Config, connector Connector) Bot { connector.RegisterMessageReceived(bot.MsgReceived) connector.RegisterEventReceived(bot.EventReceived) + connector.RegisterReplyMessageReceived(bot.ReplyMsgReceived) return bot } diff --git a/bot/handlers.go b/bot/handlers.go index 234e85e..651e0ae 100644 --- a/bot/handlers.go +++ b/bot/handlers.go @@ -22,9 +22,6 @@ func (b *bot) MsgReceived(msg msg.Message) { // msg := b.buildMessage(client, inMsg) // do need to look up user and fix it - - log.Println(msg.User.Name) - if strings.HasPrefix(msg.Body, "help ") && msg.Command { parts := strings.Fields(strings.ToLower(msg.Body)) b.checkHelp(msg.Channel, parts) @@ -55,6 +52,18 @@ func (b *bot) EventReceived(msg msg.Message) { } } +// Handle incoming replys +func (b *bot) ReplyMsgReceived(msg msg.Message, identifier string) { + log.Println("Received message: ", msg) + + for _, name := range b.pluginOrdering { + p := b.plugins[name] + if p.ReplyMessage(msg, identifier) { + break + } + } +} + func (b *bot) SendMessage(channel, message string) string { return b.conn.SendMessage(channel, message) } diff --git a/bot/interfaces.go b/bot/interfaces.go index 2e76e93..2780d29 100644 --- a/bot/interfaces.go +++ b/bot/interfaces.go @@ -22,6 +22,7 @@ type Bot interface { React(string, string, msg.Message) bool Edit(string, string, string) bool MsgReceived(msg.Message) + ReplyMsgReceived(msg.Message, string) EventReceived(msg.Message) Filter(msg.Message, string) string LastMessage(string) (msg.Message, error) @@ -33,6 +34,7 @@ type Bot interface { type Connector interface { RegisterEventReceived(func(message msg.Message)) RegisterMessageReceived(func(message msg.Message)) + RegisterReplyMessageReceived(func(msg.Message, string)) SendMessage(channel, message string) string SendAction(channel, message string) string @@ -50,6 +52,7 @@ type Connector interface { type Handler interface { Message(message msg.Message) bool Event(kind string, message msg.Message) bool + ReplyMessage(msg.Message, string) bool BotMessage(message msg.Message) bool Help(channel string, parts []string) RegisterWeb() *string diff --git a/irc/irc.go b/irc/irc.go index d348610..4c34529 100644 --- a/irc/irc.go +++ b/irc/irc.go @@ -44,6 +44,7 @@ type Irc struct { eventReceived func(msg.Message) messageReceived func(msg.Message) + replyMessageReceived func(msg.Message, string) } func New(c *config.Config) *Irc { @@ -61,6 +62,10 @@ func (i *Irc) RegisterMessageReceived(f func(msg.Message)) { i.messageReceived = f } +func (i *Irc) RegisterReplyMessageReceived(f func(msg.Message, string)) { + i.replyMessageReceived = f +} + func (i *Irc) JoinChannel(channel string) { log.Printf("Joining channel: %s", channel) i.Client.Out <- irc.Msg{Cmd: irc.JOIN, Args: []string{channel}} diff --git a/plugins/admin/admin.go b/plugins/admin/admin.go index 85988dc..1113bdd 100644 --- a/plugins/admin/admin.go +++ b/plugins/admin/admin.go @@ -117,3 +117,5 @@ func (p *AdminPlugin) BotMessage(message msg.Message) bool { func (p *AdminPlugin) RegisterWeb() *string { return nil } + +func (p *AdminPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/babbler/babbler.go b/plugins/babbler/babbler.go index e5961c9..0ae0b58 100644 --- a/plugins/babbler/babbler.go +++ b/plugins/babbler/babbler.go @@ -935,3 +935,5 @@ func (p *BabblerPlugin) babbleSeedBookends(babblerName string, start, end []stri return strings.Join(words, " "), nil } + +func (p *BabblerPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/beers/beers.go b/plugins/beers/beers.go index 1bc6521..6d1b83e 100644 --- a/plugins/beers/beers.go +++ b/plugins/beers/beers.go @@ -461,3 +461,5 @@ func (p *BeersPlugin) BotMessage(message msg.Message) bool { func (p *BeersPlugin) RegisterWeb() *string { return nil } + +func (p *BeersPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/counter/counter.go b/plugins/counter/counter.go index ebe51b0..ed70349 100644 --- a/plugins/counter/counter.go +++ b/plugins/counter/counter.go @@ -364,3 +364,5 @@ func (p *CounterPlugin) BotMessage(message msg.Message) bool { func (p *CounterPlugin) RegisterWeb() *string { return nil } + +func (p *CounterPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/dice/dice.go b/plugins/dice/dice.go index 07484ce..3ee6dc3 100644 --- a/plugins/dice/dice.go +++ b/plugins/dice/dice.go @@ -100,3 +100,5 @@ func (p *DicePlugin) BotMessage(message msg.Message) bool { func (p *DicePlugin) RegisterWeb() *string { return nil } + +func (p *DicePlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/downtime/downtime.go b/plugins/downtime/downtime.go index e553ed2..ef9df9e 100644 --- a/plugins/downtime/downtime.go +++ b/plugins/downtime/downtime.go @@ -231,3 +231,5 @@ func (p *DowntimePlugin) BotMessage(message msg.Message) bool { func (p *DowntimePlugin) RegisterWeb() *string { return nil } + +func (p *DowntimePlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/emojifyme/emojifyme.go b/plugins/emojifyme/emojifyme.go index 65b892f..e8a6119 100644 --- a/plugins/emojifyme/emojifyme.go +++ b/plugins/emojifyme/emojifyme.go @@ -112,3 +112,5 @@ func (p *EmojifyMePlugin) BotMessage(message msg.Message) bool { func (p *EmojifyMePlugin) RegisterWeb() *string { return nil } + +func (p *EmojifyMePlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/fact/factoid.go b/plugins/fact/factoid.go index be364cf..17a4e4a 100644 --- a/plugins/fact/factoid.go +++ b/plugins/fact/factoid.go @@ -764,3 +764,5 @@ func (p *Factoid) serveQuery(w http.ResponseWriter, r *http.Request) { log.Println(err) } } + +func (p *Factoid) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/fact/remember.go b/plugins/fact/remember.go index e4fc6bd..1038eb7 100644 --- a/plugins/fact/remember.go +++ b/plugins/fact/remember.go @@ -170,3 +170,5 @@ func (p *RememberPlugin) recordMsg(message msg.Message) { log.Printf("Logging message: %s: %s", message.User.Name, message.Body) p.Log[message.Channel] = append(p.Log[message.Channel], message) } + +func (p *RememberPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/first/first.go b/plugins/first/first.go index 9d63c6f..3614576 100644 --- a/plugins/first/first.go +++ b/plugins/first/first.go @@ -228,3 +228,5 @@ func (p *FirstPlugin) BotMessage(message msg.Message) bool { func (p *FirstPlugin) RegisterWeb() *string { return nil } + +func (p *FirstPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/inventory/inventory.go b/plugins/inventory/inventory.go index abbaa35..7efb807 100644 --- a/plugins/inventory/inventory.go +++ b/plugins/inventory/inventory.go @@ -236,3 +236,5 @@ func (p *InventoryPlugin) RegisterWeb() *string { // nothing to register return nil } + +func (p *InventoryPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/leftpad/leftpad.go b/plugins/leftpad/leftpad.go index 6beb410..11098d9 100644 --- a/plugins/leftpad/leftpad.go +++ b/plugins/leftpad/leftpad.go @@ -76,3 +76,5 @@ func (p *LeftpadPlugin) RegisterWeb() *string { // nothing to register return nil } + +func (p *LeftpadPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/reaction/reaction.go b/plugins/reaction/reaction.go index a52c17f..7b204b4 100644 --- a/plugins/reaction/reaction.go +++ b/plugins/reaction/reaction.go @@ -80,3 +80,5 @@ func (p *ReactionPlugin) BotMessage(message msg.Message) bool { func (p *ReactionPlugin) RegisterWeb() *string { return nil } + +func (p *ReactionPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/reminder/reminder.go b/plugins/reminder/reminder.go index 1699da2..f55cb3d 100644 --- a/plugins/reminder/reminder.go +++ b/plugins/reminder/reminder.go @@ -322,3 +322,5 @@ func reminderer(p *ReminderPlugin) { p.queueUpNextReminder() } } + +func (p *ReminderPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/rpgORdie/rpgORdie.go b/plugins/rpgORdie/rpgORdie.go index 2733d7b..d922f55 100644 --- a/plugins/rpgORdie/rpgORdie.go +++ b/plugins/rpgORdie/rpgORdie.go @@ -2,7 +2,6 @@ package rpgORdie import ( "strings" - // "log" "time" "github.com/velour/catbase/bot" @@ -11,11 +10,13 @@ import ( type RPGPlugin struct { Bot bot.Bot + listenFor map[string]bool } func New(b bot.Bot) *RPGPlugin { return &RPGPlugin{ Bot: b, + listenFor: map[string]bool{}, } } @@ -23,17 +24,20 @@ func (p *RPGPlugin) Message(message msg.Message) bool { if strings.ToLower(message.Body) == "start rpg" { ts := p.Bot.SendMessage(message.Channel, "I'll edit this.") + p.listenFor[ts] = true + time.Sleep(2 * time.Second) edited := "" for i := 0; i <= 5; i++ { p.Bot.Edit(message.Channel, edited, ts) edited += ":fire:" - time.Sleep(2 * time.Second) + time.Sleep(500 * time.Millisecond) } p.Bot.Edit(message.Channel, "HECK YES", ts) p.Bot.ReplyToMessageIdentifier(message.Channel, "How's this reply?", ts) + return true } return false } @@ -57,3 +61,13 @@ func (p *RPGPlugin) BotMessage(message msg.Message) bool { func (p *RPGPlugin) RegisterWeb() *string { return nil } + +func (p *RPGPlugin) ReplyMessage(message msg.Message, identifier string) bool { + if strings.ToLower(message.User.Name) != strings.ToLower(p.Bot.Config().Nick) { + if _, ok := p.listenFor[identifier]; ok { + p.Bot.ReplyToMessageIdentifier(message.Channel, "Pong", identifier) + return true + } + } + return false +} diff --git a/plugins/rss/rss.go b/plugins/rss/rss.go index bc7bee5..099cdf9 100644 --- a/plugins/rss/rss.go +++ b/plugins/rss/rss.go @@ -117,3 +117,5 @@ func (p *RSSPlugin) BotMessage(message msg.Message) bool { func (p *RSSPlugin) RegisterWeb() *string { return nil } + +func (p *RSSPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/stats/stats.go b/plugins/stats/stats.go index 7b7039b..81450b2 100644 --- a/plugins/stats/stats.go +++ b/plugins/stats/stats.go @@ -275,3 +275,5 @@ func (p *StatsPlugin) mkSightingStat(message msg.Message) stats { func (p *StatsPlugin) mkChannelStat(message msg.Message) stats { return stats{stat{mkDay(), "channel", message.Channel, 1}} } + +func (p *StatsPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/talker/talker.go b/plugins/talker/talker.go index 1e01b90..17e9d1b 100644 --- a/plugins/talker/talker.go +++ b/plugins/talker/talker.go @@ -119,3 +119,5 @@ func (p *TalkerPlugin) BotMessage(message msg.Message) bool { func (p *TalkerPlugin) RegisterWeb() *string { return nil } + +func (p *TalkerPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/twitch/twitch.go b/plugins/twitch/twitch.go index 4a5c1a0..941b55a 100644 --- a/plugins/twitch/twitch.go +++ b/plugins/twitch/twitch.go @@ -238,3 +238,5 @@ func (p *TwitchPlugin) checkTwitch(channel string, twitcher *Twitcher, alwaysPri twitcher.game = game } } + +func (p *TwitchPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/your/your.go b/plugins/your/your.go index 9100d37..9e24082 100644 --- a/plugins/your/your.go +++ b/plugins/your/your.go @@ -66,3 +66,5 @@ func (p *YourPlugin) BotMessage(message msg.Message) bool { func (p *YourPlugin) RegisterWeb() *string { return nil } + +func (p *YourPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/plugins/zork/zork.go b/plugins/zork/zork.go index 7bd97bd..cabc6d2 100644 --- a/plugins/zork/zork.go +++ b/plugins/zork/zork.go @@ -122,3 +122,5 @@ func (p *ZorkPlugin) Help(ch string, _ []string) { } func (p *ZorkPlugin) RegisterWeb() *string { return nil } + +func (p *ZorkPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } diff --git a/slack/slack.go b/slack/slack.go index b00ed3e..b22f622 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -41,6 +41,7 @@ type Slack struct { eventReceived func(msg.Message) messageReceived func(msg.Message) + replyMessageReceived func(msg.Message, string) } var idCounter uint64 @@ -134,6 +135,7 @@ type slackMessage struct { User string `json:"user"` Username string `json:"username"` Ts string `json:"ts"` + ThreadTs string `json:"thread_ts"` Error struct { Code uint64 `json:"code"` Msg string `json:"msg"` @@ -193,6 +195,10 @@ func (s *Slack) RegisterMessageReceived(f func(msg.Message)) { s.messageReceived = f } +func (s *Slack) RegisterReplyMessageReceived(f func(msg.Message, string)) { + s.replyMessageReceived = f +} + func (s *Slack) SendMessageType(channel, message string, meMessage bool) (string, error) { postUrl := "https://slack.com/api/chat.postMessage" if meMessage { @@ -384,19 +390,17 @@ func (s *Slack) Serve() error { } switch msg.Type { case "message": - if !msg.Hidden { + if !msg.Hidden && msg.ThreadTs == "" { m := s.buildMessage(msg) - - log.Println() - log.Println(m) - log.Println() - if m.Time.Before(s.lastRecieved) { log.Printf("Ignoring message: %+v\nlastRecieved: %v msg: %v", msg.ID, s.lastRecieved, m.Time) } else { s.lastRecieved = m.Time - s.messageReceived(s.buildMessage(msg)) + s.messageReceived(m) } + } else if msg.ThreadTs != "" { + //we're throwing away some information here by not parsing the correct reply object type, but that's okay + s.replyMessageReceived(s.buildLightReplyMessage(msg), msg.ThreadTs) } else { log.Printf("THAT MESSAGE WAS HIDDEN: %+v", msg.ID) } @@ -453,6 +457,40 @@ func (s *Slack) buildMessage(m slackMessage) msg.Message { } } +func (s *Slack) buildLightReplyMessage(m slackMessage) msg.Message { + text := html.UnescapeString(m.Text) + + text = fixText(s.getUser, text) + + isCmd, text := bot.IsCmd(s.config, text) + + isAction := m.SubType == "me_message" + + u, _ := s.getUser(m.User) + if m.Username != "" { + u = m.Username + } + + tstamp := slackTStoTime(m.Ts) + + return msg.Message{ + User: &user.User{ + ID: m.User, + Name: u, + }, + Body: text, + Raw: m.Text, + Channel: m.Channel, + Command: isCmd, + Action: isAction, + Host: string(m.ID), + Time: tstamp, + AdditionalData: map[string]string{ + "RAW_SLACK_TIMESTAMP": m.Ts, + }, + } +} + // markAllChannelsRead gets a list of all channels and marks each as read func (s *Slack) markAllChannelsRead() { chs := s.getAllChannels() From 1ac566c2c2345a9ddc029de87e34e80a11794aac Mon Sep 17 00:00:00 2001 From: skkiesel Date: Thu, 2 Nov 2017 16:32:02 -0400 Subject: [PATCH 07/28] sokobass. --- bot/mock.go | 13 +++- plugins/rpgORdie/rpgORdie.go | 140 +++++++++++++++++++++++++++++------ 2 files changed, 129 insertions(+), 24 deletions(-) diff --git a/bot/mock.go b/bot/mock.go index 1286e3f..7d6c6e6 100644 --- a/bot/mock.go +++ b/bot/mock.go @@ -39,8 +39,12 @@ func (mb *MockBot) SendAction(ch string, msg string) string { mb.Actions = append(mb.Actions, msg) return fmt.Sprintf("a-%d", len(mb.Actions)-1) } -func (mb *MockBot) ReplyToMessageIdentifier(channel, message, identifier string) (string, bool) { return "", false } -func (mb *MockBot) ReplyToMessage(channel, message string, replyTo msg.Message) (string, bool) { return "", false } +func (mb *MockBot) ReplyToMessageIdentifier(channel, message, identifier string) (string, bool) { + return "", false +} +func (mb *MockBot) ReplyToMessage(channel, message string, replyTo msg.Message) (string, bool) { + return "", false +} func (mb *MockBot) MsgReceived(msg msg.Message) {} func (mb *MockBot) EventReceived(msg msg.Message) {} func (mb *MockBot) Filter(msg msg.Message, s string) string { return "" } @@ -77,6 +81,11 @@ func (mb *MockBot) Edit(channel, newMessage, identifier string) bool { } return true } + +func (mb *MockBot) ReplyMsgReceived(msg.Message, string) { + +} + func (mb *MockBot) GetEmojiList() map[string]string { return make(map[string]string) } func (mb *MockBot) RegisterFilter(s string, f func(string) string) {} diff --git a/plugins/rpgORdie/rpgORdie.go b/plugins/rpgORdie/rpgORdie.go index d922f55..7320046 100644 --- a/plugins/rpgORdie/rpgORdie.go +++ b/plugins/rpgORdie/rpgORdie.go @@ -1,42 +1,115 @@ package rpgORdie import ( + "fmt" "strings" - "time" "github.com/velour/catbase/bot" "github.com/velour/catbase/bot/msg" ) +const ( + DUDE = ":lion_face:" + BOULDER = ":full_moon:" + HOLE = ":new_moon:" + EMPTY = ":white_large_square:" + + OK = iota + INVALID = iota + WIN = iota +) + type RPGPlugin struct { - Bot bot.Bot - listenFor map[string]bool + Bot bot.Bot + listenFor map[string]*board +} + +type board struct { + state [][]string + x, y int +} + +func NewRandomBoard() *board { + boardSize := 5 + b := board{ + state: make([][]string, boardSize), + x: boardSize - 1, + y: boardSize - 1, + } + for i := 0; i < boardSize; i++ { + b.state[i] = make([]string, boardSize) + for j := 0; j < boardSize; j++ { + b.state[i][j] = ":white_large_square:" + } + } + + b.state[boardSize-1][boardSize-1] = DUDE + b.state[boardSize/2][boardSize/2] = BOULDER + b.state[0][0] = HOLE + + return &b +} + +func (b *board) toMessageString() string { + lines := make([]string, len(b.state)) + for i := 0; i < len(b.state); i++ { + lines[i] = strings.Join(b.state[i], "") + } + return strings.Join(lines, "\n") +} + +func (b *board) checkAndMove(dx, dy int) int { + newX := b.x + dx + newY := b.y + dy + + if newX < 0 || newY < 0 || newX >= len(b.state) || newY >= len(b.state) { + return INVALID + } + + if b.state[newY][newX] == HOLE { + return INVALID + } + + win := false + if b.state[newY][newX] == BOULDER { + newBoulderX := newX + dx + newBoulderY := newY + dy + + if newBoulderX < 0 || newBoulderY < 0 || newBoulderX >= len(b.state) || newBoulderY >= len(b.state) { + return INVALID + } + + if b.state[newBoulderY][newBoulderX] != HOLE { + b.state[newBoulderY][newBoulderX] = BOULDER + } else { + win = true + } + } + + b.state[newY][newX] = DUDE + b.state[b.y][b.x] = EMPTY + b.x = newX + b.y = newY + + if win { + return WIN + } + return OK } func New(b bot.Bot) *RPGPlugin { return &RPGPlugin{ - Bot: b, - listenFor: map[string]bool{}, + Bot: b, + listenFor: map[string]*board{}, } } func (p *RPGPlugin) Message(message msg.Message) bool { if strings.ToLower(message.Body) == "start rpg" { - ts := p.Bot.SendMessage(message.Channel, "I'll edit this.") - - p.listenFor[ts] = true - - time.Sleep(2 * time.Second) - - edited := "" - for i := 0; i <= 5; i++ { - p.Bot.Edit(message.Channel, edited, ts) - edited += ":fire:" - time.Sleep(500 * time.Millisecond) - } - p.Bot.Edit(message.Channel, "HECK YES", ts) - - p.Bot.ReplyToMessageIdentifier(message.Channel, "How's this reply?", ts) + b := NewRandomBoard() + ts := p.Bot.SendMessage(message.Channel, b.toMessageString()) + p.listenFor[ts] = b + p.Bot.ReplyToMessageIdentifier(message.Channel, "Over here.", ts) return true } return false @@ -64,8 +137,31 @@ func (p *RPGPlugin) RegisterWeb() *string { func (p *RPGPlugin) ReplyMessage(message msg.Message, identifier string) bool { if strings.ToLower(message.User.Name) != strings.ToLower(p.Bot.Config().Nick) { - if _, ok := p.listenFor[identifier]; ok { - p.Bot.ReplyToMessageIdentifier(message.Channel, "Pong", identifier) + if b, ok := p.listenFor[identifier]; ok { + + var res int + + if message.Body == "left" { + res = b.checkAndMove(-1, 0) + } else if message.Body == "right" { + res = b.checkAndMove(1, 0) + } else if message.Body == "up" { + res = b.checkAndMove(0, -1) + } else if message.Body == "down" { + res = b.checkAndMove(0, 1) + } else { + return false + } + + switch res { + case OK: + p.Bot.Edit(message.Channel, b.toMessageString(), identifier) + case WIN: + p.Bot.Edit(message.Channel, b.toMessageString(), identifier) + p.Bot.ReplyToMessageIdentifier(message.Channel, "congratulations, you beat the easiest level imaginable.", identifier) + case INVALID: + p.Bot.ReplyToMessageIdentifier(message.Channel, fmt.Sprintf("you can't move %s", message.Body), identifier) + } return true } } From 345129264bdff22743d91a41fd49416990430f14 Mon Sep 17 00:00:00 2001 From: skkiesel Date: Fri, 3 Nov 2017 14:07:30 -0400 Subject: [PATCH 08/28] use 'as_user' when sending messages and replies --- slack/slack.go | 2 ++ 1 file changed, 2 insertions(+) diff --git a/slack/slack.go b/slack/slack.go index b22f622..ade25ef 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -209,6 +209,7 @@ func (s *Slack) SendMessageType(channel, message string, meMessage bool) (string url.Values{"token": {s.config.Slack.Token}, "channel": {channel}, "text": {message}, + "as_user": {"true"}, }) if err != nil { @@ -258,6 +259,7 @@ func (s *Slack) ReplyToMessageIdentifier(channel, message, identifier string) (s url.Values{"token": {s.config.Slack.Token}, "channel": {channel}, "text": {message}, + "as_user": {"true"}, "thread_ts": {identifier}, }) From 827879b72df167fcc5a010c60264a36480a40951 Mon Sep 17 00:00:00 2001 From: Chris Sexton Date: Fri, 3 Nov 2017 14:01:49 -0400 Subject: [PATCH 09/28] sisyphus: add plugin --- main.go | 2 + plugins/sisyphus/sisyphus.go | 229 ++++++++++++++++++++++++++++++ plugins/sisyphus/sisyphus_test.go | 1 + slack/slack.go | 8 +- 4 files changed, 237 insertions(+), 3 deletions(-) create mode 100644 plugins/sisyphus/sisyphus.go create mode 100644 plugins/sisyphus/sisyphus_test.go diff --git a/main.go b/main.go index 069a27f..4c34c8d 100644 --- a/main.go +++ b/main.go @@ -23,6 +23,7 @@ import ( "github.com/velour/catbase/plugins/reminder" "github.com/velour/catbase/plugins/rpgORdie" "github.com/velour/catbase/plugins/rss" + "github.com/velour/catbase/plugins/sisyphus" "github.com/velour/catbase/plugins/stats" "github.com/velour/catbase/plugins/talker" "github.com/velour/catbase/plugins/twitch" @@ -71,6 +72,7 @@ func main() { b.AddHandler("twitch", twitch.New(b)) b.AddHandler("inventory", inventory.New(b)) b.AddHandler("rpgORdie", rpgORdie.New(b)) + b.AddHandler("sisyphus", sisyphus.New(b)) // catches anything left, will always return true b.AddHandler("factoid", fact.New(b)) diff --git a/plugins/sisyphus/sisyphus.go b/plugins/sisyphus/sisyphus.go new file mode 100644 index 0000000..44f236f --- /dev/null +++ b/plugins/sisyphus/sisyphus.go @@ -0,0 +1,229 @@ +package sisyphus + +import ( + "fmt" + "log" + "math/rand" + "strconv" + "strings" + "time" + + "github.com/velour/catbase/bot" + "github.com/velour/catbase/bot/msg" +) + +const ( + BOULDER = ":full_moon:" + MOUNTAIN = ":new_moon:" +) + +type SisyphusPlugin struct { + Bot bot.Bot + listenFor map[string]*game +} + +type game struct { + id string + channel string + bot bot.Bot + who string + start time.Time + size int + hardness int + current int + nextPush time.Time + nextDec time.Time + timers [2]*time.Timer + ended bool + nextAns int +} + +func NewRandomGame(bot bot.Bot, channel, who string) *game { + size := rand.Intn(9) + 2 + g := game{ + channel: channel, + bot: bot, + who: who, + start: time.Now(), + size: size, + hardness: rand.Intn(9) + 1, + current: size / 2, + } + g.id = bot.SendMessage(channel, g.toMessageString()) + + g.schedulePush() + g.scheduleDecrement() + + return &g +} + +func (g *game) scheduleDecrement() { + if g.timers[0] != nil { + g.timers[0].Stop() + } + g.nextDec = time.Now().Add(time.Duration((60 + rand.Intn(60*5))) * time.Second) + go func() { + t := time.NewTimer(g.nextDec.Sub(time.Now())) + g.timers[0] = t + select { + case <-t.C: + g.handleDecrement() + } + }() +} + +func (g *game) schedulePush() { + if g.timers[1] != nil { + g.timers[1].Stop() + } + g.nextPush = time.Now().Add(time.Duration(rand.Intn(60)+1) * time.Minute) + go func() { + t := time.NewTimer(g.nextPush.Sub(time.Now())) + g.timers[1] = t + select { + case <-t.C: + g.handleNotify() + } + }() +} + +func (g *game) endGame() { + for _, t := range g.timers { + t.Stop() + } + g.ended = true +} + +func (g *game) handleDecrement() { + g.current++ + g.bot.Edit(g.channel, g.toMessageString(), g.id) + if g.current > g.size-2 { + g.bot.ReplyToMessageIdentifier(g.channel, "you lose", g.id) + msg := fmt.Sprintf("%s just lost the game after %s", g.who, time.Now().Sub(g.start)) + g.bot.SendMessage(g.channel, msg) + g.endGame() + } else { + g.scheduleDecrement() + } +} + +func (g *game) handleNotify() { + g.bot.ReplyToMessageIdentifier(g.channel, "You can push now.\n"+g.generateQuestion(), g.id) +} + +func (g *game) generateQuestion() string { + n1 := rand.Intn(99) + (rand.Intn(9)+1)*100 + n2 := rand.Intn(99) + (rand.Intn(9)+1)*100 + var op string + switch i := rand.Intn(3); i { + case 0: + // times + g.nextAns = n1 * n2 + op = "*" + case 1: + // plus + g.nextAns = n1 + n2 + op = "+" + case 2: + // minus + g.nextAns = n1 - n2 + op = "-" + } + return fmt.Sprintf("What is %d %s %d?", n1, op, n2) +} + +func (g *game) checkAnswer(ans string) bool { + if strings.Contains(ans, strconv.Itoa(g.nextAns)) { + g.current-- + if g.current < 0 { + g.current = 0 + } + return true + } + return false +} + +func (g *game) toMessageString() string { + out := "" + for i := 0; i < g.size; i++ { + for j := 0; j < i; j++ { + out = out + MOUNTAIN + } + if i == g.current { + out = out + BOULDER + } else if i == g.current+1 { + out = out + ":" + g.who + ":" + } + out = out + "\n" + } + return out +} + +func New(b bot.Bot) *SisyphusPlugin { + return &SisyphusPlugin{ + Bot: b, + listenFor: map[string]*game{}, + } +} + +func (p *SisyphusPlugin) Message(message msg.Message) bool { + if strings.ToLower(message.Body) == "start sisyphus" { + b := NewRandomGame(p.Bot, message.Channel, message.User.Name) + p.listenFor[b.id] = b + p.Bot.ReplyToMessageIdentifier(message.Channel, "Over here.", b.id) + return true + } + return false +} + +func (p *SisyphusPlugin) Help(channel string, parts []string) { + p.Bot.SendMessage(channel, "https://en.wikipedia.org/wiki/Sisyphus") +} + +func (p *SisyphusPlugin) Event(kind string, message msg.Message) bool { + return false +} + +func (p *SisyphusPlugin) BotMessage(message msg.Message) bool { + return false +} + +func (p *SisyphusPlugin) RegisterWeb() *string { + return nil +} + +func (p *SisyphusPlugin) ReplyMessage(message msg.Message, identifier string) bool { + if strings.ToLower(message.User.Name) != strings.ToLower(p.Bot.Config().Nick) { + if g, ok := p.listenFor[identifier]; ok { + + log.Printf("got message on %s: %+v", identifier, message) + + if g.ended { + return false + } + + if strings.ToLower(message.Body) == "end game" { + g.endGame() + return true + } + + if time.Now().After(g.nextPush) { + if g.checkAnswer(message.Body) { + p.Bot.Edit(message.Channel, g.toMessageString(), identifier) + g.schedulePush() + msg := fmt.Sprintf("Ok. You can push again in %s", g.nextPush.Sub(time.Now())) + p.Bot.ReplyToMessageIdentifier(message.Channel, msg, identifier) + } else { + p.Bot.ReplyToMessageIdentifier(message.Channel, "you lose", identifier) + msg := fmt.Sprintf("%s just lost the sisyphus game after %s", g.who, time.Now().Sub(g.start)) + p.Bot.SendMessage(message.Channel, msg) + g.endGame() + } + } else { + p.Bot.ReplyToMessageIdentifier(message.Channel, "you cannot push yet", identifier) + } + return true + } + } + return false +} diff --git a/plugins/sisyphus/sisyphus_test.go b/plugins/sisyphus/sisyphus_test.go new file mode 100644 index 0000000..f0b61a0 --- /dev/null +++ b/plugins/sisyphus/sisyphus_test.go @@ -0,0 +1 @@ +package sisyphus diff --git a/slack/slack.go b/slack/slack.go index b22f622..820efec 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -39,8 +39,8 @@ type Slack struct { emoji map[string]string - eventReceived func(msg.Message) - messageReceived func(msg.Message) + eventReceived func(msg.Message) + messageReceived func(msg.Message) replyMessageReceived func(msg.Message, string) } @@ -135,7 +135,7 @@ type slackMessage struct { User string `json:"user"` Username string `json:"username"` Ts string `json:"ts"` - ThreadTs string `json:"thread_ts"` + ThreadTs string `json:"thread_ts"` Error struct { Code uint64 `json:"code"` Msg string `json:"msg"` @@ -207,6 +207,7 @@ func (s *Slack) SendMessageType(channel, message string, meMessage bool) (string resp, err := http.PostForm(postUrl, url.Values{"token": {s.config.Slack.Token}, + "as_user": {"true"}, "channel": {channel}, "text": {message}, }) @@ -256,6 +257,7 @@ func (s *Slack) SendAction(channel, message string) string { func (s *Slack) ReplyToMessageIdentifier(channel, message, identifier string) (string, bool) { resp, err := http.PostForm("https://slack.com/api/chat.postMessage", url.Values{"token": {s.config.Slack.Token}, + "as_user": {"true"}, "channel": {channel}, "text": {message}, "thread_ts": {identifier}, From df956513c2421e41cb2834bd4b7358dce94fc176 Mon Sep 17 00:00:00 2001 From: Chris Sexton Date: Mon, 6 Nov 2017 14:32:49 -0500 Subject: [PATCH 10/28] sisyphus: config-ize timing --- config/config.go | 4 ++++ example_config.lua | 11 ++++++++++- plugins/sisyphus/sisyphus.go | 20 ++++++++++---------- 3 files changed, 24 insertions(+), 11 deletions(-) diff --git a/config/config.go b/config/config.go index 9f29b79..360e3be 100644 --- a/config/config.go +++ b/config/config.go @@ -103,6 +103,10 @@ type Config struct { Inventory struct { Max int } + Sisyphus struct { + MinDecrement int + MinPush int + } } func init() { diff --git a/example_config.lua b/example_config.lua index 969ac2e..58e3d74 100644 --- a/example_config.lua +++ b/example_config.lua @@ -105,5 +105,14 @@ config = { }, DBPath = "stats.db" }, - HttpAddr = "127.0.0.1:1337" + HttpAddr = "127.0.0.1:1337", + Inventory = { + Max = 5 + }, + Sisyphus = { + MinDecrement = 10, + MinPush = 1 + } +} + } \ No newline at end of file diff --git a/plugins/sisyphus/sisyphus.go b/plugins/sisyphus/sisyphus.go index 44f236f..2ba8343 100644 --- a/plugins/sisyphus/sisyphus.go +++ b/plugins/sisyphus/sisyphus.go @@ -29,7 +29,6 @@ type game struct { who string start time.Time size int - hardness int current int nextPush time.Time nextDec time.Time @@ -41,13 +40,12 @@ type game struct { func NewRandomGame(bot bot.Bot, channel, who string) *game { size := rand.Intn(9) + 2 g := game{ - channel: channel, - bot: bot, - who: who, - start: time.Now(), - size: size, - hardness: rand.Intn(9) + 1, - current: size / 2, + channel: channel, + bot: bot, + who: who, + start: time.Now(), + size: size, + current: size / 2, } g.id = bot.SendMessage(channel, g.toMessageString()) @@ -61,7 +59,8 @@ func (g *game) scheduleDecrement() { if g.timers[0] != nil { g.timers[0].Stop() } - g.nextDec = time.Now().Add(time.Duration((60 + rand.Intn(60*5))) * time.Second) + minDec := g.bot.Config().Sisyphus.MinDecrement + g.nextDec = time.Now().Add(time.Duration((60 + rand.Intn(60*minDec))) * time.Minute) go func() { t := time.NewTimer(g.nextDec.Sub(time.Now())) g.timers[0] = t @@ -76,7 +75,8 @@ func (g *game) schedulePush() { if g.timers[1] != nil { g.timers[1].Stop() } - g.nextPush = time.Now().Add(time.Duration(rand.Intn(60)+1) * time.Minute) + minPush := g.bot.Config().Sisyphus.MinPush + g.nextPush = time.Now().Add(time.Duration(rand.Intn(60)+minPush) * time.Minute) go func() { t := time.NewTimer(g.nextPush.Sub(time.Now())) g.timers[1] = t From 7e1c4e11065b6b67f520efb9d5f37f01dd407da7 Mon Sep 17 00:00:00 2001 From: Chris Sexton Date: Mon, 6 Nov 2017 14:40:43 -0500 Subject: [PATCH 11/28] sisyphus: modify min/max times a little more --- config/config.go | 2 ++ plugins/sisyphus/sisyphus.go | 6 ++++-- 2 files changed, 6 insertions(+), 2 deletions(-) diff --git a/config/config.go b/config/config.go index 360e3be..d00cb5c 100644 --- a/config/config.go +++ b/config/config.go @@ -105,7 +105,9 @@ type Config struct { } Sisyphus struct { MinDecrement int + MaxDecrement int MinPush int + MaxPush int } } diff --git a/plugins/sisyphus/sisyphus.go b/plugins/sisyphus/sisyphus.go index 2ba8343..926dfc1 100644 --- a/plugins/sisyphus/sisyphus.go +++ b/plugins/sisyphus/sisyphus.go @@ -60,7 +60,8 @@ func (g *game) scheduleDecrement() { g.timers[0].Stop() } minDec := g.bot.Config().Sisyphus.MinDecrement - g.nextDec = time.Now().Add(time.Duration((60 + rand.Intn(60*minDec))) * time.Minute) + maxDec := g.bot.Config().Sisyphus.MinDecrement + g.nextDec = time.Now().Add(time.Duration((minDec + rand.Intn(maxDec))) * time.Minute) go func() { t := time.NewTimer(g.nextDec.Sub(time.Now())) g.timers[0] = t @@ -76,7 +77,8 @@ func (g *game) schedulePush() { g.timers[1].Stop() } minPush := g.bot.Config().Sisyphus.MinPush - g.nextPush = time.Now().Add(time.Duration(rand.Intn(60)+minPush) * time.Minute) + maxPush := g.bot.Config().Sisyphus.MaxPush + g.nextPush = time.Now().Add(time.Duration(rand.Intn(maxPush)+minPush) * time.Minute) go func() { t := time.NewTimer(g.nextPush.Sub(time.Now())) g.timers[1] = t From 4ecbe971ace560f8e45662db8277661ea0ec6272 Mon Sep 17 00:00:00 2001 From: skkiesel Date: Thu, 9 Nov 2017 06:21:43 -0500 Subject: [PATCH 12/28] Filter out messages from bots for now --- slack/slack.go | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/slack/slack.go b/slack/slack.go index ade25ef..cca8b24 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -39,8 +39,8 @@ type Slack struct { emoji map[string]string - eventReceived func(msg.Message) - messageReceived func(msg.Message) + eventReceived func(msg.Message) + messageReceived func(msg.Message) replyMessageReceived func(msg.Message, string) } @@ -134,8 +134,9 @@ type slackMessage struct { Text string `json:"text"` User string `json:"user"` Username string `json:"username"` + BotId string `json:"bot_id"` Ts string `json:"ts"` - ThreadTs string `json:"thread_ts"` + ThreadTs string `json:"thread_ts"` Error struct { Code uint64 `json:"code"` Msg string `json:"msg"` @@ -259,7 +260,7 @@ func (s *Slack) ReplyToMessageIdentifier(channel, message, identifier string) (s url.Values{"token": {s.config.Slack.Token}, "channel": {channel}, "text": {message}, - "as_user": {"true"}, + "as_user": {"true"}, "thread_ts": {identifier}, }) @@ -392,7 +393,7 @@ func (s *Slack) Serve() error { } switch msg.Type { case "message": - if !msg.Hidden && msg.ThreadTs == "" { + if msg.BotId == "" && !msg.Hidden && msg.ThreadTs == "" { m := s.buildMessage(msg) if m.Time.Before(s.lastRecieved) { log.Printf("Ignoring message: %+v\nlastRecieved: %v msg: %v", msg.ID, s.lastRecieved, m.Time) From 11bb338e6b91dee28db8ae59845486ea3412e725 Mon Sep 17 00:00:00 2001 From: Chris Sexton Date: Thu, 9 Nov 2017 10:38:31 -0500 Subject: [PATCH 13/28] slack: remove duplicate as_user --- slack/slack.go | 2 -- 1 file changed, 2 deletions(-) diff --git a/slack/slack.go b/slack/slack.go index 42a6a65..9bd317b 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -211,7 +211,6 @@ func (s *Slack) SendMessageType(channel, message string, meMessage bool) (string "as_user": {"true"}, "channel": {channel}, "text": {message}, - "as_user": {"true"}, }) if err != nil { @@ -262,7 +261,6 @@ func (s *Slack) ReplyToMessageIdentifier(channel, message, identifier string) (s "as_user": {"true"}, "channel": {channel}, "text": {message}, - "as_user": {"true"}, "thread_ts": {identifier}, }) From 20bf8748a2ab2202d7b8e2ba76b01ce653547713 Mon Sep 17 00:00:00 2001 From: Chris Sexton Date: Sat, 11 Nov 2017 11:51:50 -0500 Subject: [PATCH 14/28] slack: add whitelist for bots --- config/config.go | 1 + slack/slack.go | 11 ++++++++--- 2 files changed, 9 insertions(+), 3 deletions(-) diff --git a/config/config.go b/config/config.go index d00cb5c..15241a2 100644 --- a/config/config.go +++ b/config/config.go @@ -109,6 +109,7 @@ type Config struct { MinPush int MaxPush int } + BotList map[string]bool } func init() { diff --git a/slack/slack.go b/slack/slack.go index 9bd317b..2af6c36 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -134,7 +134,7 @@ type slackMessage struct { Text string `json:"text"` User string `json:"user"` Username string `json:"username"` - BotId string `json:"bot_id"` + BotID string `json:"bot_id"` Ts string `json:"ts"` ThreadTs string `json:"thread_ts"` Error struct { @@ -363,7 +363,6 @@ func (s *Slack) populateEmojiList() { func (s *Slack) receiveMessage() (slackMessage, error) { var msg []byte m := slackMessage{} - //err := websocket.JSON.Receive(s.ws, &m) err := websocket.Message.Receive(s.ws, &msg) if err != nil { log.Println("Error decoding WS message") @@ -393,7 +392,13 @@ func (s *Slack) Serve() error { } switch msg.Type { case "message": - if msg.BotId == "" && !msg.Hidden && msg.ThreadTs == "" { + botOK := true + if msg.BotID != "" { + u, _ := s.getUser(msg.User) + log.Printf("User: %s, BotList: %+v", u, s.config.BotList) + botOK = s.config.BotList[strings.Title(u)] + } + if botOK && !msg.Hidden && msg.ThreadTs == "" { m := s.buildMessage(msg) if m.Time.Before(s.lastRecieved) { log.Printf("Ignoring message: %+v\nlastRecieved: %v msg: %v", msg.ID, s.lastRecieved, m.Time) From f5f70bbdc85a70cb0052ea705ac49dd8a95120f5 Mon Sep 17 00:00:00 2001 From: Chris Sexton Date: Sun, 12 Nov 2017 12:43:40 -0500 Subject: [PATCH 15/28] slack: test checking both user and username when we have a botID --- slack/slack.go | 3 +++ 1 file changed, 3 insertions(+) diff --git a/slack/slack.go b/slack/slack.go index 2af6c36..5dddef3 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -395,6 +395,9 @@ func (s *Slack) Serve() error { botOK := true if msg.BotID != "" { u, _ := s.getUser(msg.User) + if u == "" && msg.Username != "" { + u = msg.Username + } log.Printf("User: %s, BotList: %+v", u, s.config.BotList) botOK = s.config.BotList[strings.Title(u)] } From 408c37b7c711520ea35ddcde90f5c66f80df722f Mon Sep 17 00:00:00 2001 From: Chris Sexton Date: Thu, 16 Nov 2017 17:05:22 -0500 Subject: [PATCH 16/28] tell: create tell plugin --- plugins/tell/tell.go | 48 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 48 insertions(+) create mode 100644 plugins/tell/tell.go diff --git a/plugins/tell/tell.go b/plugins/tell/tell.go new file mode 100644 index 0000000..9b7d003 --- /dev/null +++ b/plugins/tell/tell.go @@ -0,0 +1,48 @@ +package tell + +import ( + "fmt" + "log" + "strings" + + "github.com/velour/catbase/bot" + "github.com/velour/catbase/bot/msg" +) + +type delayedMsg string + +type TellPlugin struct { + b bot.Bot + users map[string][]string +} + +func New(b bot.Bot) *TellPlugin { + return &TellPlugin{b, make(map[string][]string)} +} + +func (t *TellPlugin) Message(message msg.Message) bool { + if strings.HasPrefix(message.Body, "tell") { + parts := strings.Split(message.Body, " ") + target := parts[1] + newMessage := strings.Join(parts[2:], " ") + newMessage = fmt.Sprintf("Hey, %s. %s said: %s", target, message.User.Name, newMessage) + t.users[target] = append(t.users[target], newMessage) + t.b.SendMessage(message.Channel, fmt.Sprintf("Okay. I'll tell %s.", target)) + return true + } + log.Printf("current pending tells: %+v\nuser is: %s", t.users, message.User.Name) + if msg, ok := t.users[message.User.Name]; ok { + for _, m := range msg { + t.b.SendMessage(message.Channel, string(m)) + } + t.users[message.User.Name] = []string{} + return true + } + return false +} + +func (t *TellPlugin) Event(kind string, message msg.Message) bool { return false } +func (t *TellPlugin) ReplyMessage(msg.Message, string) bool { return false } +func (t *TellPlugin) BotMessage(message msg.Message) bool { return false } +func (t *TellPlugin) Help(channel string, parts []string) {} +func (t *TellPlugin) RegisterWeb() *string { return nil } From 1a420c173882401e5bff390d78681713bc021793 Mon Sep 17 00:00:00 2001 From: Chris Sexton Date: Thu, 16 Nov 2017 17:06:48 -0500 Subject: [PATCH 17/28] tell: create tell plugin --- main.go | 2 ++ 1 file changed, 2 insertions(+) diff --git a/main.go b/main.go index 4c34c8d..027a60d 100644 --- a/main.go +++ b/main.go @@ -26,6 +26,7 @@ import ( "github.com/velour/catbase/plugins/sisyphus" "github.com/velour/catbase/plugins/stats" "github.com/velour/catbase/plugins/talker" + "github.com/velour/catbase/plugins/tell" "github.com/velour/catbase/plugins/twitch" "github.com/velour/catbase/plugins/your" "github.com/velour/catbase/plugins/zork" @@ -73,6 +74,7 @@ func main() { b.AddHandler("inventory", inventory.New(b)) b.AddHandler("rpgORdie", rpgORdie.New(b)) b.AddHandler("sisyphus", sisyphus.New(b)) + b.AddHandler("tell", tell.New(b)) // catches anything left, will always return true b.AddHandler("factoid", fact.New(b)) From 9f332909b5819cf71cf7e1f1fa06d18620917f5a Mon Sep 17 00:00:00 2001 From: Chris Sexton Date: Fri, 17 Nov 2017 08:39:35 -0500 Subject: [PATCH 18/28] tell: fix empty lists and user case --- plugins/tell/tell.go | 11 +++++------ 1 file changed, 5 insertions(+), 6 deletions(-) diff --git a/plugins/tell/tell.go b/plugins/tell/tell.go index 9b7d003..05bab8a 100644 --- a/plugins/tell/tell.go +++ b/plugins/tell/tell.go @@ -2,7 +2,6 @@ package tell import ( "fmt" - "log" "strings" "github.com/velour/catbase/bot" @@ -21,21 +20,21 @@ func New(b bot.Bot) *TellPlugin { } func (t *TellPlugin) Message(message msg.Message) bool { - if strings.HasPrefix(message.Body, "tell") { + if strings.HasPrefix(strings.ToLower(message.Body), "tell") { parts := strings.Split(message.Body, " ") - target := parts[1] + target := strings.ToLower(parts[1]) newMessage := strings.Join(parts[2:], " ") newMessage = fmt.Sprintf("Hey, %s. %s said: %s", target, message.User.Name, newMessage) t.users[target] = append(t.users[target], newMessage) t.b.SendMessage(message.Channel, fmt.Sprintf("Okay. I'll tell %s.", target)) return true } - log.Printf("current pending tells: %+v\nuser is: %s", t.users, message.User.Name) - if msg, ok := t.users[message.User.Name]; ok { + uname := strings.ToLower(message.User.Name) + if msg, ok := t.users[uname]; ok && len(msg) > 0 { for _, m := range msg { t.b.SendMessage(message.Channel, string(m)) } - t.users[message.User.Name] = []string{} + t.users[uname] = []string{} return true } return false From 95d99fa94d9ada09e07d93ba4787a7cd621b8c6a Mon Sep 17 00:00:00 2001 From: skkiesel Date: Fri, 17 Nov 2017 10:16:10 -0500 Subject: [PATCH 19/28] grab the bot's bot id from the message response and use that to avoid self-reply loops --- config/config.go | 1 - slack/slack.go | 17 +++++++---------- 2 files changed, 7 insertions(+), 11 deletions(-) diff --git a/config/config.go b/config/config.go index 15241a2..d00cb5c 100644 --- a/config/config.go +++ b/config/config.go @@ -109,7 +109,6 @@ type Config struct { MinPush int MaxPush int } - BotList map[string]bool } func init() { diff --git a/slack/slack.go b/slack/slack.go index 5dddef3..94a3dc2 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -37,6 +37,8 @@ type Slack struct { users map[string]string + myBotID string + emoji map[string]string eventReceived func(msg.Message) @@ -228,6 +230,7 @@ func (s *Slack) SendMessageType(channel, message string, meMessage bool) (string type MessageResponse struct { OK bool `json:"ok"` Timestamp string `json:"ts"` + BotID string `json:"message.bot_id"` } var mr MessageResponse @@ -240,6 +243,8 @@ func (s *Slack) SendMessageType(channel, message string, meMessage bool) (string return "", errors.New("failure response received") } + s.myBotID = mr.BotID + return mr.Timestamp, err } @@ -392,16 +397,8 @@ func (s *Slack) Serve() error { } switch msg.Type { case "message": - botOK := true - if msg.BotID != "" { - u, _ := s.getUser(msg.User) - if u == "" && msg.Username != "" { - u = msg.Username - } - log.Printf("User: %s, BotList: %+v", u, s.config.BotList) - botOK = s.config.BotList[strings.Title(u)] - } - if botOK && !msg.Hidden && msg.ThreadTs == "" { + isItMe := s.myBotID != "" && msg.BotID != s.myBotID + if !isItMe && !msg.Hidden && msg.ThreadTs == "" { m := s.buildMessage(msg) if m.Time.Before(s.lastRecieved) { log.Printf("Ignoring message: %+v\nlastRecieved: %v msg: %v", msg.ID, s.lastRecieved, m.Time) From 7841da0f48b0a40131b21fcb6aa1397524753583 Mon Sep 17 00:00:00 2001 From: skkiesel Date: Fri, 17 Nov 2017 10:30:15 -0500 Subject: [PATCH 20/28] one more try --- slack/slack.go | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/slack/slack.go b/slack/slack.go index 94a3dc2..3c81389 100644 --- a/slack/slack.go +++ b/slack/slack.go @@ -230,7 +230,9 @@ func (s *Slack) SendMessageType(channel, message string, meMessage bool) (string type MessageResponse struct { OK bool `json:"ok"` Timestamp string `json:"ts"` - BotID string `json:"message.bot_id"` + Message struct { + BotID string `json:"bot_id"` + } `json:"message"` } var mr MessageResponse @@ -243,7 +245,7 @@ func (s *Slack) SendMessageType(channel, message string, meMessage bool) (string return "", errors.New("failure response received") } - s.myBotID = mr.BotID + s.myBotID = mr.Message.BotID return mr.Timestamp, err } @@ -397,7 +399,7 @@ func (s *Slack) Serve() error { } switch msg.Type { case "message": - isItMe := s.myBotID != "" && msg.BotID != s.myBotID + isItMe := msg.BotID != "" && msg.BotID == s.myBotID if !isItMe && !msg.Hidden && msg.ThreadTs == "" { m := s.buildMessage(msg) if m.Time.Before(s.lastRecieved) { From 83289d65f0394c7f8f8c34b2709dab1de31e984b Mon Sep 17 00:00:00 2001 From: Steve McCoy Date: Mon, 27 Nov 2017 22:07:59 -0500 Subject: [PATCH 21/28] Don't count "a" and "it" toward the emojification score Fixes #95. --- plugins/emojifyme/emojifyme.go | 12 +++++++++--- 1 file changed, 9 insertions(+), 3 deletions(-) diff --git a/plugins/emojifyme/emojifyme.go b/plugins/emojifyme/emojifyme.go index e8a6119..e71aede 100644 --- a/plugins/emojifyme/emojifyme.go +++ b/plugins/emojifyme/emojifyme.go @@ -71,19 +71,25 @@ func (p *EmojifyMePlugin) Message(message msg.Message) bool { tokens := strings.Fields(strings.ToLower(message.Body)) for i, token := range tokens { if _, ok := p.Emoji[token]; ok { - emojied++ + if token != "a" && token != "it" { + emojied++ + } tokens[i] = ":" + token + ":" } else if strings.HasSuffix(token, "s") { //Check to see if we can strip the trailing "es" off and get an emoji temp := strings.TrimSuffix(token, "s") if _, ok := p.Emoji[temp]; ok { - emojied++ + if token != "a" && token != "it" { + emojied++ + } tokens[i] = ":" + temp + ":s" } else if strings.HasSuffix(token, "es") { //Check to see if we can strip the trailing "es" off and get an emoji temp := strings.TrimSuffix(token, "es") if _, ok := p.Emoji[temp]; ok { - emojied++ + if token != "a" && token != "it" { + emojied++ + } tokens[i] = ":" + temp + ":es" } } From 85b649bc98ebac7be503318f6c1bd009dff793cb Mon Sep 17 00:00:00 2001 From: Steve McCoy Date: Mon, 27 Nov 2017 22:09:55 -0500 Subject: [PATCH 22/28] =?UTF-8?q?Oh,=20and=20handle=20the=20plurals?= =?UTF-8?q?=E2=80=A6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- plugins/emojifyme/emojifyme.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/plugins/emojifyme/emojifyme.go b/plugins/emojifyme/emojifyme.go index e71aede..8898b54 100644 --- a/plugins/emojifyme/emojifyme.go +++ b/plugins/emojifyme/emojifyme.go @@ -79,7 +79,7 @@ func (p *EmojifyMePlugin) Message(message msg.Message) bool { //Check to see if we can strip the trailing "es" off and get an emoji temp := strings.TrimSuffix(token, "s") if _, ok := p.Emoji[temp]; ok { - if token != "a" && token != "it" { + if temp != "a" && temp != "it" { emojied++ } tokens[i] = ":" + temp + ":s" @@ -87,7 +87,7 @@ func (p *EmojifyMePlugin) Message(message msg.Message) bool { //Check to see if we can strip the trailing "es" off and get an emoji temp := strings.TrimSuffix(token, "es") if _, ok := p.Emoji[temp]; ok { - if token != "a" && token != "it" { + if temp != "a" && temp != "it" { emojied++ } tokens[i] = ":" + temp + ":es" From 5794c73efdcb27132b4d858b8c516cebfe3a1cc7 Mon Sep 17 00:00:00 2001 From: Steve McCoy Date: Sun, 3 Dec 2017 13:04:55 -0500 Subject: [PATCH 23/28] =?UTF-8?q?Change=20to=20use=20the=20lua=20config?= =?UTF-8?q?=E2=80=A6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- config/config.go | 1 + example_config.lua | 6 +++++- plugins/emojifyme/emojifyme.go | 16 +++++++++++++--- 3 files changed, 19 insertions(+), 4 deletions(-) diff --git a/config/config.go b/config/config.go index d00cb5c..0d345c7 100644 --- a/config/config.go +++ b/config/config.go @@ -91,6 +91,7 @@ type Config struct { } Emojify struct { Chance float64 + Scoreless []string } Reaction struct { GeneralChance float64 diff --git a/example_config.lua b/example_config.lua index 58e3d74..fce3646 100644 --- a/example_config.lua +++ b/example_config.lua @@ -29,7 +29,11 @@ config = { YourChance = 0.4 }, Emojify = { - Chance = 0.02 + Chance = 0.02, + Scoreless = { + "a", + "it" + } }, DB = { File = "catbase.db", diff --git a/plugins/emojifyme/emojifyme.go b/plugins/emojifyme/emojifyme.go index 8898b54..4c1383d 100644 --- a/plugins/emojifyme/emojifyme.go +++ b/plugins/emojifyme/emojifyme.go @@ -67,11 +67,12 @@ func (p *EmojifyMePlugin) Message(message msg.Message) bool { } } + inertTokens := p.Bot.Config().Emojify.Scoreless emojied := 0.0 tokens := strings.Fields(strings.ToLower(message.Body)) for i, token := range tokens { if _, ok := p.Emoji[token]; ok { - if token != "a" && token != "it" { + if !stringsContain(inertTokens, token) { emojied++ } tokens[i] = ":" + token + ":" @@ -79,7 +80,7 @@ func (p *EmojifyMePlugin) Message(message msg.Message) bool { //Check to see if we can strip the trailing "es" off and get an emoji temp := strings.TrimSuffix(token, "s") if _, ok := p.Emoji[temp]; ok { - if temp != "a" && temp != "it" { + if !stringsContain(inertTokens, temp) { emojied++ } tokens[i] = ":" + temp + ":s" @@ -87,7 +88,7 @@ func (p *EmojifyMePlugin) Message(message msg.Message) bool { //Check to see if we can strip the trailing "es" off and get an emoji temp := strings.TrimSuffix(token, "es") if _, ok := p.Emoji[temp]; ok { - if temp != "a" && temp != "it" { + if !stringsContain(inertTokens, temp) { emojied++ } tokens[i] = ":" + temp + ":es" @@ -120,3 +121,12 @@ func (p *EmojifyMePlugin) RegisterWeb() *string { } func (p *EmojifyMePlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } + +func stringsContain(haystack []string, needle string) bool { + for _, s := range haystack { + if s == needle { + return true + } + } + return false +} From 132fdd29bef583840a2d10e63b2ebcd52cfe3865 Mon Sep 17 00:00:00 2001 From: Chris Sexton Date: Tue, 19 Dec 2017 13:37:47 -0500 Subject: [PATCH 24/28] picker: add new plugin --- main.go | 2 ++ plugins/picker/picker.go | 78 ++++++++++++++++++++++++++++++++++++++++ 2 files changed, 80 insertions(+) create mode 100644 plugins/picker/picker.go diff --git a/main.go b/main.go index 027a60d..4ab7e1b 100644 --- a/main.go +++ b/main.go @@ -19,6 +19,7 @@ import ( "github.com/velour/catbase/plugins/first" "github.com/velour/catbase/plugins/inventory" "github.com/velour/catbase/plugins/leftpad" + "github.com/velour/catbase/plugins/picker" "github.com/velour/catbase/plugins/reaction" "github.com/velour/catbase/plugins/reminder" "github.com/velour/catbase/plugins/rpgORdie" @@ -60,6 +61,7 @@ func main() { // b.AddHandler("downtime", downtime.New(b)) b.AddHandler("talker", talker.New(b)) b.AddHandler("dice", dice.New(b)) + b.AddHandler("picker", picker.New(b)) b.AddHandler("beers", beers.New(b)) b.AddHandler("remember", fact.NewRemember(b)) b.AddHandler("your", your.New(b)) diff --git a/plugins/picker/picker.go b/plugins/picker/picker.go new file mode 100644 index 0000000..7b0c3a7 --- /dev/null +++ b/plugins/picker/picker.go @@ -0,0 +1,78 @@ +// © 2013 the CatBase Authors under the WTFPL. See AUTHORS for the list of authors. + +package picker + +import ( + "strings" + "time" + + "fmt" + "math/rand" + + "github.com/velour/catbase/bot" + "github.com/velour/catbase/bot/msg" +) + +type PickerPlugin struct { + Bot bot.Bot +} + +// NewPickerPlugin creates a new PickerPlugin with the Plugin interface +func New(bot bot.Bot) *PickerPlugin { + rand.Seed(time.Now().Unix()) + + return &PickerPlugin{ + Bot: bot, + } +} + +func rollDie(sides int) int { + return rand.Intn(sides) + 1 +} + +// Message responds to the bot hook on recieving messages. +// This function returns true if the plugin responds in a meaningful way to the users message. +// Otherwise, the function returns false and the bot continues execution of other plugins. +func (p *PickerPlugin) Message(message msg.Message) bool { + if !message.Command { + return false + } + + body := message.Body + pfx, sfx := "pick {", "}" + + if strings.HasPrefix(body, pfx) && strings.HasSuffix(body, sfx) { + body = strings.TrimSuffix(strings.TrimPrefix(body, pfx), sfx) + items := strings.Split(body, ",") + item := items[rand.Intn(len(items))] + + out := fmt.Sprintf("I've chosen \"%s\" for you.", item) + + p.Bot.SendMessage(message.Channel, out) + + return true + } + return false +} + +// Help responds to help requests. Every plugin must implement a help function. +func (p *PickerPlugin) Help(channel string, parts []string) { + p.Bot.SendMessage(channel, "Choose from a list of options. Try \"pick {a,b,c}\".") +} + +// Empty event handler because this plugin does not do anything on event recv +func (p *PickerPlugin) Event(kind string, message msg.Message) bool { + return false +} + +// Handler for bot's own messages +func (p *PickerPlugin) BotMessage(message msg.Message) bool { + return false +} + +// Register any web URLs desired +func (p *PickerPlugin) RegisterWeb() *string { + return nil +} + +func (p *PickerPlugin) ReplyMessage(message msg.Message, identifier string) bool { return false } From 6bc72b93d146bcf00c3449c8247c08dac0ed2f35 Mon Sep 17 00:00:00 2001 From: Chris Sexton Date: Tue, 19 Dec 2017 13:38:02 -0500 Subject: [PATCH 25/28] dice: clean up --- plugins/dice/dice.go | 59 ++++++++++++++++++-------------------------- 1 file changed, 24 insertions(+), 35 deletions(-) diff --git a/plugins/dice/dice.go b/plugins/dice/dice.go index 3ee6dc3..92c613e 100644 --- a/plugins/dice/dice.go +++ b/plugins/dice/dice.go @@ -12,8 +12,6 @@ import ( import ( "fmt" "math/rand" - "strconv" - "strings" ) // This is a dice plugin to serve as an example and quick copy/paste for new plugins. @@ -39,46 +37,37 @@ func rollDie(sides int) int { // This function returns true if the plugin responds in a meaningful way to the users message. // Otherwise, the function returns false and the bot continues execution of other plugins. func (p *DicePlugin) Message(message msg.Message) bool { + if !message.Command { + return false + } + channel := message.Channel - parts := strings.Fields(message.Body) + nDice := 0 + sides := 0 - if len(parts) == 1 && message.Command { - var dice []string - dice = strings.Split(parts[0], "d") + if n, err := fmt.Sscanf(message.Body, "%dd%d", &nDice, &sides); n != 2 || err != nil { + return false + } - if len(dice) == 2 { - // We actually have a die roll. - nDice, err := strconv.Atoi(dice[0]) - if err != nil { - return false - } + if sides < 2 || nDice < 1 || nDice > 20 { + p.Bot.SendMessage(channel, "You're a dick.") + return true + } - sides, err := strconv.Atoi(dice[1]) - if err != nil { - return false - } + rolls := fmt.Sprintf("%s, you rolled: ", message.User.Name) - if sides < 2 || nDice < 1 || nDice > 20 { - p.Bot.SendMessage(channel, "You're a dick.") - return true - } - - rolls := fmt.Sprintf("%s, you rolled: ", message.User.Name) - - for i := 0; i < nDice; i++ { - rolls = fmt.Sprintf("%s %d", rolls, rollDie(sides)) - if i != nDice-1 { - rolls = fmt.Sprintf("%s,", rolls) - } else { - rolls = fmt.Sprintf("%s.", rolls) - } - } - - p.Bot.SendMessage(channel, rolls) - return true + for i := 0; i < nDice; i++ { + rolls = fmt.Sprintf("%s %d", rolls, rollDie(sides)) + if i != nDice-1 { + rolls = fmt.Sprintf("%s,", rolls) + } else { + rolls = fmt.Sprintf("%s.", rolls) } } - return false + + p.Bot.SendMessage(channel, rolls) + return true + } // Help responds to help requests. Every plugin must implement a help function. From 22bca5bf4d11e3f086fc6f7d09ea358d46942568 Mon Sep 17 00:00:00 2001 From: Chris Sexton Date: Tue, 19 Dec 2017 13:42:30 -0500 Subject: [PATCH 26/28] picker: less strict (no command) --- plugins/picker/picker.go | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/plugins/picker/picker.go b/plugins/picker/picker.go index 7b0c3a7..93bc237 100644 --- a/plugins/picker/picker.go +++ b/plugins/picker/picker.go @@ -34,10 +34,6 @@ func rollDie(sides int) int { // This function returns true if the plugin responds in a meaningful way to the users message. // Otherwise, the function returns false and the bot continues execution of other plugins. func (p *PickerPlugin) Message(message msg.Message) bool { - if !message.Command { - return false - } - body := message.Body pfx, sfx := "pick {", "}" @@ -46,7 +42,7 @@ func (p *PickerPlugin) Message(message msg.Message) bool { items := strings.Split(body, ",") item := items[rand.Intn(len(items))] - out := fmt.Sprintf("I've chosen \"%s\" for you.", item) + out := fmt.Sprintf("I've chosen \"%s\" for you.", strings.TrimSpace(item)) p.Bot.SendMessage(message.Channel, out) From e19d982fe5214e2d19117042b1b2f886d8d45c89 Mon Sep 17 00:00:00 2001 From: Chris Sexton Date: Thu, 4 Jan 2018 07:39:24 -0500 Subject: [PATCH 27/28] counter: add leaderboards --- plugins/counter/counter.go | 57 +++++++++++++++++++++++++++++++++++++- 1 file changed, 56 insertions(+), 1 deletion(-) diff --git a/plugins/counter/counter.go b/plugins/counter/counter.go index ed70349..b0e4828 100644 --- a/plugins/counter/counter.go +++ b/plugins/counter/counter.go @@ -45,6 +45,32 @@ func GetItems(db *sqlx.DB, nick string) ([]Item, error) { return items, nil } +func LeaderAll(db *sqlx.DB) ([]Item, error) { + s := `select id,item,nick,max(count) as count from counter group by item having count(nick) > 1 and max(count) > 1 order by count desc` + var items []Item + err := db.Select(&items, s) + if err != nil { + return nil, err + } + for i := range items { + items[i].DB = db + } + return items, nil +} + +func Leader(db *sqlx.DB, itemName string) ([]Item, error) { + s := `select * from counter where item=? order by count desc` + var items []Item + err := db.Select(&items, s, itemName) + if err != nil { + return nil, err + } + for i := range items { + items[i].DB = db + } + return items, nil +} + // GetItem returns a specific counter for a subject func GetItem(db *sqlx.DB, nick, itemName string) (Item, error) { var item Item @@ -136,7 +162,36 @@ func (p *CounterPlugin) Message(message msg.Message) bool { return false } - if tea, _ := regexp.MatchString("(?i)^tea\\. [^.]*\\. ((hot)|(iced))\\.?$", message.Body); tea { + if parts[0] == "leaderboard" { + var cmd func() ([]Item, error) + itNameTxt := "" + + if len(parts) == 1 { + cmd = func() ([]Item, error) { return LeaderAll(p.DB) } + } else { + itNameTxt = fmt.Sprintf(" for %s", parts[1]) + cmd = func() ([]Item, error) { return Leader(p.DB, parts[1]) } + } + + its, err := cmd() + if err != nil { + log.Println(err) + return false + } else if len(its) == 0 { + return false + } + + out := fmt.Sprintf("Leaderboard%s:\n", itNameTxt) + for _, it := range its { + out += fmt.Sprintf("%s with %d %s\n", + it.Nick, + it.Count, + it.Item, + ) + } + p.Bot.SendMessage(channel, out) + return true + } else if tea, _ := regexp.MatchString("(?i)^tea\\. [^.]*\\. ((hot)|(iced))\\.?$", message.Body); tea { item, err := GetItem(p.DB, nick, ":tea:") if err != nil { log.Printf("Error finding item %s.%s: %s.", nick, ":tea:", err) From a20c323ac91a778c57c2f24e5f41f9f1bfc086f1 Mon Sep 17 00:00:00 2001 From: Scott Kiesel Date: Wed, 10 Jan 2018 11:47:50 -0500 Subject: [PATCH 28/28] Fixed comment --- plugins/emojifyme/emojifyme.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/plugins/emojifyme/emojifyme.go b/plugins/emojifyme/emojifyme.go index 4c1383d..4ee9282 100644 --- a/plugins/emojifyme/emojifyme.go +++ b/plugins/emojifyme/emojifyme.go @@ -77,7 +77,7 @@ func (p *EmojifyMePlugin) Message(message msg.Message) bool { } tokens[i] = ":" + token + ":" } else if strings.HasSuffix(token, "s") { - //Check to see if we can strip the trailing "es" off and get an emoji + //Check to see if we can strip the trailing "s" off and get an emoji temp := strings.TrimSuffix(token, "s") if _, ok := p.Emoji[temp]; ok { if !stringsContain(inertTokens, temp) {