Fetch remote image using http.rb (#3114)
This commit is contained in:
parent
b11c4326d2
commit
79ef8b3653
8 changed files with 46 additions and 32 deletions
|
@ -44,6 +44,7 @@ class Account < ApplicationRecord
|
||||||
include AccountAvatar
|
include AccountAvatar
|
||||||
include AccountHeader
|
include AccountHeader
|
||||||
include Attachmentable
|
include Attachmentable
|
||||||
|
include Remotable
|
||||||
include Targetable
|
include Targetable
|
||||||
|
|
||||||
# Local users
|
# Local users
|
||||||
|
|
|
@ -26,16 +26,5 @@ module AccountAvatar
|
||||||
def avatar_static_url
|
def avatar_static_url
|
||||||
avatar_content_type == 'image/gif' ? avatar.url(:static) : avatar_original_url
|
avatar_content_type == 'image/gif' ? avatar.url(:static) : avatar_original_url
|
||||||
end
|
end
|
||||||
|
|
||||||
def avatar_remote_url=(url)
|
|
||||||
parsed_url = Addressable::URI.parse(url).normalize
|
|
||||||
|
|
||||||
return if !%w(http https).include?(parsed_url.scheme) || parsed_url.host.empty? || self[:avatar_remote_url] == url
|
|
||||||
|
|
||||||
self.avatar = URI.parse(parsed_url.to_s)
|
|
||||||
self[:avatar_remote_url] = url
|
|
||||||
rescue OpenURI::HTTPError, OpenSSL::SSL::SSLError, Paperclip::Errors::NotIdentifiedByImageMagickError => e
|
|
||||||
Rails.logger.debug "Error fetching remote avatar: #{e}"
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
|
@ -26,16 +26,5 @@ module AccountHeader
|
||||||
def header_static_url
|
def header_static_url
|
||||||
header_content_type == 'image/gif' ? header.url(:static) : header_original_url
|
header_content_type == 'image/gif' ? header.url(:static) : header_original_url
|
||||||
end
|
end
|
||||||
|
|
||||||
def header_remote_url=(url)
|
|
||||||
parsed_url = Addressable::URI.parse(url).normalize
|
|
||||||
|
|
||||||
return if !%w(http https).include?(parsed_url.scheme) || parsed_url.host.empty? || self[:header_remote_url] == url
|
|
||||||
|
|
||||||
self.header = URI.parse(parsed_url.to_s)
|
|
||||||
self[:header_remote_url] = url
|
|
||||||
rescue OpenURI::HTTPError, OpenSSL::SSL::SSLError, Paperclip::Errors::NotIdentifiedByImageMagickError => e
|
|
||||||
Rails.logger.debug "Error fetching remote header: #{e}"
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
35
app/models/concerns/remotable.rb
Normal file
35
app/models/concerns/remotable.rb
Normal file
|
@ -0,0 +1,35 @@
|
||||||
|
# frozen_string_literal: true
|
||||||
|
|
||||||
|
module Remotable
|
||||||
|
include HttpHelper
|
||||||
|
extend ActiveSupport::Concern
|
||||||
|
|
||||||
|
included do
|
||||||
|
attachment_definitions.each_key do |attachment_name|
|
||||||
|
attribute_name = "#{attachment_name}_remote_url".to_sym
|
||||||
|
method_name = "#{attribute_name}=".to_sym
|
||||||
|
|
||||||
|
define_method method_name do |url|
|
||||||
|
parsed_url = Addressable::URI.parse(url).normalize
|
||||||
|
|
||||||
|
return if !%w(http https).include?(parsed_url.scheme) || parsed_url.host.empty? || self[attribute_name] == url
|
||||||
|
|
||||||
|
begin
|
||||||
|
response = http_client.get(url)
|
||||||
|
|
||||||
|
return if response.code != 200
|
||||||
|
|
||||||
|
matches = response.headers['content-disposition']&.match(/filename="([^"]*)"/)
|
||||||
|
filename = matches.nil? ? parsed_url.path.split('/').last : matches[1]
|
||||||
|
|
||||||
|
send("#{attachment_name}=", StringIO.new(response.to_s))
|
||||||
|
send("#{attachment_name}_file_name=", filename)
|
||||||
|
|
||||||
|
self[attribute_name] = url if has_attribute?(attribute_name)
|
||||||
|
rescue HTTP::TimeoutError, OpenSSL::SSL::SSLError, Paperclip::Errors::NotIdentifiedByImageMagickError => e
|
||||||
|
Rails.logger.debug "Error fetching remote #{attachment_name}: #{e}"
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
|
@ -46,6 +46,9 @@ class MediaAttachment < ApplicationRecord
|
||||||
styles: ->(f) { file_styles f },
|
styles: ->(f) { file_styles f },
|
||||||
processors: ->(f) { file_processors f },
|
processors: ->(f) { file_processors f },
|
||||||
convert_options: { all: '-quality 90 -strip' }
|
convert_options: { all: '-quality 90 -strip' }
|
||||||
|
|
||||||
|
include Remotable
|
||||||
|
|
||||||
validates_attachment_content_type :file, content_type: IMAGE_MIME_TYPES + VIDEO_MIME_TYPES
|
validates_attachment_content_type :file, content_type: IMAGE_MIME_TYPES + VIDEO_MIME_TYPES
|
||||||
validates_attachment_size :file, less_than: 8.megabytes
|
validates_attachment_size :file, less_than: 8.megabytes
|
||||||
|
|
||||||
|
@ -59,10 +62,6 @@ class MediaAttachment < ApplicationRecord
|
||||||
remote_url.blank?
|
remote_url.blank?
|
||||||
end
|
end
|
||||||
|
|
||||||
def file_remote_url=(url)
|
|
||||||
self.file = URI.parse(Addressable::URI.parse(url).normalize.to_s)
|
|
||||||
end
|
|
||||||
|
|
||||||
def to_param
|
def to_param
|
||||||
shortcode
|
shortcode
|
||||||
end
|
end
|
||||||
|
|
|
@ -36,6 +36,7 @@ class PreviewCard < ApplicationRecord
|
||||||
has_attached_file :image, styles: { original: '120x120#' }, convert_options: { all: '-quality 80 -strip' }
|
has_attached_file :image, styles: { original: '120x120#' }, convert_options: { all: '-quality 80 -strip' }
|
||||||
|
|
||||||
include Attachmentable
|
include Attachmentable
|
||||||
|
include Remotable
|
||||||
|
|
||||||
validates :url, presence: true
|
validates :url, presence: true
|
||||||
validates_attachment_content_type :image, content_type: IMAGE_MIME_TYPES
|
validates_attachment_content_type :image, content_type: IMAGE_MIME_TYPES
|
||||||
|
|
|
@ -84,10 +84,10 @@ class FetchLinkCardService < BaseService
|
||||||
|
|
||||||
page = Nokogiri::HTML(response.to_s)
|
page = Nokogiri::HTML(response.to_s)
|
||||||
|
|
||||||
card.type = :link
|
card.type = :link
|
||||||
card.title = meta_property(page, 'og:title') || page.at_xpath('//title')&.content
|
card.title = meta_property(page, 'og:title') || page.at_xpath('//title')&.content
|
||||||
card.description = meta_property(page, 'og:description') || meta_property(page, 'description')
|
card.description = meta_property(page, 'og:description') || meta_property(page, 'description')
|
||||||
card.image = URI.parse(Addressable::URI.parse(meta_property(page, 'og:image')).normalize.to_s) if meta_property(page, 'og:image')
|
card.image_remote_url = meta_property(page, 'og:image') if meta_property(page, 'og:image')
|
||||||
|
|
||||||
return if card.title.blank?
|
return if card.title.blank?
|
||||||
|
|
||||||
|
|
|
@ -239,8 +239,8 @@ class ProcessFeedService < BaseService
|
||||||
|
|
||||||
begin
|
begin
|
||||||
media.file_remote_url = link['href']
|
media.file_remote_url = link['href']
|
||||||
media.save
|
media.save!
|
||||||
rescue OpenURI::HTTPError, OpenSSL::SSL::SSLError, Paperclip::Errors::NotIdentifiedByImageMagickError
|
rescue ActiveRecord::RecordInvalid
|
||||||
next
|
next
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
Loading…
Reference in a new issue