discourse/lib/onebox/preview.rb
Arpit Jalan 283b08d45f
DEV: Absorb onebox gem into core (#12979)
* Move onebox gem in core library

* Update template file path

* Remove warning for onebox gem caching

* Remove onebox version file

* Remove onebox gem

* Add sanitize gem

* Require onebox library in lazy-yt plugin

* Remove onebox web specific code

This code was used in standalone onebox Sinatra application

* Merge Discourse specific AllowlistedGenericOnebox engine in core

* Fix onebox engine filenames to match class name casing

* Move onebox specs from gem into core

* DEV: Rename `response` helper to `onebox_response`

Fixes a naming collision.

* Require rails_helper

* Don't use `before/after(:all)`

* Whitespace

* Remove fakeweb

* Remove poor unit tests

* DEV: Re-add fakeweb, plugins are using it

* Move onebox helpers

* Stub Instagram API

* FIX: Follow additional redirect status codes (#476)

Don’t throw errors if we encounter 303, 307 or 308 HTTP status codes in responses

* Remove an empty file

* DEV: Update the license file

Using the copy from https://choosealicense.com/licenses/gpl-2.0/#

Hopefully this will enable GitHub to show the license UI?

* DEV: Update embedded copyrights

* DEV: Add Onebox copyright notice

* DEV: Add MIT license, convert COPYRIGHT.txt to md

* DEV: Remove an incorrect copyright claim

Co-authored-by: Jarek Radosz <jradosz@gmail.com>
Co-authored-by: jbrw <jamie@goatforce5.org>
2021-05-26 15:11:35 +05:30

97 lines
2.3 KiB
Ruby

# frozen_string_literal: true
module Onebox
class Preview
# see https://bugs.ruby-lang.org/issues/14688
client_exception = defined?(Net::HTTPClientException) ? Net::HTTPClientException : Net::HTTPServerException
WEB_EXCEPTIONS ||= [client_exception, OpenURI::HTTPError, Timeout::Error, Net::HTTPError, Errno::ECONNREFUSED]
def initialize(url, options = Onebox.options)
@url = url
@options = options.dup
allowed_origins = @options[:allowed_iframe_origins] || Onebox::Engine.all_iframe_origins
@options[:allowed_iframe_regexes] = Engine.origins_to_regexes(allowed_origins)
@engine_class = Matcher.new(@url, @options).oneboxed
end
def to_s
return "" unless engine
sanitize process_html engine_html
rescue *WEB_EXCEPTIONS
""
end
def placeholder_html
return "" unless engine
sanitize process_html engine.placeholder_html
rescue *WEB_EXCEPTIONS
""
end
def errors
return {} unless engine
engine.errors
end
def data
return {} unless engine
engine.data
end
def options
OpenStruct.new(@options)
end
private
def engine_html
engine.to_html
end
def process_html(html)
return "" unless html
if @options[:max_width]
doc = Nokogiri::HTML5::fragment(html)
if doc
doc.css('[width]').each do |e|
width = e['width'].to_i
if width > @options[:max_width]
height = e['height'].to_i
if (height > 0)
ratio = (height.to_f / width.to_f)
e['height'] = (@options[:max_width] * ratio).floor
end
e['width'] = @options[:max_width]
end
end
return doc.to_html
end
end
html
end
def sanitize(html)
config = @options[:sanitize_config] || Sanitize::Config::ONEBOX
config = config.merge(allowed_iframe_regexes: @options[:allowed_iframe_regexes])
Sanitize.fragment(html, config)
end
def engine
return nil unless @engine_class
return @engine if defined?(@engine)
@engine = @engine_class.new(@url)
@engine.options = @options
@engine
end
class InvalidURI < StandardError; end
end
end