684 lines
16 KiB
Go
Raw Permalink Normal View History

2016-09-25 22:34:02 +02:00
package service
import (
"errors"
2016-09-25 22:34:02 +02:00
"fmt"
"log"
2017-07-21 14:04:16 +02:00
"regexp"
2016-09-27 22:05:44 +02:00
"strconv"
2017-07-22 14:36:55 +02:00
"strings"
2018-03-31 10:58:37 +02:00
"sync"
2016-09-27 22:05:44 +02:00
"time"
2016-09-25 22:34:02 +02:00
"github.com/nlopes/slack"
2017-07-21 14:04:16 +02:00
"github.com/erroneousboat/slack-term/components"
2017-07-21 14:04:16 +02:00
"github.com/erroneousboat/slack-term/config"
2016-09-25 22:34:02 +02:00
)
const (
ChannelTypeChannel = "channel"
ChannelTypeGroup = "group"
ChannelTypeIM = "im"
)
2016-09-25 22:34:02 +02:00
type SlackService struct {
2017-12-16 22:54:00 +01:00
Config *config.Config
2017-12-01 12:05:27 +01:00
Client *slack.Client
RTM *slack.RTM
SlackChannels []interface{}
Channels []components.ChannelItem
2017-12-01 12:05:27 +01:00
UserCache map[string]string
CurrentUserID string
CurrentUsername string
2016-09-25 22:34:02 +02:00
}
2016-10-02 16:07:35 +02:00
// NewSlackService is the constructor for the SlackService and will initialize
// the RTM and a Client
2017-12-16 22:54:00 +01:00
func NewSlackService(config *config.Config) (*SlackService, error) {
2016-09-29 21:19:09 +02:00
svc := &SlackService{
2017-12-16 22:54:00 +01:00
Config: config,
Client: slack.New(config.SlackToken),
2016-09-29 21:19:09 +02:00
UserCache: make(map[string]string),
}
2016-09-25 22:34:02 +02:00
// Get user associated with token, mainly
// used to identify user when new messages
// arrives
authTest, err := svc.Client.AuthTest()
if err != nil {
return nil, errors.New("not able to authorize client, check your connection and if your slack-token is set correctly")
}
svc.CurrentUserID = authTest.UserID
2016-09-25 22:34:02 +02:00
// Create RTM
svc.RTM = svc.Client.NewRTM()
2016-09-25 22:34:02 +02:00
go svc.RTM.ManageConnection()
2016-10-02 16:07:35 +02:00
// Creation of user cache this speeds up
// the uncovering of usernames of messages
2016-09-30 11:22:40 +02:00
users, _ := svc.Client.GetUsers()
for _, user := range users {
// only add non-deleted users
if !user.Deleted {
svc.UserCache[user.ID] = user.Name
}
2016-09-30 11:22:40 +02:00
}
2017-12-01 12:05:27 +01:00
// Get name of current user
currentUser, err := svc.Client.GetUserInfo(svc.CurrentUserID)
if err != nil {
svc.CurrentUsername = "slack-term"
}
svc.CurrentUsername = currentUser.Name
return svc, nil
2016-09-25 22:34:02 +02:00
}
2016-10-02 16:07:35 +02:00
// GetChannels will retrieve all available channels, groups, and im channels.
// Because the channels are of different types, we will append them to
// an []interface as well as to a []Channel which will give us easy access
// to the id and name of the Channel.
func (s *SlackService) GetChannels() []string {
var chans []components.ChannelItem
2016-09-25 22:34:02 +02:00
2018-03-31 10:58:37 +02:00
var wg sync.WaitGroup
// Channels
wg.Add(1)
var slackChans []slack.Channel
go func() {
var err error
slackChans, err = s.Client.GetChannels(true)
if err != nil {
chans = append(chans, components.ChannelItem{})
}
wg.Done()
}()
// Groups
wg.Add(1)
var slackGroups []slack.Group
go func() {
var err error
slackGroups, err = s.Client.GetGroups(true)
if err != nil {
chans = append(chans, components.ChannelItem{})
}
wg.Done()
}()
2018-03-31 10:58:37 +02:00
// IM
wg.Add(1)
var slackIM []slack.IM
go func() {
var err error
slackIM, err = s.Client.GetIMChannels()
if err != nil {
chans = append(chans, components.ChannelItem{})
}
wg.Done()
}()
wg.Wait()
// Channels
2016-09-30 23:52:26 +02:00
for _, chn := range slackChans {
if chn.IsMember {
s.SlackChannels = append(s.SlackChannels, chn)
chans = append(
chans, components.ChannelItem{
ID: chn.ID,
Name: chn.Name,
Topic: chn.Topic.Value,
Type: components.ChannelTypeChannel,
UserID: "",
StylePrefix: s.Config.Theme.Channel.Prefix,
StyleIcon: s.Config.Theme.Channel.Icon,
StyleText: s.Config.Theme.Channel.Text,
},
)
}
2016-09-30 23:52:26 +02:00
}
2016-09-25 22:34:02 +02:00
2016-09-30 23:52:26 +02:00
// Groups
2016-10-01 12:48:15 +02:00
for _, grp := range slackGroups {
s.SlackChannels = append(s.SlackChannels, grp)
chans = append(
chans, components.ChannelItem{
ID: grp.ID,
Name: grp.Name,
Topic: grp.Topic.Value,
Type: components.ChannelTypeGroup,
UserID: "",
StylePrefix: s.Config.Theme.Channel.Prefix,
StyleIcon: s.Config.Theme.Channel.Icon,
StyleText: s.Config.Theme.Channel.Text,
},
)
2016-10-01 12:48:15 +02:00
}
2016-09-30 23:52:26 +02:00
// IM
for _, im := range slackIM {
2016-10-01 12:48:15 +02:00
// Uncover name, when we can't uncover name for
// IM channel this is then probably a deleted
// user, because we won't add deleted users
// to the UserCache, so we skip it
2016-10-01 12:48:15 +02:00
name, ok := s.UserCache[im.User]
if ok {
chans = append(
chans,
components.ChannelItem{
ID: im.ID,
Name: name,
Topic: "",
Type: components.ChannelTypeIM,
UserID: im.User,
2018-03-31 10:58:37 +02:00
Presence: "",
StylePrefix: s.Config.Theme.Channel.Prefix,
StyleIcon: s.Config.Theme.Channel.Icon,
StyleText: s.Config.Theme.Channel.Text,
},
)
2016-10-09 16:23:44 +02:00
s.SlackChannels = append(s.SlackChannels, im)
2016-10-01 12:48:15 +02:00
}
2016-09-25 22:34:02 +02:00
}
2016-09-30 23:52:26 +02:00
s.Channels = chans
2018-03-31 10:58:37 +02:00
// We set presence of IM channels here because we need to separately
// issue an API call for every channel, this will speed up that process
s.SetPresenceChannels()
var channels []string
for _, chn := range s.Channels {
channels = append(channels, chn.ToString())
}
2018-03-31 10:58:37 +02:00
return channels
}
// ChannelsToString will relay the string representation for a channel
func (s *SlackService) ChannelsToString() []string {
var channels []string
for _, chn := range s.Channels {
channels = append(channels, chn.ToString())
}
return channels
}
2018-03-31 10:58:37 +02:00
// SetPresence will set presence for all IM channels
func (s *SlackService) SetPresenceChannels() {
var wg sync.WaitGroup
for i, channel := range s.SlackChannels {
switch channel := channel.(type) {
case slack.IM:
wg.Add(1)
go func(i int) {
presence, _ := s.GetUserPresence(channel.User)
s.Channels[i].Presence = presence
wg.Done()
}(i)
}
}
wg.Wait()
}
// SetPresenceChannelEvent will set the presence of a IM channel
func (s *SlackService) SetPresenceChannelEvent(userID string, presence string) {
// Get the correct Channel from svc.Channels
var index int
for i, channel := range s.Channels {
if userID == channel.UserID {
index = i
break
}
}
s.Channels[index].Presence = presence
2016-09-25 22:34:02 +02:00
}
2017-12-01 23:52:25 +01:00
// GetSlackChannel returns the representation of a slack channel
func (s *SlackService) GetSlackChannel(selectedChannel int) interface{} {
return s.SlackChannels[selectedChannel]
}
// GetUserPresence will get the presence of a specific user
func (s *SlackService) GetUserPresence(userID string) (string, error) {
presence, err := s.Client.GetUserPresence(userID)
if err != nil {
return "", err
}
return presence.Presence, nil
}
2016-10-29 23:59:16 +02:00
// SetChannelReadMark will set the read mark for a channel, group, and im
// channel based on the current time.
func (s *SlackService) SetChannelReadMark(channel interface{}) {
switch channel := channel.(type) {
case slack.Channel:
s.Client.SetChannelReadMark(
channel.ID, fmt.Sprintf("%f",
float64(time.Now().Unix())),
)
case slack.Group:
s.Client.SetGroupReadMark(
channel.ID, fmt.Sprintf("%f",
float64(time.Now().Unix())),
)
case slack.IM:
s.Client.MarkIMChannel(
channel.ID, fmt.Sprintf("%f",
float64(time.Now().Unix())),
)
}
}
// MarkAsRead will set the channel as read
func (s *SlackService) MarkAsRead(channelID int) {
channel := s.Channels[channelID]
if channel.Notification {
2017-12-18 21:53:27 +01:00
s.Channels[channelID].Notification = false
switch channel.Type {
case ChannelTypeChannel:
s.Client.SetChannelReadMark(
channel.ID, fmt.Sprintf("%f",
float64(time.Now().Unix())),
)
case ChannelTypeGroup:
s.Client.SetGroupReadMark(
channel.ID, fmt.Sprintf("%f",
float64(time.Now().Unix())),
)
case ChannelTypeIM:
s.Client.MarkIMChannel(
channel.ID, fmt.Sprintf("%f",
float64(time.Now().Unix())),
)
}
}
}
2018-04-01 13:03:28 +02:00
// FindChannel will loop over s.Channels to find the index where the
// channelID equals the ID
func (s *SlackService) FindChannel(channelID string) int {
2017-12-16 16:11:46 +01:00
var index int
for i, channel := range s.Channels {
if channel.ID == channelID {
index = i
break
}
}
2018-04-01 13:03:28 +02:00
return index
}
// MarkAsUnread will set the channel as unread
func (s *SlackService) MarkAsUnread(channelID string) {
index := s.FindChannel(channelID)
2017-12-16 16:11:46 +01:00
s.Channels[index].Notification = true
}
2018-04-01 13:03:28 +02:00
// GetChannelName will return the name for a specific channelID
func (s *SlackService) GetChannelName(channelID string) string {
index := s.FindChannel(channelID)
return s.Channels[index].Name
}
2016-10-29 23:59:16 +02:00
// SendMessage will send a message to a particular channel
func (s *SlackService) SendMessage(channelID int, message string) {
2016-09-27 22:05:44 +02:00
// https://godoc.org/github.com/nlopes/slack#PostMessageParameters
2016-09-28 22:10:04 +02:00
postParams := slack.PostMessageParameters{
2017-12-01 12:05:27 +01:00
AsUser: true,
Username: s.CurrentUsername,
2016-09-28 22:10:04 +02:00
}
2016-09-27 22:05:44 +02:00
// https://godoc.org/github.com/nlopes/slack#Client.PostMessage
s.Client.PostMessage(s.Channels[channelID].ID, message, postParams)
2016-09-27 22:05:44 +02:00
}
2016-09-25 22:34:02 +02:00
2016-10-29 23:59:16 +02:00
// GetMessages will get messages for a channel, group or im channel delimited
// by a count.
2017-12-03 21:43:33 +01:00
func (s *SlackService) GetMessages(channel interface{}, count int) []components.Message {
2016-09-30 23:52:26 +02:00
// https://api.slack.com/methods/channels.history
historyParams := slack.HistoryParameters{
Count: count,
Inclusive: false,
Unreads: false,
}
// https://godoc.org/github.com/nlopes/slack#History
history := new(slack.History)
var err error
switch chnType := channel.(type) {
case slack.Channel:
history, err = s.Client.GetChannelHistory(chnType.ID, historyParams)
if err != nil {
log.Fatal(err) // FIXME
}
case slack.Group:
history, err = s.Client.GetGroupHistory(chnType.ID, historyParams)
if err != nil {
log.Fatal(err) // FIXME
}
case slack.IM:
history, err = s.Client.GetIMHistory(chnType.ID, historyParams)
if err != nil {
log.Fatal(err) // FIXME
}
}
// Construct the messages
2017-12-03 21:43:33 +01:00
var messages []components.Message
2016-09-30 23:52:26 +02:00
for _, message := range history.Messages {
msg := s.CreateMessage(message)
2016-10-09 14:19:28 +02:00
messages = append(messages, msg...)
2016-09-25 22:34:02 +02:00
}
2016-10-11 09:33:56 +02:00
// Reverse the order of the messages, we want the newest in
// the last place
2017-12-03 21:43:33 +01:00
var messagesReversed []components.Message
2016-10-11 09:33:56 +02:00
for i := len(messages) - 1; i >= 0; i-- {
messagesReversed = append(messagesReversed, messages[i])
}
return messagesReversed
2016-09-25 22:34:02 +02:00
}
2016-09-29 21:19:09 +02:00
// CreateMessage will create a string formatted message that can be rendered
// in the Chat pane.
//
// [23:59] <erroneousboat> Hello world!
//
2016-10-09 14:19:28 +02:00
// This returns an array of string because we will try to uncover attachments
// associated with messages.
2017-12-03 21:43:33 +01:00
func (s *SlackService) CreateMessage(message slack.Message) []components.Message {
var msgs []components.Message
2016-09-29 21:19:09 +02:00
var name string
// Get username from cache
name, ok := s.UserCache[message.User]
// Name not in cache
if !ok {
2016-09-30 11:22:40 +02:00
if message.BotID != "" {
2016-09-29 21:19:09 +02:00
// Name not found, perhaps a bot, use Username
2016-09-30 11:22:40 +02:00
name, ok = s.UserCache[message.BotID]
if !ok {
// Not found in cache, add it
name = message.Username
s.UserCache[message.BotID] = message.Username
}
} else {
// Not a bot, not in cache, get user info
user, err := s.Client.GetUserInfo(message.User)
if err != nil {
name = "unknown"
s.UserCache[message.User] = name
} else {
name = user.Name
s.UserCache[message.User] = user.Name
2016-09-29 21:19:09 +02:00
}
}
}
if name == "" {
name = "unknown"
}
2016-10-09 14:19:28 +02:00
// When there are attachments append them
if len(message.Attachments) > 0 {
msgs = append(msgs, s.CreateMessageFromAttachments(message.Attachments)...)
2016-10-09 14:19:28 +02:00
}
2016-09-29 21:19:09 +02:00
// Parse time
floatTime, err := strconv.ParseFloat(message.Timestamp, 64)
if err != nil {
floatTime = 0.0
}
intTime := int64(floatTime)
// Format message
msg := components.Message{
Time: time.Unix(intTime, 0),
Name: name,
Content: parseMessage(s, message.Text),
StyleTime: s.Config.Theme.Message.Time,
StyleName: s.Config.Theme.Message.Name,
StyleText: s.Config.Theme.Message.Text,
}
2016-09-29 21:19:09 +02:00
2017-12-03 21:43:33 +01:00
msgs = append(msgs, msg)
2016-10-09 14:19:28 +02:00
2016-10-11 09:33:56 +02:00
return msgs
2016-09-29 21:19:09 +02:00
}
2018-03-24 13:56:55 +01:00
func (s *SlackService) CreateMessageFromMessageEvent(message *slack.MessageEvent) ([]components.Message, error) {
2016-09-29 21:19:09 +02:00
2017-12-03 21:43:33 +01:00
var msgs []components.Message
2016-09-29 21:19:09 +02:00
var name string
2018-03-24 13:56:55 +01:00
switch message.SubType {
case "message_changed":
// Append (edited) when an edited message is received
2016-10-30 15:38:13 +01:00
message = &slack.MessageEvent{Msg: *message.SubMessage}
message.Text = fmt.Sprintf("%s (edited)", message.Text)
2018-03-24 13:56:55 +01:00
case "message_replied":
// Ignore reply events
return nil, errors.New("ignoring reply events")
2016-10-30 15:38:13 +01:00
}
2016-09-29 21:19:09 +02:00
// Get username from cache
name, ok := s.UserCache[message.User]
// Name not in cache
if !ok {
2016-09-30 11:22:40 +02:00
if message.BotID != "" {
2016-09-29 21:19:09 +02:00
// Name not found, perhaps a bot, use Username
2016-09-30 11:22:40 +02:00
name, ok = s.UserCache[message.BotID]
if !ok {
// Not found in cache, add it
name = message.Username
s.UserCache[message.BotID] = message.Username
}
} else {
// Not a bot, not in cache, get user info
user, err := s.Client.GetUserInfo(message.User)
if err != nil {
name = "unknown"
s.UserCache[message.User] = name
} else {
name = user.Name
s.UserCache[message.User] = user.Name
2016-09-29 21:19:09 +02:00
}
}
}
if name == "" {
name = "unknown"
}
2016-10-09 14:19:28 +02:00
// When there are attachments append them
if len(message.Attachments) > 0 {
msgs = append(msgs, s.CreateMessageFromAttachments(message.Attachments)...)
2016-10-09 14:19:28 +02:00
}
2016-09-29 21:19:09 +02:00
// Parse time
floatTime, err := strconv.ParseFloat(message.Timestamp, 64)
if err != nil {
floatTime = 0.0
}
intTime := int64(floatTime)
// Format message
msg := components.Message{
Time: time.Unix(intTime, 0),
Name: name,
Content: parseMessage(s, message.Text),
StyleTime: s.Config.Theme.Message.Time,
StyleName: s.Config.Theme.Message.Name,
StyleText: s.Config.Theme.Message.Text,
}
2017-12-03 21:43:33 +01:00
msgs = append(msgs, msg)
2016-10-09 14:19:28 +02:00
2018-03-24 13:56:55 +01:00
return msgs, nil
2016-10-09 14:19:28 +02:00
}
// CheckNotifyMention check if the message event is either contains a
// mention or is posted on an IM channel
func (s *SlackService) CheckNotifyMention(ev *slack.MessageEvent) bool {
channel := s.Channels[s.FindChannel(ev.Channel)]
switch channel.Type {
case ChannelTypeIM:
return true
}
// Mentions have the following format:
// <@U12345|erroneousboat>
// <@U12345>
r := regexp.MustCompile(`\<@(\w+\|*\w+)\>`)
matches := r.FindAllString(ev.Text, -1)
for _, match := range matches {
if strings.Contains(match, s.CurrentUserID) {
return true
}
}
return false
}
2018-04-01 13:03:28 +02:00
func (s *SlackService) CreateNotifyMessage(channelID string) string {
channel := s.Channels[s.FindChannel(channelID)]
switch channel.Type {
case ChannelTypeChannel:
return fmt.Sprintf("Message received on channel: %s", channel.Name)
case ChannelTypeGroup:
return fmt.Sprintf("Message received in group: %s", channel.Name)
case ChannelTypeIM:
return fmt.Sprintf("Message received from: %s", channel.Name)
}
return ""
}
2017-07-22 14:36:55 +02:00
// parseMessage will parse a message string and find and replace:
// - emoji's
// - mentions
2017-07-22 14:36:55 +02:00
func parseMessage(s *SlackService, msg string) string {
// NOTE: Commented out because rendering of the emoji's
// creates artifacts from the last view because of
// double width emoji's
// msg = parseEmoji(msg)
2017-07-22 14:36:55 +02:00
msg = parseMentions(s, msg)
return msg
}
// parseMentions will try to find mention placeholders in the message
// string and replace them with the correct username with and @ symbol
//
// Mentions have the following format:
// <@U12345|erroneousboat>
// <@U12345>
2017-07-22 14:36:55 +02:00
func parseMentions(s *SlackService, msg string) string {
r := regexp.MustCompile(`\<@(\w+\|*\w+)\>`)
return r.ReplaceAllStringFunc(
msg, func(str string) string {
2017-12-01 13:24:02 +01:00
rs := r.FindStringSubmatch(str)
if len(rs) < 1 {
return str
}
2017-07-22 14:36:55 +02:00
var userID string
split := strings.Split(rs[1], "|")
if len(split) > 0 {
userID = split[0]
} else {
userID = rs[1]
}
name, ok := s.UserCache[userID]
if !ok {
user, err := s.Client.GetUserInfo(userID)
if err != nil {
name = "unknown"
s.UserCache[userID] = name
} else {
name = user.Name
s.UserCache[userID] = user.Name
}
}
if name == "" {
name = "unknown"
}
return "@" + name
},
)
}
// parseEmoji will try to find emoji placeholders in the message
// string and replace them with the correct unicode equivalent
func parseEmoji(msg string) string {
r := regexp.MustCompile("(:\\w+:)")
return r.ReplaceAllStringFunc(
msg, func(str string) string {
code, ok := config.EmojiCodemap[str]
if !ok {
return str
}
return code
},
)
}
// CreateMessageFromAttachments will construct a array of string of the Field
2016-10-09 14:19:28 +02:00
// values of Attachments from a Message.
func (s *SlackService) CreateMessageFromAttachments(atts []slack.Attachment) []components.Message {
2017-12-03 21:43:33 +01:00
var msgs []components.Message
2016-10-09 14:19:28 +02:00
for _, att := range atts {
for i := len(att.Fields) - 1; i >= 0; i-- {
2017-12-03 21:43:33 +01:00
msgs = append(msgs, components.Message{
Content: fmt.Sprintf(
2016-10-09 14:19:28 +02:00
"%s %s",
att.Fields[i].Title,
att.Fields[i].Value,
),
StyleTime: s.Config.Theme.Message.Time,
StyleName: s.Config.Theme.Message.Name,
StyleText: s.Config.Theme.Message.Text,
2017-12-03 21:43:33 +01:00
},
2016-10-09 14:19:28 +02:00
)
}
if att.Text != "" {
2017-12-03 21:43:33 +01:00
msgs = append(
msgs,
components.Message{Content: fmt.Sprintf("%s", att.Text)},
)
}
if att.Title != "" {
2017-12-03 21:43:33 +01:00
msgs = append(
msgs,
components.Message{Content: fmt.Sprintf("%s", att.Title)},
)
}
2016-10-09 14:19:28 +02:00
}
2016-10-09 14:19:28 +02:00
return msgs
2016-09-29 21:19:09 +02:00
}