@@ -587,23 +587,124 @@ def test_exclude_node_type_comment
587587 assert_equal ( "<div>text</div><b>text</b>" , safe_list_sanitize ( "<div>text</div><!-- comment --><b>text</b>" ) )
588588 end
589589
590- def test_disallow_the_dangerous_safelist_combination_of_select_and_style
591- input = "<select><style><script>alert(1)</script></style></select>"
592- tags = [ "select" , "style" ]
593- warning = /WARNING: Rails::Html::SafeListSanitizer: removing 'style' from safelist/
594- sanitized = nil
595- invocation = Proc . new { sanitized = safe_list_sanitize ( input , tags : tags ) }
596-
597- if html5_mode?
598- # if Loofah is using an HTML5 parser,
599- # then "style" should be removed by the parser as an invalid child of "select"
600- assert_silent ( &invocation )
601- else
602- # if Loofah is using an HTML4 parser,
603- # then SafeListSanitizer should remove "style" from the safelist
604- assert_output ( nil , warning , &invocation )
590+ %w[ text/plain text/css image/png image/gif image/jpeg ] . each do |mediatype |
591+ define_method "test_mediatype_#{ mediatype } _allowed" do
592+ input = %Q(<img src="data:#{ mediatype } ;base64,PHNjcmlwdD5hbGVydCgnWFNTJyk8L3NjcmlwdD4=">)
593+ expected = input
594+ actual = safe_list_sanitize ( input )
595+ assert_equal ( expected , actual )
596+
597+ input = %Q(<img src="DATA:#{ mediatype } ;base64,PHNjcmlwdD5hbGVydCgnWFNTJyk8L3NjcmlwdD4=">)
598+ expected = input
599+ actual = safe_list_sanitize ( input )
600+ assert_equal ( expected , actual )
605601 end
606- refute_includes ( sanitized , "style" )
602+ end
603+
604+ def test_mediatype_text_html_disallowed
605+ input = %q(<img src="data:text/html;base64,PHNjcmlwdD5hbGVydCgnWFNTJyk8L3NjcmlwdD4=">)
606+ expected = %q(<img>)
607+ actual = safe_list_sanitize ( input )
608+ assert_equal ( expected , actual )
609+
610+ input = %q(<img src="DATA:text/html;base64,PHNjcmlwdD5hbGVydCgnWFNTJyk8L3NjcmlwdD4=">)
611+ expected = %q(<img>)
612+ actual = safe_list_sanitize ( input )
613+ assert_equal ( expected , actual )
614+ end
615+
616+ def test_mediatype_image_svg_xml_disallowed
617+ input = %q(<img src="">)
618+ expected = %q(<img>)
619+ actual = safe_list_sanitize ( input )
620+ assert_equal ( expected , actual )
621+
622+ input = %q(<img src="DATA:image/svg+xml;base64,PHNjcmlwdD5hbGVydCgnWFNTJyk8L3NjcmlwdD4=">)
623+ expected = %q(<img>)
624+ actual = safe_list_sanitize ( input )
625+ assert_equal ( expected , actual )
626+ end
627+
628+ def test_mediatype_other_disallowed
629+ input = %q(<a href="data:foo;base64,PHNjcmlwdD5hbGVydCgnWFNTJyk8L3NjcmlwdD4=">foo</a>)
630+ expected = %q(<a>foo</a>)
631+ actual = safe_list_sanitize ( input )
632+ assert_equal ( expected , actual )
633+
634+ input = %q(<a href="DATA:foo;base64,PHNjcmlwdD5hbGVydCgnWFNTJyk8L3NjcmlwdD4=">foo</a>)
635+ expected = %q(<a>foo</a>)
636+ actual = safe_list_sanitize ( input )
637+ assert_equal ( expected , actual )
638+ end
639+
640+ def test_scrubbing_svg_attr_values_that_allow_ref
641+ input = %Q(<div fill="yellow url(http://bad.com/) #fff">hey</div>)
642+ expected = %Q(<div fill="yellow #fff">hey</div>)
643+ actual = scope_allowed_attributes %w( fill ) do
644+ safe_list_sanitize ( input )
645+ end
646+
647+ assert_equal ( expected , actual )
648+ end
649+
650+ def test_style_with_css_payload
651+ input , tags = "<style>div > span { background: \" red\" ; }</style>" , [ "style" ]
652+ expected = "<style>div > span { background: \" red\" ; }</style>"
653+ actual = safe_list_sanitize ( input , tags : tags )
654+
655+ assert_equal ( expected , actual )
656+ end
657+
658+ def test_combination_of_select_and_style_with_css_payload
659+ input , tags = "<select><style>div > span { background: \" red\" ; }</style></select>" , [ "select" , "style" ]
660+ expected = "<select><style>div > span { background: \" red\" ; }</style></select>"
661+ actual = safe_list_sanitize ( input , tags : tags )
662+
663+ assert_equal ( expected , actual )
664+ end
665+
666+ def test_combination_of_select_and_style_with_script_payload
667+ input , tags = "<select><style><script>alert(1)</script></style></select>" , [ "select" , "style" ]
668+ expected = "<select><style><script>alert(1)</script></style></select>"
669+ actual = safe_list_sanitize ( input , tags : tags )
670+
671+ assert_equal ( expected , actual )
672+ end
673+
674+ def test_combination_of_svg_and_style_with_script_payload
675+ input , tags = "<svg><style><script>alert(1)</script></style></svg>" , [ "svg" , "style" ]
676+ expected = "<svg><style><script>alert(1)</script></style></svg>"
677+ actual = safe_list_sanitize ( input , tags : tags )
678+
679+ assert_equal ( expected , actual )
680+ end
681+
682+ def test_combination_of_math_and_style_with_img_payload
683+ input , tags = "<math><style><img src=x onerror=alert(1)></style></math>" , [ "math" , "style" ]
684+ expected = "<math><style><img src=x onerror=alert(1)></style></math>"
685+ actual = safe_list_sanitize ( input , tags : tags )
686+
687+ assert_equal ( expected , actual )
688+
689+ input , tags = "<math><style><img src=x onerror=alert(1)></style></math>" , [ "math" , "style" , "img" ]
690+ expected = "<math><style><img src=x onerror=alert(1)></style></math>"
691+ actual = safe_list_sanitize ( input , tags : tags )
692+
693+ assert_equal ( expected , actual )
694+ end
695+
696+ def test_combination_of_svg_and_style_with_img_payload
697+ input , tags = "<svg><style><img src=x onerror=alert(1)></style></svg>" , [ "svg" , "style" ]
698+ expected = "<svg><style><img src=x onerror=alert(1)></style></svg>"
699+ actual = safe_list_sanitize ( input , tags : tags )
700+
701+ assert_equal ( expected , actual )
702+
703+ input , tags = "<svg><style><img src=x onerror=alert(1)></style></svg>" , [ "svg" , "style" , "img" ]
704+ expected = "<svg><style><img src=x onerror=alert(1)></style></svg>"
705+ actual = safe_list_sanitize ( input , tags : tags )
706+
707+ assert_equal ( expected , actual )
607708 end
608709
609710protected
@@ -673,8 +774,4 @@ def libxml_2_9_14_recovery_lt_bang?
673774 # then reverted in 2.10.0, see https://gitlab.gnome.org/GNOME/libxml2/-/issues/380
674775 Nokogiri . method ( :uses_libxml? ) . arity == -1 && Nokogiri . uses_libxml? ( "= 2.9.14" )
675776 end
676-
677- def html5_mode?
678- ::Loofah . respond_to? ( :html5_mode? ) && ::Loofah . html5_mode?
679- end
680777end
0 commit comments