Skip to content

Commit 4250b62

Browse files
authored
Attributes: Don't stringify attributes in the setter
Stringifying attributes in the setter was needed for IE <=9 but it breaks trusted types enforcement when setting a script `src` attribute. Note that this doesn't mean script execution works. Since jQuery disables all scripts by changing their type and then executes them by creating fresh script tags with proper `src` & possibly other attributes, this unwraps any trusted `src` wrappers, making the script not execute under strict CSP settings. We might try to fix it in the future in a separate change. Fixes gh-4948 Closes gh-4949
1 parent 4fd6912 commit 4250b62

6 files changed

+97
-1
lines changed

src/attributes/attr.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -50,7 +50,7 @@ jQuery.extend( {
5050
return ret;
5151
}
5252

53-
elem.setAttribute( name, value + "" );
53+
elem.setAttribute( name, value );
5454
return value;
5555
}
5656

test/data/mock.php

+6
Original file line numberDiff line numberDiff line change
@@ -247,6 +247,12 @@ protected function trustedHtml( $req ) {
247247
echo file_get_contents( __DIR__ . '/trusted-html.html' );
248248
}
249249

250+
protected function trustedTypesAttributes( $req ) {
251+
header( "Content-Security-Policy: require-trusted-types-for 'script'; report-uri ./mock.php?action=cspLog" );
252+
header( 'Content-type: text/html' );
253+
echo file_get_contents( __DIR__ . '/trusted-types-attributes.html' );
254+
}
255+
250256
protected function errorWithScript( $req ) {
251257
header( 'HTTP/1.0 404 Not Found' );
252258
if ( isset( $req->query['withScriptContentType'] ) ) {
+61
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,61 @@
1+
<!DOCTYPE html>
2+
<html>
3+
<head>
4+
<meta charset=utf-8 />
5+
<title>Trusted HTML attribute tests</title>
6+
</head>
7+
<body>
8+
<div id="qunit-fixture"></div>
9+
<script src="../../dist/jquery.js"></script>
10+
<script src="iframeTest.js"></script>
11+
<script>
12+
var i, input, elem, policy,
13+
results = [];
14+
15+
function runTests( messagePrefix, getTrustedScriptUrlWrapper ) {
16+
try {
17+
elem = jQuery( "<script><\/script>" )
18+
.attr( "src", getTrustedScriptUrlWrapper( "trusted-types-attributes.js" ) );
19+
elem.appendTo( document.body );
20+
21+
results.push( {
22+
actual: elem.attr( "src" ),
23+
expected: "trusted-types-attributes.js",
24+
message: messagePrefix + ": script URL properly set"
25+
} );
26+
} catch ( e ) {
27+
results.push( {
28+
actual: "error thrown",
29+
expected: "",
30+
message: messagePrefix + ": error has been thrown"
31+
} );
32+
}
33+
}
34+
35+
if ( typeof trustedTypes !== "undefined" ) {
36+
policy = trustedTypes.createPolicy( "jquery-test-policy", {
37+
createScriptURL: function( html ) {
38+
return html;
39+
}
40+
} );
41+
42+
runTests( "TrustedScriptURL", function wrapInTrustedScriptUrl( input ) {
43+
return policy.createScriptURL( input );
44+
} );
45+
} else {
46+
47+
// No TrustedScriptURL support so let's at least run tests with object wrappers
48+
// with a proper `toString` function. See trusted-html.html for more context.
49+
runTests( "Object wrapper", function( input ) {
50+
return {
51+
toString: function toString() {
52+
return input;
53+
}
54+
};
55+
} );
56+
}
57+
58+
startIframeTest( results );
59+
</script>
60+
</body>
61+
</html>

test/data/trusted-types-attributes.js

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
window.testMessage = "script run";

test/middleware-mockserver.js

+8
Original file line numberDiff line numberDiff line change
@@ -264,6 +264,14 @@ var mocks = {
264264
var body = fs.readFileSync( __dirname + "/data/trusted-html.html" ).toString();
265265
resp.end( body );
266266
},
267+
trustedTypesAttributes: function( req, resp ) {
268+
resp.writeHead( 200, {
269+
"Content-Type": "text/html",
270+
"Content-Security-Policy": "require-trusted-types-for 'script'; report-uri /base/test/data/mock.php?action=cspLog"
271+
} );
272+
var body = fs.readFileSync( __dirname + "/data/trusted-types-attributes.html" ).toString();
273+
resp.end( body );
274+
},
267275
errorWithScript: function( req, resp ) {
268276
if ( req.query.withScriptContentType ) {
269277
resp.writeHead( 404, { "Content-Type": "application/javascript" } );

test/unit/attributes.js

+20
Original file line numberDiff line numberDiff line change
@@ -1764,3 +1764,23 @@ QUnit.test( "non-lowercase boolean attribute getters should not crash", function
17641764
}
17651765
} );
17661766
} );
1767+
1768+
1769+
// Test trustedTypes support in browsers where they're supported (currently Chrome 83+).
1770+
// Browsers with no TrustedScriptURL support still run tests on object wrappers with
1771+
// a proper `toString` function.
1772+
testIframe(
1773+
"Basic TrustedScriptURL support (gh-4948)",
1774+
"mock.php?action=trustedTypesAttributes",
1775+
function( assert, jQuery, window, document, test ) {
1776+
var done = assert.async();
1777+
1778+
assert.expect( 1 );
1779+
1780+
test.forEach( function( result ) {
1781+
assert.deepEqual( result.actual, result.expected, result.message );
1782+
} );
1783+
1784+
supportjQuery.get( baseURL + "mock.php?action=cspClean" ).then( done );
1785+
}
1786+
);

0 commit comments

Comments
 (0)