Browse Source

timers: implement setImmediate

v0.9.1-release
Timothy J Fontaine 12 years ago
committed by Ben Noordhuis
parent
commit
382f22f229
  1. 17
      doc/api/timers.markdown
  2. 71
      lib/timers.js
  3. 10
      src/node.js
  4. 2
      test/common.js
  5. 53
      test/simple/test-timers-immediate.js

17
doc/api/timers.markdown

@ -46,3 +46,20 @@ event loop performance -- use wisely.
If you had previously `unref()`d a timer you can call `ref()` to explicitly
request the timer hold the program open. If the timer is already `ref`d calling
`ref` again will have no effect.
## setImmediate(callback, [arg], [...])
To schedule the "immediate" execution of `callback`. Returns an `immediateId`
for possible use with `clearImmediate()`. Optionally you can also pass
arguments to the callback.
Immediates are queued in the order created, and are popped off the queue once
per loop iteration. This is different from `process.nextTick` which will
execute `process.maxTickDepth` queued callbacks per iteration. `setImmediate`
will yield to the event loop after firing a queued callback to make sure I/O is
not being starved. While order is preserved for execution, other I/O events may
fire between any two scheduled immediate callbacks.
## clearImmediate(immediateId)
Stops an immediate from triggering.

71
lib/timers.js

@ -277,3 +277,74 @@ Timeout.prototype.close = function() {
exports.unenroll(this);
}
};
var immediateTimer = null;
var immediateQueue = { started: false };
L.init(immediateQueue);
function lazyImmediateInit() { // what's in a name?
if (immediateTimer) return;
immediateTimer = new Timer;
immediateTimer.ontimeout = processImmediate;
}
function processImmediate() {
var immediate;
if (L.isEmpty(immediateQueue)) {
immediateTimer.stop();
immediateQueue.started = false;
} else {
immediate = L.shift(immediateQueue);
if (immediate.domain) immediate.domain.enter();
immediate._onTimeout();
if (immediate.domain) immediate.domain.exit();
}
};
exports.setImmediate = function(callback) {
var immediate = {}, args;
L.init(immediate);
immediate._onTimeout = callback;
if (arguments.length > 1) {
args = Array.prototype.slice.call(arguments, 1);
immediate._onTimeout = function() {
callback.apply(null, args);
};
}
if (!immediateQueue.started) {
lazyImmediateInit();
immediateTimer.start(0, 1);
immediateQueue.started = true;
}
if (process.domain) immediate.domain = process.domain;
L.append(immediateQueue, immediate);
return immediate;
};
exports.clearImmediate = function(immediate) {
if (!immediate) return;
immediate._onTimeout = undefined;
L.remove(immediate);
if (L.isEmpty(immediateQueue)) {
immediateTimer.stop();
immediateQueue.started = false;
}
};

10
src/node.js

@ -184,6 +184,16 @@
var t = NativeModule.require('timers');
return t.clearInterval.apply(this, arguments);
};
global.setImmediate = function() {
var t = NativeModule.require('timers');
return t.setImmediate.apply(this, arguments);
};
global.clearImmediate = function() {
var t = NativeModule.require('timers');
return t.clearImmediate.apply(this, arguments);
};
};
startup.globalConsole = function() {

2
test/common.js

@ -81,8 +81,10 @@ process.on('exit', function() {
if (!exports.globalCheck) return;
var knownGlobals = [setTimeout,
setInterval,
setImmediate,
clearTimeout,
clearInterval,
clearImmediate,
console,
Buffer,
process,

53
test/simple/test-timers-immediate.js

@ -0,0 +1,53 @@
// Copyright Joyent, Inc. and other Node contributors.
//
// Permission is hereby granted, free of charge, to any person obtaining a
// copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to permit
// persons to whom the Software is furnished to do so, subject to the
// following conditions:
//
// The above copyright notice and this permission notice shall be included
// in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
// OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
// NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
// DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
// USE OR OTHER DEALINGS IN THE SOFTWARE.
var common = require('../common');
var assert = require('assert');
var immediateA = false,
immediateB = false,
immediateC = [],
before;
setImmediate(function() {
try {
immediateA = process.hrtime(before);
} catch(e) {
console.log('failed to get hrtime with offset');
}
clearImmediate(immediateB);
});
before = process.hrtime();
immediateB = setImmediate(function() {
immediateB = true;
});
setImmediate(function(x, y, z) {
immediateC = [x, y, z];
}, 1, 2, 3);
process.on('exit', function() {
assert.ok(immediateA, 'Immediate should happen after normal execution');
assert.notStrictEqual(immediateB, true, 'immediateB should not fire');
assert.deepEqual(immediateC, [1, 2, 3], 'immediateC args should match');
});
Loading…
Cancel
Save