2019-05-02 18:17:27 -04:00
|
|
|
# frozen_string_literal: true
|
|
|
|
|
2013-10-10 20:46:20 -04:00
|
|
|
# lightweight Twitter api calls
|
|
|
|
class TwitterApi
|
|
|
|
class << self
|
2022-08-21 13:26:24 -04:00
|
|
|
BASE_URL = "https://api.twitter.com"
|
2023-06-22 13:39:02 -04:00
|
|
|
URL_PARAMS = %w[
|
|
|
|
tweet.fields=id,author_id,text,created_at,entities,referenced_tweets,public_metrics
|
|
|
|
user.fields=id,name,username,profile_image_url
|
|
|
|
media.fields=type,height,width,variants,preview_image_url,url
|
|
|
|
expansions=attachments.media_keys,referenced_tweets.id.author_id
|
|
|
|
]
|
2022-08-21 13:26:24 -04:00
|
|
|
|
2013-10-10 20:46:20 -04:00
|
|
|
def prettify_tweet(tweet)
|
2023-06-22 13:39:02 -04:00
|
|
|
text = tweet[:data][:text].dup.to_s
|
|
|
|
if (entities = tweet[:data][:entities]) && (urls = entities[:urls])
|
2013-10-10 20:46:20 -04:00
|
|
|
urls.each do |url|
|
2023-07-03 18:53:12 -04:00
|
|
|
if !url[:display_url].start_with?("pic.twitter.com")
|
|
|
|
text.gsub!(
|
|
|
|
url[:url],
|
|
|
|
"<a target='_blank' href='#{url[:expanded_url]}'>#{url[:display_url]}</a>",
|
|
|
|
)
|
|
|
|
else
|
|
|
|
text.gsub!(url[:url], "")
|
|
|
|
end
|
2013-10-10 20:46:20 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
text = link_hashtags_in link_handles_in text
|
2016-05-11 16:11:26 -04:00
|
|
|
result = Rinku.auto_link(text, :all, 'target="_blank"').to_s
|
|
|
|
|
2023-06-22 13:39:02 -04:00
|
|
|
if tweet[:includes] && media = tweet[:includes][:media]
|
2016-05-11 16:11:26 -04:00
|
|
|
media.each do |m|
|
2023-06-22 13:39:02 -04:00
|
|
|
if m[:type] == "photo"
|
|
|
|
result << "<div class='tweet-images'><img class='tweet-image' src='#{m[:url]}' width='#{m[:width]}' height='#{m[:height]}'></div>"
|
|
|
|
elsif m[:type] == "video" || m[:type] == "animated_gif"
|
2020-08-17 15:37:36 -04:00
|
|
|
video_to_display =
|
2023-06-22 13:39:02 -04:00
|
|
|
m[:variants]
|
|
|
|
.select { |v| v[:content_type] == "video/mp4" }
|
|
|
|
.sort { |v| v[:bit_rate] }
|
2020-08-17 15:37:36 -04:00
|
|
|
.last # choose highest bitrate
|
|
|
|
|
2023-06-22 13:39:02 -04:00
|
|
|
if video_to_display && url = video_to_display[:url]
|
|
|
|
width = m[:width]
|
|
|
|
height = m[:height]
|
2020-08-17 15:37:36 -04:00
|
|
|
|
2020-08-17 18:56:41 -04:00
|
|
|
attributes =
|
2023-06-22 13:39:02 -04:00
|
|
|
if m[:type] == "animated_gif"
|
2020-08-17 18:56:41 -04:00
|
|
|
%w[playsinline loop muted autoplay disableRemotePlayback disablePictureInPicture]
|
|
|
|
else
|
|
|
|
%w[controls playsinline]
|
|
|
|
end.join(" ")
|
|
|
|
|
2020-08-17 15:37:36 -04:00
|
|
|
result << <<~HTML
|
|
|
|
<div class='tweet-images'>
|
|
|
|
<div class='aspect-image-full-size' style='--aspect-ratio:#{width}/#{height};'>
|
2020-08-17 18:56:41 -04:00
|
|
|
<video class='tweet-video' #{attributes}
|
2020-10-30 10:04:29 -04:00
|
|
|
width='#{width}'
|
2020-08-17 15:37:36 -04:00
|
|
|
height='#{height}'
|
2023-06-22 13:39:02 -04:00
|
|
|
poster='#{m[:preview_image_url]}'>
|
2020-08-17 15:37:36 -04:00
|
|
|
<source src='#{url}' type="video/mp4">
|
|
|
|
</video>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
HTML
|
2017-03-17 16:49:29 -04:00
|
|
|
end
|
2016-05-11 16:11:26 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
result
|
2013-10-10 20:46:20 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def tweet_for(id)
|
2022-08-21 13:26:24 -04:00
|
|
|
JSON.parse(twitter_get(tweet_uri_for(id)))
|
2013-10-10 20:46:20 -04:00
|
|
|
end
|
2014-01-28 15:51:29 -05:00
|
|
|
alias_method :status, :tweet_for
|
|
|
|
|
2013-10-10 20:46:20 -04:00
|
|
|
def twitter_credentials_missing?
|
|
|
|
consumer_key.blank? || consumer_secret.blank?
|
|
|
|
end
|
|
|
|
|
|
|
|
protected
|
|
|
|
|
|
|
|
def link_handles_in(text)
|
2022-02-09 06:54:02 -05:00
|
|
|
text
|
|
|
|
.gsub(/(?:^|\s)@\w+/) do |match|
|
2022-06-10 01:20:26 -04:00
|
|
|
whitespace = match[0] == " " ? " " : ""
|
|
|
|
handle = match.strip[1..]
|
|
|
|
"#{whitespace}<a href='https://twitter.com/#{handle}' target='_blank'>@#{handle}</a>"
|
2022-02-09 06:54:02 -05:00
|
|
|
end
|
|
|
|
.strip
|
2013-10-10 20:46:20 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def link_hashtags_in(text)
|
2022-02-09 06:54:02 -05:00
|
|
|
text
|
|
|
|
.gsub(/(?:^|\s)#\w+/) do |match|
|
2022-06-10 01:20:26 -04:00
|
|
|
whitespace = match[0] == " " ? " " : ""
|
|
|
|
hashtag = match.strip[1..]
|
|
|
|
"#{whitespace}<a href='https://twitter.com/search?q=%23#{hashtag}' target='_blank'>##{hashtag}</a>"
|
2022-02-09 06:54:02 -05:00
|
|
|
end
|
|
|
|
.strip
|
2013-10-10 20:46:20 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def tweet_uri_for(id)
|
2023-06-22 13:39:02 -04:00
|
|
|
URI.parse "#{BASE_URL}/2/tweets/#{id}?#{URL_PARAMS.join("&")}"
|
2013-10-10 20:46:20 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def twitter_get(uri)
|
|
|
|
request = Net::HTTP::Get.new(uri)
|
|
|
|
request.add_field "Authorization", "Bearer #{bearer_token}"
|
2022-08-17 11:32:48 -04:00
|
|
|
response = http(uri).request(request)
|
|
|
|
|
|
|
|
if response.kind_of?(Net::HTTPTooManyRequests)
|
|
|
|
Rails.logger.warn("Twitter API rate limit has been reached")
|
|
|
|
end
|
|
|
|
|
|
|
|
response.body
|
2013-10-10 20:46:20 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def authorization
|
|
|
|
request = Net::HTTP::Post.new(auth_uri)
|
|
|
|
|
|
|
|
request.add_field "Authorization", "Basic #{bearer_token_credentials}"
|
|
|
|
request.add_field "Content-Type", "application/x-www-form-urlencoded;charset=UTF-8"
|
|
|
|
|
|
|
|
request.set_form_data "grant_type" => "client_credentials"
|
|
|
|
|
|
|
|
http(auth_uri).request(request).body
|
|
|
|
end
|
|
|
|
|
|
|
|
def bearer_token
|
|
|
|
@access_token ||= JSON.parse(authorization).fetch("access_token")
|
|
|
|
end
|
|
|
|
|
|
|
|
def bearer_token_credentials
|
|
|
|
Base64.strict_encode64(
|
2019-12-11 23:54:26 -05:00
|
|
|
"#{UrlHelper.encode_component(consumer_key)}:#{UrlHelper.encode_component(consumer_secret)}",
|
2013-10-10 20:46:20 -04:00
|
|
|
)
|
|
|
|
end
|
|
|
|
|
|
|
|
def auth_uri
|
|
|
|
URI.parse "#{BASE_URL}/oauth2/token"
|
|
|
|
end
|
|
|
|
|
|
|
|
def http(uri)
|
|
|
|
Net::HTTP.new(uri.host, uri.port).tap { |http| http.use_ssl = true }
|
|
|
|
end
|
|
|
|
|
|
|
|
def consumer_key
|
|
|
|
SiteSetting.twitter_consumer_key
|
|
|
|
end
|
|
|
|
|
|
|
|
def consumer_secret
|
|
|
|
SiteSetting.twitter_consumer_secret
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|