timers: do not use user object call/apply
Timers should work even if the user has monkey-patched `.call()` and `.apply()` to undesirable values. PR-URL: https://github.com/nodejs/node/pull/12960 Ref: https://github.com/nodejs/node/issues/12956 Reviewed-By: Refael Ackermann <refack@gmail.com> Reviewed-By: Jeremiah Senkpiel <fishrock123@rocketmail.com> Reviewed-By: James M Snell <jasnell@gmail.com> Reviewed-By: Joyee Cheung <joyeec9h3@gmail.com> Reviewed-By: Rich Trott <rtrott@gmail.com>
This commit is contained in:
parent
7c3a23b9c1
commit
98609fc1c4
@ -485,20 +485,20 @@ function ontimeout(timer) {
|
|||||||
if (typeof callback !== 'function')
|
if (typeof callback !== 'function')
|
||||||
return promiseResolve(callback, args[0]);
|
return promiseResolve(callback, args[0]);
|
||||||
if (!args)
|
if (!args)
|
||||||
callback.call(timer);
|
timer._onTimeout();
|
||||||
else {
|
else {
|
||||||
switch (args.length) {
|
switch (args.length) {
|
||||||
case 1:
|
case 1:
|
||||||
callback.call(timer, args[0]);
|
timer._onTimeout(args[0]);
|
||||||
break;
|
break;
|
||||||
case 2:
|
case 2:
|
||||||
callback.call(timer, args[0], args[1]);
|
timer._onTimeout(args[0], args[1]);
|
||||||
break;
|
break;
|
||||||
case 3:
|
case 3:
|
||||||
callback.call(timer, args[0], args[1], args[2]);
|
timer._onTimeout(args[0], args[1], args[2]);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
callback.apply(timer, args);
|
Function.prototype.apply.call(callback, timer, args);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (timer._repeat)
|
if (timer._repeat)
|
||||||
@ -806,7 +806,7 @@ function runCallback(timer) {
|
|||||||
return timer._callback(argv[0], argv[1], argv[2]);
|
return timer._callback(argv[0], argv[1], argv[2]);
|
||||||
// more than 3 arguments run slower with .apply
|
// more than 3 arguments run slower with .apply
|
||||||
default:
|
default:
|
||||||
return timer._callback.apply(timer, argv);
|
return Function.prototype.apply.call(timer._callback, timer, argv);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
40
test/parallel/test-timers-user-call.js
Normal file
40
test/parallel/test-timers-user-call.js
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
// Make sure `setTimeout()` and friends don't throw if the user-supplied
|
||||||
|
// function has .call() and .apply() monkey-patched to undesirable values.
|
||||||
|
|
||||||
|
// Refs: https://github.com/nodejs/node/issues/12956
|
||||||
|
|
||||||
|
'use strict';
|
||||||
|
|
||||||
|
const common = require('../common');
|
||||||
|
|
||||||
|
{
|
||||||
|
const fn = common.mustCall(10);
|
||||||
|
fn.call = 'not a function';
|
||||||
|
fn.apply = 'also not a function';
|
||||||
|
setTimeout(fn, 1);
|
||||||
|
setTimeout(fn, 1, 'oneArg');
|
||||||
|
setTimeout(fn, 1, 'two', 'args');
|
||||||
|
setTimeout(fn, 1, 'three', '(3)', 'args');
|
||||||
|
setTimeout(fn, 1, 'more', 'than', 'three', 'args');
|
||||||
|
|
||||||
|
setImmediate(fn, 1);
|
||||||
|
setImmediate(fn, 1, 'oneArg');
|
||||||
|
setImmediate(fn, 1, 'two', 'args');
|
||||||
|
setImmediate(fn, 1, 'three', '(3)', 'args');
|
||||||
|
setImmediate(fn, 1, 'more', 'than', 'three', 'args');
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
const testInterval = (...args) => {
|
||||||
|
const fn = common.mustCall(() => { clearInterval(interval); });
|
||||||
|
fn.call = 'not a function';
|
||||||
|
fn.apply = 'also not a function';
|
||||||
|
const interval = setInterval(fn, 1, ...args);
|
||||||
|
};
|
||||||
|
|
||||||
|
testInterval();
|
||||||
|
testInterval('oneArg');
|
||||||
|
testInterval('two', 'args');
|
||||||
|
testInterval('three', '(3)', 'args');
|
||||||
|
testInterval('more', 'than', 'three', 'args');
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user