package service import ( "fmt" "log" "strconv" "time" "github.com/nlopes/slack" ) type SlackService struct { Client *slack.Client RTM *slack.RTM SlackChannels []interface{} Channels []Channel UserCache map[string]string CurrentUserID string } type Channel struct { ID string Name string } // NewSlackService is the constructor for the SlackService and will initialize // the RTM and a Client func NewSlackService(token string) *SlackService { svc := &SlackService{ Client: slack.New(token), UserCache: make(map[string]string), } svc.RTM = svc.Client.NewRTM() go svc.RTM.ManageConnection() // Creation of user cache this speeds up // the uncovering of usernames of messages users, _ := svc.Client.GetUsers() for _, user := range users { // only add non-deleted users if !user.Deleted { svc.UserCache[user.ID] = user.Name } } // Get user associated with token, mainly // used to identify user when new messages // arrives authTest, err := svc.Client.AuthTest() if err != nil { log.Fatal("Client.AuthTest() failed") } svc.CurrentUserID = authTest.UserID return svc } // 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() []Channel { var chans []Channel // Channel slackChans, err := s.Client.GetChannels(true) if err != nil { chans = append(chans, Channel{}) } for _, chn := range slackChans { s.SlackChannels = append(s.SlackChannels, chn) chans = append(chans, Channel{chn.ID, chn.Name}) } // Groups slackGroups, err := s.Client.GetGroups(true) if err != nil { chans = append(chans, Channel{}) } for _, grp := range slackGroups { s.SlackChannels = append(s.SlackChannels, grp) chans = append(chans, Channel{grp.ID, grp.Name}) } // IM slackIM, err := s.Client.GetIMChannels() if err != nil { chans = append(chans, Channel{}) } for _, im := range slackIM { // Uncover name, when we can't uncover name for // IM channel this is then probably a deleted // user, because we wont add deleted users // to the UserCache, so we skip it name, ok := s.UserCache[im.User] if ok { chans = append(chans, Channel{im.ID, name}) s.SlackChannels = append(s.SlackChannels, im) } } s.Channels = chans return chans } func (s *SlackService) SendMessage(channel string, message string) { // https://godoc.org/github.com/nlopes/slack#PostMessageParameters postParams := slack.PostMessageParameters{ AsUser: true, } // https://godoc.org/github.com/nlopes/slack#Client.PostMessage s.Client.PostMessage(channel, message, postParams) } func (s *SlackService) GetMessages(channel interface{}, count int) []string { // 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: // TODO: json: cannot unmarshal number into Go value of type string 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 var messages []string for _, message := range history.Messages { msg := s.CreateMessage(message) messages = append(messages, msg...) } // Reverse the order of the messages, we want the newest in // the last place var messagesReversed []string for i := len(messages) - 1; i >= 0; i-- { messagesReversed = append(messagesReversed, messages[i]) } return messagesReversed } // CreateMessage will create a string formatted message that can be rendered // in the Chat pane. // // [23:59] Hello world! // // This returns an array of string because we will try to uncover attachments // associated with messages. func (s *SlackService) CreateMessage(message slack.Message) []string { var msgs []string var name string // Get username from cache name, ok := s.UserCache[message.User] // Name not in cache if !ok { if message.BotID != "" { // Name not found, perhaps a bot, use Username 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 } } } if name == "" { name = "unknown" } // When there are attachments append them if len(message.Attachments) > 0 { msgs = append(msgs, createMessageFromAttachments(message.Attachments)...) } // Parse time floatTime, err := strconv.ParseFloat(message.Timestamp, 64) if err != nil { floatTime = 0.0 } intTime := int64(floatTime) // Format message msg := fmt.Sprintf( "[%s] <%s> %s", time.Unix(intTime, 0).Format("15:04"), name, message.Text, ) msgs = append(msgs, msg) return msgs } func (s *SlackService) CreateMessageFromMessageEvent(message *slack.MessageEvent) []string { var msgs []string var name string // Get username from cache name, ok := s.UserCache[message.User] // Name not in cache if !ok { if message.BotID != "" { // Name not found, perhaps a bot, use Username 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 } } } if name == "" { name = "unknown" } // When there are attachments append them if len(message.Attachments) > 0 { msgs = append(msgs, createMessageFromAttachments(message.Attachments)...) } // Parse time floatTime, err := strconv.ParseFloat(message.Timestamp, 64) if err != nil { floatTime = 0.0 } intTime := int64(floatTime) // Format message msg := fmt.Sprintf( "[%s] <%s> %s", time.Unix(intTime, 0).Format("15:04"), name, message.Text, ) msgs = append(msgs, msg) return msgs } // createMessageFromAttachments will construct a array of string of the Field // values of Attachments from a Message. func createMessageFromAttachments(atts []slack.Attachment) []string { var msgs []string for _, att := range atts { for i := len(att.Fields) - 1; i >= 0; i-- { msgs = append(msgs, fmt.Sprintf( "%s %s", att.Fields[i].Title, att.Fields[i].Value, ), ) } } return msgs }