Mercurial > rsstweet
changeset 245:2935c5e52a71
No more official twitter api
author | nanaya <me@nanaya.net> |
---|---|
date | Mon, 17 Jul 2023 04:23:09 +0900 |
parents | 0f0cc55ff11b (current diff) ebb65a26f070 (diff) |
children | 1cf8291962a2 |
files | |
diffstat | 13 files changed, 245 insertions(+), 252 deletions(-) [+] |
line wrap: on
line diff
--- a/Gemfile Fri Jul 14 01:45:40 2023 +0900 +++ b/Gemfile Mon Jul 17 04:23:09 2023 +0900 @@ -8,7 +8,6 @@ gem "redis" -gem "twitter" gem "twitter-text" gem "newrelic_rpm"
--- a/Gemfile.lock Fri Jul 14 01:45:40 2023 +0900 +++ b/Gemfile.lock Mon Jul 17 04:23:09 2023 +0900 @@ -19,46 +19,20 @@ i18n (>= 1.6, < 2) minitest (>= 5.1) tzinfo (~> 2.0) - addressable (2.8.1) - public_suffix (>= 2.0.2, < 6.0) - buftok (0.2.0) builder (3.2.4) concurrent-ruby (1.1.10) connection_pool (2.3.0) crass (1.0.6) - domain_name (0.5.20190701) - unf (>= 0.0.5, < 1.0.0) - equalizer (0.0.11) erubi (1.12.0) - ffi (1.15.5) - ffi-compiler (1.0.1) - ffi (>= 1.0.0) - rake - http (4.4.1) - addressable (~> 2.3) - http-cookie (~> 1.0) - http-form_data (~> 2.2) - http-parser (~> 1.2.0) - http-cookie (1.0.5) - domain_name (~> 0.5) - http-form_data (2.3.0) - http-parser (1.2.3) - ffi-compiler (>= 1.0, < 2.0) - http_parser.rb (0.6.0) - http_parser.rb (0.6.0-java) i18n (1.12.0) concurrent-ruby (~> 1.0) idn-ruby (0.1.5) loofah (2.19.1) crass (~> 1.0.2) nokogiri (>= 1.5.9) - memoizable (0.4.2) - thread_safe (~> 0.3, >= 0.3.1) method_source (1.0.0) mini_portile2 (2.8.1) minitest (5.17.0) - multipart-post (2.2.3) - naught (1.1.0) newrelic_rpm (8.15.0) nio4r (2.5.8) nio4r (2.5.8-java) @@ -69,7 +43,6 @@ racc (~> 1.4) nokogiri (1.14.0-x86-mingw32) racc (~> 1.4) - public_suffix (5.0.1) puma (6.0.2) nio4r (~> 2.0) puma (6.0.2-java) @@ -96,21 +69,7 @@ redis-client (>= 0.9.0) redis-client (0.12.0) connection_pool - simple_oauth (0.3.1) thor (1.2.1) - thread_safe (0.3.6) - thread_safe (0.3.6-java) - twitter (7.0.0) - addressable (~> 2.3) - buftok (~> 0.2.0) - equalizer (~> 0.0.11) - http (~> 4.0) - http-form_data (~> 2.0) - http_parser.rb (~> 0.6.0) - memoizable (~> 0.4.0) - multipart-post (~> 2.0) - naught (~> 1.0) - simple_oauth (~> 0.3.0) twitter-text (3.1.0) idn-ruby unf (~> 0.1.0) @@ -134,7 +93,6 @@ puma railties (~> 7.0.1) redis - twitter twitter-text BUNDLED WITH
--- a/app/controllers/tweets_controller.rb Fri Jul 14 01:45:40 2023 +0900 +++ b/app/controllers/tweets_controller.rb Mon Jul 17 04:23:09 2023 +0900 @@ -1,39 +1,43 @@ class TweetsController < ApplicationController def index - return redirect if params[:id].present? || params[:name].present? + return redirect if params[:name].present? end def show return redirect if params[:id][/\D/].present? - client = Tweet.new(params[:id].to_i) - @user = client.user + @user = CachedFetch.user_by_id params[:id] + + if @user.nil? + head :not_found + return + end + + if @user[:protected] + head :forbidden + return + end return redirect if normalized_screen_name != params[:name] - @tweets = client.timeline - rescue Twitter::Error::Forbidden - head :forbidden - rescue Twitter::Error::NotFound - head :not_found - rescue Twitter::Error::Unauthorized - head :forbidden + @tweets = CachedFetch.timeline params[:id] + + head :not_found if @tweets.nil? end def redirect - @user ||= Tweet.new(params[:id].presence || params[:name]).user - redirect_to tweet_path(@user.id, normalized_screen_name) - rescue Twitter::Error::Forbidden - head :forbidden - rescue Twitter::Error::NotFound - head :not_found - rescue Twitter::Error::Unauthorized - head :forbidden + @user ||= CachedFetch.user_by_username(params[:name]) + + if @user.nil? + head :not_found + else + redirect_to tweet_path(@user[:id], normalized_screen_name) + end end private def normalized_screen_name - @user.screen_name.presence || '_' + @user[:username].presence || '_' end end
--- a/app/helpers/application_helper.rb Fri Jul 14 01:45:40 2023 +0900 +++ b/app/helpers/application_helper.rb Mon Jul 17 04:23:09 2023 +0900 @@ -5,27 +5,23 @@ "tag:rsstweet@nanaya.pro,2014:#{id}" end - def expand_url(text, *urls) - urls.flatten! + def expand_url(text, urls) + text.gsub /https?:\/\/t\.co\/[A-Za-z0-9]+/ do |url| + expanded = urls[url] - urls = urls.reduce({}) do |result, u| - if u.try(:[], :url) - result[u[:url]] = u[:expanded_url] + case expanded + when nil then url + when Hash then expanded[:url] + else expanded end - - result - end - - text.gsub /https?:\/\/t\.co\/[A-Za-z0-9]+/ do |url| - urls[url] || url end end def status_url(tweet) - status_url_base tweet.user.screen_name, tweet.id + status_url_base tweet[:user][:username], tweet[:id] end - def status_url_base(screen_name, tweet_id) - "https://twitter.com/#{screen_name.presence || '_'}/status/#{tweet_id}" + def status_url_base(username, id) + "https://twitter.com/#{username.presence || '_'}/status/#{id}" end end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/app/lib/cached_fetch.rb Mon Jul 17 04:23:09 2023 +0900 @@ -0,0 +1,17 @@ +module CachedFetch + def self.timeline(user_id) + cached("timeline:#{user_id}") { LegitClient.timeline(user_id)&.[](:timeline) } + end + + def self.user_by_id(user_id) + cached("user_by_id:#{user_id}") { LegitClient.user_by_id(user_id)&.[](:user) } + end + + def self.user_by_username(username) + cached("user_by_username:#{username}") { LegitClient.user_by_username(username)&.[](:user) } + end + + def self.cached(key, &block) + Rails.cache.fetch(key, expires_in: (15 + rand(60)).minutes, &block) + end +end
--- a/app/lib/clients.rb Fri Jul 14 01:45:40 2023 +0900 +++ /dev/null Thu Jan 01 00:00:00 1970 +0000 @@ -1,23 +0,0 @@ -class Clients - def self.client_options(id) - { - :timeouts => { - :connect => 5, - :read => 5, - :write => 5, - }, - }.merge $cfg[:twitter][id] - end - - def self.instance - @@instance ||= self.new - end - - def initialize - @clients = {} - end - - def get(id) - @clients[id] ||= Twitter::REST::Client.new(self.class.client_options id) - end -end
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/app/lib/legit_client.rb Mon Jul 17 04:23:09 2023 +0900 @@ -0,0 +1,152 @@ +module LegitClient + def self.timeline(user_id) + resp = fetch("https://twitter.com/i/api/graphql/1-5o8Qhfc2kWlu_2rWNcug/UserTweetsAndReplies?variables=%7B%22userId%22%3A#{escape_param user_id}%2C%22count%22%3A50%2C%22includePromotedContent%22%3Atrue%2C%22withCommunity%22%3Atrue%2C%22withVoice%22%3Atrue%2C%22withV2Timeline%22%3Atrue%7D&features=%7B%22rweb_lists_timeline_redesign_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%22tweetypie_unmention_optimization_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%3Afalse%2C%22tweet_awards_web_tipping_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_media_download_video_enabled%22%3Afalse%2C%22responsive_web_enhance_cards_enabled%22%3Afalse%7D&fieldToggles=%7B%22withAuxiliaryUserLabels%22%3Afalse%2C%22withArticleRichContentState%22%3Afalse%7D") + + handle_response resp, :timeline, "timeline(#{user_id})", ->(json) do + normalize_timeline json['data']['user']['result']['timeline_v2']['timeline']['instructions'], user_id + end + end + + def self.user_by_id(user_id) + resp = fetch("https://twitter.com/i/api/graphql/i_0UQ54YrCyqLUvgGzXygA/UserByRestId?variables=%7B%22userId%22%3A#{escape_param user_id}%2C%22withSafetyModeUserFields%22%3Atrue%7D&features=%7B%22hidden_profile_likes_enabled%22%3Afalse%2C%22hidden_profile_subscriptions_enabled%22%3Afalse%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%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%3Afalse%7D") + + handle_response resp, :user, "user_by_id(#{user_id})", ->(json) do + normalize_user json['data']['user']['result'] + end + end + + def self.user_by_username(username) + resp = fetch("https://twitter.com/i/api/graphql/xc8f1g7BYqr6VTzTbvNlGw/UserByScreenName?variables=%7B%22screen_name%22%3A#{escape_param username}%2C%22withSafetyModeUserFields%22%3Atrue%7D&features=%7B%22hidden_profile_likes_enabled%22%3Afalse%2C%22hidden_profile_subscriptions_enabled%22%3Afalse%2C%22responsive_web_graphql_exclude_directive_enabled%22%3Atrue%2C%22verified_phone_label_enabled%22%3Afalse%2C%22subscriptions_verification_info_verified_since_enabled%22%3Atrue%2C%22highlights_tweets_tab_ui_enabled%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%3Afalse%7D") + + handle_response resp, :user, "user_by_username(#{username})", ->(json) do + normalize_user json['data']['user']['result'] + end + end + + def self.escape_param(param) + CGI.escape JSON.dump(param) + end + + def self.fetch(uri) + Net::HTTP.get(URI(uri), $cfg[:headers].sample) + end + + def self.handle_response(resp, key, error_key, callback) + json = JSON.parse(resp) + { + key => callback.call(json), + raw: resp, + } + rescue => e + if json.is_a? Hash + if json['errors'].is_a? Array + return rate_limit_check(json) + elsif json['data'].is_a? Hash + return + end + end + Rails.logger.error("#{error_key} fail: #{resp}") + + raise e + end + + def self.normalize_entity_media(json) + ret = {} + + json.each do |entity_media| + val = {} + + if entity_media['type'] == 'photo' + val[:image_url] = entity_media['media_url_https'] + elsif entity_media['type'] == 'video' + val[:variants] = entity_media['video_info']['variants'] + .filter { |variant| variant['bitrate'].present? } + .map do |variant| + { + bitrate: variant['bitrate'], + url: variant['url'], + } + end + end + + if !val.empty? + val[:url] = entity_media['expanded_url'] + val[:type] = entity_media['type'] + val[:id] = entity_media['media_key'] + end + + key = if ret[entity_media['url']].nil? + entity_media['url'] + else + entity_media['media_key'] + end + + ret[key] = val + end + + ret + end + + def self.normalize_entity_urls(json) + ret = {} + + (json || {}).each do |entity_url| + ret[entity_url['url']] = entity_url['expanded_url'] + end + + ret + end + + def self.normalize_timeline(json, user_id) + json.find { |instruction| instruction['type'] == 'TimelineAddEntries' }['entries'] + .filter { |entry| entry['entryId'] =~ /\A(profile-conversation|tweet)-/ } + .reduce([]) do |acc, entry| + if entry['content']['entryType'] == 'TimelineTimelineItem' + acc.push(entry['content']) + else + entry['content']['items'].each do |item| + acc.push(item['item']) + end + end + acc + end.map { |rawTweet| normalize_tweet(rawTweet['itemContent']['tweet_results']['result']) } + .filter { |tweet| !tweet.nil? && tweet.dig(:user, :id) == user_id } + end + + def self.normalize_tweet(json) + return nil if json.nil? + + return normalize_tweet(json['tweet']) if json['__typename'] == 'TweetWithVisibilityResults' + + { + id: json['rest_id'], + created_at: Time.parse(json['legacy']['created_at']), + user: normalize_user(json['core']['user_results']['result']), + message: json.dig('note_tweet', 'note_tweet_results', 'result', 'text') || json['legacy']['full_text'], + retweet: normalize_tweet(json.dig('legacy', 'retweeted_status_result', 'result')), + quote: normalize_tweet(json.dig('quoted_status_result', 'result')), + quote_id: json['legacy']['quoted_status_id_str'], + reply_to_id: json['legacy']['in_reply_to_status_id_str'], + reply_to_user_id: json['legacy']['in_reply_to_user_id_str'], + reply_to_username: json['legacy']['in_reply_to_screen_name'], + entity_urls: { **normalize_entity_urls(json['legacy']['entities']['urls']), **normalize_entity_urls(json.dig('note_tweet', 'note_tweet_results', 'result', 'entity_set', 'urls')) }, + entity_media: normalize_entity_media(json.dig('legacy', 'extended_entities', 'media') || []), + } + end + + def self.normalize_user(json) + { + avatar_url: json['legacy']['profile_image_url_https'], + id: json['rest_id'], + name: json['legacy']['name'], + protected: json['legacy']['protected'] == true, + username: json['legacy']['screen_name'], + } + end + + def self.rate_limit_check(json) + return unless json['errors'].any? { |err| err['code'] == 88 } + + raise 'Rate limited!' + end +end
--- a/app/models/tweet.rb Fri Jul 14 01:45:40 2023 +0900 +++ /dev/null Thu Jan 01 00:00:00 1970 +0000 @@ -1,100 +0,0 @@ -class Tweet - TIMELINE_OPTIONS = { - :count => 100, - :exclude_replies => false, - :include_rts => true, - :tweet_mode => :extended, - } - - def self.cache_expires_time - (15 + rand(15)).minutes - end - - def initialize(twitter_id) - @twitter_id = twitter_id - end - - def id - user.id - end - - def timeline - if @timeline.nil? - cache_key = "timeline:v2:#{id}/#{Base64.urlsafe_encode64 id.to_s}" - raw = Rails.cache.fetch(cache_key, :expires_in => self.class.cache_expires_time) do - client_try(:user_timeline, id, TIMELINE_OPTIONS).tap do |data| - if data[:result] == :ok - if data[:data].any? && data[:data].first.user.id != id - wrong_user = data[:data].first.user - throw "Wrong timeline data. Requested: #{id}, got: #{wrong_user.id} (#{wrong_user.screen_name.printable})" - end - - data[:data] = data[:data].select do |tweet| - tweet.retweeted_status.nil? || tweet.user.id != tweet.retweeted_status.user.id - end.map { |tweet| tweet.to_h } - end - end - end - - raise Twitter::Error::NotFound if raw[:result] == :not_found - - @timeline = raw[:data].map { |tweet_hash| Twitter::Tweet.new(tweet_hash) } - end - - @timeline - end - - def user - if @user.nil? - cache_key = "user:v1:#{@twitter_id.is_a?(Integer) ? 'id' : 'lookup'}:#{@twitter_id}" - raw = Rails.cache.fetch(cache_key, :expires_in => self.class.cache_expires_time) do - client_try(:user, @twitter_id).tap do |data| - if data[:result] == :ok - user = data[:data] - - if user.id != @twitter_id && user.screen_name.downcase != @twitter_id.try(:downcase) - throw "Wrong user data. Requested: #{@twitter_id}, got: #{user.id} (#{user.screen_name.printable})" - end - end - end - end - - raise Twitter::Error::NotFound if raw[:result] == :not_found - - @user = raw[:data] - end - - @user - end - - def client - Clients.instance.get client_config_id - end - - def client_try(method, *args) - initial_config_id = client_config_id - - begin - data = client.public_send method, *args - rescue Twitter::Error::TooManyRequests - @client_config_id = (1 + @client_config_id) % @client_config_count - - if initial_config_id == client_config_id - raise - else - retry - end - rescue Twitter::Error::NotFound - return { :result => :not_found } - end - - { :result => :ok, :data => data } - end - - def client_config_id - @client_config_count ||= $cfg[:twitter].size - @client_config_id ||= rand(@client_config_count) - - @client_config_id - end -end
--- a/app/views/tweets/_tweet.atom.erb Fri Jul 14 01:45:40 2023 +0900 +++ b/app/views/tweets/_tweet.atom.erb Mon Jul 17 04:23:09 2023 +0900 @@ -1,13 +1,16 @@ <entry> - <id><%= atom_id "#{tweet.user.id}/#{tweet.id}" %></id> - <published><%= tweet.created_at.xmlschema %></published> - <updated><%= tweet.created_at.xmlschema %></updated> + <% + created_at = tweet[:created_at].xmlschema + %> + <id><%= atom_id "#{tweet[:user][:id]}/#{tweet[:id]}" %></id> + <published><%= created_at %></published> + <updated><%= created_at %></updated> <link rel="alternate" type="text/html" href="<%= status_url(tweet) %>"/> - <title><%= truncate tweet.unescaped_text, :length => 30 %></title> + <title><%= truncate tweet[:message], :length => 30 %></title> <content type="html"> <%= render(:partial => "tweet", :formats => :html, :locals => { :tweet => tweet }).to_str %> </content> <author> - <name><%= tweet.user.screen_name %></name> + <name><%= tweet[:user][:username] %></name> </author> </entry>
--- a/app/views/tweets/_tweet.html.erb Fri Jul 14 01:45:40 2023 +0900 +++ b/app/views/tweets/_tweet.html.erb Mon Jul 17 04:23:09 2023 +0900 @@ -1,34 +1,32 @@ -<% if tweet.retweeted_status.present? %> +<% if tweet[:retweet].present? %> <p> - <%= link_to status_url(tweet.retweeted_status) do %> + <%= link_to status_url(tweet[:retweet]) do %> <em>Retweeted:</em> <% end %> </p> - <%= render "tweet", :tweet => tweet.retweeted_status, :with_time => true %> -<% else%> + <%= render "tweet", :tweet => tweet[:retweet], :with_time => true %> +<% else %> <% if defined?(with_time) && with_time %> <p> - <small>Originally tweeted at <%= tweet.created_at.rfc822 %></small> + <small>Originally tweeted at <%= tweet[:created_at].rfc822 %></small> </p> <% end %> - <% if tweet.in_reply_to_status_id.present? %> + <% if tweet[:reply_to_id].present? %> <p> <small> Replying to - <%= link_to 'tweet', status_url_base(tweet.in_reply_to_screen_name, tweet.in_reply_to_status_id) %> - by <%= link_to tweet.in_reply_to_screen_name, "https://twitter.com/#{tweet.in_reply_to_screen_name}" %> + <%= link_to 'tweet', status_url_base(tweet[:reply_to_username], tweet[:reply_to_id]) %> + by <%= link_to "@#{tweet[:reply_to_username]}", "https://twitter.com/#{tweet[:reply_to_username]}" %> </small> </p> <% end %> <p> - <%# FIXME: Twitter gem doesn't support extended mode when writing this %> <%= auto_link(expand_url( - tweet.full_text_extended, - tweet.attrs[:entities][:urls], - tweet.attrs[:entities][:media] + tweet[:message], + { **tweet[:entity_urls], **tweet[:entity_media] } )) .gsub("\n", "<br />") .html_safe @@ -36,31 +34,36 @@ </p> <p> - <%= link_to "https://twitter.com/#{tweet.user.screen_name}" do %> - <%= image_tag tweet.user.profile_image_url_https.to_s, :alt => "profile image for #{tweet.user.name.printable}" %> - <%= tweet.user.name.printable -%> + <%= link_to "https://twitter.com/#{tweet[:user][:username]}" do %> + <%= image_tag tweet[:user][:avatar_url].to_s, :alt => "profile image for #{tweet[:user][:name].printable}" %> + <%= tweet[:user][:name].printable -%> <% end %> </p> <p> - <% tweet.media.each_with_index do |media, i| %> - <% if media.is_a? Twitter::Media::Photo %> - <%= link_to "#{media.media_url_https}?name=orig" do %> - <%= image_tag "#{media.media_url_https}?name=small", :alt => "attachment #{i + 1}" -%> + <% tweet[:entity_media].each do |_short_url, media| %> + <% if media[:type] == 'photo' %> + <%= link_to "#{media[:image_url]}?name=orig" do %> + <%= image_tag "#{media[:image_url]}?name=small", :alt => "attachment #{media[:id]}" -%> <% end %> - <% elsif media.is_a? Twitter::Media::Video %> - <%= video_tag media.video_info.variants - .select { |i| i.bitrate.is_a? Integer } - .sort_by { |i| -i.bitrate } - .map(&:url), width: '100%', controls: true + <% elsif media[:type] == 'video' %> + <%= video_tag media[:variants] + .sort_by { |variant| -variant[:bitrate] } + .map { |variant| variant[:url] }, width: '100%', controls: true %> <% end %> <% end %> </p> - <% if tweet.quoted_status.present? %> + <% if tweet[:quote_id].present? %> <blockquote> - <%= render "tweet", :tweet => tweet.quoted_status, :with_time => true %> + <% if tweet[:quote].present? %> + <%= render "tweet", :tweet => tweet[:quote], :with_time => true %> + <% else %> + <%= link_to status_url_base(nil, tweet[:quote_id]) do %> + <em>Preview not available, view tweet directly.</em> + <% end %> + <% end %> </blockquote> <% end %> <% end %>
--- a/app/views/tweets/show.atom.erb Fri Jul 14 01:45:40 2023 +0900 +++ b/app/views/tweets/show.atom.erb Mon Jul 17 04:23:09 2023 +0900 @@ -1,15 +1,15 @@ <?xml version="1.0" encoding="UTF-8"?> <feed xml:lang="en-US" xmlns="http://www.w3.org/2005/Atom"> - <id><%= atom_id(@user.id) %></id> + <id><%= atom_id(@user[:id]) %></id> - <link rel="alternate" type="text/html" href="https://twitter.com/<%= @user.screen_name %>" /> + <link rel="alternate" type="text/html" href="https://twitter.com/<%= @user[:username] %>" /> <link rel="self" type="application/atom+xml" href="<%= request.url %>" /> - <title><%= "#{@user.name.printable} (#{@user.screen_name})" %></title> - <icon><%= @user.profile_image_url_https %></icon> - <logo><%= @user.profile_image_url_https %></logo> + <title><%= "#{@user[:name].printable} (#{@user[:username]})" %></title> + <icon><%= @user[:avatar_url] %></icon> + <logo><%= @user[:avatar_url] %></logo> - <updated><%= (@tweets.first.try(:created_at) || Time.at(0)).xmlschema %></updated> + <updated><%= (@tweets.first.try(:[], :created_at) || Time.at(0)).xmlschema %></updated> <%= render :partial => "tweet", :collection => @tweets, :cached => true %> </feed>
--- a/config/initializers/ext_twitter_tweet.rb Fri Jul 14 01:45:40 2023 +0900 +++ /dev/null Thu Jan 01 00:00:00 1970 +0000 @@ -1,11 +0,0 @@ -class Twitter::Tweet - def full_text_extended - attrs[:full_text].printable - end - memoize :full_text_extended - - def unescaped_text - CGI.unescapeHTML full_text_extended - end - memoize :unescaped_text -end