Skip to content

Commit 9c98e4e

Browse files
mgolgibson042
andauthored
Manipulation: Avoid concatenating strings in buildFragment
Concatenating HTML strings in buildFragment is a possible security risk as it creates an opportunity of escaping the concatenated wrapper. It also makes it impossible to support secure HTML wrappers like [trusted types](https://web.dev/trusted-types/). It's safer to create wrapper elements using `document.createElement` & `appendChild`. The previous way was needed in jQuery <4 because IE <10 doesn't accept table parts set via `innerHTML`, even if the element which contents are set is a proper table element, e.g.: ```js tr.innerHTML = "<td></td>"; ``` The whole structure needs to be passed in one HTML string. jQuery 4 drops support for IE <11 so this is no longer an issue; in older version we'd have to duplicate the code paths. IE <10 needed to have `<option>` elements wrapped in `<select multiple="multiple">` but we no longer need that on master which makes the `document.createElement` way shorter as we don't have to call `setAttribute`. All these improvements, apart from making logic more secure, decrease the gzipped size by 58 bytes. Closes gh-4724 Ref gh-4409 Ref angular/angular.js#17028 Co-authored-by: Richard Gibson <[email protected]>
1 parent 7a6fae6 commit 9c98e4e

File tree

3 files changed

+23
-13
lines changed

3 files changed

+23
-13
lines changed

src/manipulation/buildFragment.js

+8-6
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
import jQuery from "../core.js";
22
import toType from "../core/toType.js";
33
import isAttached from "../core/isAttached.js";
4+
import arr from "../var/arr.js";
45
import rtagName from "./var/rtagName.js";
56
import rscriptType from "./var/rscriptType.js";
67
import wrapMap from "./wrapMap.js";
@@ -35,15 +36,16 @@ function buildFragment( elems, context, scripts, selection, ignored ) {
3536

3637
// Deserialize a standard representation
3738
tag = ( rtagName.exec( elem ) || [ "", "" ] )[ 1 ].toLowerCase();
38-
wrap = wrapMap[ tag ] || wrapMap._default;
39-
tmp.innerHTML = wrap[ 1 ] + jQuery.htmlPrefilter( elem ) + wrap[ 2 ];
39+
wrap = wrapMap[ tag ] || arr;
4040

41-
// Descend through wrappers to the right content
42-
j = wrap[ 0 ];
43-
while ( j-- ) {
44-
tmp = tmp.lastChild;
41+
// Create wrappers & descend into them.
42+
j = wrap.length;
43+
while ( --j > -1 ) {
44+
tmp = tmp.appendChild( context.createElement( wrap[ j ] ) );
4545
}
4646

47+
tmp.innerHTML = jQuery.htmlPrefilter( elem );
48+
4749
jQuery.merge( nodes, tmp.childNodes );
4850

4951
// Remember the top-level container

src/manipulation/wrapMap.js

+4-7
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,14 @@
1-
// We have to close these tags to support XHTML (#13200)
21
var wrapMap = {
32

43
// Table parts need to be wrapped with `<table>` or they're
54
// stripped to their contents when put in a div.
65
// XHTML parsers do not magically insert elements in the
76
// same way that tag soup parsers do, so we cannot shorten
87
// this by omitting <tbody> or other required elements.
9-
thead: [ 1, "<table>", "</table>" ],
10-
col: [ 2, "<table><colgroup>", "</colgroup></table>" ],
11-
tr: [ 2, "<table><tbody>", "</tbody></table>" ],
12-
td: [ 3, "<table><tbody><tr>", "</tr></tbody></table>" ],
13-
14-
_default: [ 0, "", "" ]
8+
thead: [ "table" ],
9+
col: [ "colgroup", "table" ],
10+
tr: [ "tbody", "table" ],
11+
td: [ "tr", "tbody", "table" ]
1512
};
1613

1714
wrapMap.tbody = wrapMap.tfoot = wrapMap.colgroup = wrapMap.caption = wrapMap.thead;

test/unit/manipulation.js

+11
Original file line numberDiff line numberDiff line change
@@ -2969,3 +2969,14 @@ QUnit.test( "Sanitized HTML doesn't get unsanitized", function( assert ) {
29692969
test( "<noembed><noembed/><img src=url404 onerror=xss(12)>" );
29702970
}
29712971
} );
2972+
2973+
QUnit.test( "Works with invalid attempts to close the table wrapper", function( assert ) {
2974+
assert.expect( 3 );
2975+
2976+
// This test case attempts to close the tags which wrap input
2977+
// based on matching done in wrapMap which should be ignored.
2978+
var elem = jQuery( "<td></td></tr></tbody></table><td></td>" );
2979+
assert.strictEqual( elem.length, 2, "Two elements created" );
2980+
assert.strictEqual( elem[ 0 ].nodeName.toLowerCase(), "td", "First element is td" );
2981+
assert.strictEqual( elem[ 1 ].nodeName.toLowerCase(), "td", "Second element is td" );
2982+
} );

0 commit comments

Comments
 (0)