Skip to content

Commit

Permalink
Merge pull request from GHSA-cfjv-5498-mph5
Browse files Browse the repository at this point in the history
Prior to this commit, when a translation key indicated that the
translation text was HTML, the value returned by `I18n.translate` would
always be marked as `html_safe`.  However, the value returned by
`I18n.translate` could be an untrusted value directly from
`options[:default]`.

This commit ensures values directly from `options[:default]` are not
marked as `html_safe`.

Co-authored-by: Jonathan Hefner <jonathan@hefner.pro>
  • Loading branch information
georgeclaghorn and jonathanhefner committed Sep 9, 2020
1 parent ea74a7d commit 82a91fd
Show file tree
Hide file tree
Showing 2 changed files with 19 additions and 1 deletion.
13 changes: 12 additions & 1 deletion actionview/lib/action_view/helpers/translation_helper.rb
Original file line number Diff line number Diff line change
Expand Up @@ -79,14 +79,22 @@ def translate(key, options = {})

if html_safe_translation_key?(key)
html_safe_options = options.dup

options.except(*I18n::RESERVED_KEYS).each do |name, value|
unless name == :count && value.is_a?(Numeric)
html_safe_options[name] = ERB::Util.html_escape(value.to_s)
end
end

html_safe_options[:default] = MISSING_TRANSLATION unless html_safe_options[:default].blank?

translation = I18n.translate(scope_key_by_partial(key), html_safe_options.merge(raise: i18n_raise))

translation.respond_to?(:html_safe) ? translation.html_safe : translation
if translation.equal?(MISSING_TRANSLATION)
options[:default].first
else
translation.respond_to?(:html_safe) ? translation.html_safe : translation
end
else
I18n.translate(scope_key_by_partial(key), options.merge(raise: i18n_raise))
end
Expand Down Expand Up @@ -121,6 +129,9 @@ def localize(*args)
alias :l :localize

private
MISSING_TRANSLATION = Object.new
private_constant :MISSING_TRANSLATION

def scope_key_by_partial(key)
if key.to_s.first == "."
if @virtual_path
Expand Down
7 changes: 7 additions & 0 deletions actionview/test/template/translation_helper_test.rb
Original file line number Diff line number Diff line change
Expand Up @@ -205,6 +205,13 @@ def test_translate_with_last_default_not_named_html
assert_equal false, translation.html_safe?
end

def test_translate_does_not_mark_unsourced_string_default_as_html_safe
untrusted_string = "<script>alert()</script>"
translation = translate(:"translations.missing", default: [:"translations.missing_html", untrusted_string])
assert_equal untrusted_string, translation
assert_not_predicate translation, :html_safe?
end

def test_translate_with_string_default
translation = translate(:'translations.missing', default: "A Generic String")
assert_equal "A Generic String", translation
Expand Down

0 comments on commit 82a91fd

Please sign in to comment.