You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

352 lines
8.5 KiB

'use strict';
function Kareem() {
this._pres = {};
this._posts = {};
}
Kareem.prototype.execPre = function(name, context, args, callback) {
if (arguments.length === 3) {
callback = args;
args = [];
}
var pres = this._pres[name] || [];
var numPres = pres.length;
var numAsyncPres = pres.numAsync || 0;
var currentPre = 0;
var asyncPresLeft = numAsyncPres;
var done = false;
var $args = args;
if (!numPres) {
return process.nextTick(function() {
callback(null);
});
}
var next = function() {
if (currentPre >= numPres) {
return;
}
var pre = pres[currentPre];
if (pre.isAsync) {
pre.fn.call(
context,
function(error) {
if (error) {
if (done) {
return;
}
done = true;
return callback(error);
}
++currentPre;
next.apply(context, arguments);
},
function(error) {
if (error) {
if (done) {
return;
}
done = true;
return callback(error);
}
if (--numAsyncPres === 0) {
return callback(null);
}
});
} else if (pre.fn.length > 0) {
var args = [function(error) {
if (error) {
if (done) {
return;
}
done = true;
return callback(error);
}
if (++currentPre >= numPres) {
if (asyncPresLeft > 0) {
// Leave parallel hooks to run
return;
} else {
return callback(null);
}
}
next.apply(context, arguments);
}];
var _args = arguments.length >= 2 ? arguments : [null].concat($args);
for (var i = 1; i < _args.length; ++i) {
args.push(_args[i]);
}
pre.fn.apply(context, args);
} else {
pre.fn.call(context);
if (++currentPre >= numPres) {
if (asyncPresLeft > 0) {
// Leave parallel hooks to run
return;
} else {
return process.nextTick(function() {
callback(null);
});
}
}
next();
}
};
next.apply(null, [null].concat(args));
};
Kareem.prototype.execPreSync = function(name, context) {
var pres = this._pres[name] || [];
var numPres = pres.length;
for (var i = 0; i < numPres; ++i) {
pres[i].fn.call(context);
}
};
Kareem.prototype.execPost = function(name, context, args, options, callback) {
if (arguments.length < 5) {
callback = options;
options = null;
}
var posts = this._posts[name] || [];
var numPosts = posts.length;
var currentPost = 0;
var firstError = null;
if (options && options.error) {
firstError = options.error;
}
if (!numPosts) {
return process.nextTick(function() {
callback.apply(null, [firstError].concat(args));
});
}
var next = function() {
var post = posts[currentPost];
var numArgs = 0;
var argLength = args.length;
var newArgs = [];
for (var i = 0; i < argLength; ++i) {
numArgs += args[i] && args[i]._kareemIgnore ? 0 : 1;
if (!args[i] || !args[i]._kareemIgnore) {
newArgs.push(args[i]);
}
}
if (firstError) {
if (post.length === numArgs + 2) {
post.apply(context, [firstError].concat(newArgs).concat(function(error) {
if (error) {
firstError = error;
}
if (++currentPost >= numPosts) {
return callback.call(null, firstError);
}
next();
}));
} else {
if (++currentPost >= numPosts) {
return callback.call(null, firstError);
}
next();
}
} else {
if (post.length === numArgs + 2) {
// Skip error handlers if no error
if (++currentPost >= numPosts) {
return callback.apply(null, [null].concat(args));
}
return next();
}
if (post.length === numArgs + 1) {
post.apply(context, newArgs.concat(function(error) {
if (error) {
firstError = error;
return next();
}
if (++currentPost >= numPosts) {
return callback.apply(null, [null].concat(args));
}
next();
}));
} else {
post.apply(context, newArgs);
if (++currentPost >= numPosts) {
return callback.apply(null, [null].concat(args));
}
next();
}
}
};
next();
};
Kareem.prototype.execPostSync = function(name, context) {
var posts = this._posts[name] || [];
var numPosts = posts.length;
for (var i = 0; i < numPosts; ++i) {
posts[i].call(context);
}
};
function _handleWrapError(instance, error, name, context, args, options, callback) {
if (options.useErrorHandlers) {
var _options = { error: error };
return instance.execPost(name, context, args, _options, function(error) {
return typeof callback === 'function' && callback(error);
});
} else {
return typeof callback === 'function' ?
callback(error) :
undefined;
}
}
Kareem.prototype.wrap = function(name, fn, context, args, options) {
var lastArg = (args.length > 0 ? args[args.length - 1] : null);
var argsWithoutCb = typeof lastArg === 'function' ?
args.slice(0, args.length - 1) :
args;
var _this = this;
var useLegacyPost;
if (typeof options === 'object') {
useLegacyPost = options && options.useLegacyPost;
} else {
useLegacyPost = options;
}
options = options || {};
this.execPre(name, context, args, function(error) {
if (error) {
var numCallbackParams = options.numCallbackParams || 0;
var nulls = [];
for (var i = 0; i < numCallbackParams; ++i) {
nulls.push(null);
}
return _handleWrapError(_this, error, name, context, nulls,
options, lastArg);
}
var end = (typeof lastArg === 'function' ? args.length - 1 : args.length);
fn.apply(context, args.slice(0, end).concat(function() {
var args = arguments;
var argsWithoutError = Array.prototype.slice.call(arguments, 1);
if (options.nullResultByDefault && argsWithoutError.length === 0) {
argsWithoutError.push(null);
}
if (arguments[0]) {
// Assume error
return _handleWrapError(_this, arguments[0], name, context,
argsWithoutError, options, lastArg);
} else {
if (useLegacyPost && typeof lastArg === 'function') {
lastArg.apply(context, arguments);
}
_this.execPost(name, context, argsWithoutError, function() {
if (arguments[0]) {
return typeof lastArg === 'function' ?
lastArg(arguments[0]) :
undefined;
}
return typeof lastArg === 'function' && !useLegacyPost ?
lastArg.apply(context, arguments) :
undefined;
});
}
}));
});
};
Kareem.prototype.createWrapper = function(name, fn, context, options) {
var _this = this;
return function() {
var args = Array.prototype.slice.call(arguments);
_this.wrap(name, fn, context, args, options);
};
};
Kareem.prototype.pre = function(name, isAsync, fn, error) {
if (typeof arguments[1] !== 'boolean') {
error = fn;
fn = isAsync;
isAsync = false;
}
this._pres[name] = this._pres[name] || [];
var pres = this._pres[name];
if (isAsync) {
pres.numAsync = pres.numAsync || 0;
++pres.numAsync;
}
pres.push({ fn: fn, isAsync: isAsync });
return this;
};
Kareem.prototype.post = function(name, fn) {
(this._posts[name] = this._posts[name] || []).push(fn);
return this;
};
Kareem.prototype.clone = function() {
var n = new Kareem();
for (var key in this._pres) {
if (!this._pres.hasOwnProperty(key)) {
continue;
}
n._pres[key] = this._pres[key].slice();
n._pres[key].numAsync = this._pres[key].numAsync;
}
for (var key in this._posts) {
if (!this._posts.hasOwnProperty(key)) {
continue;
}
n._posts[key] = this._posts[key].slice();
}
return n;
};
Kareem.prototype.merge = function(other) {
var ret = this.clone();
for (var key in other._pres) {
if (!other._pres.hasOwnProperty(key)) {
continue;
}
ret._pres[key] = (ret._pres[key] || []).concat(other._pres[key].slice());
ret._pres[key].numAsync += other._pres[key].numAsync;
}
for (var key in other._posts) {
if (!other._posts.hasOwnProperty(key)) {
continue;
}
ret._posts[key] = (ret._posts[key] || []).concat(other._posts[key].slice());
}
return ret;
};
module.exports = Kareem;