Using mocks as return values for mocks is impossible
jcdickinson opened this issue · 0 comments
jcdickinson commented
I'm trying to mock the IndexedDB IDBFactory
and it's return values (this is a real scenario for which I have no control over the interface). IDBOpenDBRequest has a property (result
) that returns an object (IDBDatabase
) that needs to be mocked. Because everything is cloned (including return values from properties):
MockBehavior.Strict
throws immediately on properties such aslength
(as lodash probes the object for array likeness etc. during clone).verifyAll()
doesn't work, as the clone targets some unknown (likely) cloned interceptor.
var behavior = MockBehavior.Strict; // Or Loose for (2)
interface Foo {
someMethod(): void;
}
interface Bar {
readonly foo: Foo;
}
var foo = Mock.ofType<Foo>(undefined, behavior);
foo.setup(x => x.someMethod()).verifiable(Times.once());
var bar = Mock.ofType<Bar>(undefined, behavior);
bar.setup(x => x.foo).returns(() => foo.object);
// Act
// 1. MockException is thrown as `length` is not mocked.
console.log(bar.object.foo);
// Assert
// 2. The method is called on the clone, which means it is not called on the value provided by foo.
foo.verifyAll();
There is no other way to solve the specific scenario I'm testing, and this does work in Moq.