See this comment for a revised version.
The override mistake is a problem introduced with ES5's ability to mark properties as non-writable. When a property on the prototype of an object is marked non-writable, it makes the property unaddable on the object.
A notable example of this is if we freeze (which marks all properties as non-writable) on Object.prototype
.
Object.freeze(Object.prototype);
When we do this, it becomes impossible to set toString()
on any objects that don't already have a .toString()
.
e.g. the following fails:
"use strict";
Object.freeze(Object.prototype);
const newObject = {};
newObject.toString = () => "Some string"; // TypeError
This is of particular concern to the SES proposal, which needs to freeze all builtins in order to guarantee security properties.
In this proposal we propose a new descriptor member called overridable
,
this member may only be present when writable
is set to false
.
When overridable
is present, non-writable properties on the prototype will be ignored when attempting to set them.
For example the following will print 10
:
"use strict";
const proto = {};
Object.defineProperty(proto, 'example', { value: 5, writable: false, overridable: true });
const object = { __proto__: proto };
object.example = 10;
console.log(object.example);
For SES, we need to be able to able to freeze entire objects, but without creating the problems of the override mistake.
As such a new function will be provided called Object.harden
, this method acts just like Object.freeze
,
except it sets overridable: true
on all descriptors (that are configurable).
For example, the following will just work now:
"use strict";
Object.harden(Object.prototype);
const object = {};
object.toString = () => "Hello world!";
console.log(object.toString()); // Hello world!
QUESTION: Should this simply be an option to freeze
rather than a new method?
If we expose overridable: boolean
as a new property on descriptors returned from Object.getOwnPropertyDescriptor(s)
,
will existing code break?
If so, should we add a new method to get overridable
to Object
(e.g. Object.getIsOverridable(object, name)
), or should we expose overridable
only when it is set to true
.
Based on the above comment, this is a possible candidate for v2:
Harden - Fixing the override mistake
The override mistake
The override mistake is a problem introduced with ES5's ability to mark properties as non-writable.
When a property on the prototype of an object is marked non-writable, it makes the property unaddable on the object.
A notable example of this is if we freeze (which marks all properties as non-writable) on
Object.prototype
.When we do this, it becomes impossible to set
toString()
on any objects that don't already have a.toString()
.e.g. the following fails:
This is of particular concern to the SES proposal, which needs to freeze all builtins in order to guarantee security properties.
Solution: Add
[[OverridableProperties]]
internal slot, and set it duringharden
The override mistake as it turns out is not a fundmental rule in the meta-object model, but rather a consequence of how implementations of
[[Set]]
handle non-writable descriptors. In particularOrdinarySetWithOwnDescriptor
returnsfalse
if the descriptor is non-writable, regardless of whether the receiver could be written to or not.A previous pull request showed how the override mistake may be turned off entirely, however this turns out to not be web compatible. What the PR does show us though is that we could change
OrdinarySetWithOwnDescriptor
to have an OPT-IN mechanism for fixing the override mistake.As such this proposal proposes a single new internal slot on ordinary objects
[[OverridableProperties]]
, this slot is a list of properties that are allowed to be overriden inOrdinarySetWithOwnDescriptor
. In order to populate this list, we add a new global methodharden
, this function performsObject.freeze
on the object and populates[[OverridableProperties]]
with all properties that use avalue
(notget
/set
) descriptor.For example the following will print
10
:In this example, once
proto
is hardened it's[[OverridableProperties]]
will contain"x"
andSymbol.iterator
, this allows them to be overriden. This would be accomplished effectively the same as in the previous pull request to fix the override mistake, except we check first that the propertyP
is within[[OverridableProperties]]
.FAQ
What does
harden
do on exotic objects?This is TBD, but either simply behaving like
Object.freeze
or throwing an error are two possible strategies.What about
[[OverridableProperties]]
on builtin exotic objects?This is TBD.
What about
[[OveriddableProperties]]
on other exotic objects?We don't propose changing those objects in this proposal, most objects in typical environments are already ordinary objects so they will automatically gain this behaviour. If there are any signficant host exotic objects that need this behaviour they will need to be changed by the host.
Can we override individual properties, rather than all or nothing?
This is also TBD, the proposal currently only does all properties as this is probably the most common case (e.g. in SES
lockdown
will simply callharden
on all transitively reachable objects). If there are use cases for individual properties, or excluding properties this might be open to change.