2019-05-12 22:37:49 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2021-07-13 15:36:16 -04:00
|
|
|
module DiscourseChatIntegration::Provider::SlackProvider
|
2017-08-03 10:48:33 -04:00
|
|
|
class SlackTranscript
|
2022-03-15 09:32:28 -04:00
|
|
|
class UserFetchError < RuntimeError; end
|
|
|
|
|
2017-08-15 10:44:51 -04:00
|
|
|
attr_reader :users, :channel_id, :messages
|
2017-08-03 10:48:33 -04:00
|
|
|
|
2018-04-07 22:16:36 -04:00
|
|
|
def initialize(channel_name:, channel_id:, requested_thread_ts: nil)
|
2017-08-15 03:26:03 -04:00
|
|
|
@channel_name = channel_name
|
2017-08-15 10:44:51 -04:00
|
|
|
@channel_id = channel_id
|
2018-04-07 22:16:36 -04:00
|
|
|
@requested_thread_ts = requested_thread_ts
|
2017-08-03 19:47:04 -04:00
|
|
|
|
2017-08-15 03:26:03 -04:00
|
|
|
@first_message_index = 0
|
|
|
|
@last_message_index = -1 # We can use negative array indicies to select the last message - fancy!
|
|
|
|
end
|
2017-08-03 19:47:04 -04:00
|
|
|
|
2017-08-15 03:26:03 -04:00
|
|
|
def set_first_message_by_ts(ts)
|
|
|
|
message_index = @messages.find_index { |m| m.ts == ts }
|
|
|
|
@first_message_index = message_index if message_index
|
|
|
|
end
|
2017-08-03 19:47:04 -04:00
|
|
|
|
2017-08-15 03:26:03 -04:00
|
|
|
def set_last_message_by_ts(ts)
|
|
|
|
message_index = @messages.find_index { |m| m.ts == ts }
|
|
|
|
@last_message_index = message_index if message_index
|
|
|
|
end
|
2017-08-03 19:47:04 -04:00
|
|
|
|
2017-08-15 03:26:03 -04:00
|
|
|
def set_first_message_by_index(val)
|
|
|
|
@first_message_index = val if @messages[val]
|
|
|
|
end
|
2017-08-03 10:48:33 -04:00
|
|
|
|
2017-08-15 03:26:03 -04:00
|
|
|
def set_last_message_by_index(val)
|
|
|
|
@last_message_index = val if @messages[val]
|
|
|
|
end
|
2017-08-03 19:47:04 -04:00
|
|
|
|
2017-08-15 11:19:24 -04:00
|
|
|
# Apply a heuristic to decide which is the first message in the current conversation
|
|
|
|
def guess_first_message(skip_messages: 5) # Can skip the last n messages
|
2018-04-07 22:16:36 -04:00
|
|
|
return true if @requested_thread_ts # Always start thread on first message
|
2019-08-07 16:59:34 -04:00
|
|
|
return false if @messages.blank? || @messages.size < skip_messages
|
2017-08-15 11:19:24 -04:00
|
|
|
|
|
|
|
possible_first_messages = @messages[0..-skip_messages]
|
|
|
|
|
|
|
|
# Work through the messages in order. If a gap is found, this could be the first message
|
|
|
|
new_first_message_index = nil
|
|
|
|
previous_message_ts = @messages[-skip_messages].ts.split('.').first.to_i
|
|
|
|
possible_first_messages.each_with_index do |message, index|
|
|
|
|
|
|
|
|
# Calculate the time since the last message
|
|
|
|
this_ts = message.ts.split('.').first.to_i
|
|
|
|
time_since_previous_message = this_ts - previous_message_ts
|
|
|
|
|
|
|
|
# If greater than 3 minutes, this could be the first message
|
|
|
|
if time_since_previous_message > 3.minutes
|
|
|
|
new_first_message_index = index
|
|
|
|
end
|
|
|
|
|
|
|
|
previous_message_ts = this_ts
|
|
|
|
end
|
|
|
|
|
|
|
|
if new_first_message_index
|
|
|
|
@first_message_index = new_first_message_index
|
2017-10-10 00:02:27 -04:00
|
|
|
true
|
2017-08-15 11:19:24 -04:00
|
|
|
else
|
2017-10-10 00:02:27 -04:00
|
|
|
false
|
2017-08-15 11:19:24 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2017-08-15 03:26:03 -04:00
|
|
|
def first_message
|
2017-10-10 00:02:27 -04:00
|
|
|
@messages[@first_message_index]
|
2017-08-15 03:26:03 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def last_message
|
2017-10-10 00:02:27 -04:00
|
|
|
@messages[@last_message_index]
|
2017-08-15 03:26:03 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
# These two methods convert potentially negative array indices into positive ones
|
|
|
|
def first_message_number
|
2017-10-10 00:02:27 -04:00
|
|
|
@first_message_index < 0 ? @messages.length + @first_message_index : @first_message_index
|
2017-08-15 03:26:03 -04:00
|
|
|
end
|
|
|
|
def last_message_number
|
2017-10-10 00:02:27 -04:00
|
|
|
@last_message_index < 0 ? @messages.length + @last_message_index : @last_message_index
|
2017-08-03 10:48:33 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def build_transcript
|
2019-05-12 22:37:49 -04:00
|
|
|
post_content = +"[quote]\n"
|
2017-08-15 03:26:03 -04:00
|
|
|
post_content << "[**#{I18n.t('chat_integration.provider.slack.transcript.view_on_slack', name: @channel_name)}**](#{first_message.url})\n"
|
2017-08-03 10:48:33 -04:00
|
|
|
|
|
|
|
all_avatars = {}
|
|
|
|
|
|
|
|
last_username = ''
|
|
|
|
|
2017-08-03 19:47:04 -04:00
|
|
|
transcript_messages = @messages[@first_message_index..@last_message_index]
|
|
|
|
|
|
|
|
transcript_messages.each do |m|
|
2017-08-03 10:48:33 -04:00
|
|
|
same_user = m.username == last_username
|
|
|
|
last_username = m.username
|
|
|
|
|
|
|
|
unless same_user
|
|
|
|
if avatar = m.avatar
|
|
|
|
all_avatars[m.username] ||= avatar
|
|
|
|
end
|
|
|
|
|
|
|
|
post_content << "\n"
|
|
|
|
post_content << "![#{m.username}] " if m.avatar
|
|
|
|
post_content << "**@#{m.username}:** "
|
|
|
|
end
|
|
|
|
|
|
|
|
post_content << m.text
|
|
|
|
|
|
|
|
m.attachments.each do |attachment|
|
|
|
|
post_content << "\n> #{attachment}\n"
|
|
|
|
end
|
|
|
|
|
|
|
|
post_content << "\n"
|
|
|
|
end
|
|
|
|
|
|
|
|
post_content << "[/quote]\n\n"
|
|
|
|
|
|
|
|
all_avatars.each do |username, url|
|
|
|
|
post_content << "[#{username}]: #{url}\n"
|
|
|
|
end
|
|
|
|
|
2020-06-15 11:45:25 -04:00
|
|
|
if not @requested_thread_ts.nil?
|
|
|
|
post_content << "<!--SLACK_CHANNEL_ID=#{@channel_id};SLACK_TS=#{@requested_thread_ts}-->"
|
|
|
|
end
|
|
|
|
|
2017-10-10 00:02:27 -04:00
|
|
|
post_content
|
2017-08-03 10:48:33 -04:00
|
|
|
end
|
|
|
|
|
2021-04-22 13:50:11 -04:00
|
|
|
def build_modal_ui
|
|
|
|
data = {
|
|
|
|
type: "modal",
|
|
|
|
title: {
|
|
|
|
type: "plain_text",
|
|
|
|
text: I18n.t("chat_integration.provider.slack.transcript.modal_title")
|
|
|
|
},
|
|
|
|
blocks: [
|
|
|
|
{
|
|
|
|
"type": "section",
|
|
|
|
"text": {
|
|
|
|
"type": "mrkdwn",
|
|
|
|
"text": I18n.t("chat_integration.provider.slack.transcript.modal_description")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
|
|
|
|
|
|
|
if @messages
|
|
|
|
post_content = build_transcript
|
2021-07-13 15:36:16 -04:00
|
|
|
secret = DiscourseChatIntegration::Helper.save_transcript(post_content)
|
2021-04-22 13:50:11 -04:00
|
|
|
link = "#{Discourse.base_url}/chat-transcript/#{secret}"
|
|
|
|
|
|
|
|
data[:blocks] << {
|
|
|
|
"type": "section",
|
|
|
|
"text": {
|
|
|
|
"type": "mrkdwn",
|
|
|
|
"text": ":writing_hand: *#{I18n.t("chat_integration.provider.slack.transcript.transcript_ready")}*"
|
|
|
|
},
|
|
|
|
"accessory": {
|
|
|
|
"type": "button",
|
|
|
|
"text": {
|
|
|
|
"type": "plain_text",
|
|
|
|
"text": I18n.t("chat_integration.provider.slack.transcript.continue_on_discourse"),
|
|
|
|
"emoji": true
|
|
|
|
},
|
|
|
|
"style": "primary",
|
|
|
|
"url": link,
|
|
|
|
"action_id": "null_action"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
data[:blocks] << {
|
|
|
|
"type": "section",
|
|
|
|
"text": {
|
|
|
|
"type": "mrkdwn",
|
|
|
|
"text": ":writing_hand: #{I18n.t("chat_integration.provider.slack.transcript.loading")}"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
end
|
|
|
|
|
|
|
|
data
|
|
|
|
end
|
|
|
|
|
2017-08-03 12:24:49 -04:00
|
|
|
def build_slack_ui
|
|
|
|
post_content = build_transcript
|
2021-07-13 15:36:16 -04:00
|
|
|
secret = DiscourseChatIntegration::Helper.save_transcript(post_content)
|
2017-08-03 12:24:49 -04:00
|
|
|
link = "#{Discourse.base_url}/chat-transcript/#{secret}"
|
|
|
|
|
2018-04-07 22:16:36 -04:00
|
|
|
return { text: "<#{link}|#{I18n.t("chat_integration.provider.slack.transcript.post_to_discourse")}>" } if @requested_thread_ts
|
|
|
|
|
2017-10-10 00:02:27 -04:00
|
|
|
{
|
|
|
|
text: "<#{link}|#{I18n.t("chat_integration.provider.slack.transcript.post_to_discourse")}>",
|
|
|
|
attachments: [
|
|
|
|
{
|
|
|
|
pretext: I18n.t(
|
|
|
|
"chat_integration.provider.slack.transcript.first_message_pretext",
|
|
|
|
n: @messages.length - first_message_number
|
|
|
|
),
|
|
|
|
fallback: "#{first_message.username} - #{first_message.raw_text}",
|
|
|
|
color: "#007AB8",
|
|
|
|
author_name: first_message.username,
|
|
|
|
author_icon: first_message.avatar,
|
|
|
|
text: first_message.raw_text,
|
|
|
|
footer: I18n.t(
|
|
|
|
"chat_integration.provider.slack.transcript.posted_in",
|
|
|
|
name: @channel_name
|
|
|
|
),
|
|
|
|
ts: first_message.ts,
|
|
|
|
callback_id: last_message.ts,
|
|
|
|
actions: [
|
|
|
|
{
|
|
|
|
name: "first_message",
|
|
|
|
text: I18n.t(
|
|
|
|
"chat_integration.provider.slack.transcript.change_first_message"
|
|
|
|
),
|
|
|
|
type: "select",
|
|
|
|
options: first_message_options = @messages[ [(first_message_number - 20), 0].max .. last_message_number]
|
2017-11-27 03:07:54 -05:00
|
|
|
.map { |m| { text: "#{m.username}: #{m.processed_text_with_attachments}", value: m.ts } }
|
2017-10-10 00:02:27 -04:00
|
|
|
}
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
pretext: I18n.t(
|
|
|
|
"chat_integration.provider.slack.transcript.last_message_pretext",
|
|
|
|
n: @messages.length - last_message_number
|
|
|
|
),
|
|
|
|
fallback: "#{last_message.username} - #{last_message.raw_text}",
|
|
|
|
color: "#007AB8",
|
|
|
|
author_name: last_message.username,
|
|
|
|
author_icon: last_message.avatar,
|
|
|
|
text: last_message.raw_text,
|
|
|
|
footer: I18n.t(
|
|
|
|
"chat_integration.provider.slack.transcript.posted_in",
|
|
|
|
name: @channel_name
|
|
|
|
),
|
|
|
|
ts: last_message.ts,
|
|
|
|
callback_id: first_message.ts,
|
|
|
|
actions: [
|
|
|
|
{
|
|
|
|
name: "last_message",
|
|
|
|
text: I18n.t(
|
|
|
|
"chat_integration.provider.slack.transcript.change_last_message"
|
|
|
|
),
|
|
|
|
type: "select",
|
|
|
|
options: @messages[first_message_number..(last_message_number + 20)]
|
2017-11-27 03:07:54 -05:00
|
|
|
.map { |m| { text: "#{m.username}: #{m.processed_text_with_attachments}", value: m.ts } }
|
2017-10-10 00:02:27 -04:00
|
|
|
}
|
|
|
|
],
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
2017-08-03 12:24:49 -04:00
|
|
|
end
|
|
|
|
|
2017-08-15 03:26:03 -04:00
|
|
|
def load_user_data
|
2022-03-15 09:32:28 -04:00
|
|
|
key = "slack_user_info_#{Digest::SHA1.hexdigest(SiteSetting.chat_integration_slack_access_token)}"
|
|
|
|
@users = Discourse.cache.fetch(key, expires_in: 10.minutes) do
|
|
|
|
fetch_user_data
|
|
|
|
end
|
|
|
|
true
|
|
|
|
rescue UserFetchError
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
|
|
|
def fetch_user_data
|
2021-07-13 15:36:16 -04:00
|
|
|
http = ::DiscourseChatIntegration::Provider::SlackProvider.slack_api_http
|
2017-08-03 12:24:49 -04:00
|
|
|
|
2018-12-13 20:03:49 -05:00
|
|
|
cursor = nil
|
2017-08-03 12:24:49 -04:00
|
|
|
req = Net::HTTP::Post.new(URI('https://slack.com/api/users.list'))
|
2018-12-13 20:03:49 -05:00
|
|
|
|
2022-03-15 09:32:28 -04:00
|
|
|
users = {}
|
2018-12-13 20:03:49 -05:00
|
|
|
loop do
|
|
|
|
break if cursor == ""
|
|
|
|
req.set_form_data(token: SiteSetting.chat_integration_slack_access_token, limit: 200, cursor: cursor)
|
|
|
|
response = http.request(req)
|
2022-03-15 09:32:28 -04:00
|
|
|
raise UserFetchError.new unless response.kind_of? Net::HTTPSuccess
|
2020-09-25 15:08:28 -04:00
|
|
|
json = JSON.parse(response.body)
|
2022-03-15 09:32:28 -04:00
|
|
|
raise UserFetchError.new unless json['ok']
|
2018-12-13 20:03:49 -05:00
|
|
|
cursor = json['response_metadata']['next_cursor']
|
2019-07-04 14:50:07 -04:00
|
|
|
json['members'].each do |user|
|
|
|
|
# Slack uses display_name and falls back to real_name if it is not set
|
|
|
|
if user['profile']['display_name'].blank?
|
|
|
|
user['_transcript_username'] = user['profile']['real_name']
|
|
|
|
else
|
|
|
|
user['_transcript_username'] = user['profile']['display_name']
|
|
|
|
end
|
|
|
|
user['_transcript_username'] = user['_transcript_username'].gsub(' ', '_')
|
2022-03-15 09:32:28 -04:00
|
|
|
users[user['id']] = user
|
2019-07-04 14:50:07 -04:00
|
|
|
end
|
2018-12-13 20:03:49 -05:00
|
|
|
end
|
2022-03-15 09:32:28 -04:00
|
|
|
users
|
2017-08-03 12:24:49 -04:00
|
|
|
end
|
|
|
|
|
2017-08-15 10:44:51 -04:00
|
|
|
def load_chat_history(count: 500)
|
2021-07-13 15:36:16 -04:00
|
|
|
http = DiscourseChatIntegration::Provider::SlackProvider.slack_api_http
|
2017-08-03 12:24:49 -04:00
|
|
|
|
2018-04-07 22:16:36 -04:00
|
|
|
endpoint = @requested_thread_ts ? "replies" : "history"
|
|
|
|
|
|
|
|
req = Net::HTTP::Post.new(URI("https://slack.com/api/conversations.#{endpoint}"))
|
2017-08-03 12:24:49 -04:00
|
|
|
|
|
|
|
data = {
|
|
|
|
token: SiteSetting.chat_integration_slack_access_token,
|
2017-08-15 10:44:51 -04:00
|
|
|
channel: @channel_id,
|
2018-04-07 21:27:49 -04:00
|
|
|
limit: count
|
2017-08-03 12:24:49 -04:00
|
|
|
}
|
|
|
|
|
2018-04-07 22:16:36 -04:00
|
|
|
data[:ts] = @requested_thread_ts if @requested_thread_ts
|
|
|
|
|
2017-08-03 12:24:49 -04:00
|
|
|
req.set_form_data(data)
|
|
|
|
response = http.request(req)
|
|
|
|
return false unless response.kind_of? Net::HTTPSuccess
|
2020-09-25 15:08:28 -04:00
|
|
|
json = JSON.parse(response.body)
|
2017-08-03 12:24:49 -04:00
|
|
|
return false unless json['ok']
|
|
|
|
|
2018-04-07 22:16:36 -04:00
|
|
|
raw_messages = json['messages']
|
|
|
|
raw_messages = raw_messages.reverse unless @requested_thread_ts
|
2017-08-03 12:24:49 -04:00
|
|
|
|
2017-08-15 03:26:03 -04:00
|
|
|
# Build some message objects
|
|
|
|
@messages = []
|
|
|
|
raw_messages.each_with_index do |message, index|
|
2017-11-27 03:07:25 -05:00
|
|
|
# Only load messages
|
2017-08-15 03:26:03 -04:00
|
|
|
next unless message["type"] == "message"
|
2018-04-07 22:16:36 -04:00
|
|
|
|
|
|
|
# Don't load responses to threads unless specifically requested (if ts==thread_ts then it's the thread parent)
|
|
|
|
next if !@requested_thread_ts && message["thread_ts"] && message["thread_ts"] != message["ts"]
|
2017-11-27 03:07:25 -05:00
|
|
|
|
2017-08-15 03:26:03 -04:00
|
|
|
this_message = SlackMessage.new(message, self)
|
|
|
|
@messages << this_message
|
|
|
|
end
|
2017-08-03 19:47:04 -04:00
|
|
|
|
2017-08-03 12:24:49 -04:00
|
|
|
end
|
2017-08-03 10:48:33 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
end
|