123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488 |
- var wrap = require('shimmer').wrap;
- var HAS_CREATE_AL = 1 << 0;
- var HAS_BEFORE_AL = 1 << 1;
- var HAS_AFTER_AL = 1 << 2;
- var HAS_ERROR_AL = 1 << 3;
- var listeners = [];
- var uid = 0;
- var inAsyncTick = false;
- var listenerStack = [];
- var asyncCatcher;
- var asyncWrap;
- function union(dest, added) {
- var destLength = dest.length;
- var addedLength = added.length;
- var returned = [];
- if (destLength === 0 && addedLength === 0) return returned;
- for (var j = 0; j < destLength; j++) returned[j] = dest[j];
- if (addedLength === 0) return returned;
- for (var i = 0; i < addedLength; i++) {
- var missing = true;
- for (j = 0; j < destLength; j++) {
- if (dest[j].uid === added[i].uid) {
- missing = false;
- break;
- }
- }
- if (missing) returned.push(added[i]);
- }
- return returned;
- }
- if (process._fatalException) {
-
- var inErrorTick = false;
-
- var errorValues;
- asyncCatcher = function asyncCatcher(er) {
- var length = listeners.length;
- if (inErrorTick || length === 0) return false;
- var handled = false;
-
- inErrorTick = true;
- for (var i = 0; i < length; ++i) {
- var listener = listeners[i];
- if ((listener.flags & HAS_ERROR_AL) === 0) continue;
- var value = errorValues && errorValues[listener.uid];
- handled = listener.error(value, er) || handled;
- }
- inErrorTick = false;
-
- if (listenerStack.length > 0) listeners = listenerStack.pop();
- errorValues = undefined;
- return handled && !inAsyncTick;
- };
- asyncWrap = function asyncWrap(original, list, length) {
- var values = [];
-
- inAsyncTick = true;
- for (var i = 0; i < length; ++i) {
- var listener = list[i];
- values[listener.uid] = listener.data;
- if ((listener.flags & HAS_CREATE_AL) === 0) continue;
- var value = listener.create(listener.data);
- if (value !== undefined) values[listener.uid] = value;
- }
- inAsyncTick = false;
-
- return function () {
-
- errorValues = values;
-
- listenerStack.push(listeners);
-
- listeners = union(list, listeners);
-
- inAsyncTick = true;
- for (var i = 0; i < length; ++i) {
- if ((list[i].flags & HAS_BEFORE_AL) > 0) {
- list[i].before(this, values[list[i].uid]);
- }
- }
- inAsyncTick = false;
-
- var returned = original.apply(this, arguments);
-
- inAsyncTick = true;
- for (i = 0; i < length; ++i) {
- if ((list[i].flags & HAS_AFTER_AL) > 0) {
- list[i].after(this, values[list[i].uid]);
- }
- }
- inAsyncTick = false;
-
- listeners = listenerStack.pop();
- errorValues = undefined;
- return returned;
- };
- };
- wrap(process, '_fatalException', function (_fatalException) {
- return function _asyncFatalException(er) {
- return asyncCatcher(er) || _fatalException(er);
- };
- });
- }
- else {
-
- var errorThrew = false;
-
- asyncCatcher = function uncaughtCatcher(er) {
-
- if (errorThrew) throw er;
- var handled = false;
-
- var length = listeners.length;
- for (var i = 0; i < length; ++i) {
- var listener = listeners[i];
- if ((listener.flags & HAS_ERROR_AL) === 0) continue;
- handled = listener.error(null, er) || handled;
- }
-
- if (!handled && inAsyncTick) throw er;
- };
- asyncWrap = function asyncWrap(original, list, length) {
- var values = [];
-
- inAsyncTick = true;
- for (var i = 0; i < length; ++i) {
- var listener = list[i];
- values[listener.uid] = listener.data;
- if ((listener.flags & HAS_CREATE_AL) === 0) continue;
- var value = listener.create(listener.data);
- if (value !== undefined) values[listener.uid] = value;
- }
- inAsyncTick = false;
-
- return function () {
-
-
- var threw = false;
-
- var handled = false;
-
- listenerStack.push(listeners);
-
- listeners = union(list, listeners);
-
- inAsyncTick = true;
- for (var i = 0; i < length; ++i) {
- if ((list[i].flags & HAS_BEFORE_AL) > 0) {
- list[i].before(this, values[list[i].uid]);
- }
- }
- inAsyncTick = false;
-
- var returned;
- try {
- returned = original.apply(this, arguments);
- }
- catch (er) {
- threw = true;
- for (var i = 0; i < length; ++i) {
- if ((listeners[i].flags & HAS_ERROR_AL) == 0) continue;
- try {
- handled = listeners[i].error(values[list[i].uid], er) || handled;
- }
- catch (x) {
- errorThrew = true;
- throw x;
- }
- }
- if (!handled) {
-
- process.removeListener('uncaughtException', asyncCatcher);
- process._originalNextTick(function () {
- process.addListener('uncaughtException', asyncCatcher);
- });
- throw er;
- }
- }
- finally {
-
- if (!threw || handled) {
- inAsyncTick = true;
- for (i = 0; i < length; ++i) {
- if ((list[i].flags & HAS_AFTER_AL) > 0) {
- list[i].after(this, values[list[i].uid]);
- }
- }
- inAsyncTick = false;
- }
-
- listeners = listenerStack.pop();
- }
- return returned;
- };
- };
-
- process.addListener('uncaughtException', asyncCatcher);
- }
- function simpleWrap(original, list, length) {
- inAsyncTick = true;
- for (var i = 0; i < length; ++i) {
- var listener = list[i];
- if (listener.create) listener.create(listener.data);
- }
- inAsyncTick = false;
-
-
- return function () {
- listenerStack.push(listeners);
- listeners = union(list, listeners);
- var returned = original.apply(this, arguments);
- listeners = listenerStack.pop();
- return returned;
- };
- }
- function wrapCallback(original) {
- var length = listeners.length;
-
- if (length === 0) return original;
-
- var list = listeners.slice();
- for (var i = 0; i < length; ++i) {
- if (list[i].flags > 0) return asyncWrap(original, list, length);
- }
- return simpleWrap(original, list, length);
- }
- function AsyncListener(callbacks, data) {
- if (typeof callbacks.create === 'function') {
- this.create = callbacks.create;
- this.flags |= HAS_CREATE_AL;
- }
- if (typeof callbacks.before === 'function') {
- this.before = callbacks.before;
- this.flags |= HAS_BEFORE_AL;
- }
- if (typeof callbacks.after === 'function') {
- this.after = callbacks.after;
- this.flags |= HAS_AFTER_AL;
- }
- if (typeof callbacks.error === 'function') {
- this.error = callbacks.error;
- this.flags |= HAS_ERROR_AL;
- }
- this.uid = ++uid;
- this.data = data === undefined ? null : data;
- }
- AsyncListener.prototype.create = undefined;
- AsyncListener.prototype.before = undefined;
- AsyncListener.prototype.after = undefined;
- AsyncListener.prototype.error = undefined;
- AsyncListener.prototype.data = undefined;
- AsyncListener.prototype.uid = 0;
- AsyncListener.prototype.flags = 0;
- function createAsyncListener(callbacks, data) {
- if (typeof callbacks !== 'object' || !callbacks) {
- throw new TypeError('callbacks argument must be an object');
- }
- if (callbacks instanceof AsyncListener) {
- return callbacks;
- }
- else {
- return new AsyncListener(callbacks, data);
- }
- }
- function addAsyncListener(callbacks, data) {
- var listener;
- if (!(callbacks instanceof AsyncListener)) {
- listener = createAsyncListener(callbacks, data);
- }
- else {
- listener = callbacks;
- }
-
- var registered = false;
- for (var i = 0; i < listeners.length; i++) {
- if (listener === listeners[i]) {
- registered = true;
- break;
- }
- }
- if (!registered) listeners.push(listener);
- return listener;
- }
- function removeAsyncListener(listener) {
- for (var i = 0; i < listeners.length; i++) {
- if (listener === listeners[i]) {
- listeners.splice(i, 1);
- break;
- }
- }
- }
- process.createAsyncListener = createAsyncListener;
- process.addAsyncListener = addAsyncListener;
- process.removeAsyncListener = removeAsyncListener;
- module.exports = wrapCallback;
|