Skip to content

Commit 868e11d

Browse files
authored
Data: Patch & warn about props from Object.prototype in data objects
Closes gh-561 Ref gh-559
1 parent b276616 commit 868e11d

File tree

5 files changed

+118
-0
lines changed

5 files changed

+118
-0
lines changed

src/jquery/data.js

+86
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,86 @@
1+
import { migratePatchFunc, migrateWarn } from "../main.js";
2+
3+
function patchDataProto( original, options ) {
4+
var i,
5+
apiName = options.apiName,
6+
isInstanceMethod = options.isInstanceMethod,
7+
8+
// `Object.prototype` keys are not enumerable so list the
9+
// official ones here. An alternative would be wrapping
10+
// data objects with a Proxy but that creates additional issues
11+
// like breaking object identity on subsequent calls.
12+
objProtoKeys = [
13+
"__proto__",
14+
"__defineGetter__",
15+
"__defineSetter__",
16+
"__lookupGetter__",
17+
"__lookupSetter__",
18+
"hasOwnProperty",
19+
"isPrototypeOf",
20+
"propertyIsEnumerable",
21+
"toLocaleString",
22+
"toString",
23+
"valueOf"
24+
],
25+
26+
// Use a null prototype at the beginning so that we can define our
27+
// `__proto__` getter & setter. We'll reset the prototype afterwards.
28+
intermediateDataObj = Object.create( null );
29+
30+
for ( i = 0; i < objProtoKeys.length; i++ ) {
31+
( function( key ) {
32+
Object.defineProperty( intermediateDataObj, key, {
33+
get: function() {
34+
migrateWarn( "data-null-proto",
35+
"Accessing properties from " + apiName +
36+
" inherited from Object.prototype is removed" );
37+
return ( key + "__cache" ) in intermediateDataObj ?
38+
intermediateDataObj[ key + "__cache" ] :
39+
Object.prototype[ key ];
40+
},
41+
set: function( value ) {
42+
migrateWarn( "data-null-proto",
43+
"Setting properties from " + apiName +
44+
" inherited from Object.prototype is removed" );
45+
intermediateDataObj[ key + "__cache" ] = value;
46+
}
47+
} );
48+
} )( objProtoKeys[ i ] );
49+
}
50+
51+
Object.setPrototypeOf( intermediateDataObj, Object.prototype );
52+
53+
return function jQueryDataProtoPatched() {
54+
var result = original.apply( this, arguments );
55+
56+
if ( arguments.length !== ( isInstanceMethod ? 0 : 1 ) || result === undefined ) {
57+
return result;
58+
}
59+
60+
// Insert an additional object in the prototype chain between `result`
61+
// and `Object.prototype`; that intermediate object proxies properties
62+
// to `Object.prototype`, warning about their usage first.
63+
Object.setPrototypeOf( result, intermediateDataObj );
64+
65+
return result;
66+
};
67+
}
68+
69+
// Yes, we are patching jQuery.data twice; here & above. This is necessary
70+
// so that each of the two patches can be independently disabled.
71+
migratePatchFunc( jQuery, "data",
72+
patchDataProto( jQuery.data, {
73+
apiName: "jQuery.data()",
74+
isPrivateData: false,
75+
isInstanceMethod: false
76+
} ),
77+
"data-null-proto" );
78+
migratePatchFunc( jQuery.fn, "data",
79+
patchDataProto( jQuery.fn.data, {
80+
apiName: "jQuery.fn.data()",
81+
isPrivateData: true,
82+
isInstanceMethod: true
83+
} ),
84+
"data-null-proto" );
85+
86+
// TODO entry in warnings.md

src/migrate.js

+1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ import "./jquery/selector.js";
66
import "./jquery/ajax.js";
77
import "./jquery/attributes.js";
88
import "./jquery/css.js";
9+
import "./jquery/data.js";
910
import "./jquery/effects.js";
1011
import "./jquery/event.js";
1112
import "./jquery/manipulation.js";

test/data/testinit.js

+1
Original file line numberDiff line numberDiff line change
@@ -62,6 +62,7 @@
6262
"unit/jquery/ajax.js",
6363
"unit/jquery/attributes.js",
6464
"unit/jquery/css.js",
65+
"unit/jquery/data.js",
6566
"unit/jquery/deferred.js",
6667
"unit/jquery/effects.js",
6768
"unit/jquery/event.js",

test/unit/jquery/data.js

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
QUnit.module( "data" );
2+
3+
QUnit.test( "properties from Object.prototype", function( assert ) {
4+
assert.expect( 6 );
5+
6+
var div = jQuery( "<div>" ).appendTo( "#qunit-fixture" );
7+
8+
div.data( "foo", "bar" );
9+
10+
expectNoMessage( assert, "Regular properties", function() {
11+
assert.strictEqual( div.data( "foo" ), "bar", "data access" );
12+
assert.strictEqual( jQuery.data( div[ 0 ], "foo" ), "bar", "data access (static method)" );
13+
} );
14+
15+
expectMessage( assert, "Properties from Object.prototype", 2, function() {
16+
assert.ok( div.data().hasOwnProperty( "foo" ),
17+
"hasOwnProperty works" );
18+
assert.ok( jQuery.data( div[ 0 ] ).hasOwnProperty( "foo" ),
19+
"hasOwnProperty works (static method)" );
20+
} );
21+
} );

warnings.md

+9
Original file line numberDiff line numberDiff line change
@@ -144,6 +144,15 @@ This is _not_ a warning, but a console log message the plugin shows when it firs
144144

145145
**Solution:** Always pass string values to `.css()`, and explicitly add units where required. For example, use `$.css("line-height", "2")` to specify 200% of the current line height or `$.css("line-height", "2px")` to specify pixels. When the numeric value is in a variable, ensure the value is converted to string, e.g. `$.css("line-height", String(height))` and `$.css("line-height", height+"px")`.
146146

147+
### \[data-null-proto\] Accessing properties from jQuery.data() inherited from Object.prototype is removed
148+
### \[data-null-proto\] Setting properties from jQuery.data() inherited from Object.prototype is removed
149+
### \[data-null-proto\] Accessing properties from jQuery.fn.data() inherited from Object.prototype is removed
150+
### \[data-null-proto\] Setting properties from jQuery.fn.data() inherited from Object.prototype is removed
151+
152+
**Cause:** As of jQuery 4.0.0, data objects no longer inherit from `Object.prototype`. This includes properties like `__proto__` or `hasOwnProperty`.
153+
154+
**Solution:** Don't use properties inherited from `Object.prototype` on data objects. Instead of `jQuery.data( node ).hasOwnProperty( "foo" )` use `Object.hasOwn( jQuery.data( node ), "foo" )` or, if you need to support older browsers like IE 11, use `Object.prototype.hasOwnProperty.call( jQuery.data( node ), "foo" )`.
155+
147156
### \[self-closed-tags\] JQMIGRATE: HTML tags must be properly nested and closed: _(HTML string)_
148157

149158
**Cause:** jQuery 3.5.0 changed the way it processes HTML strings. Previously, jQuery would attempt to fix self-closed tags like `<i class="test" />` that the HTML5 specification says are not self-closed, turning it into `<i class="test"></i>`. This processing can create a [security problem](https://nvd.nist.gov/vuln/detail/CVE-2020-11022) with malicious strings, so the functionality had to be removed.

0 commit comments

Comments
 (0)