-
-
Notifications
You must be signed in to change notification settings - Fork 6.5k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
mock.calls[][] holds reference instead of actual value when function called twice with object #434
Comments
This is because you are only doing the require once. |
That does not solve the problem because I want to test if the function has been called with an object and then with the same object mutated in the same test. So the beforeEach will run only once. jest.dontMock('../unmocked');
describe('getSearchResults', function(){
var unmocked;
var mocked;
beforeEach(function() {
unmocked = require('../unmocked');
mocked = require('../mocked');
});
it('mock.calls should deep copy objects', function(){
unmocked.func();
var firstCall = mocked.func.mock.calls[0][0];
var secondCall = mocked.func.mock.calls[1][0];
expect(firstCall).toEqual({ content: 'first' });
expect(secondCall).toEqual({ content: 'second' });
});
}); |
Cloning here would be too expensive for performance. I understand this is an issue but the side-effect in your function is probably not something you'd want in your application anyway, I'll assume? While I do not know what your application is doing, shallow copying the object in your function |
I'm facing the same issue. I'm constructing a nested object, validating it midway in the process, and use some final formatting in the end (date -> string). I'm mocking the validator, and have to do deep copy the object being passed to it. It would be nice that |
I've just stumbled onto this as being an issue, using I build an object with 'old data', call a function to act on it, change 1 property, and call a function to act on it. I feel like a mocked function should deep copy the parameters passed in. |
Stumbled upon the same thing here. Any chance this gets revisited / reopened? Right now our production code is doing unnecessary copies, just so the test framework works as expected. |
@cpojer - Why would cloning be too expensive? Can't there be some kind of reasonable setting that only does it if we are asking about multiple calls? Or can we just tell jest somehow via a setting: "in this situation, I want you to clone". As to your second question about why someone would do this without copying the object. I use typeorm and I have my code running in a loop to update a user's status from somewhere else -- this is from a 3rd party so I keep asking the 3rd party system what user's status is and save the result in my own db for caching purposes. The code more or less looks like this: async updateUserStatus(user) {
const maxTries = 5;
let tryNum = 1;
while (tryNum <= maxTries) {
await sleep(sleepTimeout);
const resp = await client.getStatus(user.id);
user.status = resp.status;
await this.userRespository.save(user);
if (user.status === 'success') {
return Promise.resolve();
}
}
this.log.info('Failed to get user\'s status in a sucess state');
} I ran into this and spent some time trying to debug the situation. I haven't used I totally get the perf issues but surely there's a reasonable middle ground. |
I wonder if it makes sense to add the ability to opt in to cloning |
I want to add my vote for an option to clone between calls. Deep cloning may be expensive, but it is better to have slightly slower tests than to have needless copies in production or code that cannot be tested. |
I like my own idea above, and would love to see a PR.
|
Lets say you agree with this statement, with the current implementation of jest there is no way to write a test that verifies you don't do this.
That does not help if you have a single call with the wrong value, then after that the value gets mutated to the correct value. The test will pass, even though the mock was called with the wrong value. |
We recently encountered the same problem. I think instead of cloning the object in my source code, I'd prefer to waste some extra memory in the test runner. @SimenB So having some methods like One can even call So, @cpojer would you consider to add this functionality. Or is there already a solution for that? |
I stumpled across this as well today and nearly lost my mind, before I found out it really is an issue of jest itself. I do understand the argument about performance, even though it is a clear side-effect caused by jest and not a real solution to add unnecessary The idea of @SimenB sounds like a solution which could fulfill performance requirements where object copies are not required. |
This issue has been automatically locked since there has not been any recent activity after it was closed. Please open a new issue for related bugs. |
Consider the following test:
With some arbitrary function being mocked in the test:
And a function which will be tested:
The tested function calls the mocked function with an object, mutates the object and calls the mocked function a second time.
mocked.func.mock.calls[0][0]
andmocked.func.mock.calls[1][0]
will now hold the same reference toobj
and the above test fails.I'm not sure if that's intended.
If it's not, shouldn't jest make a deep copy of the argument if it's an object?
I'm using
facebook/jest#0.5.x
.The text was updated successfully, but these errors were encountered: