Mercurial > rsstweet
annotate app/lib/legit_client.rb @ 267:5326ed3fa7c3
Adjust api
author | nanaya <me@nanaya.net> |
---|---|
date | Thu, 17 Apr 2025 20:08:20 +0900 |
parents | 4f86037f6e6a |
children |
rev | line source |
---|---|
260 | 1 require "net/http" |
259 | 2 |
264 | 3 class LegitClient |
4 def initialize | |
5 @id, @headers = $cfg[:headers].sample | |
234 | 6 end |
7 | |
8 def self.escape_param(param) | |
9 CGI.escape JSON.dump(param) | |
10 end | |
11 | |
12 def self.normalize_entity_media(json) | |
13 ret = {} | |
14 | |
15 json.each do |entity_media| | |
16 val = {} | |
17 | |
260 | 18 case entity_media["type"] |
19 when "animated_gif", "video" | |
20 val[:variants] = entity_media["video_info"]["variants"] | |
21 .filter { |variant| variant["bitrate"].present? } | |
253
d726e8b92dd1
Support animated gif (same as video)
nanaya <me@nanaya.net>
parents:
252
diff
changeset
|
22 .map do |variant| |
d726e8b92dd1
Support animated gif (same as video)
nanaya <me@nanaya.net>
parents:
252
diff
changeset
|
23 { |
260 | 24 bitrate: variant["bitrate"], |
25 url: variant["url"] | |
253
d726e8b92dd1
Support animated gif (same as video)
nanaya <me@nanaya.net>
parents:
252
diff
changeset
|
26 } |
d726e8b92dd1
Support animated gif (same as video)
nanaya <me@nanaya.net>
parents:
252
diff
changeset
|
27 end |
260 | 28 when "photo" |
29 val[:image_url] = entity_media["media_url_https"].sub(/\.([^.]+)$/, '?format=\1') | |
234 | 30 end |
31 | |
32 if !val.empty? | |
260 | 33 val[:url] = entity_media["expanded_url"] |
34 val[:type] = entity_media["type"] | |
35 val[:id] = entity_media["media_key"] | |
234 | 36 end |
37 | |
260 | 38 key = if ret[entity_media["url"]].nil? |
39 entity_media["url"] | |
237
961d362e42c7
The url in entity media isn't unique as they all point to the same thing
nanaya <me@nanaya.net>
parents:
236
diff
changeset
|
40 else |
260 | 41 entity_media["media_key"] |
237
961d362e42c7
The url in entity media isn't unique as they all point to the same thing
nanaya <me@nanaya.net>
parents:
236
diff
changeset
|
42 end |
961d362e42c7
The url in entity media isn't unique as they all point to the same thing
nanaya <me@nanaya.net>
parents:
236
diff
changeset
|
43 |
961d362e42c7
The url in entity media isn't unique as they all point to the same thing
nanaya <me@nanaya.net>
parents:
236
diff
changeset
|
44 ret[key] = val |
234 | 45 end |
46 | |
47 ret | |
48 end | |
49 | |
50 def self.normalize_entity_urls(json) | |
51 ret = {} | |
52 | |
240
c454ea4f7b34
Add support for note tweets (with no formatting)
nanaya <me@nanaya.net>
parents:
238
diff
changeset
|
53 (json || {}).each do |entity_url| |
260 | 54 ret[entity_url["url"]] = entity_url["expanded_url"] |
234 | 55 end |
56 | |
57 ret | |
58 end | |
59 | |
238
a04b4830eef2
Filter out non-own tweets included for replies
nanaya <me@nanaya.net>
parents:
237
diff
changeset
|
60 def self.normalize_timeline(json, user_id) |
252 | 61 json |
62 .reduce([]) do |acc, instruction| | |
260 | 63 case instruction["type"] |
64 when "TimelineAddEntries" then acc += instruction["entries"] | |
65 when "TimelinePinEntry" then acc << instruction["entry"] | |
252 | 66 end |
67 | |
68 acc | |
260 | 69 end.filter { |entry| entry["entryId"] =~ /\A(profile-conversation|tweet)-/ } |
234 | 70 .reduce([]) do |acc, entry| |
260 | 71 if entry["content"]["entryType"] == "TimelineTimelineItem" |
72 acc.push(entry["content"]) | |
234 | 73 else |
260 | 74 entry["content"]["items"].each do |item| |
75 acc.push(item["item"]) | |
234 | 76 end |
77 end | |
78 acc | |
260 | 79 end.map { |raw_tweet| normalize_tweet(raw_tweet["itemContent"]["tweet_results"]["result"]) } |
238
a04b4830eef2
Filter out non-own tweets included for replies
nanaya <me@nanaya.net>
parents:
237
diff
changeset
|
80 .filter { |tweet| !tweet.nil? && tweet.dig(:user, :id) == user_id } |
234 | 81 end |
82 | |
83 def self.normalize_tweet(json) | |
84 return nil if json.nil? | |
85 | |
260 | 86 return normalize_tweet(json["tweet"]) if json["__typename"] == "TweetWithVisibilityResults" |
234 | 87 |
88 { | |
260 | 89 id: json["rest_id"], |
90 created_at: Time.parse(json["legacy"]["created_at"]), | |
91 user: normalize_user(json["core"]["user_results"]["result"]), | |
92 message: json.dig("note_tweet", "note_tweet_results", "result", "text") || json["legacy"]["full_text"], | |
93 retweet: normalize_tweet(json.dig("legacy", "retweeted_status_result", "result")), | |
94 quote: normalize_tweet(json.dig("quoted_status_result", "result")), | |
95 quote_id: json["legacy"]["quoted_status_id_str"], | |
96 reply_to_id: json["legacy"]["in_reply_to_status_id_str"], | |
97 reply_to_user_id: json["legacy"]["in_reply_to_user_id_str"], | |
98 reply_to_username: json["legacy"]["in_reply_to_screen_name"], | |
99 entity_urls: { **normalize_entity_urls(json["legacy"]["entities"]["urls"]), **normalize_entity_urls(json.dig("note_tweet", "note_tweet_results", "result", "entity_set", "urls")) }, | |
100 entity_media: normalize_entity_media(json.dig("legacy", "extended_entities", "media") || []) | |
234 | 101 } |
102 end | |
103 | |
104 def self.normalize_user(json) | |
105 { | |
260 | 106 avatar_url: json["legacy"]["profile_image_url_https"], |
107 id: json["rest_id"], | |
108 name: json["legacy"]["name"], | |
109 protected: json["legacy"]["protected"] == true, | |
110 username: json["legacy"]["screen_name"] | |
234 | 111 } |
112 end | |
243
bc2f45058c9e
Prevent caching of rate limited error and combine response handling
nanaya <me@nanaya.net>
parents:
241
diff
changeset
|
113 |
264 | 114 def timeline(user_id) |
267 | 115 resp = fetch("https://x.com/i/api/graphql/pz0IHaV_t7T4HJavqqqcIA/UserTweetsAndReplies?variables=%7B%22userId%22%3A#{self.class.escape_param user_id}%2C%22count%22%3A50%2C%22includePromotedContent%22%3Atrue%2C%22withCommunity%22%3Atrue%2C%22withVoice%22%3Atrue%7D&features=%7B%22rweb_video_screen_enabled%22%3Afalse%2C%22profile_label_improvements_pcf_label_in_post_enabled%22%3Atrue%2C%22rweb_tipjar_consumption_enabled%22%3Atrue%2C%22responsive_web_graphql_exclude_directive_enabled%22%3Atrue%2C%22verified_phone_label_enabled%22%3Afalse%2C%22creator_subscriptions_tweet_preview_api_enabled%22%3Atrue%2C%22responsive_web_graphql_timeline_navigation_enabled%22%3Atrue%2C%22responsive_web_graphql_skip_user_profile_image_extensions_enabled%22%3Afalse%2C%22premium_content_api_read_enabled%22%3Afalse%2C%22communities_web_enable_tweet_community_results_fetch%22%3Atrue%2C%22c9s_tweet_anatomy_moderator_badge_enabled%22%3Atrue%2C%22responsive_web_grok_analyze_button_fetch_trends_enabled%22%3Afalse%2C%22responsive_web_grok_analyze_post_followups_enabled%22%3Atrue%2C%22responsive_web_jetfuel_frame%22%3Afalse%2C%22responsive_web_grok_share_attachment_enabled%22%3Atrue%2C%22articles_preview_enabled%22%3Atrue%2C%22responsive_web_edit_tweet_api_enabled%22%3Atrue%2C%22graphql_is_translatable_rweb_tweet_is_translatable_enabled%22%3Atrue%2C%22view_counts_everywhere_api_enabled%22%3Atrue%2C%22longform_notetweets_consumption_enabled%22%3Atrue%2C%22responsive_web_twitter_article_tweet_consumption_enabled%22%3Atrue%2C%22tweet_awards_web_tipping_enabled%22%3Afalse%2C%22responsive_web_grok_show_grok_translated_post%22%3Afalse%2C%22responsive_web_grok_analysis_button_from_backend%22%3Atrue%2C%22creator_subscriptions_quote_tweet_preview_enabled%22%3Afalse%2C%22freedom_of_speech_not_reach_fetch_enabled%22%3Atrue%2C%22standardized_nudges_misinfo%22%3Atrue%2C%22tweet_with_visibility_results_prefer_gql_limited_actions_policy_enabled%22%3Atrue%2C%22longform_notetweets_rich_text_read_enabled%22%3Atrue%2C%22longform_notetweets_inline_media_enabled%22%3Atrue%2C%22responsive_web_grok_image_annotation_enabled%22%3Atrue%2C%22responsive_web_enhance_cards_enabled%22%3Afalse%7D&fieldToggles=%7B%22withArticlePlainText%22%3Afalse%7D", true) |
264 | 116 |
117 handle_response resp, :timeline, "timeline(#{user_id})", ->(json) do | |
267 | 118 self.class.normalize_timeline json["data"]["user"]["result"]["timeline"]["timeline"]["instructions"], user_id |
264 | 119 end |
120 end | |
121 | |
122 def user_by_id(user_id) | |
267 | 123 # obtain by going to /i/user/#{user_id} |
124 resp = fetch("https://x.com/i/api/graphql/5vdJ5sWkbSRDiiNZvwc2Yg/UserByRestId?variables=%7B%22userId%22%3A#{self.class.escape_param user_id}%7D&features=%7B%22hidden_profile_subscriptions_enabled%22%3Atrue%2C%22profile_label_improvements_pcf_label_in_post_enabled%22%3Atrue%2C%22rweb_tipjar_consumption_enabled%22%3Atrue%2C%22responsive_web_graphql_exclude_directive_enabled%22%3Atrue%2C%22verified_phone_label_enabled%22%3Afalse%2C%22highlights_tweets_tab_ui_enabled%22%3Atrue%2C%22responsive_web_twitter_article_notes_tab_enabled%22%3Atrue%2C%22subscriptions_feature_can_gift_premium%22%3Atrue%2C%22creator_subscriptions_tweet_preview_api_enabled%22%3Atrue%2C%22responsive_web_graphql_skip_user_profile_image_extensions_enabled%22%3Afalse%2C%22responsive_web_graphql_timeline_navigation_enabled%22%3Atrue%7D") | |
264 | 125 |
126 handle_response resp, :user, "user_by_id(#{user_id})", ->(json) do | |
127 self.class.normalize_user json["data"]["user"]["result"] | |
128 end | |
129 end | |
130 | |
131 def user_by_username(username) | |
267 | 132 resp = fetch("https://x.com/i/api/graphql/32pL5BWe9WKeSK1MoPvFQQ/UserByScreenName?variables=%7B%22screen_name%22%3A#{self.class.escape_param username}%7D&features=%7B%22hidden_profile_subscriptions_enabled%22%3Atrue%2C%22profile_label_improvements_pcf_label_in_post_enabled%22%3Atrue%2C%22rweb_tipjar_consumption_enabled%22%3Atrue%2C%22responsive_web_graphql_exclude_directive_enabled%22%3Atrue%2C%22verified_phone_label_enabled%22%3Afalse%2C%22subscriptions_verification_info_is_identity_verified_enabled%22%3Atrue%2C%22subscriptions_verification_info_verified_since_enabled%22%3Atrue%2C%22highlights_tweets_tab_ui_enabled%22%3Atrue%2C%22responsive_web_twitter_article_notes_tab_enabled%22%3Atrue%2C%22subscriptions_feature_can_gift_premium%22%3Atrue%2C%22creator_subscriptions_tweet_preview_api_enabled%22%3Atrue%2C%22responsive_web_graphql_skip_user_profile_image_extensions_enabled%22%3Afalse%2C%22responsive_web_graphql_timeline_navigation_enabled%22%3Atrue%7D&fieldToggles=%7B%22withAuxiliaryUserLabels%22%3Atrue%7D") |
264 | 133 |
134 handle_response resp, :user, "user_by_username(#{username})", ->(json) do | |
135 self.class.normalize_user json["data"]["user"]["result"] | |
136 end | |
137 end | |
138 | |
267 | 139 private def fetch(uri, with_tid = false) |
140 headers = if with_tid then @headers else @headers.except('x-client-transaction-id') end | |
141 | |
142 Net::HTTP.get(URI(uri), headers) | |
264 | 143 end |
144 | |
145 private def handle_response(resp, key, error_key, callback) | |
146 json = JSON.parse(resp) | |
147 { | |
148 key => callback.call(json), | |
149 raw: resp | |
150 } | |
151 rescue => e | |
152 if json.is_a? Hash | |
153 if json["errors"].is_a? Array | |
154 return rate_limit_check(json) | |
155 elsif json["data"].is_a? Hash | |
156 return | |
157 end | |
158 end | |
159 Rails.logger.error("#{error_key} fail: #{resp}") | |
160 | |
161 raise e | |
162 end | |
163 | |
164 private def rate_limit_check(json) | |
260 | 165 return unless json["errors"].any? { |err| err["code"] == 88 } |
243
bc2f45058c9e
Prevent caching of rate limited error and combine response handling
nanaya <me@nanaya.net>
parents:
241
diff
changeset
|
166 |
264 | 167 raise "Rate limited! Client: #{@id}" |
243
bc2f45058c9e
Prevent caching of rate limited error and combine response handling
nanaya <me@nanaya.net>
parents:
241
diff
changeset
|
168 end |
234 | 169 end |