| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727 | 
							- // Rig weak dependencies
 
- if (typeof MicroQueue === 'undefined' && Package['micro-queue']) {
 
-   MicroQueue = Package['micro-queue'].MicroQueue;
 
- }
 
- if (typeof ReactiveList === 'undefined' && Package['reactive-list']) {
 
-   ReactiveList = Package['reactive-list'].ReactiveList;
 
- }
 
- // Rig weak dependencies in +0.9.1
 
- if (typeof MicroQueue === 'undefined' && Package['wekan-cfs-micro-queue']) {
 
-   MicroQueue = Package['wekan-cfs-micro-queue'].MicroQueue;
 
- }
 
- if (typeof ReactiveList === 'undefined' && Package['wekan-cfs-reactive-list']) {
 
-   ReactiveList = Package['wekan-cfs-reactive-list'].ReactiveList;
 
- }
 
- /**
 
-  * Creates an instance of a power queue // Testing inline comment
 
-  * [Check out demo](http://power-queue-test.meteor.com/)
 
-  *
 
-  * @constructor
 
-  * @self powerqueue
 
-  * @param {object} [options] Settings
 
-  * @param {boolean} [options.filo=false] Make it a first in last out queue
 
-  * @param {boolean} [options.isPaused=false] Set queue paused
 
-  * @param {boolean} [options.autostart=true] May adding a task start the queue
 
-  * @param {string} [options.name="Queue"] Name of the queue
 
-  * @param {number} [options.maxProcessing=1] Limit of simultanous running tasks
 
-  * @param {number} [options.maxFailures = 5] Limit retries of failed tasks, if 0 or below we allow infinite failures
 
-  * @param {number} [options.jumpOnFailure = true] Jump to next task and retry failed task later
 
-  * @param {boolean} [options.debug=false] Log verbose messages to the console
 
-  * @param {boolean} [options.reactive=true] Set whether or not this queue should be reactive
 
-  * @param {boolean} [options.onAutostart] Callback for the queue autostart event
 
-  * @param {boolean} [options.onPaused] Callback for the queue paused event
 
-  * @param {boolean} [options.onReleased] Callback for the queue release event
 
-  * @param {boolean} [options.onEnded] Callback for the queue end event
 
-  * @param {[SpinalQueue](spinal-queue.spec.md)} [options.spinalQueue] Set spinal queue uses pr. default `MicroQueue` or `ReactiveList` if added to the project
 
-  */
 
- PowerQueue = function(options) {
 
-   var self = this;
 
-   var test = 5;
 
-   self.reactive = (options && options.reactive === false) ? false :  true;
 
-   // Allow user to use another micro-queue #3
 
-   // We try setting the ActiveQueue to MicroQueue if installed in the app
 
-   var ActiveQueue = (typeof MicroQueue !== 'undefined') && MicroQueue || undefined;
 
-   // If ReactiveList is added to the project we use this over MicroQueue
 
-   ActiveQueue = (typeof ReactiveList !== 'undefined') && ReactiveList || ActiveQueue;
 
-   // We allow user to overrule and set a custom spinal-queue spec complient queue
 
-   if (options && typeof options.spinalQueue !== 'undefined') {
 
-     ActiveQueue = options.spinalQueue;
 
-   }
 
-   if (typeof ActiveQueue === 'undefined') {
 
-     console.log('Error: You need to add a spinal queue to the project');
 
-     console.log('Please add "micro-queue", "reactive-list" to the project');
 
-     throw new Error('Please add "micro-queue", "reactive-list" or other spinalQueue compatible packages');
 
-   }
 
-   // Default is fifo lilo
 
-   self.invocations = new ActiveQueue({
 
-     //
 
-     sort: (options && (options.filo || options.lifo)),
 
-     reactive: self.reactive
 
-   });
 
-   //var self.invocations = new ReactiveList(queueOrder);
 
-   // List of current tasks being processed
 
-   self._processList = new ActiveQueue({
 
-     reactive: self.reactive
 
-   }); //ReactiveList();
 
-   // Max number of simultanious tasks being processed
 
-   self._maxProcessing = new ReactiveProperty(options && options.maxProcessing || 1, self.reactive);
 
-   // Reactive number of tasks being processed
 
-   self._isProcessing = new ReactiveProperty(0, self.reactive);
 
-   // Boolean indicating if queue is paused or not
 
-   self._paused = new ReactiveProperty((options && options.isPaused || false), self.reactive);
 
-   // Boolean indicator for queue status active / running (can still be paused)
 
-   self._running = new ReactiveProperty(false, self.reactive);
 
-   // Counter for errors, errors are triggered if maxFailures is exeeded
 
-   self._errors = new ReactiveProperty(0, self.reactive);
 
-   // Counter for task failures, contains error count
 
-   self._failures = new ReactiveProperty(0, self.reactive);
 
-   // On failure jump to new task - if false the current task is rerun until error
 
-   self._jumpOnFailure = (options && options.jumpOnFailure === false) ? false : true;
 
-   // Count of all added tasks
 
-   self._maxLength = new ReactiveProperty(0, self.reactive);
 
-   // Boolean indicate whether or not a "add" task is allowed to start the queue
 
-   self._autostart = new ReactiveProperty( ((options && options.autostart === false) ? false : true), self.reactive);
 
-   // Limit times a task is allowed to fail and be rerun later before triggering an error
 
-   self._maxFailures = new ReactiveProperty( (options && options.maxFailures || 5), self.reactive);
 
-   // Name / title of this queue - Not used - should deprecate
 
-   self.title = options && options.name || 'Queue';
 
-   // debug - will print error / failures passed to next
 
-   self.debug = !!(options && options.debug);
 
-   /** @method PowerQueue.total
 
-    * @reactive
 
-    * @returns {number} The total number of tasks added to this queue
 
-    */
 
-   self.total = self._maxLength.get;
 
-   /** @method PowerQueue.isPaused
 
-    * @reactive
 
-    * @returns {boolean} Status of the paused state of the queue
 
-    */
 
-   self.isPaused = self._paused.get;
 
-   /** @method PowerQueue.processing
 
-    * @reactive
 
-    * @returns {number} Number of tasks currently being processed
 
-    */
 
-   self.processing = self._isProcessing.get;
 
-   /** @method PowerQueue.errors
 
-    * @reactive
 
-    * @returns {number} The total number of errors
 
-    * Errors are triggered when [maxFailures](PowerQueue.maxFailures) are exeeded
 
-    */
 
-   self.errors = self._errors.get;
 
-   /** @method PowerQueue.failures
 
-    * @reactive
 
-    * @returns {number} The total number of failed tasks
 
-    */
 
-   self.failures = self._failures.get;
 
-   /** @method PowerQueue.isRunning
 
-    * @reactive
 
-    * @returns {boolean} True if the queue is running
 
-    * > NOTE: The task can be paused but marked as running
 
-    */
 
-   self.isRunning = self._running.get;
 
-   /** @method PowerQueue.maxProcessing Get setter for maxProcessing
 
-    * @param {number} [max] If not used this function works as a getter
 
-    * @reactive
 
-    * @returns {number} Maximum number of simultaneous processing tasks
 
-    *
 
-    * Example:
 
-    * ```js
 
-    *   foo.maxProcessing();    // Works as a getter and returns the current value
 
-    *   foo.maxProcessing(20);  // This sets the value to 20
 
-    * ```
 
-    */
 
-   self.maxProcessing = self._maxProcessing.getset;
 
-   self._maxProcessing.onChange = function() {
 
-     // The user can change the max allowed processing tasks up or down here...
 
-     // Update the throttle up
 
-     self.updateThrottleUp();
 
-     // Update the throttle down
 
-     self.updateThrottleDown();
 
-   };
 
-   /** @method PowerQueue.autostart Get setter for autostart
 
-    * @param {boolean} [autorun] If not used this function works as a getter
 
-    * @reactive
 
-    * @returns {boolean} If adding a task may trigger the queue to start
 
-    *
 
-    * Example:
 
-    * ```js
 
-    *   foo.autostart();    // Works as a getter and returns the current value
 
-    *   foo.autostart(true);  // This sets the value to true
 
-    * ```
 
-    */
 
-   self.autostart = self._autostart.getset;
 
-   /** @method PowerQueue.maxFailures Get setter for maxFailures
 
-    * @param {number} [max] If not used this function works as a getter
 
-    * @reactive
 
-    * @returns {number} The maximum for failures pr. task before triggering an error
 
-    *
 
-    * Example:
 
-    * ```js
 
-    *   foo.maxFailures();    // Works as a getter and returns the current value
 
-    *   foo.maxFailures(10);  // This sets the value to 10
 
-    * ```
 
-    */
 
-   self.maxFailures = self._maxFailures.getset;
 
-   /** @callback PowerQueue.onPaused
 
-    * Is called when queue is ended
 
-    */
 
-   self.onPaused = options && options.onPaused || function() {
 
-     self.debug && console.log(self.title + ' ENDED');
 
-   };
 
-   /** @callback PowerQueue.onEnded
 
-    * Is called when queue is ended
 
-    */
 
-   self.onEnded = options && options.onEnded || function() {
 
-     self.debug && console.log(self.title + ' ENDED');
 
-   };
 
-   /** @callback PowerQueue.onRelease
 
-    * Is called when queue is released
 
-    */
 
-   self.onRelease = options && options.onRelease || function() {
 
-     self.debug && console.log(self.title + ' RELEASED');
 
-   };
 
-   /** @callback PowerQueue.onAutostart
 
-    * Is called when queue is auto started
 
-    */
 
-   self.onAutostart = options && options.onAutostart || function() {
 
-     self.debug && console.log(self.title + ' Autostart');
 
-   };
 
- };
 
-   /** @method PowerQueue.prototype.processList
 
-    * @reactive
 
-    * @returns {array} List of tasks currently being processed
 
-    */
 
-   PowerQueue.prototype.processingList = function() {
 
-     var self = this;
 
-     return self._processList.fetch();
 
-   };
 
-   /** @method PowerQueue.prototype.isHalted
 
-    * @reactive
 
-    * @returns {boolean} True if the queue is not running or paused
 
-    */
 
-   PowerQueue.prototype.isHalted = function() {
 
-     var self = this;
 
-     return (!self._running.get() || self._paused.get());
 
-   };
 
-   /** @method PowerQueue.prototype.length
 
-    * @reactive
 
-    * @returns {number} Number of tasks left in queue to be processed
 
-    */
 
-   PowerQueue.prototype.length = function() {
 
-     var self = this;
 
-     return self.invocations.length();
 
-   };
 
-   /** @method PowerQueue.prototype.progress
 
-    * @reactive
 
-    * @returns {number} 0 .. 100 % Indicates the status of the queue
 
-    */
 
-   PowerQueue.prototype.progress = function() {
 
-     var self = this;
 
-     var progress = self._maxLength.get() - self.invocations.length() - self._isProcessing.get();
 
-     if (self._maxLength.value > 0) {
 
-       return Math.round(progress / self._maxLength.value * 100);
 
-     }
 
-     return 0;
 
-   };
 
-   /** @method PowerQueue.prototype.usage
 
-    * @reactive
 
-    * @returns {number} 0 .. 100 % Indicates resource usage of the queue
 
-    */
 
-   PowerQueue.prototype.usage = function() {
 
-     var self = this;
 
-     return Math.round(self._isProcessing.get() / self._maxProcessing.get() * 100);
 
-   };
 
-   /** @method PowerQueue.prototype.reset Reset the queue
 
-    * Calling this will:
 
-    * * stop the queue
 
-    * * paused to false
 
-    * * Discart all queue data
 
-    *
 
-    * > NOTE: At the moment if the queue has processing tasks they can change
 
-    * > the `errors` and `failures` counters. This could change in the future or
 
-    * > be prevented by creating a whole new instance of the `PowerQueue`
 
-    */
 
-   PowerQueue.prototype.reset = function() {
 
-     var self = this;
 
-     self.debug && console.log(self.title + ' RESET');
 
-     self._running.set(false);
 
-     self._paused.set(false);
 
-     self.invocations.reset();
 
-     self._processList.reset();
 
-     // // Loop through the processing tasks and reset these
 
-     // self._processList.forEach(function(data) {
 
-     //   if (data.queue instanceof PowerQueue) {
 
-     //     data.queue.reset();
 
-     //   }
 
-     // }, true);
 
-     self._maxLength.set(0);
 
-     self._failures.set(0);
 
-     self._errors.set(0);
 
-   };
 
-   /** @method PowerQueue._autoStartTasks
 
-    * @private
 
-    *
 
-    * This method defines the autostart algorithm that allows add task to trigger
 
-    * a start of the queue if queue is not paused.
 
-    */
 
-   PowerQueue.prototype._autoStartTasks = function() {
 
-     var self = this;
 
-     // We dont start anything by ourselfs if queue is paused
 
-     if (!self._paused.value) {
 
-       // Queue is not running and we are set to autostart so we start the queue
 
-       if (!self._running.value && self._autostart.value) {
 
-         // Trigger callback / event
 
-         self.onAutostart();
 
-         // Set queue as running
 
-         self._running.set(true);
 
-       }
 
-       // Make sure that we use all available resources
 
-       if (self._running.value) {
 
-         // Call next to start up the queue
 
-         self.next(null);
 
-       }
 
-     }
 
-   };
 
-   /** @method PowerQueue.prototype.add
 
-    * @param {any} data The task to be handled
 
-    * @param {number} [failures] Used internally to Pass on number of failures.
 
-    */
 
-   PowerQueue.prototype.add = function(data, failures, id) {
 
-     var self = this;
 
-     // Assign new id to task
 
-     var assignNewId = self._jumpOnFailure || typeof id === 'undefined';
 
-     // Set the task id
 
-     var taskId = (assignNewId) ? self._maxLength.value + 1 : id;
 
-     // self.invocations.add({ _id: currentId, data: data, failures: failures || 0 }, reversed);
 
-     self.invocations.insert(taskId, { _id: taskId, data: data, failures: failures || 0 });
 
-     // If we assigned new id then increase length
 
-     if (assignNewId) self._maxLength.inc();
 
-     self._autoStartTasks();
 
-   };
 
-   /** @method PowerQueue.prototype.updateThrottleUp
 
-    * @private
 
-    *
 
-    * Calling this method will update the throttle on the queue adding tasks.
 
-    *
 
-    * > Note: Currently we only support the PowerQueue - but we could support
 
-    * > a more general interface for pauseable tasks or other usecases.
 
-    */
 
-   PowerQueue.prototype.updateThrottleUp = function() {
 
-     var self = this;
 
-     // How many additional tasks can we handle?
 
-     var availableSlots = self._maxProcessing.value - self._isProcessing.value;
 
-     // If we can handle more, we have more, we're running, and we're not paused
 
-     if (!self._paused.value && self._running.value && availableSlots > 0 && self.invocations._length > 0) {
 
-       // Increase counter of current number of tasks being processed
 
-       self._isProcessing.inc();
 
-       // Run task
 
-       self.runTask(self.invocations.getFirstItem());
 
-       // Repeat recursively; this is better than a for loop to avoid blocking the UI
 
-       self.updateThrottleUp();
 
-     }
 
-   };
 
-   /** @method PowerQueue.prototype.updateThrottleDown
 
-    * @private
 
-    *
 
-    * Calling this method will update the throttle on the queue pause tasks.
 
-    *
 
-    * > Note: Currently we only support the PowerQueue - but we could support
 
-    * > a more general interface for pauseable tasks or other usecases.
 
-    */
 
-   PowerQueue.prototype.updateThrottleDown = function() {
 
-     var self = this;
 
-     // Calculate the differece between acutuall processing tasks and target
 
-     var diff = self._isProcessing.value - self._maxProcessing.value;
 
-     // If the diff is more than 0 then we have many tasks processing.
 
-     if (diff > 0) {
 
-       // We pause the latest added tasks
 
-       self._processList.forEachReverse(function(data) {
 
-         if (diff > 0 && data.queue instanceof PowerQueue) {
 
-           diff--;
 
-           // We dont mind calling pause on multiple times on each task
 
-           // theres a simple check going on preventing any duplicate actions
 
-           data.queue.pause();
 
-         }
 
-       }, true);
 
-     }
 
-   };
 
-   /** @method PowerQueue.prototype.next
 
-    * @param {string} [err] Error message if task failed
 
-    * > * Can pass in `null` to start the queue
 
-    * > * Passing in a string to `next` will trigger a failure
 
-    * > * Passing nothing will simply let the next task run
 
-    * `next` is handed into the [taskHandler](PowerQueue.taskHandler) as a
 
-    * callback to mark an error or end of current task
 
-    */
 
-   PowerQueue.prototype.next = function(err) {
 
-     var self = this;
 
-     // Primary concern is to throttle up because we are either:
 
-     // 1. Starting the queue
 
-     // 2. Starting next task
 
-     //
 
-     // This function does not shut down running tasks
 
-     self.updateThrottleUp();
 
-     // We are running, no tasks are being processed even we just updated the
 
-     // throttle up and we got no errors.
 
-     // 1. We are paused and releasing tasks
 
-     // 2. We are done
 
-     if (self._running.value && self._isProcessing.value === 0 && err !== null) {
 
-       // We have no tasks processing so this queue is now releasing resources
 
-       // this could be that the queue is paused or stopped, in that case the
 
-       // self.invocations._length would be > 0
 
-       // If on the other hand the self.invocations._length is 0 then we have no more
 
-       // tasks in the queue so the queue has ended
 
-       self.onRelease(self.invocations._length);
 
-       if (!self.invocations._length) { // !self._paused.value &&
 
-         // Check if queue is done working
 
-         // Stop the queue
 
-         self._running.set(false);
 
-         // self.invocations.reset(); // This should be implicit
 
-         self.onEnded();
 
-       }
 
-     }
 
-   };
 
-   /** @callback done
 
-    * @param {Meteor.Error | Error | String | null} [feedback] This allows the task to communicate with the queue
 
-    *
 
-    * Explaination of `feedback`
 
-    * * `Meteor.Error` This means that the task failed in a controlled manner and is allowed to rerun
 
-    * * `Error` This will throw the passed error - as its an unitended error
 
-    * * `null` The task is not done yet, rerun later
 
-    * * `String` The task can perform certain commands on the queue
 
-    *    * "pause" - pause the queue
 
-    *    * "stop" - stop the queue
 
-    *    * "reset" - reset the queue
 
-    *    * "cancel" - cancel the queue
 
-    *
 
-    */
 
-   /** @method PowerQueue.prototype.runTaskDone
 
-    * @private
 
-    * @param {Meteor.Error | Error | String | null} [feedback] This allows the task to communicate with the queue
 
-    * @param {object} invocation
 
-    *
 
-    * > Note: `feedback` is explained in [Done callback](#done)
 
-    *
 
-    */
 
-   // Rig the callback function
 
-   PowerQueue.prototype.runTaskDone = function(feedback, invocation) {
 
-     var self = this;
 
-     // If the task handler throws an error then add it to the queue again
 
-     // we allow this for a max of self._maxFailures
 
-     // If the error is null then we add the task silently back into the
 
-     // microQueue in reverse... This could be due to pause or throttling
 
-     if (feedback instanceof Meteor.Error) {
 
-       // We only count failures if maxFailures are above 0
 
-       if (self._maxFailures.value > 0) invocation.failures++;
 
-       self._failures.inc();
 
-       // If the user has set the debug flag we print out failures/errors
 
-       self.debug && console.error('Error: "' + self.title + '" ' + feedback.message + ', ' + feedback.stack);
 
-       if (invocation.failures < self._maxFailures.value) {
 
-         // Add the task again with the increased failures
 
-         self.add(invocation.data, invocation.failures, invocation._id);
 
-       } else {
 
-         self._errors.inc();
 
-         self.errorHandler(invocation.data, self.add, invocation.failures);
 
-       }
 
-       // If a error is thrown we assume its not intended
 
-     } else if (feedback instanceof Error) throw feedback;
 
-     if (feedback)
 
-     // We use null to throttle pauseable tasks
 
-     if (feedback === null) {
 
-       // We add this task into the queue, no questions asked
 
-       self.invocations.insert(invocation._id, { data: invocation.data, failures: invocation.failures, _id: invocation._id });
 
-     }
 
-     // If the user returns a string we got a command
 
-     if (feedback === ''+feedback) {
 
-       var command = {
 
-         'pause': function() { self.pause(); },
 
-         'stop': function() { self.stop(); },
 
-         'reset': function() { self.reset(); },
 
-         'cancel': function() { self.cancel(); },
 
-       };
 
-       if (typeof command[feedback] === 'function') {
 
-         // Run the command on this queue
 
-         command[feedback]();
 
-       } else {
 
-         // We dont recognize this command, throw an error
 
-         throw new Error('Unknown queue command "' + feedback + '"');
 
-       }
 
-     }
 
-     // Decrease the number of tasks being processed
 
-     // make sure we dont go below 0
 
-     if (self._isProcessing.value > 0) self._isProcessing.dec();
 
-     // Task has ended we remove the task from the process list
 
-     self._processList.remove(invocation._id);
 
-     invocation.data = null;
 
-     invocation.failures = null;
 
-     invocation._id = null;
 
-     invocation = null;
 
-     delete invocation;
 
-     // Next task
 
-     Meteor.setTimeout(function() {
 
-       self.next();
 
-     }, 0);
 
-   };
 
-   /** @method PowerQueue.prototype.runTask
 
-    * @private // This is not part of the open api
 
-    * @param {object} invocation The object stored in the micro-queue
 
-    */
 
-   PowerQueue.prototype.runTask = function(invocation) {
 
-     var self = this;
 
-     // We start the fitting task handler
 
-     // Currently we only support the PowerQueue but we could have a more general
 
-     // interface for tasks that allow throttling
 
-     try {
 
-       if (invocation.data instanceof PowerQueue) {
 
-         // Insert PowerQueue into process list
 
-         self._processList.insert(invocation._id, { id: invocation._id, queue: invocation.data });
 
-         // Handle task
 
-         self.queueTaskHandler(invocation.data, function subQueueCallbackDone(feedback) {
 
-           self.runTaskDone(feedback, invocation);
 
-         }, invocation.failures);
 
-       } else {
 
-         // Insert task into process list
 
-         self._processList.insert(invocation._id, invocation.data);
 
-         // Handle task
 
-         self.taskHandler(invocation.data, function taskCallbackDone(feedback) {
 
-           self.runTaskDone(feedback, invocation);
 
-         }, invocation.failures);
 
-       }
 
-     } catch(err) {
 
-       throw new Error('Error while running taskHandler for queue, Error: ' + err.message);
 
-     }
 
-   };
 
-   /** @method PowerQueue.prototype.queueTaskHandler
 
-    * This method handles tasks that are sub queues
 
-    */
 
-   PowerQueue.prototype.queueTaskHandler = function(subQueue, next, failures) {
 
-     var self = this;
 
-     // Monitor sub queue task releases
 
-     subQueue.onRelease = function(remaining) {
 
-       // Ok, we were paused - this could be throttling so we respect this
 
-       // So when the queue is halted we add it back into the main queue
 
-       if (remaining > 0) {
 
-         // We get out of the queue but dont repport error and add to run later
 
-         next(null);
 
-       } else {
 
-         // Queue has ended
 
-         // We simply trigger next task when the sub queue is complete
 
-         next();
 
-         // When running subqueues it doesnt make sense to track failures and retry
 
-         // the sub queue - this is sub queue domain
 
-       }
 
-     };
 
-     // Start the queue
 
-     subQueue.run();
 
-   };
 
-   /** @callback PowerQueue.prototype.taskHandler
 
-    * @param {any} data This can be data or functions
 
-    * @param {function} next Function `next` call this to end task
 
-    * @param {number} failures Number of failures on this task
 
-    *
 
-    * Default task handler expects functions as data:
 
-    * ```js
 
-    *   self.taskHandler = function(data, next, failures) {
 
-    *     // This default task handler expects invocation to be a function to run
 
-    *     if (typeof data !== 'function') {
 
-    *       throw new Error('Default task handler expects a function');
 
-    *     }
 
-    *     try {
 
-    *       // Have the function call next
 
-    *       data(next, failures);
 
-    *     } catch(err) {
 
-    *       // Throw to fail this task
 
-    *       next(err);
 
-    *     }
 
-    *   };
 
-    * ```
 
-    */
 
-   // Can be overwrittin by the user
 
-   PowerQueue.prototype.taskHandler = function(data, next, failures) {
 
-     var self = this;
 
-     // This default task handler expects invocation to be a function to run
 
-     if (typeof data !== 'function') {
 
-       throw new Error('Default task handler expects a function');
 
-     }
 
-     try {
 
-       // Have the function call next
 
-       data(next, failures);
 
-     } catch(err) {
 
-       // Throw to fail this task
 
-       next(err);
 
-     }
 
-   };
 
-   /** @callback PowerQueue.prototype.errorHandler
 
-    * @param {any} data This can be data or functions
 
-    * @param {function} addTask Use this function to insert the data into the queue again
 
-    * @param {number} failures Number of failures on this task
 
-    *
 
-    * The default callback:
 
-    * ```js
 
-    *   var foo = new PowerQueue();
 
-    *
 
-    *   // Overwrite the default action
 
-    *   foo.errorHandler = function(data, addTask, failures) {
 
-    *     // This could be overwritten the data contains the task data and addTask
 
-    *     // is a helper for adding the task to the queue
 
-    *     // try again: addTask(data);
 
-    *     // console.log('Terminate at ' + failures + ' failures');
 
-    *   };
 
-    * ```
 
-    */
 
-   PowerQueue.prototype.errorHandler = function(data, addTask, failures) {
 
-     var self = this;
 
-     // This could be overwritten the data contains the task data and addTask
 
-     // is a helper for adding the task to the queue
 
-     // try again: addTask(data);
 
-     self.debug && console.log('Terminate at ' + failures + ' failures');
 
-   };
 
-   /** @method PowerQueue.prototype.pause Pause the queue
 
-    * @todo We should have it pause all processing tasks
 
-    */
 
-   PowerQueue.prototype.pause = function() {
 
-     var self = this;
 
-     if (!self._paused.value) {
 
-       self._paused.set(true);
 
-       // Loop through the processing tasks and pause these
 
-       self._processList.forEach(function(data) {
 
-         if (data.queue instanceof PowerQueue) {
 
-           // Pause the sub queue
 
-           data.queue.pause();
 
-         }
 
-       }, true);
 
-       // Trigger callback
 
-       self.onPaused();
 
-     }
 
-   };
 
-   /** @method PowerQueue.prototype.resume Start a paused queue
 
-    * @todo We should have it resume all processing tasks
 
-    *
 
-    * > This will not start a stopped queue
 
-    */
 
-   PowerQueue.prototype.resume = function() {
 
-     var self = this;
 
-     self.run();
 
-   };
 
-   /** @method PowerQueue.prototype.run Starts the queue
 
-    * > Using this command will resume a paused queue and will
 
-    * > start a stopped queue.
 
-    */
 
-   PowerQueue.prototype.run = function() {
 
-     var self = this;
 
-     //not paused and already running or queue empty or paused subqueues
 
-     if (!self._paused.value && self._running.value || !self.invocations._length) {
 
-       return;
 
-     }
 
-     self._paused.set(false);
 
-     self._running.set(true);
 
-     self.next(null);
 
-   };
 
-   /** @method PowerQueue.prototype.stop Stops the queue
 
-    */
 
-   PowerQueue.prototype.stop = function() {
 
-     var self = this;
 
-     self._running.set(false);
 
-   };
 
-   /** @method PowerQueue.prototype.cancel Cancel the queue
 
-    */
 
-   PowerQueue.prototype.cancel = function() {
 
-     var self = this;
 
-     self.reset();
 
-   };
 
 
  |