From 55bca6b8e85cbe719e748f4cd6acd2a8c2accb14 Mon Sep 17 00:00:00 2001 From: Mike Dalessio Date: Mon, 19 Jun 2023 14:32:07 -0400 Subject: [PATCH] Update Action Text to use HTML5 when available The change from `#clone` to `#dup` is necessary to work around an issue in Nokogiri where `#clone` is not defined properly for HTML5 fragment and the fragment does not have a parent Document. `#dup` behaves the way we expect, so this should be fine. --- .../app/helpers/action_text/content_helper.rb | 2 +- actiontext/lib/action_text.rb | 14 ++++++++++++++ actiontext/lib/action_text/fragment.rb | 4 ++-- actiontext/lib/action_text/html_conversion.rb | 2 +- .../test/integration/controller_render_test.rb | 2 +- actiontext/test/integration/job_render_test.rb | 2 +- 6 files changed, 20 insertions(+), 6 deletions(-) diff --git a/actiontext/app/helpers/action_text/content_helper.rb b/actiontext/app/helpers/action_text/content_helper.rb index e63bd781df221..0980bf0f00e31 100644 --- a/actiontext/app/helpers/action_text/content_helper.rb +++ b/actiontext/app/helpers/action_text/content_helper.rb @@ -4,7 +4,7 @@ module ActionText module ContentHelper - mattr_accessor(:sanitizer) { Rails::Html::Sanitizer.safe_list_sanitizer.new } + mattr_accessor(:sanitizer) { Rails::Html::Sanitizer.best_supported_vendor.safe_list_sanitizer.new } mattr_accessor(:allowed_tags) { sanitizer.class.allowed_tags + [ ActionText::Attachment.tag_name, "figure", "figcaption" ] } mattr_accessor(:allowed_attributes) { sanitizer.class.allowed_attributes + ActionText::Attachment::ATTRIBUTES } mattr_accessor(:scrubber) diff --git a/actiontext/lib/action_text.rb b/actiontext/lib/action_text.rb index d6038a939f003..6d6b1126ad79c 100644 --- a/actiontext/lib/action_text.rb +++ b/actiontext/lib/action_text.rb @@ -42,4 +42,18 @@ module Attachments autoload :Minification autoload :TrixConversion end + + class << self + def html_document_class + return @html_document_class if defined?(@html_document_class) + @html_document_class = + defined?(Nokogiri::HTML5) ? Nokogiri::HTML5::Document : Nokogiri::HTML4::Document + end + + def html_document_fragment_class + return @html_document_fragment_class if defined?(@html_document_fragment_class) + @html_document_fragment_class = + defined?(Nokogiri::HTML5) ? Nokogiri::HTML5::DocumentFragment : Nokogiri::HTML4::DocumentFragment + end + end end diff --git a/actiontext/lib/action_text/fragment.rb b/actiontext/lib/action_text/fragment.rb index 129344f90f0fb..21a6897b8f5fe 100644 --- a/actiontext/lib/action_text/fragment.rb +++ b/actiontext/lib/action_text/fragment.rb @@ -7,7 +7,7 @@ def wrap(fragment_or_html) case fragment_or_html when self fragment_or_html - when Nokogiri::HTML::DocumentFragment + when Nokogiri::XML::DocumentFragment # base class for all fragments new(fragment_or_html) else from_html(fragment_or_html) @@ -30,7 +30,7 @@ def find_all(selector) end def update - yield source = self.source.clone + yield source = self.source.dup self.class.new(source) end diff --git a/actiontext/lib/action_text/html_conversion.rb b/actiontext/lib/action_text/html_conversion.rb index 1e1062ea3fff6..bef7730640b02 100644 --- a/actiontext/lib/action_text/html_conversion.rb +++ b/actiontext/lib/action_text/html_conversion.rb @@ -18,7 +18,7 @@ def create_element(tag_name, attributes = {}) private def document - Nokogiri::HTML::Document.new.tap { |doc| doc.encoding = "UTF-8" } + ActionText.html_document_class.new.tap { |doc| doc.encoding = "UTF-8" } end end end diff --git a/actiontext/test/integration/controller_render_test.rb b/actiontext/test/integration/controller_render_test.rb index c9d8817d8f31a..afedaae17c520 100644 --- a/actiontext/test/integration/controller_render_test.rb +++ b/actiontext/test/integration/controller_render_test.rb @@ -20,7 +20,7 @@ class ActionText::ControllerRenderTest < ActionDispatch::IntegrationTest host! "loocalhoost" get message_path(message, format: :json) - content = Nokogiri::HTML::DocumentFragment.parse(response.parsed_body["content"]) + content = ActionText.html_document_fragment_class.parse(response.parsed_body["content"]) assert_select content, "img:match('src', ?)", %r"//loocalhoost/.+/racecar" end diff --git a/actiontext/test/integration/job_render_test.rb b/actiontext/test/integration/job_render_test.rb index 86955b3794bf5..d0f4211226b9c 100644 --- a/actiontext/test/integration/job_render_test.rb +++ b/actiontext/test/integration/job_render_test.rb @@ -18,7 +18,7 @@ class ActionText::JobRenderTest < ActiveJob::TestCase perform_enqueued_jobs end - rendered = Nokogiri::HTML::DocumentFragment.parse(File.read(file)) + rendered = ActionText.html_document_fragment_class.parse(File.read(file)) assert_select rendered, "img:match('src', ?)", %r"//foo.example.com:9001/.+/racecar" end end