Skip to content

Commit e9042c4

Browse files
authoredApr 20, 2023
Handling non-configurable object descriptors on the prototype (#2508)
* regression test for #2491 * fix: ensure we can always restore our own spies * Add tests for mocks, fakes and stubs This shows an incoherent appraoch to how we deal with object descriptors across different code paths. * fix: only bother with unconfigurable descriptors if they are our own * remove test for sandbox.replace never supported undefined or protypal props in the first place. See #2195 for backing discussion on creating sinon.define() * fix linting
1 parent 3b41aff commit e9042c4

File tree

3 files changed

+53
-1
lines changed

3 files changed

+53
-1
lines changed
 

‎lib/sinon/stub.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -134,7 +134,7 @@ function assertValidPropertyDescriptor(descriptor, property) {
134134
if (!descriptor || !property) {
135135
return;
136136
}
137-
if (!descriptor.configurable && !descriptor.writable) {
137+
if (descriptor.isOwn && !descriptor.configurable && !descriptor.writable) {
138138
throw new TypeError(
139139
`Descriptor for property ${property} is non-configurable and non-writable`
140140
);

‎lib/sinon/util/core/wrap-method.js

+1
Original file line numberDiff line numberDiff line change
@@ -137,6 +137,7 @@ module.exports = function wrapMethod(object, property, method) {
137137
for (i = 0; i < types.length; i++) {
138138
mirrorProperties(methodDesc[types[i]], wrappedMethodDesc[types[i]]);
139139
}
140+
methodDesc.configurable = true;
140141
Object.defineProperty(object, property, methodDesc);
141142

142143
// catch failing assignment

‎test/issues/issues-test.js

+51
Original file line numberDiff line numberDiff line change
@@ -822,4 +822,55 @@ describe("issues", function () {
822822
assert.isTrue(fooStubInstance.wasCalled);
823823
});
824824
});
825+
826+
describe("#2491 - unable to restore spies on an instance where the prototype has an unconfigurable property descriptor", function () {
827+
function createInstanceFromClassWithReadOnlyPropertyDescriptor() {
828+
class BaseClass {}
829+
Object.defineProperty(BaseClass.prototype, "aMethod", {
830+
value: function () {
831+
return 42;
832+
},
833+
});
834+
835+
// anchor
836+
const instance = new BaseClass();
837+
assert.equals(instance.aMethod(), 42);
838+
839+
return instance;
840+
}
841+
842+
it("should ensure copied object descriptors are always configurable for spies", function () {
843+
const instance =
844+
createInstanceFromClassWithReadOnlyPropertyDescriptor();
845+
this.sandbox.spy(instance, "aMethod");
846+
847+
refute.exception(() => {
848+
this.sandbox.restore(); // #2491: this throws TypeError: Cannot assign to read only property 'myMethod' of object '#<BaseClass>'
849+
});
850+
});
851+
852+
it("should not throw if the unconfigurable object descriptor to be used for a Stub is on the prototype", function () {
853+
const instance =
854+
createInstanceFromClassWithReadOnlyPropertyDescriptor();
855+
856+
// per #2491 this throws 'TypeError: Descriptor for property aMethod is non-configurable and non-writable'
857+
// that makes sense for descriptors taken from the object, but not its prototype, as we are free to change
858+
// the latter when setting it
859+
refute.exception(() => {
860+
this.sandbox.stub(instance, "aMethod").returns("a stub");
861+
});
862+
});
863+
864+
it("should not throw if the unconfigurable object descriptor to be used for a Mock is on the prototype", function () {
865+
const instance =
866+
createInstanceFromClassWithReadOnlyPropertyDescriptor();
867+
868+
const mock = this.sandbox.mock(instance);
869+
870+
// per #2491 this throws 'TypeError: Attempted to wrap undefined property myMethod as function
871+
refute.exception(() => {
872+
mock.expects("aMethod").once();
873+
});
874+
});
875+
});
825876
});

0 commit comments

Comments
 (0)
Please sign in to comment.