admin管理员组文章数量:1344708
I have a rateLimit
function (which is just a modified version of this code):
function rateLimit(func, wait) {
var timeout;
return function () {
var context = this;
var args = arguments;
var later = function () {
timeout = null;
func.apply(context, args);
};
clearTimeout(timeout);
timeout = setTimeout(later, wait);
};
}
This function works perfectly well in my app, so I'm fairly certain the implementation is fine. However, the following test fails:
jest.useFakeTimers();
test('rateLimit', () => {
const action = jest.fn();
const doAction = rateLimit(action, 100);
doAction(); // This should increment the call count
doAction(); // This shouldn't, because 100ms hasn't elapsed yet
jest.advanceTimersByTime(101);
doAction(); // This should increment the count again
expect(action).toHaveBeenCalledTimes(2);
});
With the error:
Expected mock function to have been called two times, but it was called one time.
Here is a runnable version of this code on repl.it.
What am I doing wrong here?
I have a rateLimit
function (which is just a modified version of this code):
function rateLimit(func, wait) {
var timeout;
return function () {
var context = this;
var args = arguments;
var later = function () {
timeout = null;
func.apply(context, args);
};
clearTimeout(timeout);
timeout = setTimeout(later, wait);
};
}
This function works perfectly well in my app, so I'm fairly certain the implementation is fine. However, the following test fails:
jest.useFakeTimers();
test('rateLimit', () => {
const action = jest.fn();
const doAction = rateLimit(action, 100);
doAction(); // This should increment the call count
doAction(); // This shouldn't, because 100ms hasn't elapsed yet
jest.advanceTimersByTime(101);
doAction(); // This should increment the count again
expect(action).toHaveBeenCalledTimes(2);
});
With the error:
Expected mock function to have been called two times, but it was called one time.
Here is a runnable version of this code on repl.it.
What am I doing wrong here?
Share Improve this question asked Mar 27, 2019 at 6:41 Mark BellMark Bell 29.8k26 gold badges121 silver badges150 bronze badges2 Answers
Reset to default 8Your rate limiter uses a trailing
method where it cancels any calls currently in progress when new ones e in...until the wait time has expired at which point the function is called.
You just need to advance the timers again so the function gets called again:
jest.useFakeTimers();
test('rateLimit', () => {
const action = jest.fn();
const doAction = rateLimit(action, 100);
doAction(); // This should increment the call count
doAction(); // This shouldn't, because 100ms hasn't elapsed yet
jest.advanceTimersByTime(101);
doAction(); // This should increment the count again
jest.advanceTimersByTime(101); // <= advance the timers again
expect(action).toHaveBeenCalledTimes(2); // Success!
});
You should use the second call to fire the timers:
More info here
test('rateLimit', () => {
const action = jest.fn();
const doAction = rateLimit(action, 100);
doAction(); // This should increment the call count
doAction(); // This shouldn't, because 100ms hasn't elapsed yet
jest.runAllTimers();
doAction(); // This should increment the count again
jest.runAllTimers();
expect(action).toHaveBeenCalledTimes(2);
});
本文标签: javascriptTesting a function which uses setTimeout with Jest why is this test failingStack Overflow
版权声明:本文标题:javascript - Testing a function which uses setTimeout with Jest: why is this test failing? - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743706621a2525225.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
发表评论