Gate is an utility to await multiple asynchronous calls in Node environment.
$ npm install gate
You can get each asynchronous result by index or name.
var gate = require('gate');
var fs = require('fs');
var g = gate.create();
fs.readFile('file1', 'utf8', g.latch(1));
fs.readFile('file2', 'utf8', g.latch(1));
g.await(function (err, results) {
if (err) throw err;
console.log(results[0]); // content for file1
console.log(results[1]); // content for file2
});
var gate = require('gate');
var fs = require('fs');
var g = gate.create();
fs.readFile('file1', 'utf8', g.latch('file1Result', 1));
fs.readFile('file2', 'utf8', g.latch('file2Result', 1));
g.await(function (err, results) {
if (err) throw err;
console.log(results.file1Result); // content for file1
console.log(results.file2Result); // content for file2
});
A extra field is added to an Error object.
error.gate_locatioin
: The asynchronous call location that the error occurred.
gate
module provides following API.
Returns a Gate object.
options
: Optional. Theoptions
can have followng keys.
KEY | TYPE | DEFAULT VALUE | DESCRIPTION |
---|---|---|---|
count | Number | -1 | A number of times the returned function must be called before an awaiting callback can start. Negative value means that count is not specified. |
failFast | Boolean | true | Indicates whether an awaiting callback is invoked as soon as possible when any error is found. If failFast is true, the found error is set as first argument of the awaiting callback. |
var g = gate.create();
var g = gate.create({count: 5, failFast: false}});
--
Gate
objects provide following API.
Returns a callback. The callback arguments are mapped with a mapping
definition.
If a count is given to gate.create()
, the count is decremented.
name
: Optional. A name for callback arguments. If not specified, an index number is used as name.
var g = gate.create();
fs.readFile('file1', 'utf8', g.latch('file1Result', {data: 1})); // name is specified
fs.readFile('file2', 'utf8', g.latch({data: 1})); // name is not specified
g.await(function (err, results) {
if (err) throw err;
console.log(results.file1Result.data); // get by name
console.log(results[1].data); // get by index
});
mapping
: Optional. An argument mapping definition. Themapping
gives names to callback arguments. Themappipng
must be a number or an object.- If the
mapping
is a number, single argument is mapped. - If the
mapping
is an object, multiple arguments can be mapped. - If the
mapping
isnull
orundefined
, all arguments are mapped as Array.
var g = gate.create();
fs.readFile('file1', 'utf8', g.latch(1)); // single argument
fs.readFile('file2', 'utf8', g.latch({data: 1, name: 'file2'})); // multiple arguments
fs.readFile('file3', 'utf8', g.latch()); // all arguments
g.await(function (err, results) {
if (err) throw err;
console.log(results[0]); // content for file1
console.log(results[1].data); // content for file2
console.log(results[1].name); // arbitrary value for file2
console.log(results[2][0]); // read error for file3 (1st argument of fs.readFile callback)
console.log(results[2][1]); // content for file3 (2nd argument of fs.readFile callback)
});
Indicates that a value is a plain value and it's not a mapping index.
value
: Required. A plain value.
var g = gate.create();
// a number for a `data` property is a mapping index, but a number for `g.val()` is a plain value
fs.readFile('file1', 'utf8', g.latch({data: 1, i: g.val(1)}));
fs.readFile('file2', 'utf8', g.latch({data: 1, i: g.val(2)}));
g.await(function (err, results) {
if (err) throw err;
console.log(results[0].data); // content for file1
console.log(results[0].i); // 1
console.log(results[1].data); // content for file2
console.log(results[1].i); // 2
});
Awaits all asynchronous calls completion and then runs a callback
.
callback
: Required. A callback to run after all asynchronous calls are done.err
: An error to indicate any asynhronous calls are failed.results
: An object to contain each asynchronous result as property.gate
: A new gate object.
var g = gate.create();
fs.readFile('file1', 'utf8', g.latch(1));
fs.readFile('file2', 'utf8', g.latch(1));
g.await(function (err, results) {
if (err) {
console.log(err);
} else {
console.log(results[0]);
console.log(results[1]);
}
});
Gets a current count, if a count is given to gate.latch()
.
Otherwise, -1
is returned.
This is a readonly property.
var g = gate.create(2);
console.log(g.count); // 2
fs.readFile('file1', 'utf8', g.latch(1));
console.log(g.count); // 1
fs.readFile('file2', 'utf8', g.latch(1));
console.log(g.count); // 0
Pass an argument index or an object includes argument indexs to a function being returned from gate.latch()
.
In the object, values except whose type is number
are recognized arguments.
To pass an number as argument, wrap it with val
function.
var gate = require('gate');
var fs = require('fs');
var exec = require('child_process').exec;
var g = gate.create();
// single mapping: arguments[1] in the callback will be result
fs.readFile('file1', 'utf8', g.latch(1));
// multiple mapping: arguments[1] and argments[2] in the callback will be result
exec('cat *.js bad_file | wc -l', g.latch({stdout: 1, stderr: 2}));
// all mapping: arguments will be result
fs.readFile('file2', 'utf8', g.latch());
g.await(function (err, results) {
if (err !== null) {
console.log('exec error: ' + err);
}
console.log('file1: ' + results[0]);
console.log('stdout: ' + results[1].stdout);
console.log('stderr: ' + results[1].stderr);
console.log('file2: ' + results[2]);
});
Pass a count number to gate.create()
to wait until a set of callbacks are done.
var gate = require('gate');
var fs = require('fs');
var files = ['file1', 'file2'];
var g = gate.create(files.length);
g.await(function (err, results) {
if (err) throw err;
console.log(results[0]);
console.log(results[1]);
});
setTimeout(function () {
files.forEach(function (file) {
fs.readFile(file, 'utf8', g.latch({name: file, data: 1}));
});
}, 0);
Check the first argument of the awaiting callback. If the argument is not null, it is any error object.
var gate = require('gate');
var fs = require('fs');
var g = gate.create();
fs.readFile('file1', 'utf8', g.latch({name: 'file1', data: 1}));
fs.readFile('non-existent', 'utf8', g.latch({name: 'non-existent', data: 1}));
g.await(function (err, results) {
// handle any error
if (err) {
console.log(err);
} else {
console.log(results);
}
});
Turn off failFaslt
option and include an error object in each result.
Then you can handle all errors by yourself.
var gate = require('gate');
var fs = require('fs');
var g = gate.create({failFast: false});
fs.readFile('non-existent1', 'utf8', g.latch({err: 0, data: 1})); // include an error object
fs.readFile('non-existent2', 'utf8', g.latch({err: 0, data: 1})); // include an error object
g.await(function (err, results) {
// handle all errors
Object.keys(results).forEach(function (key) {
var result = results[key];
if (result.err) {
console.log(result.err);
}
});
});
You can use third argument of an awaiting callback to nest 'gate.await()'.
var gate = require('gate');
var fs = require('fs');
var g = gate.create();
fs.readFile('file1', 'utf8', g.latch(1));
fs.readFile('file2', 'utf8', g.latch(1));
g.await(function (err, results, g) {
if (err) throw err;
var name1 = results[0]; // content for file1
var name2 = results[1]; // content for file2
fs.readFile(name1, 'utf8', g.latch(1));
fs.readFile(name2, 'utf8', g.latch(1));
g.await(function (err, results, g) {
if (err) throw err;
console.log(results[0]); // content for name1
console.log(results[1]); // content for name2
});
});