require 'cgi' require 'time' require 'json' require 'slack' require 'ruboty/adapters/base' module Ruboty module Adapters class SlackRTM < Base env :SLACK_TOKEN, "Account's token. get one on https://api.slack.com/web#basics" env :SLACK_EXPOSE_CHANNEL_NAME, "if this set to 1, message.to will be channel name instead of id", optional: true env :SLACK_IGNORE_BOT_MESSAGE, "If this set to 1, bot ignores bot_messages", optional: true env :SLACK_IGNORE_GENERAL, "if this set to 1, bot ignores all messages on #general channel", optional: true env :SLACK_GENERAL_NAME, "Set general channel name if your Slack changes general name", optional: true env :SLACK_AUTO_RECONNECT, "Enable auto reconnect", optional: true def run init bind connect end def say(message) channel = message[:to] if channel[0] == '#' channel = resolve_channel_id(channel[1..-1]) end return unless channel if message[:attachments] && !message[:attachments].empty? client.chat_postMessage( channel: channel, text: message[:code] ? "```\n#{message[:body]}\n```" : message[:body], parse: message[:parse] || 'full', unfurl_links: true, as_user: true, attachments: message[:attachments].to_json ) else realtime.send_message( type: 'message', channel: channel, text: message[:code] ? "```\n#{message[:body]}\n```" : resolve_send_mention(message[:body]), mrkdwn: true ) end end def add_reaction(reaction, channel_id, timestamp) client.reactions_add(name: reaction, channel: channel_id, timestamp: timestamp) end private def init response = client.auth_test @user_info_caches = {} @channel_info_caches = {} @usergroup_info_caches = {} ENV['RUBOTY_NAME'] ||= response['user'] make_users_cache make_channels_cache make_usergroups_cache end def bind realtime.on_text do |data| method_name = "on_#{data['type']}".to_sym send(method_name, data) if respond_to?(method_name, true) end end def connect Thread.start do loop do sleep 5 set_active end end loop do realtime.main_loop rescue nil break unless ENV['SLACK_AUTO_RECONNECT'] @url = null @realtime = null end end def url @url ||= begin response = Net::HTTP.post_form(URI.parse('https://slack.com/api/rtm.connect'), token: ENV['SLACK_TOKEN']) body = JSON.parse(response.body) URI.parse(body['url']) end end def client @client ||= ::Slack::Client.new(token: ENV['SLACK_TOKEN']) end def realtime @realtime ||= ::Ruboty::SlackRTM::Client.new(websocket_url: url) end def expose_channel_name? if @expose_channel_name.nil? @expose_channel_name = ENV['SLACK_EXPOSE_CHANNEL_NAME'] == '1' else @expose_channel_name end end def set_active client.users_setActive end # event handlers def on_message(data) user = user_info(data['user']) || {} channel = channel_info(data['channel']) if (data['subtype'] == 'bot_message' || user['is_bot']) && ENV['SLACK_IGNORE_BOT_MESSAGE'] == '1' return end if channel return if channel['name'] == (ENV['SLACK_GENERAL_NAME'] || 'general') && ENV['SLACK_IGNORE_GENERAL'] == '1' channel_to = expose_channel_name? ? "##{channel['name']}" : channel['id'] else # direct message channel_to = data['channel'] end message_info = { from: data['channel'], from_name: user['name'], to: channel_to, channel: channel, user: user, time: Time.at(data['ts'].to_f) } text, mention_to = extract_mention(data['text']) robot.receive(message_info.merge(body: text, mention_to: mention_to)) (data['attachments'] || []).each do |attachment| body, body_mention_to = extract_mention(attachment['fallback'] || "#{attachment['text']} #{attachment['pretext']}".strip) unless body.empty? robot.receive(message_info.merge(body: body, mention_to: body_mention_to)) end end end def on_channel_change(data) make_channels_cache end alias_method :on_channel_deleted, :on_channel_change alias_method :on_channel_renamed, :on_channel_change alias_method :on_channel_archived, :on_channel_change alias_method :on_channel_unarchived, :on_channel_change def on_user_change(data) user = data['user'] || data['bot'] @user_info_caches[user['id']] = user end alias_method :on_bot_added, :on_user_change alias_method :on_bot_changed, :on_user_change def extract_mention(text) mention_to = [] text = (text || '').gsub(/\<\@(?[0-9A-Z]+)(?:\|(?[^>]+))?\>/) do |_| name = Regexp.last_match[:name] unless name user = user_info(Regexp.last_match[:uid]) mention_to << user name = user['name'] end "@#{name}" end text.gsub!(/\[0-9A-Z]+)(?:\|(?[^>]+))?\>/) do |_| handle = Regexp.last_match[:handle] unless handle handle = usergroup_info(Regexp.last_match[:usergroup_id]) end "#{handle}" end text.gsub!(/\[^>|@]+)(\|\@[^>]+)?\>/) do |_| "@#{Regexp.last_match[:special]}" end text.gsub!(/\<((?[^>|]+)(?:\|(?[^>]*))?)\>/) do |_| Regexp.last_match[:ref] || Regexp.last_match[:link] end text.gsub!(/\#(?[A-Z0-9]+)/) do |_| room_id = Regexp.last_match[:room_id] msg = "##{room_id}" if channel = channel_info(room_id) msg = "##{channel['name']}" end msg end [CGI.unescapeHTML(text), mention_to] end def resolve_send_mention(text) text = text.to_s text.gsub!(/@(?[0-9a-z._-]+)/) do |_| mention = Regexp.last_match[:mention] msg = "@#{mention}" @user_info_caches.each_pair do |id, user| if user['name'].downcase == mention.downcase msg = "<@#{id}>" end end msg end text.gsub!(/@(?(?:everyone|group|channel|here))/) do |_| "" end text.gsub!(/@(?[0-9a-z._-]+)/) do |_| subteam_name = Regexp.last_match[:subteam_name] msg = "@#{subteam_name}" @usergroup_info_caches.each_pair do |id, usergroup| if usergroup && usergroup['handle'] == subteam_name msg = "" end end msg end text.gsub!(/\#(?[a-z0-9_-]+)/) do |_| room_id = Regexp.last_match[:room_id] msg = "##{room_id}" @channel_info_caches.each_pair do |id, channel| if channel && channel['name'] == room_id msg = "<##{id}|#{room_id}>" end end msg end text end def make_users_cache resp = client.users_list if resp['ok'] resp['members'].each do |user| @user_info_caches[user['id']] = user end end end def make_channels_cache resp = client.channels_list if resp['ok'] resp['channels'].each do |channel| @channel_info_caches[channel['id']] = channel end end end def make_usergroups_cache resp = client.get("usergroups.list") if resp['ok'] resp['usergroups'].each do |usergroup| @usergroup_info_caches[usergroup['id']] = usergroup end end end def user_info(user_id) return {} if user_id.to_s.empty? @user_info_caches[user_id] ||= begin resp = client.users_info(user: user_id) resp['user'] end end def channel_info(channel_id) @channel_info_caches[channel_id] ||= begin resp = case channel_id when /^C/ client.channels_info(channel: channel_id) else {} end resp['channel'] end end def resolve_channel_id(name) ret_id = nil @channel_info_caches.each_pair do |id, channel| if channel['name'] == name ret_id = id break end end return ret_id end def usergroup_info(usergroup_id) @usergroup_info_caches[usergroup_id] || begin make_usergroups_cache @usergroup_info_caches[usergroup_id] end end end end end