sanitize 5.2.3 → 6.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/HISTORY.md +80 -0
- data/LICENSE +1 -1
- data/README.md +26 -42
- data/lib/sanitize/config/default.rb +5 -0
- data/lib/sanitize/transformers/clean_css.rb +1 -0
- data/lib/sanitize/transformers/clean_element.rb +45 -0
- data/lib/sanitize/version.rb +1 -3
- data/lib/sanitize.rb +1 -1
- data/test/test_clean_comment.rb +16 -16
- data/test/test_clean_css.rb +5 -5
- data/test/test_clean_doctype.rb +15 -15
- data/test/test_clean_element.rb +99 -92
- data/test/test_config.rb +9 -9
- data/test/test_malicious_css.rb +20 -7
- data/test/test_malicious_html.rb +135 -31
- data/test/test_parser.rb +8 -8
- data/test/test_sanitize.rb +28 -28
- data/test/test_sanitize_css.rb +53 -53
- data/test/test_transformers.rb +37 -37
- metadata +10 -24
    
        checksums.yaml
    CHANGED
    
    | @@ -1,7 +1,7 @@ | |
| 1 1 | 
             
            ---
         | 
| 2 2 | 
             
            SHA256:
         | 
| 3 | 
            -
              metadata.gz:  | 
| 4 | 
            -
              data.tar.gz:  | 
| 3 | 
            +
              metadata.gz: 93adca1e155370d138ccb7c500b618e2ed218297d21593ec8937638f4d99731b
         | 
| 4 | 
            +
              data.tar.gz: 740b6d84113a0945928601b6cad03e36b4ee76f7c3098c72ddb1a1b01ec5d0ec
         | 
| 5 5 | 
             
            SHA512:
         | 
| 6 | 
            -
              metadata.gz:  | 
| 7 | 
            -
              data.tar.gz:  | 
| 6 | 
            +
              metadata.gz: 4d3e9852ec92ac961c2e35d4a04e7d967dd2eac364e656837b93daf95c1b653da53d4ef7f104af83887e46d08237ddca1efa945facde3efbfcfce0164c0fe334
         | 
| 7 | 
            +
              data.tar.gz: 05a56334e5cdbbee7b165b19245b90a8acdd82bcd72bbc9f84e2780d914f8b040d19d9ff71934b0c1bd71df4b55f407f460c76dffdbd275b183ecaffb2fa6c38
         | 
    
        data/HISTORY.md
    CHANGED
    
    | @@ -1,5 +1,85 @@ | |
| 1 1 | 
             
            # Sanitize History
         | 
| 2 2 |  | 
| 3 | 
            +
            ## 6.0.2 (2023-07-06)
         | 
| 4 | 
            +
             | 
| 5 | 
            +
            ### Bug Fixes
         | 
| 6 | 
            +
             | 
| 7 | 
            +
            * CVE-2023-36823: Fixed an HTML+CSS sanitization bypass that could allow XSS
         | 
| 8 | 
            +
              (cross-site scripting). This issue affects Sanitize versions 3.0.0 through
         | 
| 9 | 
            +
              6.0.1.
         | 
| 10 | 
            +
             | 
| 11 | 
            +
              When using Sanitize's relaxed config or a custom config that allows `<style>`
         | 
| 12 | 
            +
              elements and one or more CSS at-rules, carefully crafted input could be used
         | 
| 13 | 
            +
              to sneak arbitrary HTML through Sanitize.
         | 
| 14 | 
            +
             | 
| 15 | 
            +
              See the following security advisory for additional details:
         | 
| 16 | 
            +
              [GHSA-f5ww-cq3m-q3g7](https://github.com/rgrove/sanitize/security/advisories/GHSA-f5ww-cq3m-q3g7)
         | 
| 17 | 
            +
             | 
| 18 | 
            +
              Thanks to @cure53 for finding this issue.
         | 
| 19 | 
            +
             | 
| 20 | 
            +
            ## 6.0.1 (2023-01-27)
         | 
| 21 | 
            +
             | 
| 22 | 
            +
            ### Bug Fixes
         | 
| 23 | 
            +
             | 
| 24 | 
            +
            * Sanitize now always removes `<noscript>` elements and their contents, even
         | 
| 25 | 
            +
              when `noscript` is in the allowlist.
         | 
| 26 | 
            +
             | 
| 27 | 
            +
              This fixes a sanitization bypass that could occur when `noscript` was allowed
         | 
| 28 | 
            +
              by a custom allowlist. In this scenario, carefully crafted input could sneak
         | 
| 29 | 
            +
              arbitrary HTML through Sanitize, potentially enabling an XSS (cross-site
         | 
| 30 | 
            +
              scripting) attack.
         | 
| 31 | 
            +
             | 
| 32 | 
            +
              Sanitize's default configs don't allow `<noscript>` elements and are not
         | 
| 33 | 
            +
              vulnerable. This issue only affects users who are using a custom config that
         | 
| 34 | 
            +
              adds `noscript` to the element allowlist.
         | 
| 35 | 
            +
             | 
| 36 | 
            +
              The root cause of this issue is that HTML parsing rules treat the contents of
         | 
| 37 | 
            +
              a `<noscript>` element differently depending on whether scripting is enabled
         | 
| 38 | 
            +
              in the user agent. Nokogiri doesn't support scripting so it follows the
         | 
| 39 | 
            +
              "scripting disabled" rules, but a web browser with scripting enabled will
         | 
| 40 | 
            +
              follow the "scripting enabled" rules. This means that Sanitize can't reliably
         | 
| 41 | 
            +
              make the contents of a `<noscript>` element safe for scripting enabled
         | 
| 42 | 
            +
              browsers, so the safest thing to do is to remove the element and its contents
         | 
| 43 | 
            +
              entirely.
         | 
| 44 | 
            +
             | 
| 45 | 
            +
              See the following security advisory for additional details:
         | 
| 46 | 
            +
              [GHSA-fw3g-2h3j-qmm7](https://github.com/rgrove/sanitize/security/advisories/GHSA-fw3g-2h3j-qmm7)
         | 
| 47 | 
            +
             | 
| 48 | 
            +
              Thanks to David Klein from [TU Braunschweig](https://www.tu-braunschweig.de/en/ias)
         | 
| 49 | 
            +
              (@leeN) for reporting this issue.
         | 
| 50 | 
            +
             | 
| 51 | 
            +
            * Fixed an edge case in which the contents of an "unescaped text" element (such
         | 
| 52 | 
            +
              as `<noembed>` or `<xmp>`) were not properly escaped if that element was
         | 
| 53 | 
            +
              allowlisted and was also inside an allowlisted `<math>` or `<svg>` element.
         | 
| 54 | 
            +
             | 
| 55 | 
            +
              The only way to encounter this situation was to ignore multiple warnings in
         | 
| 56 | 
            +
              the readme and create a custom config that allowlisted all the elements
         | 
| 57 | 
            +
              involved, including `<math>` or `<svg>`. If you're using a default config or
         | 
| 58 | 
            +
              if you heeded the warnings about MathML and SVG not being supported, you're
         | 
| 59 | 
            +
              not affected by this issue.
         | 
| 60 | 
            +
             | 
| 61 | 
            +
              Please let this be a reminder that Sanitize cannot safely sanitize MathML or
         | 
| 62 | 
            +
              SVG content and does not support this use case. The default configs don't
         | 
| 63 | 
            +
              allow MathML or SVG elements, and allowlisting MathML or SVG elements in a
         | 
| 64 | 
            +
              custom config may create a security vulnerability in your application.
         | 
| 65 | 
            +
             | 
| 66 | 
            +
              Documentation has been updated to add more warnings and to make the existing
         | 
| 67 | 
            +
              warnings about this more prominent.
         | 
| 68 | 
            +
             | 
| 69 | 
            +
              Thanks to David Klein from [TU Braunschweig](https://www.tu-braunschweig.de/en/ias)
         | 
| 70 | 
            +
              (@leeN) for reporting this issue.
         | 
| 71 | 
            +
             | 
| 72 | 
            +
            ## 6.0.0 (2021-08-03)
         | 
| 73 | 
            +
             | 
| 74 | 
            +
            ### Potentially Breaking Changes
         | 
| 75 | 
            +
             | 
| 76 | 
            +
            * Ruby 2.5.0 is now the oldest officially supported Ruby version.
         | 
| 77 | 
            +
             | 
| 78 | 
            +
            * Sanitize now requires Nokogiri 1.12.0 or higher, which includes Nokogumbo.
         | 
| 79 | 
            +
              The separate dependency on Nokogumbo has been removed. [@lis2 - #211][211]
         | 
| 80 | 
            +
             | 
| 81 | 
            +
            [211]:https://github.com/rgrove/sanitize/pull/211
         | 
| 82 | 
            +
             | 
| 3 83 | 
             
            ## 5.2.3 (2021-01-11)
         | 
| 4 84 |  | 
| 5 85 | 
             
            ### Bug Fixes
         | 
    
        data/LICENSE
    CHANGED
    
    
    
        data/README.md
    CHANGED
    
    | @@ -11,27 +11,26 @@ protocols within attributes that contain URLs. You can also allow specific CSS | |
| 11 11 | 
             
            properties, @ rules, and URL protocols in elements or attributes containing CSS.
         | 
| 12 12 | 
             
            Any HTML or CSS that you don't explicitly allow will be removed.
         | 
| 13 13 |  | 
| 14 | 
            -
            Sanitize is based on [ | 
| 15 | 
            -
             | 
| 16 | 
            -
             | 
| 17 | 
            -
             | 
| 18 | 
            -
             | 
| 14 | 
            +
            Sanitize is based on the [Nokogiri HTML5 parser][nokogiri], which parses HTML
         | 
| 15 | 
            +
            the same way modern browsers do, and [Crass][crass], which parses CSS the same
         | 
| 16 | 
            +
            way modern browsers do. As long as your allowlist config only allows safe markup
         | 
| 17 | 
            +
            and CSS, even the most malformed or malicious input will be transformed into
         | 
| 18 | 
            +
            safe output.
         | 
| 19 19 |  | 
| 20 20 | 
             
            [](http://badge.fury.io/rb/sanitize)
         | 
| 21 21 | 
             
            [](https://github.com/rgrove/sanitize/actions?query=workflow%3ATests)
         | 
| 22 22 |  | 
| 23 23 | 
             
            [crass]:https://github.com/rgrove/crass
         | 
| 24 | 
            -
            [ | 
| 24 | 
            +
            [nokogiri]:https://github.com/sparklemotion/nokogiri
         | 
| 25 25 |  | 
| 26 26 | 
             
            Links
         | 
| 27 27 | 
             
            -----
         | 
| 28 28 |  | 
| 29 29 | 
             
            * [Home](https://github.com/rgrove/sanitize/)
         | 
| 30 | 
            -
            * [API Docs]( | 
| 30 | 
            +
            * [API Docs](https://rubydoc.info/github/rgrove/sanitize/Sanitize)
         | 
| 31 31 | 
             
            * [Issues](https://github.com/rgrove/sanitize/issues)
         | 
| 32 | 
            -
            * [Release History](https://github.com/rgrove/sanitize/ | 
| 33 | 
            -
            * [Online Demo](https://sanitize. | 
| 34 | 
            -
            * [Biased comparison of Ruby HTML sanitization libraries](https://github.com/rgrove/sanitize/blob/master/COMPARISON.md)
         | 
| 32 | 
            +
            * [Release History](https://github.com/rgrove/sanitize/releases)
         | 
| 33 | 
            +
            * [Online Demo](https://sanitize-web.fly.dev/)
         | 
| 35 34 |  | 
| 36 35 | 
             
            Installation
         | 
| 37 36 | 
             
            -------------
         | 
| @@ -72,10 +71,11 @@ Sanitize can sanitize the following types of input: | |
| 72 71 | 
             
            * Standalone CSS stylesheets
         | 
| 73 72 | 
             
            * Standalone CSS properties
         | 
| 74 73 |  | 
| 75 | 
            -
             | 
| 76 | 
            -
             | 
| 77 | 
            -
             | 
| 78 | 
            -
             | 
| 74 | 
            +
            > **Warning**
         | 
| 75 | 
            +
            >
         | 
| 76 | 
            +
            > Sanitize cannot fully sanitize the contents of `<math>` or `<svg>` elements. MathML and SVG elements are [foreign elements](https://html.spec.whatwg.org/multipage/syntax.html#foreign-elements) that don't follow normal HTML parsing rules.
         | 
| 77 | 
            +
            >
         | 
| 78 | 
            +
            > By default, Sanitize will remove all MathML and SVG elements. If you add MathML or SVG elements to a custom element allowlist, you may create a security vulnerability in your application.
         | 
| 79 79 |  | 
| 80 80 | 
             
            ### HTML Fragments
         | 
| 81 81 |  | 
| @@ -420,15 +420,21 @@ elements not in this array will be removed. | |
| 420 420 | 
             
            ]
         | 
| 421 421 | 
             
            ```
         | 
| 422 422 |  | 
| 423 | 
            -
            **Warning | 
| 424 | 
            -
             | 
| 425 | 
            -
             | 
| 426 | 
            -
             | 
| 427 | 
            -
            removed by Sanitize.
         | 
| 423 | 
            +
            > **Warning**
         | 
| 424 | 
            +
            >
         | 
| 425 | 
            +
            > Sanitize cannot fully sanitize the contents of `<math>` or `<svg>` elements. MathML and SVG elements are [foreign elements](https://html.spec.whatwg.org/multipage/syntax.html#foreign-elements) that don't follow normal HTML parsing rules.
         | 
| 426 | 
            +
            >
         | 
| 427 | 
            +
            > By default, Sanitize will remove all MathML and SVG elements. If you add MathML or SVG elements to a custom element allowlist, you must assume that any content inside them will be allowed, even if that content would otherwise be removed or escaped by Sanitize. This may create a security vulnerability in your application.
         | 
| 428 | 
            +
             | 
| 429 | 
            +
            > **Note**
         | 
| 430 | 
            +
            >
         | 
| 431 | 
            +
            > Sanitize always removes `<noscript>` elements and their contents, even if `noscript` is in the allowlist.
         | 
| 432 | 
            +
            >
         | 
| 433 | 
            +
            > This is because a `<noscript>` element's content is parsed differently in browsers depending on whether or not scripting is enabled. Since Nokogiri doesn't support scripting, it always parses `<noscript>` elements as if scripting is disabled. This results in edge cases where it's not possible to reliably sanitize the contents of a `<noscript>` element because Nokogiri can't fully replicate the parsing behavior of a scripting-enabled browser.
         | 
| 428 434 |  | 
| 429 435 | 
             
            #### :parser_options (Hash)
         | 
| 430 436 |  | 
| 431 | 
            -
            [Parsing options](https://github.com/rubys/nokogumbo/tree/ | 
| 437 | 
            +
            [Parsing options](https://github.com/rubys/nokogumbo/tree/master#parsing-options) to be supplied to `nokogumbo`.
         | 
| 432 438 |  | 
| 433 439 | 
             
            ```ruby
         | 
| 434 440 | 
             
            :parser_options => {
         | 
| @@ -667,25 +673,3 @@ html = %[ | |
| 667 673 | 
             
            Sanitize.fragment(html, :transformers => youtube_transformer)
         | 
| 668 674 | 
             
            # => '<iframe width="420" height="315" src="//www.youtube.com/embed/dQw4w9WgXcQ" frameborder="0" allowfullscreen=""></iframe>'
         | 
| 669 675 | 
             
            ```
         | 
| 670 | 
            -
             | 
| 671 | 
            -
            License
         | 
| 672 | 
            -
            -------
         | 
| 673 | 
            -
             | 
| 674 | 
            -
            Copyright (c) 2015 Ryan Grove (ryan@wonko.com)
         | 
| 675 | 
            -
             | 
| 676 | 
            -
            Permission is hereby granted, free of charge, to any person obtaining a copy of
         | 
| 677 | 
            -
            this software and associated documentation files (the 'Software'), to deal in
         | 
| 678 | 
            -
            the Software without restriction, including without limitation the rights to
         | 
| 679 | 
            -
            use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
         | 
| 680 | 
            -
            the Software, and to permit persons to whom the Software is furnished to do so,
         | 
| 681 | 
            -
            subject to the following conditions:
         | 
| 682 | 
            -
             | 
| 683 | 
            -
            The above copyright notice and this permission notice shall be included in all
         | 
| 684 | 
            -
            copies or substantial portions of the Software.
         | 
| 685 | 
            -
             | 
| 686 | 
            -
            THE SOFTWARE IS PROVIDED 'AS IS', WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         | 
| 687 | 
            -
            IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
         | 
| 688 | 
            -
            FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
         | 
| 689 | 
            -
            COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
         | 
| 690 | 
            -
            IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
         | 
| 691 | 
            -
            CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
         | 
| @@ -54,6 +54,11 @@ class Sanitize | |
| 54 54 |  | 
| 55 55 | 
             
                  # HTML elements to allow. By default, no elements are allowed (which means
         | 
| 56 56 | 
             
                  # that all HTML will be stripped).
         | 
| 57 | 
            +
                  #
         | 
| 58 | 
            +
                  # Warning: Sanitize cannot safely sanitize the contents of foreign
         | 
| 59 | 
            +
                  # elements (elements in the MathML or SVG namespaces). Do not add `math`
         | 
| 60 | 
            +
                  # or `svg` to this list! If you do, you may create a security
         | 
| 61 | 
            +
                  # vulnerability in your application.
         | 
| 57 62 | 
             
                  :elements => [],
         | 
| 58 63 |  | 
| 59 64 | 
             
                  # HTML parsing options to pass to Nokogumbo.
         | 
| @@ -1,5 +1,6 @@ | |
| 1 1 | 
             
            # encoding: utf-8
         | 
| 2 2 |  | 
| 3 | 
            +
            require 'cgi'
         | 
| 3 4 | 
             
            require 'set'
         | 
| 4 5 |  | 
| 5 6 | 
             
            class Sanitize; module Transformers; class CleanElement
         | 
| @@ -18,6 +19,18 @@ class Sanitize; module Transformers; class CleanElement | |
| 18 19 | 
             
              # http://www.whatwg.org/specs/web-apps/current-work/multipage/elements.html#embedding-custom-non-visible-data-with-the-data-*-attributes
         | 
| 19 20 | 
             
              REGEX_DATA_ATTR = /\Adata-(?!xml)[a-z_][\w.\u00E0-\u00F6\u00F8-\u017F\u01DD-\u02AF-]*\z/u
         | 
| 20 21 |  | 
| 22 | 
            +
              # Elements whose content is treated as unescaped text by HTML parsers.
         | 
| 23 | 
            +
              UNESCAPED_TEXT_ELEMENTS = Set.new(%w[
         | 
| 24 | 
            +
                iframe
         | 
| 25 | 
            +
                noembed
         | 
| 26 | 
            +
                noframes
         | 
| 27 | 
            +
                noscript
         | 
| 28 | 
            +
                plaintext
         | 
| 29 | 
            +
                script
         | 
| 30 | 
            +
                style
         | 
| 31 | 
            +
                xmp
         | 
| 32 | 
            +
              ])
         | 
| 33 | 
            +
             | 
| 21 34 | 
             
              # Attributes that need additional escaping on `<a>` elements due to unsafe
         | 
| 22 35 | 
             
              # libxml2 behavior.
         | 
| 23 36 | 
             
              UNSAFE_LIBXML_ATTRS_A = Set.new(%w[
         | 
| @@ -185,6 +198,28 @@ class Sanitize; module Transformers; class CleanElement | |
| 185 198 | 
             
                  @add_attributes[name].each {|key, val| node[key] = val }
         | 
| 186 199 | 
             
                end
         | 
| 187 200 |  | 
| 201 | 
            +
                # Make a best effort to ensure that text nodes in invalid "unescaped text"
         | 
| 202 | 
            +
                # elements that are inside a math or svg namespace are properly escaped so
         | 
| 203 | 
            +
                # that they don't get parsed as HTML.
         | 
| 204 | 
            +
                #
         | 
| 205 | 
            +
                # Sanitize is explicitly documented as not supporting MathML or SVG, but
         | 
| 206 | 
            +
                # people sometimes allow `<math>` and `<svg>` elements in their custom
         | 
| 207 | 
            +
                # configs without realizing that it's not safe. This workaround makes it
         | 
| 208 | 
            +
                # slightly less unsafe, but you still shouldn't allow `<math>` or `<svg>`
         | 
| 209 | 
            +
                # because Nokogiri doesn't parse them the same way browsers do and Sanitize
         | 
| 210 | 
            +
                # can't guarantee that their contents are safe.
         | 
| 211 | 
            +
                unless node.namespace.nil?
         | 
| 212 | 
            +
                  prefix = node.namespace.prefix
         | 
| 213 | 
            +
             | 
| 214 | 
            +
                  if (prefix == 'math' || prefix == 'svg') && UNESCAPED_TEXT_ELEMENTS.include?(name)
         | 
| 215 | 
            +
                    node.children.each do |child|
         | 
| 216 | 
            +
                      if child.type == Nokogiri::XML::Node::TEXT_NODE
         | 
| 217 | 
            +
                        child.content = CGI.escapeHTML(child.content)
         | 
| 218 | 
            +
                      end
         | 
| 219 | 
            +
                    end
         | 
| 220 | 
            +
                  end
         | 
| 221 | 
            +
                end
         | 
| 222 | 
            +
             | 
| 188 223 | 
             
                # Element-specific special cases.
         | 
| 189 224 | 
             
                case name
         | 
| 190 225 |  | 
| @@ -217,6 +252,16 @@ class Sanitize; module Transformers; class CleanElement | |
| 217 252 |  | 
| 218 253 | 
             
                    node['content'] = node['content'].gsub(/;\s*charset\s*=.+\z/, ';charset=utf-8')
         | 
| 219 254 | 
             
                  end
         | 
| 255 | 
            +
             | 
| 256 | 
            +
                # A `<noscript>` element's content is parsed differently in browsers
         | 
| 257 | 
            +
                # depending on whether or not scripting is enabled. Since Nokogiri doesn't
         | 
| 258 | 
            +
                # support scripting, it always parses `<noscript>` elements as if scripting
         | 
| 259 | 
            +
                # is disabled. This results in edge cases where it's not possible to
         | 
| 260 | 
            +
                # reliably sanitize the contents of a `<noscript>` element because Nokogiri
         | 
| 261 | 
            +
                # can't fully replicate the parsing behavior of a scripting-enabled browser.
         | 
| 262 | 
            +
                # The safest thing to do is to simply remove all `<noscript>` elements.
         | 
| 263 | 
            +
                when 'noscript'
         | 
| 264 | 
            +
                  node.unlink
         | 
| 220 265 | 
             
                end
         | 
| 221 266 | 
             
              end
         | 
| 222 267 |  | 
    
        data/lib/sanitize/version.rb
    CHANGED
    
    
    
        data/lib/sanitize.rb
    CHANGED
    
    
    
        data/test/test_clean_comment.rb
    CHANGED
    
    | @@ -11,18 +11,18 @@ describe 'Sanitize::Transformers::CleanComment' do | |
| 11 11 | 
             
                end
         | 
| 12 12 |  | 
| 13 13 | 
             
                it 'should remove comments' do
         | 
| 14 | 
            -
                  @s.fragment('foo <!-- comment --> bar').must_equal 'foo  bar'
         | 
| 15 | 
            -
                  @s.fragment('foo <!-- ').must_equal 'foo '
         | 
| 16 | 
            -
                  @s.fragment('foo <!-- - -> bar').must_equal 'foo '
         | 
| 17 | 
            -
                  @s.fragment("foo <!--\n\n\n\n-->bar").must_equal 'foo bar'
         | 
| 18 | 
            -
                  @s.fragment("foo <!-- <!-- <!-- --> --> -->bar").must_equal 'foo  --> -->bar'
         | 
| 19 | 
            -
                  @s.fragment("foo <div <!-- comment -->>bar</div>").must_equal 'foo <div>>bar</div>'
         | 
| 14 | 
            +
                  _(@s.fragment('foo <!-- comment --> bar')).must_equal 'foo  bar'
         | 
| 15 | 
            +
                  _(@s.fragment('foo <!-- ')).must_equal 'foo '
         | 
| 16 | 
            +
                  _(@s.fragment('foo <!-- - -> bar')).must_equal 'foo '
         | 
| 17 | 
            +
                  _(@s.fragment("foo <!--\n\n\n\n-->bar")).must_equal 'foo bar'
         | 
| 18 | 
            +
                  _(@s.fragment("foo <!-- <!-- <!-- --> --> -->bar")).must_equal 'foo  --> -->bar'
         | 
| 19 | 
            +
                  _(@s.fragment("foo <div <!-- comment -->>bar</div>")).must_equal 'foo <div>>bar</div>'
         | 
| 20 20 |  | 
| 21 21 | 
             
                  # Special case: the comment markup is inside a <script>, which makes it
         | 
| 22 22 | 
             
                  # text content and not an actual HTML comment.
         | 
| 23 | 
            -
                  @s.fragment("<script><!-- comment --></script>").must_equal ''
         | 
| 23 | 
            +
                  _(@s.fragment("<script><!-- comment --></script>")).must_equal ''
         | 
| 24 24 |  | 
| 25 | 
            -
                  Sanitize.fragment("<script><!-- comment --></script>", :allow_comments => false, :elements => ['script'])
         | 
| 25 | 
            +
                  _(Sanitize.fragment("<script><!-- comment --></script>", :allow_comments => false, :elements => ['script']))
         | 
| 26 26 | 
             
                    .must_equal '<script><!-- comment --></script>'
         | 
| 27 27 | 
             
                end
         | 
| 28 28 | 
             
              end
         | 
| @@ -33,14 +33,14 @@ describe 'Sanitize::Transformers::CleanComment' do | |
| 33 33 | 
             
                end
         | 
| 34 34 |  | 
| 35 35 | 
             
                it 'should allow comments' do
         | 
| 36 | 
            -
                  @s.fragment('foo <!-- comment --> bar').must_equal 'foo <!-- comment --> bar'
         | 
| 37 | 
            -
                  @s.fragment('foo <!-- ').must_equal 'foo <!-- -->'
         | 
| 38 | 
            -
                  @s.fragment('foo <!-- - -> bar').must_equal 'foo <!-- - -> bar-->'
         | 
| 39 | 
            -
                  @s.fragment("foo <!--\n\n\n\n-->bar").must_equal "foo <!--\n\n\n\n-->bar"
         | 
| 40 | 
            -
                  @s.fragment("foo <!-- <!-- <!-- --> --> -->bar").must_equal 'foo <!-- <!-- <!-- --> --> -->bar'
         | 
| 41 | 
            -
                  @s.fragment("foo <div <!-- comment -->>bar</div>").must_equal 'foo <div>>bar</div>'
         | 
| 42 | 
            -
             | 
| 43 | 
            -
                  Sanitize.fragment("<script><!-- comment --></script>", :allow_comments => true, :elements => ['script'])
         | 
| 36 | 
            +
                  _(@s.fragment('foo <!-- comment --> bar')).must_equal 'foo <!-- comment --> bar'
         | 
| 37 | 
            +
                  _(@s.fragment('foo <!-- ')).must_equal 'foo <!-- -->'
         | 
| 38 | 
            +
                  _(@s.fragment('foo <!-- - -> bar')).must_equal 'foo <!-- - -> bar-->'
         | 
| 39 | 
            +
                  _(@s.fragment("foo <!--\n\n\n\n-->bar")).must_equal "foo <!--\n\n\n\n-->bar"
         | 
| 40 | 
            +
                  _(@s.fragment("foo <!-- <!-- <!-- --> --> -->bar")).must_equal 'foo <!-- <!-- <!-- --> --> -->bar'
         | 
| 41 | 
            +
                  _(@s.fragment("foo <div <!-- comment -->>bar</div>")).must_equal 'foo <div>>bar</div>'
         | 
| 42 | 
            +
             | 
| 43 | 
            +
                  _(Sanitize.fragment("<script><!-- comment --></script>", :allow_comments => true, :elements => ['script']))
         | 
| 44 44 | 
             
                    .must_equal '<script><!-- comment --></script>'
         | 
| 45 45 | 
             
                end
         | 
| 46 46 | 
             
              end
         | 
    
        data/test/test_clean_css.rb
    CHANGED
    
    | @@ -10,15 +10,15 @@ describe 'Sanitize::Transformers::CSS::CleanAttribute' do | |
| 10 10 | 
             
              end
         | 
| 11 11 |  | 
| 12 12 | 
             
              it 'should sanitize CSS properties in style attributes' do
         | 
| 13 | 
            -
                @s.fragment(%[
         | 
| 13 | 
            +
                _(@s.fragment(%[
         | 
| 14 14 | 
             
                  <div style="color: #fff; width: expression(alert(1)); /* <-- evil! */"></div>
         | 
| 15 | 
            -
                ].strip).must_equal %[
         | 
| 15 | 
            +
                ].strip)).must_equal %[
         | 
| 16 16 | 
             
                  <div style="color: #fff;  /* <-- evil! */"></div>
         | 
| 17 17 | 
             
                ].strip
         | 
| 18 18 | 
             
              end
         | 
| 19 19 |  | 
| 20 20 | 
             
              it 'should remove the style attribute if the sanitized CSS is empty' do
         | 
| 21 | 
            -
                @s.fragment('<div style="width: expression(alert(1))"></div>').
         | 
| 21 | 
            +
                _(@s.fragment('<div style="width: expression(alert(1))"></div>')).
         | 
| 22 22 | 
             
                  must_equal '<div></div>'
         | 
| 23 23 | 
             
              end
         | 
| 24 24 | 
             
            end
         | 
| @@ -46,7 +46,7 @@ describe 'Sanitize::Transformers::CSS::CleanElement' do | |
| 46 46 | 
             
                  </style>
         | 
| 47 47 | 
             
                ].strip
         | 
| 48 48 |  | 
| 49 | 
            -
                @s.fragment(html).must_equal %[
         | 
| 49 | 
            +
                _(@s.fragment(html)).must_equal %[
         | 
| 50 50 | 
             
                  <style>
         | 
| 51 51 | 
             
                  /* Yay CSS! */
         | 
| 52 52 | 
             
                  .foo { color: #fff; }
         | 
| @@ -62,6 +62,6 @@ describe 'Sanitize::Transformers::CSS::CleanElement' do | |
| 62 62 | 
             
              end
         | 
| 63 63 |  | 
| 64 64 | 
             
              it 'should remove the <style> element if the sanitized CSS is empty' do
         | 
| 65 | 
            -
                @s.fragment('<style></style>').must_equal ''
         | 
| 65 | 
            +
                _(@s.fragment('<style></style>')).must_equal ''
         | 
| 66 66 | 
             
              end
         | 
| 67 67 | 
             
            end
         | 
    
        data/test/test_clean_doctype.rb
    CHANGED
    
    | @@ -11,18 +11,18 @@ describe 'Sanitize::Transformers::CleanDoctype' do | |
| 11 11 | 
             
                end
         | 
| 12 12 |  | 
| 13 13 | 
             
                it 'should remove doctype declarations' do
         | 
| 14 | 
            -
                  @s.document('<!DOCTYPE html><html>foo</html>').must_equal "<html>foo</html>"
         | 
| 15 | 
            -
                  @s.fragment('<!DOCTYPE html>foo').must_equal 'foo'
         | 
| 14 | 
            +
                  _(@s.document('<!DOCTYPE html><html>foo</html>')).must_equal "<html>foo</html>"
         | 
| 15 | 
            +
                  _(@s.fragment('<!DOCTYPE html>foo')).must_equal 'foo'
         | 
| 16 16 | 
             
                end
         | 
| 17 17 |  | 
| 18 18 | 
             
                it 'should not allow doctype definitions in fragments' do
         | 
| 19 | 
            -
                  @s.fragment('<!DOCTYPE html><html>foo</html>')
         | 
| 19 | 
            +
                  _(@s.fragment('<!DOCTYPE html><html>foo</html>'))
         | 
| 20 20 | 
             
                    .must_equal "foo"
         | 
| 21 21 |  | 
| 22 | 
            -
                  @s.fragment('<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN"><html>foo</html>')
         | 
| 22 | 
            +
                  _(@s.fragment('<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN"><html>foo</html>'))
         | 
| 23 23 | 
             
                    .must_equal "foo"
         | 
| 24 24 |  | 
| 25 | 
            -
                  @s.fragment("<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Strict//EN\"\n    \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\"><html>foo</html>")
         | 
| 25 | 
            +
                  _(@s.fragment("<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Strict//EN\"\n    \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\"><html>foo</html>"))
         | 
| 26 26 | 
             
                    .must_equal "foo"
         | 
| 27 27 | 
             
                end
         | 
| 28 28 | 
             
              end
         | 
| @@ -33,38 +33,38 @@ describe 'Sanitize::Transformers::CleanDoctype' do | |
| 33 33 | 
             
                end
         | 
| 34 34 |  | 
| 35 35 | 
             
                it 'should allow doctype declarations in documents' do
         | 
| 36 | 
            -
                  @s.document('<!DOCTYPE html><html>foo</html>')
         | 
| 36 | 
            +
                  _(@s.document('<!DOCTYPE html><html>foo</html>'))
         | 
| 37 37 | 
             
                    .must_equal "<!DOCTYPE html><html>foo</html>"
         | 
| 38 38 |  | 
| 39 | 
            -
                  @s.document('<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN"><html>foo</html>')
         | 
| 39 | 
            +
                  _(@s.document('<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN"><html>foo</html>'))
         | 
| 40 40 | 
             
                    .must_equal "<!DOCTYPE html><html>foo</html>"
         | 
| 41 41 |  | 
| 42 | 
            -
                  @s.document("<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Strict//EN\"\n    \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\"><html>foo</html>")
         | 
| 42 | 
            +
                  _(@s.document("<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Strict//EN\"\n    \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\"><html>foo</html>"))
         | 
| 43 43 | 
             
                    .must_equal "<!DOCTYPE html><html>foo</html>"
         | 
| 44 44 | 
             
                end
         | 
| 45 45 |  | 
| 46 46 | 
             
                it 'should not allow obviously invalid doctype declarations in documents' do
         | 
| 47 | 
            -
                  @s.document('<!DOCTYPE blah blah blah><html>foo</html>')
         | 
| 47 | 
            +
                  _(@s.document('<!DOCTYPE blah blah blah><html>foo</html>'))
         | 
| 48 48 | 
             
                    .must_equal "<!DOCTYPE html><html>foo</html>"
         | 
| 49 49 |  | 
| 50 | 
            -
                  @s.document('<!DOCTYPE blah><html>foo</html>')
         | 
| 50 | 
            +
                  _(@s.document('<!DOCTYPE blah><html>foo</html>'))
         | 
| 51 51 | 
             
                    .must_equal "<!DOCTYPE html><html>foo</html>"
         | 
| 52 52 |  | 
| 53 | 
            -
                  @s.document('<!DOCTYPE html BLAH "-//W3C//DTD HTML 4.01//EN"><html>foo</html>')
         | 
| 53 | 
            +
                  _(@s.document('<!DOCTYPE html BLAH "-//W3C//DTD HTML 4.01//EN"><html>foo</html>'))
         | 
| 54 54 | 
             
                    .must_equal "<!DOCTYPE html><html>foo</html>"
         | 
| 55 55 |  | 
| 56 | 
            -
                  @s.document('<!whatever><html>foo</html>')
         | 
| 56 | 
            +
                  _(@s.document('<!whatever><html>foo</html>'))
         | 
| 57 57 | 
             
                    .must_equal "<html>foo</html>"
         | 
| 58 58 | 
             
                end
         | 
| 59 59 |  | 
| 60 60 | 
             
                it 'should not allow doctype definitions in fragments' do
         | 
| 61 | 
            -
                  @s.fragment('<!DOCTYPE html><html>foo</html>')
         | 
| 61 | 
            +
                  _(@s.fragment('<!DOCTYPE html><html>foo</html>'))
         | 
| 62 62 | 
             
                    .must_equal "foo"
         | 
| 63 63 |  | 
| 64 | 
            -
                  @s.fragment('<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN"><html>foo</html>')
         | 
| 64 | 
            +
                  _(@s.fragment('<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01//EN"><html>foo</html>'))
         | 
| 65 65 | 
             
                    .must_equal "foo"
         | 
| 66 66 |  | 
| 67 | 
            -
                  @s.fragment("<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Strict//EN\"\n    \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\"><html>foo</html>")
         | 
| 67 | 
            +
                  _(@s.fragment("<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Strict//EN\"\n    \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd\"><html>foo</html>"))
         | 
| 68 68 | 
             
                    .must_equal "foo"
         | 
| 69 69 | 
             
                end
         | 
| 70 70 | 
             
              end
         |