New Case Study:See how Anthropic automated 95% of dependency reviews with Socket.Learn More
Socket
Sign inDemoInstall
Socket

redibox-hook-job

Package Overview
Dependencies
Maintainers
2
Versions
23
Alerts
File Explorer

Advanced tools

Socket logo

Install Socket

Detect and block malicious and high-risk dependencies

Install

redibox-hook-job - npm Package Compare versions

Comparing version

to
1.1.0

6

lib/hook.js

@@ -98,5 +98,3 @@ 'use strict';

*/
toKey() {
let key = arguments.length <= 0 || arguments[0] === undefined ? '' : arguments[0];
toKey(key = '') {
return `${ this.options.keyPrefix }:${ key }`;

@@ -115,2 +113,2 @@ }

exports.default = JobHook;
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9ob29rLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7O0FBQUE7Ozs7QUFDQTs7OztBQUNBOzs7O0FBQ0E7Ozs7QUFDQTs7OztBQUNBOzs7O0FBRWUsTUFBTSxPQUFOLDJCQUErQjtBQUM1QyxnQkFBYztBQUNaLFVBQU0sS0FBTjtBQUNBLFNBQUssTUFBTCxHQUFjLEVBQWQ7QUFDRDs7Ozs7OztBQU9ELGFBQVc7QUFDVDtBQUNEOzs7Ozs7QUFNRCxZQUFVO0FBQ1I7QUFDRDs7Ozs7QUFLRCxlQUFhO0FBQ1gsUUFBSSxDQUFDLEtBQUssT0FBTCxDQUFhLE9BQWxCLEVBQTJCO0FBQ3pCLGFBQU8sbUJBQVEsT0FBUixFQUFQO0FBQ0Q7O0FBRUQsU0FBSyxJQUFJLElBQUksQ0FBUixFQUFXLE1BQU0sS0FBSyxPQUFMLENBQWEsTUFBYixDQUFvQixNQUExQyxFQUFrRCxJQUFJLEdBQXRELEVBQTJELEdBQTNELEVBQWdFO0FBQzlELFlBQU0sUUFBUSxLQUFLLE9BQUwsQ0FBYSxNQUFiLENBQW9CLENBQXBCLENBQWQ7QUFDQSxXQUFLLFdBQUwsQ0FBaUIsS0FBakI7QUFDRDs7QUFFRCxXQUFPLG1CQUFRLE9BQVIsRUFBUDtBQUNEOzs7Ozs7Ozs7QUFTRCxTQUFPLEtBQVAsRUFBYyxJQUFkLEVBQW9CLE9BQXBCLEVBQTZCO0FBQzNCLFNBQUssR0FBTCxDQUFTLE9BQVQsQ0FBaUIsQ0FBQyxrQkFBRCxHQUFxQixLQUFyQixFQUEyQixBQUEzQixDQUFqQjtBQUNBLFdBQU8sa0JBQVEsS0FBSyxJQUFiLEVBQW1CLElBQW5CLEVBQXlCLElBQXpCLEVBQStCLE9BQS9CLEVBQXdDLEtBQXhDLEVBQStDLElBQS9DLENBQVA7QUFDRDs7Ozs7O0FBTUQsY0FBWSxLQUFaLEVBQW1CO0FBQ2pCLFNBQUssR0FBTCxDQUFTLE9BQVQsQ0FBaUIsQ0FBQyxPQUFELEdBQVUsTUFBTSxJQUFoQixFQUFxQixVQUFyQixDQUFqQjtBQUNBLFNBQUssTUFBTCxDQUFZLE1BQU0sSUFBbEIsSUFBMEIsaUNBQWUsS0FBSyxPQUFwQixFQUFnQyxLQUFoQyxHQUF5QyxLQUFLLElBQTlDLENBQTFCO0FBQ0Q7Ozs7Ozs7QUFPRCxVQUFnQjtBQUFBLFFBQVYsR0FBVSx5REFBSixFQUFJOztBQUNkLFdBQU8sQ0FBQyxBQUFELEdBQUcsS0FBSyxPQUFMLENBQWEsU0FBaEIsRUFBMEIsQ0FBMUIsR0FBNkIsR0FBN0IsRUFBaUMsQUFBakMsQ0FBUDtBQUNEOzs7Ozs7O0FBT0QsVUFBUSxJQUFSLEVBQWM7QUFDWixTQUFLLE9BQUwsQ0FBYSxPQUFiLEdBQXVCLFFBQVEsQ0FBQyxLQUFLLE9BQUwsQ0FBYSxPQUE3QztBQUNEO0FBNUUyQztrQkFBekIsTyIsImZpbGUiOiJob29rLmpzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IEpvYiBmcm9tICcuL2pvYic7XG5pbXBvcnQgUXVldWUgZnJvbSAnLi9xdWV1ZSc7XG5pbXBvcnQgUHJvbWlzZSBmcm9tICdibHVlYmlyZCc7XG5pbXBvcnQgZGVmYXVsdHMgZnJvbSAnLi9kZWZhdWx0cyc7XG5pbXBvcnQgc2NyaXB0cyBmcm9tICcuL3NjcmlwdHMnO1xuaW1wb3J0IHsgQmFzZUhvb2sgfSBmcm9tICdyZWRpYm94JztcblxuZXhwb3J0IGRlZmF1bHQgY2xhc3MgSm9iSG9vayBleHRlbmRzIEJhc2VIb29rIHtcbiAgY29uc3RydWN0b3IoKSB7XG4gICAgc3VwZXIoJ2pvYicpO1xuICAgIHRoaXMucXVldWVzID0ge307XG4gIH1cblxuICAvLyBub2luc3BlY3Rpb24gSlNVbnVzZWRHbG9iYWxTeW1ib2xzLEpTTWV0aG9kQ2FuQmVTdGF0aWNcbiAgLyoqXG4gICAqIERlZmF1bHQgY29uZmlnIGZvciBzY2hlZHVsZXJcbiAgICogQHJldHVybnMge3tzb21lRGVmYXVsdFRoaW5nOiBzdHJpbmd9fVxuICAgKi9cbiAgZGVmYXVsdHMoKSB7XG4gICAgcmV0dXJuIGRlZmF1bHRzO1xuICB9XG5cbiAgLyoqXG4gICAqXG4gICAqIEByZXR1cm5zIHt7YWRkSm9iLCBjaGVja1N0YWxsZWRKb2JzLCByZW1vdmVKb2J9fVxuICAgKi9cbiAgc2NyaXB0cygpIHtcbiAgICByZXR1cm4gc2NyaXB0cztcbiAgfVxuXG4gIC8qKlxuICAgKlxuICAgKi9cbiAgaW5pdGlhbGl6ZSgpIHtcbiAgICBpZiAoIXRoaXMub3B0aW9ucy5lbmFibGVkKSB7XG4gICAgICByZXR1cm4gUHJvbWlzZS5yZXNvbHZlKCk7XG4gICAgfVxuXG4gICAgZm9yIChsZXQgaSA9IDAsIGxlbiA9IHRoaXMub3B0aW9ucy5xdWV1ZXMubGVuZ3RoOyBpIDwgbGVuOyBpKyspIHtcbiAgICAgIGNvbnN0IHF1ZXVlID0gdGhpcy5vcHRpb25zLnF1ZXVlc1tpXTtcbiAgICAgIHRoaXMucXVldWVDcmVhdGUocXVldWUpO1xuICAgIH1cblxuICAgIHJldHVybiBQcm9taXNlLnJlc29sdmUoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDcmVhdGVzIGEgbmV3IGpvYiBmb3IgdGhlIHNwZWNpZmllZCBxdWV1ZVxuICAgKiBAcGFyYW0gcXVldWVcbiAgICogQHBhcmFtIGRhdGFcbiAgICogQHBhcmFtIG9wdGlvbnNcbiAgICogQHJldHVybnMgeyp8Sm9ifVxuICAgKi9cbiAgY3JlYXRlKHF1ZXVlLCBkYXRhLCBvcHRpb25zKSB7XG4gICAgdGhpcy5sb2cudmVyYm9zZShgQ3JlYXRpbmcgdGFzayBmb3IgJHtxdWV1ZX1gKTtcbiAgICByZXR1cm4gbmV3IEpvYih0aGlzLmNvcmUsIG51bGwsIGRhdGEsIG9wdGlvbnMsIHF1ZXVlLCB0cnVlKTtcbiAgfVxuXG4gIC8qKlxuICAgKlxuICAgKiBAcGFyYW0gcXVldWVcbiAgICovXG4gIHF1ZXVlQ3JlYXRlKHF1ZXVlKSB7XG4gICAgdGhpcy5sb2cudmVyYm9zZShgUXVldWUgJyR7cXVldWUubmFtZX0nIGNyZWF0ZWQhYCk7XG4gICAgdGhpcy5xdWV1ZXNbcXVldWUubmFtZV0gPSBuZXcgUXVldWUoeyAuLi50aGlzLm9wdGlvbnMsIC4uLnF1ZXVlIH0sIHRoaXMuY29yZSk7XG4gIH1cblxuICAvKipcbiAgICogQ29udmVydHMgdGhlIHVzZXJzIGtleSB0byB0aGUgZnVsbCByZWRpcyBrZXkgd2l0aCBtb2R1bGUgcHJlZml4LlxuICAgKiBAcGFyYW0ga2V5XG4gICAqIEByZXR1cm5zIHtzdHJpbmd9XG4gICAqL1xuICB0b0tleShrZXkgPSAnJykge1xuICAgIHJldHVybiBgJHt0aGlzLm9wdGlvbnMua2V5UHJlZml4fToke2tleX1gO1xuICB9XG5cbiAgLyoqXG4gICAqIFRvIGVuYWJsZSBieXBhc3Npbmcgb2YgY2FjaGUgZm9yIHdyYXAgZnVuY3Rpb25zXG4gICAqIFRvZ2dsZXMgYnkgZGVmYXVsdCBvciBwYXNzIGluIHRydWUvZmFsc2VcbiAgICogQHBhcmFtIGJvb2xcbiAgICovXG4gIGVuYWJsZWQoYm9vbCkge1xuICAgIHRoaXMub3B0aW9ucy5lbmFibGVkID0gYm9vbCB8fCAhdGhpcy5vcHRpb25zLmVuYWJsZWQ7XG4gIH1cbn1cbiJdfQ==
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9ob29rLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7O0FBQUE7Ozs7QUFDQTs7OztBQUNBOzs7O0FBQ0E7Ozs7QUFDQTs7OztBQUNBOzs7O0FBRWUsTUFBTSxPQUFOLDJCQUErQjtBQUM1QyxnQkFBYztBQUNaLFVBQU0sS0FBTjtBQUNBLFNBQUssTUFBTCxHQUFjLEVBQWQ7QUFDRDs7Ozs7OztBQU9ELGFBQVc7QUFDVDtBQUNEOzs7Ozs7QUFNRCxZQUFVO0FBQ1I7QUFDRDs7Ozs7QUFLRCxlQUFhO0FBQ1gsUUFBSSxDQUFDLEtBQUssT0FBTCxDQUFhLE9BQWxCLEVBQTJCO0FBQ3pCLGFBQU8sbUJBQVEsT0FBUixFQUFQO0FBQ0Q7O0FBRUQsU0FBSyxJQUFJLElBQUksQ0FBUixFQUFXLE1BQU0sS0FBSyxPQUFMLENBQWEsTUFBYixDQUFvQixNQUExQyxFQUFrRCxJQUFJLEdBQXRELEVBQTJELEdBQTNELEVBQWdFO0FBQzlELFlBQU0sUUFBUSxLQUFLLE9BQUwsQ0FBYSxNQUFiLENBQW9CLENBQXBCLENBQWQ7QUFDQSxXQUFLLFdBQUwsQ0FBaUIsS0FBakI7QUFDRDs7QUFFRCxXQUFPLG1CQUFRLE9BQVIsRUFBUDtBQUNEOzs7Ozs7Ozs7QUFTRCxTQUFPLEtBQVAsRUFBYyxJQUFkLEVBQW9CLE9BQXBCLEVBQTZCO0FBQzNCLFNBQUssR0FBTCxDQUFTLE9BQVQsQ0FBaUIsQ0FBQyxrQkFBRCxHQUFxQixLQUFyQixFQUEyQixBQUEzQixDQUFqQjtBQUNBLFdBQU8sa0JBQVEsS0FBSyxJQUFiLEVBQW1CLElBQW5CLEVBQXlCLElBQXpCLEVBQStCLE9BQS9CLEVBQXdDLEtBQXhDLEVBQStDLElBQS9DLENBQVA7QUFDRDs7Ozs7O0FBTUQsY0FBWSxLQUFaLEVBQW1CO0FBQ2pCLFNBQUssR0FBTCxDQUFTLE9BQVQsQ0FBaUIsQ0FBQyxPQUFELEdBQVUsTUFBTSxJQUFoQixFQUFxQixVQUFyQixDQUFqQjtBQUNBLFNBQUssTUFBTCxDQUFZLE1BQU0sSUFBbEIsSUFBMEIsaUNBQWUsS0FBSyxPQUFwQixFQUFnQyxLQUFoQyxHQUF5QyxLQUFLLElBQTlDLENBQTFCO0FBQ0Q7Ozs7Ozs7QUFPRCxRQUFNLE1BQU0sRUFBWixFQUFnQjtBQUNkLFdBQU8sQ0FBQyxBQUFELEdBQUcsS0FBSyxPQUFMLENBQWEsU0FBaEIsRUFBMEIsQ0FBMUIsR0FBNkIsR0FBN0IsRUFBaUMsQUFBakMsQ0FBUDtBQUNEOzs7Ozs7O0FBT0QsVUFBUSxJQUFSLEVBQWM7QUFDWixTQUFLLE9BQUwsQ0FBYSxPQUFiLEdBQXVCLFFBQVEsQ0FBQyxLQUFLLE9BQUwsQ0FBYSxPQUE3QztBQUNEO0FBNUUyQztrQkFBekIsTyIsImZpbGUiOiJob29rLmpzIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IEpvYiBmcm9tICcuL2pvYic7XG5pbXBvcnQgUXVldWUgZnJvbSAnLi9xdWV1ZSc7XG5pbXBvcnQgUHJvbWlzZSBmcm9tICdibHVlYmlyZCc7XG5pbXBvcnQgZGVmYXVsdHMgZnJvbSAnLi9kZWZhdWx0cyc7XG5pbXBvcnQgc2NyaXB0cyBmcm9tICcuL3NjcmlwdHMnO1xuaW1wb3J0IHsgQmFzZUhvb2sgfSBmcm9tICdyZWRpYm94JztcblxuZXhwb3J0IGRlZmF1bHQgY2xhc3MgSm9iSG9vayBleHRlbmRzIEJhc2VIb29rIHtcbiAgY29uc3RydWN0b3IoKSB7XG4gICAgc3VwZXIoJ2pvYicpO1xuICAgIHRoaXMucXVldWVzID0ge307XG4gIH1cblxuICAvLyBub2luc3BlY3Rpb24gSlNVbnVzZWRHbG9iYWxTeW1ib2xzLEpTTWV0aG9kQ2FuQmVTdGF0aWNcbiAgLyoqXG4gICAqIERlZmF1bHQgY29uZmlnIGZvciBzY2hlZHVsZXJcbiAgICogQHJldHVybnMge3tzb21lRGVmYXVsdFRoaW5nOiBzdHJpbmd9fVxuICAgKi9cbiAgZGVmYXVsdHMoKSB7XG4gICAgcmV0dXJuIGRlZmF1bHRzO1xuICB9XG5cbiAgLyoqXG4gICAqXG4gICAqIEByZXR1cm5zIHt7YWRkSm9iLCBjaGVja1N0YWxsZWRKb2JzLCByZW1vdmVKb2J9fVxuICAgKi9cbiAgc2NyaXB0cygpIHtcbiAgICByZXR1cm4gc2NyaXB0cztcbiAgfVxuXG4gIC8qKlxuICAgKlxuICAgKi9cbiAgaW5pdGlhbGl6ZSgpIHtcbiAgICBpZiAoIXRoaXMub3B0aW9ucy5lbmFibGVkKSB7XG4gICAgICByZXR1cm4gUHJvbWlzZS5yZXNvbHZlKCk7XG4gICAgfVxuXG4gICAgZm9yIChsZXQgaSA9IDAsIGxlbiA9IHRoaXMub3B0aW9ucy5xdWV1ZXMubGVuZ3RoOyBpIDwgbGVuOyBpKyspIHtcbiAgICAgIGNvbnN0IHF1ZXVlID0gdGhpcy5vcHRpb25zLnF1ZXVlc1tpXTtcbiAgICAgIHRoaXMucXVldWVDcmVhdGUocXVldWUpO1xuICAgIH1cblxuICAgIHJldHVybiBQcm9taXNlLnJlc29sdmUoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDcmVhdGVzIGEgbmV3IGpvYiBmb3IgdGhlIHNwZWNpZmllZCBxdWV1ZVxuICAgKiBAcGFyYW0gcXVldWVcbiAgICogQHBhcmFtIGRhdGFcbiAgICogQHBhcmFtIG9wdGlvbnNcbiAgICogQHJldHVybnMgeyp8Sm9ifVxuICAgKi9cbiAgY3JlYXRlKHF1ZXVlLCBkYXRhLCBvcHRpb25zKSB7XG4gICAgdGhpcy5sb2cudmVyYm9zZShgQ3JlYXRpbmcgdGFzayBmb3IgJHtxdWV1ZX1gKTtcbiAgICByZXR1cm4gbmV3IEpvYih0aGlzLmNvcmUsIG51bGwsIGRhdGEsIG9wdGlvbnMsIHF1ZXVlLCB0cnVlKTtcbiAgfVxuXG4gIC8qKlxuICAgKlxuICAgKiBAcGFyYW0gcXVldWVcbiAgICovXG4gIHF1ZXVlQ3JlYXRlKHF1ZXVlKSB7XG4gICAgdGhpcy5sb2cudmVyYm9zZShgUXVldWUgJyR7cXVldWUubmFtZX0nIGNyZWF0ZWQhYCk7XG4gICAgdGhpcy5xdWV1ZXNbcXVldWUubmFtZV0gPSBuZXcgUXVldWUoeyAuLi50aGlzLm9wdGlvbnMsIC4uLnF1ZXVlIH0sIHRoaXMuY29yZSk7XG4gIH1cblxuICAvKipcbiAgICogQ29udmVydHMgdGhlIHVzZXJzIGtleSB0byB0aGUgZnVsbCByZWRpcyBrZXkgd2l0aCBtb2R1bGUgcHJlZml4LlxuICAgKiBAcGFyYW0ga2V5XG4gICAqIEByZXR1cm5zIHtzdHJpbmd9XG4gICAqL1xuICB0b0tleShrZXkgPSAnJykge1xuICAgIHJldHVybiBgJHt0aGlzLm9wdGlvbnMua2V5UHJlZml4fToke2tleX1gO1xuICB9XG5cbiAgLyoqXG4gICAqIFRvIGVuYWJsZSBieXBhc3Npbmcgb2YgY2FjaGUgZm9yIHdyYXAgZnVuY3Rpb25zXG4gICAqIFRvZ2dsZXMgYnkgZGVmYXVsdCBvciBwYXNzIGluIHRydWUvZmFsc2VcbiAgICogQHBhcmFtIGJvb2xcbiAgICovXG4gIGVuYWJsZWQoYm9vbCkge1xuICAgIHRoaXMub3B0aW9ucy5lbmFibGVkID0gYm9vbCB8fCAhdGhpcy5vcHRpb25zLmVuYWJsZWQ7XG4gIH1cbn1cbiJdfQ==

@@ -34,18 +34,18 @@ 'use strict';

*/
constructor(core, id) {
let data = arguments.length <= 2 || arguments[2] === undefined ? {} : arguments[2];
let options = arguments.length <= 3 || arguments[3] === undefined ? {
unique: false,
timeout: 60000 } : arguments[3];
let // 1 minute default timeout
queueName = arguments[4];
let isNew = arguments[5];
constructor(core, id, data = {}, options = {
unique: false,
timeout: 60000 }, // 1 minute default timeout
queueName, isNew) {
this.id = id;
this.core = core;
this.data = data;
this.options = options;
this.status = 'created';
this.options = options;
this.subscriptions = [];
this.queueName = queueName;
this.subscriptions = [];
this.type = data.runs && Array.isArray(data.runs) ? 'relay' : 'single';
// parent relay job timeout should cover all child job timeouts
if (this.type === 'relay') this.options.timeout = this.options.timeout * data.runs.length;
if (isNew) {

@@ -69,4 +69,6 @@ // this Proxy allows chaining methods while still keeping the

});
return this.proxy;
}
return this;

@@ -116,7 +118,7 @@ }

* @returns {*}
*/
toObject() {
*/
toObject(excludeData) {
return {
id: this.id,
data: this.data,
data: excludeData ? 'hidden' : this.data,
status: this.status,

@@ -141,4 +143,3 @@ options: this.options

this.status = 'saved';
// this.queue.jobs[id] = this;
return _bluebird2.default.resolve(this.toObject());
return _bluebird2.default.resolve(this.toObject(true));
});

@@ -172,18 +173,10 @@ }

// on message received
const channel = message.channel;
// remove the pubsub data
if (message.data) {
message = message.data;
}
if (message.data) message = message.data;
// if there's an error the assume failed.
if (message.error) {
return this.onFailureCallback(message);
}
// if there's an error then assume failed.
if (message.error) return this.onFailureCallback(message);
// is it from the success channel.
if (this.subscriptions[0] === channel) {
return this.onSuccessCallback(message);
}
if (this.subscriptions[0] === message.channel) return this.onSuccessCallback(message);

@@ -194,3 +187,3 @@ return this.onFailureCallback(message);

type: 'job',
error: new Error('Error while subscribing to job events, however this job will still be queued - ' + 'you may be unable to receive onComplete / onFailure events for this job.'),
error: new Error('Error while subscribing to job events, however this job will still be queued - ' + 'you may be unable to receive onSuccess / onFailure events for this job.'),
error_actual: error

@@ -209,5 +202,3 @@ }));

retries(n) {
if (n < 0) {
throw Error('Retries cannot be negative');
}
if (n < 0) throw Error('Retries cannot be negative');
this.options.retries = n - 1;

@@ -252,3 +243,3 @@ return this.proxy;

/**
* Set how long this job can run before it times out.
* Set how long this job can remain running for before it times out.
* @param ms

@@ -263,52 +254,2 @@ * @returns {Job}

/**
*
* @returns {Job.initialJob|*}
*/
initialJob() {
return this._internalData.initialJob;
}
/**
*
* @returns {Job.initialQueue|*}
*/
initialQueue() {
return this._internalData.initialQueue;
}
/**
* Remove this job from all sets.
* @param cb
*/
remove() {
let cb = arguments.length <= 0 || arguments[0] === undefined ? _redibox.noop : arguments[0];
this.core.client.removejob(this._toQueueKey('succeeded'), this._toQueueKey('failed'), this._toQueueKey('waiting'), this._toQueueKey('active'), this._toQueueKey('stalling'), this._toQueueKey('jobs'), this.id, cb);
}
/**
* Re-save this job for the purpose of retrying it.
* @param cb
*/
retry() {
let cb = arguments.length <= 0 || arguments[0] === undefined ? _redibox.noop : arguments[0];
this.core.client.multi().srem(this._toQueueKey('failed'), this.id).lpush(this._toQueueKey('waiting'), this.id).exec(cb);
}
/**
* Callbacks true of false if this job exists in the specified set.
* @param set
* @param cb
*/
isInSet(set) {
let cb = arguments.length <= 1 || arguments[1] === undefined ? _redibox.noop : arguments[1];
this.core.client.sismember(this._toQueueKey(set), this.id, (err, result) => {
if (err) return cb(err);
return cb(null, result === 1);
});
}
/**
* Generates a queue prefixed key based on the provided string.

@@ -353,2 +294,74 @@ * @param str

exports.default = Job;
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/job.js"],"names":[],"mappings":"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","file":"job.js","sourcesContent":["/**\n *\n * The MIT License (MIT)\n *\n * Copyright (c) 2015 Salakar\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n *\n */\n\nimport cuid from 'cuid';\nimport Promise from 'bluebird';\nimport { noop, sha1sum, tryJSONStringify, tryJSONParse } from 'redibox';\n\n/**\n * @class Job\n */\nclass Job {\n\n  /**\n   *\n   * @param core\n   * @param id\n   * @param data\n   * @param options\n   * @param queueName\n   * @param isNew\n   * @returns {*}\n   */\n  constructor(core, id, data = {}, options = {\n    unique: false,\n    timeout: 60000, // 1 minute default timeout\n  }, queueName, isNew) {\n    this.id = id;\n    this.core = core;\n    this.data = data;\n    this.status = 'created';\n    this.options = options;\n    this.queueName = queueName;\n    this.subscriptions = [];\n    if (isNew) {\n      // this Proxy allows chaining methods while still keeping the\n      // save() promise valid\n      this.proxy = new Proxy(this, {\n        get(target, name) {\n          if (name in target) {\n            return target[name];\n          }\n\n          // haxxors\n          if (name === 'then') {\n            target.promise = target.save();\n            return target.promise.then.bind(target.promise);\n          }\n\n          return undefined;\n        },\n      });\n      return this.proxy;\n    }\n    return this;\n  }\n\n  /**\n   * Query redis for the specified job id and converts it to a new instance of Job.\n   * @static\n   * @param queue\n   * @param id\n   */\n  static fromId(queue, id) {\n    return queue.client.hget(queue.toKey('jobs'), id).then((data) =>\n      Job.fromData(queue, id, data)\n    );\n  }\n\n  /**\n   * Converts a JSON string of a job's data to a new instance of Job\n   * @static\n   * @param queue\n   * @param id\n   * @param data\n   * @returns {Job | null}\n   */\n  static fromData(queue, id, data) {\n    const obj = tryJSONParse(data);\n    if (!obj) return null;\n    const job = new Job(queue.core, id, obj.data, obj.options, queue.name);\n    job.status = obj.data.status;\n    return job;\n  }\n\n  /**\n   * Convert this Job instance to a json string.\n   */\n  toData() {\n    return tryJSONStringify({\n      id: this.id,\n      data: this.data,\n      status: this.status,\n      options: this.options,\n    });\n  }\n\n  /**\n   *\n   * @returns {*}\n\t */\n  toObject() {\n    return {\n      id: this.id,\n      data: this.data,\n      status: this.status,\n      options: this.options,\n    };\n  }\n\n  /**\n   * Internal save that pushes to redis.\n   * @private\n   */\n  _save() {\n    this.core.log.verbose(`Saving new job ${this.id} for ${this.queueName}`);\n    return this.core.client.addjob(\n      this._toQueueKey('jobs'),\n      this._toQueueKey('waiting'),\n      this._toQueueKey('id'),\n      this.toData(),\n      !!this.options.unique,\n      this.id).then((id) => {\n        if (this.options.unique && id === 0) {\n          this.status = 'duplicate';\n          return Promise.reject(new Error(`ERR_DUPLICATE: Job ${this.id} already exists, save has been aborted.`));\n        }\n        this.core.log.verbose(`Saved job for ${this.queueName}`);\n        this.id = id;\n        this.status = 'saved';\n        // this.queue.jobs[id] = this;\n        return Promise.resolve(this.toObject());\n      }\n    );\n  }\n\n  /**\n   * Save this instance of Job to redis. Any active queues will pick it up\n   * immediately for processing.\n   * @returns {*}\n   */\n  save() {\n    this.id = `${this.queueName}-${(this.options.unique ? sha1sum(this.data) : cuid())}`;\n\n    if (this.options.notifySuccess) {\n      this.options.notifySuccess = `job:${this.id}:success`;\n      this.subscriptions.push(`job:${this.id}:success`);\n    }\n\n    if (this.options.notifyFailure) {\n      this.options.notifyFailure = `job:${this.id}:failure`;\n      this.subscriptions.push(`job:${this.id}:failure`);\n    }\n\n    if (this.options.notifySuccess || this.options.notifyFailure) {\n      if (!this.core.pubsub.options.subscriber) {\n        return Promise.reject(\n          new Error('Cannot subscribe to job events when RediBox.pubsub \\'subscriber\\' config is set to disabled.')\n        );\n      }\n\n      return this.core.pubsub.subscribeOnceOf(this.subscriptions, (message) => { // on message received\n        const channel = message.channel;\n\n        // remove the pubsub data\n        if (message.data) {\n          message = message.data;\n        }\n\n        // if there's an error the assume failed.\n        if (message.error) {\n          return this.onFailureCallback(message);\n        }\n\n        // is it from the success channel.\n        if (this.subscriptions[0] === channel) {\n          return this.onSuccessCallback(message);\n        }\n\n        return this.onFailureCallback(message);\n      }, this.options.timeout + 2000).then(() =>  // subscribed callback\n        this._save()\n      ).catch(error =>\n        this.onFailureCallback({\n          type: 'job',\n          error: new Error('Error while subscribing to job events, however this job will still be queued - ' +\n            'you may be unable to receive onComplete / onFailure events for this job.'),\n          error_actual: error,\n        })\n      );\n    }\n\n    return this._save();\n  }\n\n  /**\n   * Set the number of times this job will retry on failure\n   * @param n\n   * @returns {Job}\n   */\n  retries(n) {\n    if (n < 0) {\n      throw Error('Retries cannot be negative');\n    }\n    this.options.retries = n - 1;\n    return this.proxy;\n  }\n\n  /**\n   * Set the onSuccess callback and notify option\n   * @param notify\n   * @returns {Job}\n   */\n  onSuccess(notify) {\n    this.options.notifySuccess = true;\n    this.onSuccessCallback = notify;\n    if (!this.onFailureCallback) this.onFailureCallback = noop;\n    return this.proxy;\n  }\n\n  /**\n   * Set the onFailure callback and notify option\n   * @param notify\n   * @returns {Job}\n   */\n  onFailure(notify) {\n    this.options.notifyFailure = true;\n    this.onFailureCallback = notify;\n    if (!this.onSuccessCallback) this.onSuccessCallback = noop;\n    return this.proxy;\n  }\n\n  /**\n   *\n   * @param bool\n   * @returns {Job}\n   */\n  unique(bool) {\n    this.options.unique = bool;\n    return this.proxy;\n  }\n\n  /**\n   * Set how long this job can run before it times out.\n   * @param ms\n   * @returns {Job}\n   */\n  timeout(ms) {\n    this.options.timeout = ms;\n    return this.proxy;\n  }\n\n  /**\n   *\n   * @returns {Job.initialJob|*}\n   */\n  initialJob() {\n    return this._internalData.initialJob;\n  }\n\n  /**\n   *\n   * @returns {Job.initialQueue|*}\n   */\n  initialQueue() {\n    return this._internalData.initialQueue;\n  }\n\n  /**\n   * Remove this job from all sets.\n   * @param cb\n   */\n  remove(cb = noop) {\n    this.core.client.removejob(\n      this._toQueueKey('succeeded'), this._toQueueKey('failed'), this._toQueueKey('waiting'),\n      this._toQueueKey('active'), this._toQueueKey('stalling'), this._toQueueKey('jobs'),\n      this.id, cb);\n  }\n\n  /**\n   * Re-save this job for the purpose of retrying it.\n   * @param cb\n   */\n  retry(cb = noop) {\n    this.core.client.multi()\n        .srem(this._toQueueKey('failed'), this.id)\n        .lpush(this._toQueueKey('waiting'), this.id)\n        .exec(cb);\n  }\n\n  /**\n   * Callbacks true of false if this job exists in the specified set.\n   * @param set\n   * @param cb\n   */\n  isInSet(set, cb = noop) {\n    this.core.client.sismember(this._toQueueKey(set), this.id, (err, result) => {\n      if (err) return cb(err);\n      return cb(null, result === 1);\n    });\n  }\n\n  /**\n   * Generates a queue prefixed key based on the provided string.\n   * @param str\n   * @returns {string}\n   * @private\n   */\n  _toQueueKey(str) {\n    if (this.core.cluster.isCluster()) {\n      return `${this.core.hooks.job.options.keyPrefix}:{${this.queueName}}:${str}`;\n    }\n    return `${this.core.hooks.job.options.keyPrefix}:${this.queueName}:${str}`;\n  }\n\n}\n\nexport default Job;\n"]}
// Experimental code below is TODO
// /**
// *
// * @returns {Job.initialJob|*}
// */
// initialJob() {
// return this._internalData.initialJob;
// }
//
// /**
// *
// * @returns {Job.initialQueue|*}
// */
// initialQueue() {
// return this._internalData.initialQueue;
// }
/**
*
* @param jobId
* @returns {Promise}
*/
// getJob(jobId) {
// return new Promise((resolve, reject) => {
// if (jobId in this.jobs) {
// // we have the job locally
// return resolve(this.jobs[jobId]);
// }
// // not local so gather from redis
// Job.fromId(this, jobId)
// .then(job => {
// this.jobs[jobId] = job;
// return resolve(job);
// }).catch(reject);
// });
// }
//
// /**
// * Remove this job from all sets.
// * @param cb
// */
// remove(cb = noop) {
// this.core.client.removejob(
// this._toQueueKey('succeeded'), this._toQueueKey('failed'), this._toQueueKey('waiting'),
// this._toQueueKey('active'), this._toQueueKey('stalling'), this._toQueueKey('jobs'),
// this.id, cb);
// }
//
// /**
// * Re-save this job for the purpose of retrying it.
// * @param cb
// */
// retry(cb = noop) {
// this.core.client.multi()
// .srem(this._toQueueKey('failed'), this.id)
// .lpush(this._toQueueKey('waiting'), this.id)
// .exec(cb);
// }
//
// /**
// * Callbacks true of false if this job exists in the specified set.
// * @param set
// * @param cb
// */
// isInSet(set, cb = noop) {
// this.core.client.sismember(this._toQueueKey(set), this.id, (err, result) => {
// if (err) return cb(err);
// return cb(null, result === 1);
// });
// }
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/job.js"],"names":[],"mappings":"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","file":"job.js","sourcesContent":["/**\n *\n * The MIT License (MIT)\n *\n * Copyright (c) 2015 Salakar\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n *\n */\n\nimport cuid from 'cuid';\nimport Promise from 'bluebird';\nimport { noop, sha1sum, tryJSONStringify, tryJSONParse } from 'redibox';\n\n/**\n * @class Job\n */\nclass Job {\n\n  /**\n   *\n   * @param core\n   * @param id\n   * @param data\n   * @param options\n   * @param queueName\n   * @param isNew\n   * @returns {*}\n   */\n  constructor(core, id, data = {}, options = {\n    unique: false,\n    timeout: 60000, // 1 minute default timeout\n  }, queueName, isNew) {\n    this.id = id;\n    this.core = core;\n    this.data = data;\n    this.options = options;\n    this.status = 'created';\n    this.subscriptions = [];\n    this.queueName = queueName;\n    this.type = data.runs && Array.isArray(data.runs) ? 'relay' : 'single';\n\n    // parent relay job timeout should cover all child job timeouts\n    if (this.type === 'relay') this.options.timeout = this.options.timeout * data.runs.length;\n\n    if (isNew) {\n      // this Proxy allows chaining methods while still keeping the\n      // save() promise valid\n      this.proxy = new Proxy(this, {\n        get(target, name) {\n          if (name in target) {\n            return target[name];\n          }\n\n          // haxxors\n          if (name === 'then') {\n            target.promise = target.save();\n            return target.promise.then.bind(target.promise);\n          }\n\n          return undefined;\n        },\n      });\n\n      return this.proxy;\n    }\n\n    return this;\n  }\n\n  /**\n   * Query redis for the specified job id and converts it to a new instance of Job.\n   * @static\n   * @param queue\n   * @param id\n   */\n  static fromId(queue, id) {\n    return queue.client.hget(queue.toKey('jobs'), id).then((data) =>\n      Job.fromData(queue, id, data)\n    );\n  }\n\n  /**\n   * Converts a JSON string of a job's data to a new instance of Job\n   * @static\n   * @param queue\n   * @param id\n   * @param data\n   * @returns {Job | null}\n   */\n  static fromData(queue, id, data) {\n    const obj = tryJSONParse(data);\n    if (!obj) return null;\n    const job = new Job(queue.core, id, obj.data, obj.options, queue.name);\n    job.status = obj.data.status;\n    return job;\n  }\n\n  /**\n   * Convert this Job instance to a json string.\n   */\n  toData() {\n    return tryJSONStringify({\n      id: this.id,\n      data: this.data,\n      status: this.status,\n      options: this.options,\n    });\n  }\n\n  /**\n   *\n   * @returns {*}\n   */\n  toObject(excludeData) {\n    return {\n      id: this.id,\n      data: excludeData ? 'hidden' : this.data,\n      status: this.status,\n      options: this.options,\n    };\n  }\n\n  /**\n   * Internal save that pushes to redis.\n   * @private\n   */\n  _save() {\n    this.core.log.verbose(`Saving new job ${this.id} for ${this.queueName}`);\n    return this.core.client.addjob(\n      this._toQueueKey('jobs'),\n      this._toQueueKey('waiting'),\n      this._toQueueKey('id'),\n      this.toData(),\n      !!this.options.unique,\n      this.id).then((id) => {\n        if (this.options.unique && id === 0) {\n          this.status = 'duplicate';\n          return Promise.reject(new Error(`ERR_DUPLICATE: Job ${this.id} already exists, save has been aborted.`));\n        }\n        this.core.log.verbose(`Saved job for ${this.queueName}`);\n        this.id = id;\n        this.status = 'saved';\n        return Promise.resolve(this.toObject(true));\n      }\n    );\n  }\n\n  /**\n   * Save this instance of Job to redis. Any active queues will pick it up\n   * immediately for processing.\n   * @returns {*}\n   */\n  save() {\n    this.id = `${this.queueName}-${(this.options.unique ? sha1sum(this.data) : cuid())}`;\n\n    if (this.options.notifySuccess) {\n      this.options.notifySuccess = `job:${this.id}:success`;\n      this.subscriptions.push(`job:${this.id}:success`);\n    }\n\n    if (this.options.notifyFailure) {\n      this.options.notifyFailure = `job:${this.id}:failure`;\n      this.subscriptions.push(`job:${this.id}:failure`);\n    }\n\n    if (this.options.notifySuccess || this.options.notifyFailure) {\n      if (!this.core.pubsub.options.subscriber) {\n        return Promise.reject(\n          new Error('Cannot subscribe to job events when RediBox.pubsub \\'subscriber\\' config is set to disabled.')\n        );\n      }\n\n      return this.core.pubsub.subscribeOnceOf(\n        this.subscriptions,\n        (message) => { // on message received\n          // remove the pubsub data\n          if (message.data) message = message.data;\n\n          // if there's an error then assume failed.\n          if (message.error) return this.onFailureCallback(message);\n\n          // is it from the success channel.\n          if (this.subscriptions[0] === message.channel) return this.onSuccessCallback(message);\n\n          return this.onFailureCallback(message);\n        },\n        this.options.timeout + 2000\n      ).then(() =>  // subscribed callback\n        this._save()\n      ).catch(error =>\n        this.onFailureCallback({\n          type: 'job',\n          error: new Error('Error while subscribing to job events, however this job will still be queued - ' +\n            'you may be unable to receive onSuccess / onFailure events for this job.'),\n          error_actual: error,\n        })\n      );\n    }\n\n    return this._save();\n  }\n\n  /**\n   * Set the number of times this job will retry on failure\n   * @param n\n   * @returns {Job}\n   */\n  retries(n) {\n    if (n < 0) throw Error('Retries cannot be negative');\n    this.options.retries = n - 1;\n    return this.proxy;\n  }\n\n  /**\n   * Set the onSuccess callback and notify option\n   * @param notify\n   * @returns {Job}\n   */\n  onSuccess(notify) {\n    this.options.notifySuccess = true;\n    this.onSuccessCallback = notify;\n    if (!this.onFailureCallback) this.onFailureCallback = noop;\n    return this.proxy;\n  }\n\n  /**\n   * Set the onFailure callback and notify option\n   * @param notify\n   * @returns {Job}\n   */\n  onFailure(notify) {\n    this.options.notifyFailure = true;\n    this.onFailureCallback = notify;\n    if (!this.onSuccessCallback) this.onSuccessCallback = noop;\n    return this.proxy;\n  }\n\n  /**\n   *\n   * @param bool\n   * @returns {Job}\n   */\n  unique(bool) {\n    this.options.unique = bool;\n    return this.proxy;\n  }\n\n  /**\n   * Set how long this job can remain running for before it times out.\n   * @param ms\n   * @returns {Job}\n   */\n  timeout(ms) {\n    this.options.timeout = ms;\n    return this.proxy;\n  }\n\n  /**\n   * Generates a queue prefixed key based on the provided string.\n   * @param str\n   * @returns {string}\n   * @private\n   */\n  _toQueueKey(str) {\n    if (this.core.cluster.isCluster()) {\n      return `${this.core.hooks.job.options.keyPrefix}:{${this.queueName}}:${str}`;\n    }\n    return `${this.core.hooks.job.options.keyPrefix}:${this.queueName}:${str}`;\n  }\n\n}\n\nexport default Job;\n\n// Experimental code below is TODO\n\n// /**\n//  *\n//  * @returns {Job.initialJob|*}\n//  */\n// initialJob() {\n//   return this._internalData.initialJob;\n// }\n//\n// /**\n//  *\n//  * @returns {Job.initialQueue|*}\n//  */\n// initialQueue() {\n//   return this._internalData.initialQueue;\n// }\n\n/**\n *\n * @param jobId\n * @returns {Promise}\n */\n// getJob(jobId) {\n//   return new Promise((resolve, reject) => {\n//     if (jobId in this.jobs) {\n//       // we have the job locally\n//       return resolve(this.jobs[jobId]);\n//     }\n//     // not local so gather from redis\n//     Job.fromId(this, jobId)\n//        .then(job => {\n//          this.jobs[jobId] = job;\n//          return resolve(job);\n//        }).catch(reject);\n//   });\n// }\n//\n// /**\n//  * Remove this job from all sets.\n//  * @param cb\n//  */\n// remove(cb = noop) {\n//   this.core.client.removejob(\n//     this._toQueueKey('succeeded'), this._toQueueKey('failed'), this._toQueueKey('waiting'),\n//     this._toQueueKey('active'), this._toQueueKey('stalling'), this._toQueueKey('jobs'),\n//     this.id, cb);\n// }\n//\n// /**\n//  * Re-save this job for the purpose of retrying it.\n//  * @param cb\n//  */\n// retry(cb = noop) {\n//   this.core.client.multi()\n//       .srem(this._toQueueKey('failed'), this.id)\n//       .lpush(this._toQueueKey('waiting'), this.id)\n//       .exec(cb);\n// }\n//\n// /**\n//  * Callbacks true of false if this job exists in the specified set.\n//  * @param set\n//  * @param cb\n//  */\n// isInSet(set, cb = noop) {\n//   this.core.client.sismember(this._toQueueKey(set), this.id, (err, result) => {\n//     if (err) return cb(err);\n//     return cb(null, result === 1);\n//   });\n// }\n"]}

@@ -40,33 +40,48 @@ 'use strict';

this._jobTick = () => {
this._onLocalTickComplete = () => {
this.running--;
this.queued--;
if (!this.options.throttle) return setImmediate(this._queueTick);
return this.client.throttle(this.toKey('throttle'), this.options.throttle.limit, this.options.throttle.seconds).then(throttle => {
const shouldThrottle = throttle[0] === 1;
if (!shouldThrottle) {
this.throttled = false;
return setImmediate(this._queueTick);
}
this.throttled = true;
const timeRemaining = throttle[2] === 0 ? 1 : throttle[2];
this.log.verbose(`'${ this.name }' queue reached it's throttle limit, resuming in ${ timeRemaining } seconds.`);
return setTimeout(this._queueTick, timeRemaining * 1000);
}).catch(this._queueTick);
};
this._onLocalTickError = error => {
this.queued--;
this.log.error(error);
setImmediate(this._queueTick);
};
this._throttleQueue = () => {};
this._queueTick = () => {
if (this.paused || !this.options.enabled) {
return void 0;
}
this.queued++;
return this._getNextJob().then(job => {
this.running++;
if (this.running + this.queued < this.concurrency) {
setImmediate(this._jobTick);
// queue more jobs if within limit
if (this.running + this.queued < this.options.concurrency) {
// concurrency is a little pointless right now if we're throttling jobs
if (!this.options.throttle) setImmediate(this._queueTick);
}
return this._runJob(job).then(() => {
this.running--;
this.queued--;
setImmediate(this._jobTick);
}).catch(() => {
this.running--;
this.queued--;
setImmediate(this._jobTick);
});
}).catch(error => {
this.log.error(error);
setImmediate(this._jobTick);
});
return this._runJob(job).then(this._onLocalTickComplete).catch(this._onLocalTickComplete);
}).catch(this._onLocalTickError);
};
this._restartProcessing = () => {
this.clients.block.once('ready', this._jobTick);
this.clients.block.once('ready', this._queueTick);
};

@@ -76,3 +91,2 @@

this.jobs = {};
this.core = core;

@@ -82,2 +96,3 @@ this.client = core.client;

this.started = false;
this.throttled = false;
this.log = this.core.log;

@@ -89,3 +104,3 @@ this.name = options.name;

this.log.verbose(`Blocking client for queue '${ this.name }' is ready. Starting queue processor.`);
this.process(this.options.concurrency);
this.beginWorking();
});

@@ -119,35 +134,6 @@ }

// /**
// *
// * @param data
// * @returns {Job}
// */
// createJob(data) {
// return new Job(this, null, data);
// }
/**
*
* @param jobId
* @returns {Promise}
*/
// getJob(jobId) {
// return new Promise((resolve, reject) => {
// if (jobId in this.jobs) {
// // we have the job locally
// return resolve(this.jobs[jobId]);
// }
// // not local so gather from redis
// Job.fromId(this, jobId)
// .then(job => {
// this.jobs[jobId] = job;
// return resolve(job);
// }).catch(reject);
// });
// }
/**
*
* @returns {Promise}
*/
_getNextJob() {

@@ -163,2 +149,33 @@ this.log.verbose(`Getting next job for queue '${ this.name }'.`);

* @param job
* @param jobError
* @private
*/
_logJobFailure(job, jobError) {
if (process.env.KUBERNETES_PORT || process.env.KUBERNETES_SERVICE_HOST) {
/* eslint no-console: 0 */
console.log(JSON.stringify({
level: 'error',
type: 'redibox_job_failure',
data: {
runs: job.data.runs,
queue: this.name,
stack: jobError.stack ? jobError.stack.split('\n').slice(0, 5) : []
}
}));
} else {
this.log.error('');
this.log.error('--------------- RDB JOB ERROR/FAILURE ---------------');
this.log.error(`Job: ${ job.data.runs }` || this.name);
if (jobError.stack) {
this.log.error(jobError.stack.split('\n').slice(0, 5));
}
this.log.error(jobError);
this.log.error('------------------------------------------------------');
this.log.error('');
}
}
/**
*
* @param job
* @returns {Promise}

@@ -177,5 +194,3 @@ */

// silently ignore any multiple calls
if (handled) {
return void 0;
}
if (handled) return void 0;

@@ -190,6 +205,4 @@ clearTimeout(preventStallingTimeout);

if (job.data.runs && Array.isArray(job.data.runs) && data !== false) {
return this._finishMultiJob(null, data, job);
}
// only relay to next job if user did not resolve 'false' on current job
if (job.type === 'relay' && data !== false) return this._finishRelayJob(null, data, job);
return this._finishSingleJob(null, data, job);

@@ -199,3 +212,3 @@ };

// Handle any errors returned
const handleError = err => {
const handleError = jobError => {
clearTimeout(preventStallingTimeout);

@@ -217,19 +230,7 @@

if ((!job.data.initialJob || !job.data.initialJob.options.notifyFailure) && !Array.isArray(job.data.runs)) {
this.log.error('');
this.log.error('--------------- RDB JOB ERROR/FAILURE ---------------');
this.log.error(`Job: ${ job.data.runs }` || this.name);
if (err.stack) {
err.stack.split('\n').forEach(error => {
this.log.error(error);
});
}
this.log.error(err);
this.log.error('------------------------------------------------------');
this.log.error('');
this._logJobFailure(job, jobError);
}
if (job.data.runs && Array.isArray(job.data.runs)) {
return this._finishMultiJob(err, null, job);
}
return this._finishSingleJob(err, null, job);
if (job.type === 'relay') return this._finishRelayJob(jobError, null, job);
return this._finishSingleJob(jobError, null, job);
};

@@ -258,4 +259,3 @@

if (job.options.timeout) {
const msg = `Job ${ job.id } timed out (${ job.options.timeout }ms)`;
setTimeout(handleError.bind(null, Error(msg)), job.options.timeout);
setTimeout(handleError.bind(null, Error(`Job ${ job.id } timed out (${ job.options.timeout }ms)`)), job.options.timeout);
}

@@ -271,97 +271,78 @@

/**
* Completes a multi job or continues to the next stage.
*
* @param error
* @param data
* @param job
* @returns {Promise}
* @returns {{job: {id: *, worker_id: (*|String|string), status: string}, error: *, output: *}}
* @private
*/
_finishMultiJob(error, data, job) {
delete this.jobs[job.id];
_createJobEvent(error, data, job) {
return {
job: _extends({
id: job.id,
worker_id: this.core.id,
status: error ? 'failed' : 'succeeded'
}, job.data),
error,
output: data
};
}
return new _bluebird2.default((resolve, reject) => {
const status = error ? 'failed' : 'succeeded';
/**
*
* @param error
* @param data
* @param job
* @param multi
* @private
*/
_updateJobStatus(error, data, job, multi) {
const status = error ? 'failed' : 'succeeded';
const multi = this.client.multi();
multi.lrem(this.toKey('active'), 0, job.id);
multi.srem(this.toKey('stalling'), job.id);
multi.lrem(this.toKey('active'), 0, job.id);
multi.srem(this.toKey('stalling'), job.id);
const event = {
job: _extends({
id: job.id,
worker_id: this.core.id,
status
}, job.data),
error,
output: data
};
const currentJob = job.data.runs.shift();
const nextJob = job.data.runs[0];
let nextQueue = this.name;
// keep a record of the first job in this relay instance
// ssssh JSON ;p
if (!job.data.initialJob) {
job.data.initialJob = (0, _redibox.tryJSONParse)(job.toData());
if (status === 'failed') {
if (job.options.retries > 0) {
job.options.retries = job.options.retries - 1;
job.status = 'retrying';
multi.hset(this.toKey('jobs'), job.id, job.toData());
multi.lpush(this.toKey('waiting'), job.id);
} else {
job.status = 'failed';
multi.hdel(this.toKey('jobs'), job.id);
// TODO track failures and their data somewhere else for reviewing
// multi.hset(this.toKey('jobs'), job.id, job.toData());
// multi.sadd(this.toKey('failed'), job.id);
}
// keep a record of the first queue in this relay instance
if (!job.data.initialQueue) {
job.data.initialQueue = this.name;
}
if (status === 'failed') {
if (job.options.retries > 0) {
job.options.retries = job.options.retries - 1;
job.status = 'retrying';
event.event = 'retrying';
multi.hset(this.toKey('jobs'), job.id, job.toData());
multi.lpush(this.toKey('waiting'), job.id);
} else {
job.status = 'failed';
multi.hset(this.toKey('jobs'), job.id, job.toData());
multi.sadd(this.toKey('failed'), job.id);
}
} else {
} else {
job.status = 'succeeded';
multi.hset(this.toKey('jobs'), job.id, job.toData());
multi.hdel(this.toKey('jobs'), job.id);
// multi.sadd(this.toKey('succeeded'), job.id); // TODO LOG JOBS
// TODO track successes and their data somewhere else for reviewing
// multi.hset(this.toKey('jobs'), job.id, job.toData());
// multi.sadd(this.toKey('succeeded'), job.id);
}
}
// check if we need to relay to another job
if (!(job.data.runs.length === 0 || !!error)) {
if ((0, _redibox.isObject)(nextJob)) {
nextQueue = nextJob.queue;
job.data.runs[0] = nextJob.runs;
} else if (job.data.initialQueue) {
nextQueue = job.data.initialQueue;
}
/**
*
* @param error
* @param data
* @param job
* @returns {Promise}
*/
_finishSingleJob(error, data, job) {
const multi = this.client.multi();
const status = error ? 'failed' : 'succeeded';
this._updateJobStatus(error, data, job, multi);
// add some debug data for the next job
// so it can tell where its call originated from
job.data.from_job = currentJob;
job.data.from_queue = this.name;
job.data.from_timestamp = (0, _redibox.getTimeStamp)();
job.data.data = data;
// emit success or failure event if we have listeners
if (error && job.options.notifyFailure) {
this.core.pubsub.publish(job.options.notifyFailure, this._createJobEvent(error, data, job));
} else if (job.options.notifySuccess) {
this.core.pubsub.publish(job.options.notifySuccess, this._createJobEvent(error, data, job));
}
return this.core.hooks.job.create(nextQueue, job.data).then(() => {
multi.exec(errMulti => {
if (errMulti) return reject(errMulti);
return resolve({ status, result: error || data });
});
});
}
// we've just finished the last job in the relay
if (event.error) {
if (job.data.initialJob.options.notifyFailure) {
this.core.pubsub.publish(job.data.initialJob.options.notifyFailure, event);
}
} else if (job.data.initialJob.options.notifySuccess) {
this.core.pubsub.publish(job.data.initialJob.options.notifySuccess, event);
}
return multi.exec(errMulti => {
return new _bluebird2.default((resolve, reject) => {
multi.exec(errMulti => {
if (errMulti) return reject(errMulti);

@@ -374,3 +355,3 @@ return resolve({ status, result: error || data });

/**
*
* Completes a multi job or continues to the next stage.
* @param error

@@ -380,46 +361,60 @@ * @param data

* @returns {Promise}
* @private
*/
_finishSingleJob(error, data, job) {
delete this.jobs[job.id];
_finishRelayJob(error, data, job) {
let nextQueue = this.name;
const nextJob = job.data.runs[0];
const multi = this.client.multi();
const currentJob = job.data.runs.shift();
const status = error ? 'failed' : 'succeeded';
return new _bluebird2.default((resolve, reject) => {
const status = error ? 'failed' : 'succeeded';
const multi = this.client.multi();
// keep a record of the first job in this relay instance
if (!job.data.initialJob) {
job.data.initialJob = (0, _redibox.tryJSONParse)(job.toData());
}
multi.lrem(this.toKey('active'), 0, job.id);
multi.srem(this.toKey('stalling'), job.id);
// keep a record of the first queue in this relay instance
if (!job.data.initialQueue) {
job.data.initialQueue = this.name;
}
if (status === 'failed') {
if (job.options.retries > 0) {
job.options.retries = job.options.retries - 1;
job.status = 'retrying';
multi.hset(this.toKey('jobs'), job.id, job.toData());
multi.lpush(this.toKey('waiting'), job.id);
} else {
job.status = 'failed';
multi.hset(this.toKey('jobs'), job.id, job.toData());
multi.sadd(this.toKey('failed'), job.id);
}
} else {
job.status = 'succeeded';
multi.hset(this.toKey('jobs'), job.id, job.toData());
multi.hdel(this.toKey('jobs'), job.id);
// multi.sadd(this.toKey('succeeded'), job.id); // TODO LOG JOBS
}
this._updateJobStatus(error, data, job, multi);
if (error && (job.options.notifySuccess || job.notifyFailure)) {
const event = {
job: _extends({
id: job.id,
worker_id: this.core.id,
status
}, job.data),
error,
output: data
};
if (job.options.notifyFailure) this.core.pubsub.publish(job.options.notifyFailure, event);
if (job.options.notifySuccess) this.core.pubsub.publish(job.options.notifySuccess, event);
// check if we need to relay to another job
if (!(job.data.runs.length === 0 || !!error)) {
if ((0, _redibox.isObject)(nextJob)) {
nextQueue = nextJob.queue;
job.data.runs[0] = nextJob.runs;
} else if (job.data.initialQueue) {
nextQueue = job.data.initialQueue;
}
multi.exec(errMulti => {
// add some debug data for the next job
// so it can tell where the relay originated from
job.data.from_job = currentJob;
job.data.from_queue = this.name;
job.data.from_timestamp = (0, _redibox.getTimeStamp)();
// relay resolved data
job.data.data = data;
return new _bluebird2.default((resolve, reject) => {
return this.core.hooks.job.create(nextQueue, job.data).then(() => {
multi.exec(errMulti => {
if (errMulti) return reject(errMulti);
return resolve({ status, result: error || data });
});
});
});
}
// we've just finished the last job in the relay
// emit success or failure event if we have listeners
if (error && job.data.initialJob.options.notifyFailure) {
this.core.pubsub.publish(job.data.initialJob.options.notifyFailure, this._createJobEvent(error, data, job));
} else if (job.data.initialJob.options.notifySuccess) {
this.core.pubsub.publish(job.data.initialJob.options.notifySuccess, this._createJobEvent(error, data, job));
}
return new _bluebird2.default((resolve, reject) => {
return multi.exec(errMulti => {
if (errMulti) return reject(errMulti);

@@ -433,2 +428,15 @@ return resolve({ status, result: error || data });

*
* @private
*/
/**
*
* @param error
* @private
*/
/**
*
* @returns {*}

@@ -447,7 +455,4 @@ * @private

* Start the queue.
* @param concurrency
*/
process() {
let concurrency = arguments.length <= 0 || arguments[0] === undefined ? 1 : arguments[0];
beginWorking() {
if (this.started || !this.options.enabled) {

@@ -458,8 +463,7 @@ this.log.info(`Queue ${ this.name } is currently disabled.`);

this.queued = 0;
this.running = 0;
this.started = true;
this.running = 0;
this.queued = 0;
this.concurrency = concurrency;
this.log.verbose(`Queue '${ this.name }' - started with a concurrency of ${ this.concurrency }.`);
this.log.verbose(`Queue '${ this.name }' - started with a concurrency of ${ this.options.concurrency }.`);

@@ -473,3 +477,3 @@ this.clients.block.once('error', this._restartProcessing);

return this._jobTick();
return this._queueTick();
}

@@ -484,6 +488,3 @@

return this.client.checkstalledjobs(this.toKey('stallTime'), this.toKey('stalling'), this.toKey('waiting'), this.toKey('active'), (0, _redibox.getTimeStamp)(), this.options.stallInterval).then(() => {
if (!this.options.enabled || this.paused) {
return _bluebird2.default.resolve();
}
if (!this.options.enabled || this.paused) return _bluebird2.default.resolve();
return _bluebird2.default.delay(this.options.stallInterval).then(this.checkStalledJobs);

@@ -512,2 +513,2 @@ });

exports.default = Queue;
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/queue.js"],"names":[],"mappings":"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","file":"queue.js","sourcesContent":["import Job from './job';\nimport Promise from 'bluebird';\nimport defaults from './defaults';\nimport EventEmitter from 'eventemitter3';\nimport { deepGet, isObject, getTimeStamp, tryJSONParse } from 'redibox';\n\nexport default class Queue extends EventEmitter {\n\n  /**\n   *\n   * @param options\n   * @param core\n   * @returns {Queue}\n   */\n  constructor(options, core) {\n    super();\n    this.jobs = {};\n    this.core = core;\n    this.client = core.client;\n    this.paused = false;\n    this.started = false;\n    this.log = this.core.log;\n    this.name = options.name;\n    this.handler = options.handler || null;\n    this.options = Object.assign({}, defaults.queue, options || {});\n    this.core.createClient('block', this).then(() => {\n      this.log.verbose(`Blocking client for queue '${this.name}' is ready. Starting queue processor.`);\n      this.process(this.options.concurrency);\n    });\n  }\n\n  /**\n   *\n   * @returns {*}\n   */\n  destroy() {\n    const keys = [\n      'id', 'jobs', 'stallTime', 'stalling', 'waiting', 'active', 'succeeded', 'failed',\n    ].map(key => this.toKey(key));\n    return this.client.del(...keys);\n  }\n\n  /**\n   *\n   * @returns {Promise}\n   */\n  checkHealth() {\n    return this\n      .client.multi()\n      .llen(this.toKey('waiting'))\n      .llen(this.toKey('active'))\n      .scard(this.toKey('succeeded'))\n      .scard(this.toKey('failed'))\n      .then(results => { /* eslint arrow-body-style: 0 */\n        return {\n          waiting: results[0][1],\n          active: results[1][1],\n          succeeded: results[2][1],\n          failed: results[3][1],\n        };\n      });\n  }\n\n  // /**\n  //  *\n  //  * @param data\n  //  * @returns {Job}\n  //  */\n  // createJob(data) {\n  //   return new Job(this, null, data);\n  // }\n\n  /**\n   *\n   * @param jobId\n   * @returns {Promise}\n   */\n  // getJob(jobId) {\n  //   return new Promise((resolve, reject) => {\n  //     if (jobId in this.jobs) {\n  //       // we have the job locally\n  //       return resolve(this.jobs[jobId]);\n  //     }\n  //     // not local so gather from redis\n  //     Job.fromId(this, jobId)\n  //        .then(job => {\n  //          this.jobs[jobId] = job;\n  //          return resolve(job);\n  //        }).catch(reject);\n  //   });\n  // }\n\n  /**\n   *\n   * @returns {Promise}\n   */\n  _getNextJob() {\n    this.log.verbose(`Getting next job for queue '${this.name}'.`);\n    return this.clients.block.brpoplpush(\n      this.toKey('waiting'),\n      this.toKey('active'), 0\n    ).then(jobId =>\n      Job.fromId(this, jobId).then(job => {\n        return job;\n      })\n    );\n  }\n\n  /**\n   *\n   * @param job\n   * @returns {Promise}\n   */\n  _runJob(job) {\n    if (!job || !job.data) return Promise.resolve();\n    const runs = job.data && job.data.runs && Array.isArray(job.data.runs) ? job.data.runs[0] : job.data.runs;\n    const handler = (typeof this.handler === 'string' ?\n        deepGet(global, this.handler) : this.handler) || deepGet(global, runs);\n\n    let preventStallingTimeout;\n    let handled = false;\n\n    // Handle an \"OK\" response from the promise\n    const handleOK = data => {\n      // silently ignore any multiple calls\n      if (handled) {\n        return void 0;\n      }\n\n      clearTimeout(preventStallingTimeout);\n      handled = true;\n\n      // set the data back to internal data\n      if (job._internalData) {\n        job.data = job._internalData;\n      }\n\n      if (job.data.runs && Array.isArray(job.data.runs) && data !== false) {\n        return this._finishMultiJob(null, data, job);\n      }\n\n      return this._finishSingleJob(null, data, job);\n    };\n\n    // Handle any errors returned\n    const handleError = err => {\n      clearTimeout(preventStallingTimeout);\n\n      // silently ignore any multiple calls\n      if (handled) {\n        return void 0;\n      }\n\n      handled = true;\n\n      // set the data back to internal job data\n      if (job._internalData) {\n        job.data = job._internalData;\n      }\n\n      // only log the error if no notifyFailure pubsub set\n      if ((!job.data.initialJob || !job.data.initialJob.options.notifyFailure) && !Array.isArray(job.data.runs)) {\n        this.log.error('');\n        this.log.error('--------------- RDB JOB ERROR/FAILURE ---------------');\n        this.log.error(`Job: ${job.data.runs}` || this.name);\n        if (err.stack) {\n          err.stack.split('\\n').forEach(error => {\n            this.log.error(error);\n          });\n        }\n        this.log.error(err);\n        this.log.error('------------------------------------------------------');\n        this.log.error('');\n      }\n\n      if (job.data.runs && Array.isArray(job.data.runs)) {\n        return this._finishMultiJob(err, null, job);\n      }\n      return this._finishSingleJob(err, null, job);\n    };\n\n    const preventStalling = () => {\n      this.client.srem(this.toKey('stalling'), job.id, () => {\n        if (!handled) {\n          preventStallingTimeout = setTimeout(preventStalling, this.options.stallInterval / 2);\n        }\n      });\n    };\n\n    if (!handler) {\n      return handleError(\n        new Error(\n          `\"${job.data.runs || 'No Job Handler Specified'}\" was not found. Skipping job. To fix this\n             you must either specify a handler function via queue.process() or provide a valid handler\n             node global path in your job options 'handler', e.g. if you had a global function in\n            'global.sails.services.myservice' you'd specify the handler as 'sails.services.myservice.myHandler'.`\n        )\n      );\n    }\n\n    preventStalling(); // start stalling monitor\n\n    job._internalData = job.data;\n    job.data = job.data.data || job.data;\n\n    if (job.options.timeout) {\n      const msg = `Job ${job.id} timed out (${job.options.timeout}ms)`;\n      setTimeout(handleError.bind(null, Error(msg)), job.options.timeout);\n    }\n\n    if (job.options.noBind || this.options.noBind) {\n      return handler(job).then(handleOK, handleError).catch(handleError);\n    }\n\n    return handler.bind(job, job)(job).then(handleOK, handleError).catch(handleError);\n  }\n\n  /**\n   * Completes a multi job or continues to the next stage.\n   * @param error\n   * @param data\n   * @param job\n   * @returns {Promise}\n   * @private\n   */\n  _finishMultiJob(error, data, job) {\n    delete this.jobs[job.id];\n\n    return new Promise((resolve, reject) => {\n      const status = error ? 'failed' : 'succeeded';\n\n      const multi = this.client.multi();\n      multi.lrem(this.toKey('active'), 0, job.id);\n      multi.srem(this.toKey('stalling'), job.id);\n\n      const event = {\n        job: {\n          id: job.id,\n          worker_id: this.core.id,\n          status,\n          ...job.data,\n        },\n        error,\n        output: data,\n      };\n\n      const currentJob = job.data.runs.shift();\n      const nextJob = job.data.runs[0];\n      let nextQueue = this.name;\n\n      // keep a record of the first job in this relay instance\n      // ssssh JSON ;p\n      if (!job.data.initialJob) {\n        job.data.initialJob = tryJSONParse(job.toData());\n      }\n\n      // keep a record of the first queue in this relay instance\n      if (!job.data.initialQueue) {\n        job.data.initialQueue = this.name;\n      }\n\n      if (status === 'failed') {\n        if (job.options.retries > 0) {\n          job.options.retries = job.options.retries - 1;\n          job.status = 'retrying';\n          event.event = 'retrying';\n          multi.hset(this.toKey('jobs'), job.id, job.toData());\n          multi.lpush(this.toKey('waiting'), job.id);\n        } else {\n          job.status = 'failed';\n          multi.hset(this.toKey('jobs'), job.id, job.toData());\n          multi.sadd(this.toKey('failed'), job.id);\n        }\n      } else {\n        job.status = 'succeeded';\n        multi.hset(this.toKey('jobs'), job.id, job.toData());\n        multi.hdel(this.toKey('jobs'), job.id);\n        // multi.sadd(this.toKey('succeeded'), job.id); // TODO LOG JOBS\n      }\n\n      // check if we need to relay to another job\n      if (!(job.data.runs.length === 0 || !!error)) {\n        if (isObject(nextJob)) {\n          nextQueue = nextJob.queue;\n          job.data.runs[0] = nextJob.runs;\n        } else if (job.data.initialQueue) {\n          nextQueue = job.data.initialQueue;\n        }\n\n        // add some debug data for the next job\n        // so it can tell where its call originated from\n        job.data.from_job = currentJob;\n        job.data.from_queue = this.name;\n        job.data.from_timestamp = getTimeStamp();\n        job.data.data = data;\n\n        return this.core.hooks.job.create(nextQueue, job.data).then(() => {\n          multi.exec(errMulti => {\n            if (errMulti) return reject(errMulti);\n            return resolve({ status, result: error || data });\n          });\n        });\n      }\n\n      // we've just finished the last job in the relay\n      if (event.error) {\n        if (job.data.initialJob.options.notifyFailure) {\n          this.core.pubsub.publish(job.data.initialJob.options.notifyFailure, event);\n        }\n      } else if (job.data.initialJob.options.notifySuccess) {\n        this.core.pubsub.publish(job.data.initialJob.options.notifySuccess, event);\n      }\n\n      return multi.exec(errMulti => {\n        if (errMulti) return reject(errMulti);\n        return resolve({ status, result: error || data });\n      });\n    });\n  }\n\n  /**\n   *\n   * @param error\n   * @param data\n   * @param job\n   * @returns {Promise}\n   */\n  _finishSingleJob(error, data, job) {\n    delete this.jobs[job.id];\n\n    return new Promise((resolve, reject) => {\n      const status = error ? 'failed' : 'succeeded';\n      const multi = this.client.multi();\n\n      multi.lrem(this.toKey('active'), 0, job.id);\n      multi.srem(this.toKey('stalling'), job.id);\n\n      if (status === 'failed') {\n        if (job.options.retries > 0) {\n          job.options.retries = job.options.retries - 1;\n          job.status = 'retrying';\n          multi.hset(this.toKey('jobs'), job.id, job.toData());\n          multi.lpush(this.toKey('waiting'), job.id);\n        } else {\n          job.status = 'failed';\n          multi.hset(this.toKey('jobs'), job.id, job.toData());\n          multi.sadd(this.toKey('failed'), job.id);\n        }\n      } else {\n        job.status = 'succeeded';\n        multi.hset(this.toKey('jobs'), job.id, job.toData());\n        multi.hdel(this.toKey('jobs'), job.id);\n        // multi.sadd(this.toKey('succeeded'), job.id); // TODO LOG JOBS\n      }\n\n      if (error && (job.options.notifySuccess || job.notifyFailure)) {\n        const event = {\n          job: {\n            id: job.id,\n            worker_id: this.core.id,\n            status,\n            ...job.data,\n          },\n          error,\n          output: data,\n        };\n        if (job.options.notifyFailure) this.core.pubsub.publish(job.options.notifyFailure, event);\n        if (job.options.notifySuccess) this.core.pubsub.publish(job.options.notifySuccess, event);\n      }\n\n      multi.exec(errMulti => {\n        if (errMulti) return reject(errMulti);\n        return resolve({ status, result: error || data });\n      });\n    });\n  }\n\n  /**\n   *\n   * @returns {*}\n   * @private\n   */\n  _jobTick = () => {\n    if (this.paused || !this.options.enabled) {\n      return void 0;\n    }\n\n    this.queued++;\n\n    return this._getNextJob().then(job => {\n      this.running++;\n\n      if ((this.running + this.queued) < this.concurrency) {\n        setImmediate(this._jobTick);\n      }\n\n      return this._runJob(job).then(() => {\n        this.running--;\n        this.queued--;\n        setImmediate(this._jobTick);\n      }).catch(() => {\n        this.running--;\n        this.queued--;\n        setImmediate(this._jobTick);\n      });\n    }).catch(error => {\n      this.log.error(error);\n      setImmediate(this._jobTick);\n    });\n  };\n\n  /**\n   *\n   * @private\n   */\n  _restartProcessing = () => {\n    this.clients.block.once('ready', this._jobTick);\n  };\n\n  /**\n   * Start the queue.\n   * @param concurrency\n   */\n  process(concurrency = 1) {\n    if (this.started || !this.options.enabled) {\n      this.log.info(`Queue ${this.name} is currently disabled.`);\n      return void 0;\n    }\n\n    this.started = true;\n    this.running = 0;\n    this.queued = 0;\n    this.concurrency = concurrency;\n\n    this.log.verbose(`Queue '${this.name}' - started with a concurrency of ${this.concurrency}.`);\n\n    this.clients.block.once('error', this._restartProcessing);\n    this.clients.block.once('close', this._restartProcessing);\n\n    this.checkStalledJobs().then(() => {\n      this.log.verbose('checkStalledJobs completed');\n    }).catch(() => {\n    });\n\n    return this._jobTick();\n  }\n\n  /**\n   *\n   * @returns {*}\n   */\n  checkStalledJobs() {\n    this.log.verbose('checkStalledJobs');\n    return this.client.checkstalledjobs(\n      this.toKey('stallTime'),\n      this.toKey('stalling'),\n      this.toKey('waiting'),\n      this.toKey('active'),\n      getTimeStamp(),\n      this.options.stallInterval\n    ).then(() => {\n      if (!this.options.enabled || this.paused) {\n        return Promise.resolve();\n      }\n\n      return Promise.delay(this.options.stallInterval).then(this.checkStalledJobs);\n    });\n  }\n\n  /**\n   *\n   * @param str\n   * @returns {*}\n   */\n  toKey(str) {\n    if (this.core.cluster.isCluster()) {\n      return `${this.options.keyPrefix}:{${this.name}}:${str}`;\n    }\n    return `${this.options.keyPrefix}:${this.name}:${str}`;\n  }\n\n  /**\n   * Add the eventPrefix to an event name\n   * @param eventName\n   * @returns {string}\n   */\n  toEventName = eventName => `queue:${this.name}:${eventName}`;\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/queue.js"],"names":[],"mappings":"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","file":"queue.js","sourcesContent":["import Job from './job';\nimport Promise from 'bluebird';\nimport defaults from './defaults';\nimport EventEmitter from 'eventemitter3';\nimport { deepGet, isObject, getTimeStamp, tryJSONParse } from 'redibox';\n\nexport default class Queue extends EventEmitter {\n\n  /**\n   *\n   * @param options\n   * @param core\n   * @returns {Queue}\n   */\n  constructor(options, core) {\n    super();\n    this.core = core;\n    this.client = core.client;\n    this.paused = false;\n    this.started = false;\n    this.throttled = false;\n    this.log = this.core.log;\n    this.name = options.name;\n    this.handler = options.handler || null;\n    this.options = Object.assign({}, defaults.queue, options || {});\n    this.core.createClient('block', this).then(() => {\n      this.log.verbose(`Blocking client for queue '${this.name}' is ready. Starting queue processor.`);\n      this.beginWorking();\n    });\n  }\n\n  /**\n   *\n   * @returns {*}\n   */\n  destroy() {\n    const keys = [\n      'id', 'jobs', 'stallTime', 'stalling', 'waiting', 'active', 'succeeded', 'failed',\n    ].map(key => this.toKey(key));\n    return this.client.del(...keys);\n  }\n\n  /**\n   *\n   * @returns {Promise}\n   */\n  checkHealth() {\n    return this\n      .client.multi()\n      .llen(this.toKey('waiting'))\n      .llen(this.toKey('active'))\n      .scard(this.toKey('succeeded'))\n      .scard(this.toKey('failed'))\n      .then(results => { /* eslint arrow-body-style: 0 */\n        return {\n          waiting: results[0][1],\n          active: results[1][1],\n          succeeded: results[2][1],\n          failed: results[3][1],\n        };\n      });\n  }\n\n  /**\n   *\n   * @returns {Promise}\n   */\n  _getNextJob() {\n    this.log.verbose(`Getting next job for queue '${this.name}'.`);\n    return this.clients.block.brpoplpush(\n      this.toKey('waiting'),\n      this.toKey('active'), 0\n    ).then(jobId =>\n      Job.fromId(this, jobId).then(job => {\n        return job;\n      })\n    );\n  }\n\n  /**\n   *\n   * @param job\n   * @param jobError\n   * @private\n   */\n  _logJobFailure(job, jobError) {\n    if (process.env.KUBERNETES_PORT || process.env.KUBERNETES_SERVICE_HOST) {\n      /* eslint no-console: 0 */\n      console.log(JSON.stringify({\n        level: 'error',\n        type: 'redibox_job_failure',\n        data: {\n          runs: job.data.runs,\n          queue: this.name,\n          stack: jobError.stack ? jobError.stack.split('\\n').slice(0, 5) : [],\n        },\n      }));\n    } else {\n      this.log.error('');\n      this.log.error('--------------- RDB JOB ERROR/FAILURE ---------------');\n      this.log.error(`Job: ${job.data.runs}` || this.name);\n      if (jobError.stack) {\n        this.log.error(jobError.stack.split('\\n').slice(0, 5));\n      }\n      this.log.error(jobError);\n      this.log.error('------------------------------------------------------');\n      this.log.error('');\n    }\n  }\n\n  /**\n   *\n   * @param job\n   * @returns {Promise}\n   */\n  _runJob(job) {\n    if (!job || !job.data) return Promise.resolve();\n    const runs = job.data && job.data.runs && Array.isArray(job.data.runs) ? job.data.runs[0] : job.data.runs;\n    const handler = (typeof this.handler === 'string' ?\n        deepGet(global, this.handler) : this.handler) || deepGet(global, runs);\n\n    let preventStallingTimeout;\n    let handled = false;\n\n    // Handle an \"OK\" response from the promise\n    const handleOK = data => {\n      // silently ignore any multiple calls\n      if (handled) return void 0;\n\n      clearTimeout(preventStallingTimeout);\n      handled = true;\n\n      // set the data back to internal data\n      if (job._internalData) {\n        job.data = job._internalData;\n      }\n\n      // only relay to next job if user did not resolve 'false' on current job\n      if (job.type === 'relay' && data !== false) return this._finishRelayJob(null, data, job);\n      return this._finishSingleJob(null, data, job);\n    };\n\n    // Handle any errors returned\n    const handleError = jobError => {\n      clearTimeout(preventStallingTimeout);\n\n      // silently ignore any multiple calls\n      if (handled) {\n        return void 0;\n      }\n\n      handled = true;\n\n      // set the data back to internal job data\n      if (job._internalData) {\n        job.data = job._internalData;\n      }\n\n      // only log the error if no notifyFailure pubsub set\n      if ((!job.data.initialJob || !job.data.initialJob.options.notifyFailure) && !Array.isArray(job.data.runs)) {\n        this._logJobFailure(job, jobError);\n      }\n\n      if (job.type === 'relay') return this._finishRelayJob(jobError, null, job);\n      return this._finishSingleJob(jobError, null, job);\n    };\n\n    const preventStalling = () => {\n      this.client.srem(this.toKey('stalling'), job.id, () => {\n        if (!handled) {\n          preventStallingTimeout = setTimeout(preventStalling, this.options.stallInterval / 2);\n        }\n      });\n    };\n\n    if (!handler) {\n      return handleError(\n        new Error(\n          `\"${job.data.runs || 'No Job Handler Specified'}\" was not found. Skipping job. To fix this\n             you must either specify a handler function via queue.process() or provide a valid handler\n             node global path in your job options 'handler', e.g. if you had a global function in\n            'global.sails.services.myservice' you'd specify the handler as 'sails.services.myservice.myHandler'.`\n        )\n      );\n    }\n\n    preventStalling(); // start stalling monitor\n\n    job._internalData = job.data;\n    job.data = job.data.data || job.data;\n\n    if (job.options.timeout) {\n      setTimeout(handleError.bind(null, Error(`Job ${job.id} timed out (${job.options.timeout}ms)`)), job.options.timeout);\n    }\n\n    if (job.options.noBind || this.options.noBind) {\n      return handler(job).then(handleOK, handleError).catch(handleError);\n    }\n\n    return handler.bind(job, job)(job).then(handleOK, handleError).catch(handleError);\n  }\n\n  /**\n   *\n   * @param error\n   * @param data\n   * @param job\n   * @returns {{job: {id: *, worker_id: (*|String|string), status: string}, error: *, output: *}}\n   * @private\n   */\n  _createJobEvent(error, data, job) {\n    return {\n      job: {\n        id: job.id,\n        worker_id: this.core.id,\n        status: error ? 'failed' : 'succeeded',\n        ...job.data,\n      },\n      error,\n      output: data,\n    };\n  }\n\n  /**\n   *\n   * @param error\n   * @param data\n   * @param job\n   * @param multi\n   * @private\n   */\n  _updateJobStatus(error, data, job, multi) {\n    const status = error ? 'failed' : 'succeeded';\n\n    multi.lrem(this.toKey('active'), 0, job.id);\n    multi.srem(this.toKey('stalling'), job.id);\n\n    if (status === 'failed') {\n      if (job.options.retries > 0) {\n        job.options.retries = job.options.retries - 1;\n        job.status = 'retrying';\n        multi.hset(this.toKey('jobs'), job.id, job.toData());\n        multi.lpush(this.toKey('waiting'), job.id);\n      } else {\n        job.status = 'failed';\n        multi.hdel(this.toKey('jobs'), job.id);\n        // TODO track failures and their data somewhere else for reviewing\n        // multi.hset(this.toKey('jobs'), job.id, job.toData());\n        // multi.sadd(this.toKey('failed'), job.id);\n      }\n    } else {\n      job.status = 'succeeded';\n      multi.hdel(this.toKey('jobs'), job.id);\n      // TODO track successes and their data somewhere else for reviewing\n      // multi.hset(this.toKey('jobs'), job.id, job.toData());\n      // multi.sadd(this.toKey('succeeded'), job.id);\n    }\n  }\n\n  /**\n   *\n   * @param error\n   * @param data\n   * @param job\n   * @returns {Promise}\n   */\n  _finishSingleJob(error, data, job) {\n    const multi = this.client.multi();\n    const status = error ? 'failed' : 'succeeded';\n    this._updateJobStatus(error, data, job, multi);\n\n    // emit success or failure event if we have listeners\n    if (error && job.options.notifyFailure) {\n      this.core.pubsub.publish(job.options.notifyFailure, this._createJobEvent(error, data, job));\n    } else if (job.options.notifySuccess) {\n      this.core.pubsub.publish(job.options.notifySuccess, this._createJobEvent(error, data, job));\n    }\n\n    return new Promise((resolve, reject) => {\n      multi.exec(errMulti => {\n        if (errMulti) return reject(errMulti);\n        return resolve({ status, result: error || data });\n      });\n    });\n  }\n\n  /**\n   * Completes a multi job or continues to the next stage.\n   * @param error\n   * @param data\n   * @param job\n   * @returns {Promise}\n   * @private\n   */\n  _finishRelayJob(error, data, job) {\n    let nextQueue = this.name;\n    const nextJob = job.data.runs[0];\n    const multi = this.client.multi();\n    const currentJob = job.data.runs.shift();\n    const status = error ? 'failed' : 'succeeded';\n\n    // keep a record of the first job in this relay instance\n    if (!job.data.initialJob) {\n      job.data.initialJob = tryJSONParse(job.toData());\n    }\n\n    // keep a record of the first queue in this relay instance\n    if (!job.data.initialQueue) {\n      job.data.initialQueue = this.name;\n    }\n\n    this._updateJobStatus(error, data, job, multi);\n\n    // check if we need to relay to another job\n    if (!(job.data.runs.length === 0 || !!error)) {\n      if (isObject(nextJob)) {\n        nextQueue = nextJob.queue;\n        job.data.runs[0] = nextJob.runs;\n      } else if (job.data.initialQueue) {\n        nextQueue = job.data.initialQueue;\n      }\n\n      // add some debug data for the next job\n      // so it can tell where the relay originated from\n      job.data.from_job = currentJob;\n      job.data.from_queue = this.name;\n      job.data.from_timestamp = getTimeStamp();\n      // relay resolved data\n      job.data.data = data;\n\n      return new Promise((resolve, reject) => {\n        return this.core.hooks.job.create(nextQueue, job.data).then(() => {\n          multi.exec(errMulti => {\n            if (errMulti) return reject(errMulti);\n            return resolve({ status, result: error || data });\n          });\n        });\n      });\n    }\n\n    // we've just finished the last job in the relay\n    // emit success or failure event if we have listeners\n    if (error && job.data.initialJob.options.notifyFailure) {\n      this.core.pubsub.publish(job.data.initialJob.options.notifyFailure, this._createJobEvent(error, data, job));\n    } else if (job.data.initialJob.options.notifySuccess) {\n      this.core.pubsub.publish(job.data.initialJob.options.notifySuccess, this._createJobEvent(error, data, job));\n    }\n\n    return new Promise((resolve, reject) => {\n      return multi.exec(errMulti => {\n        if (errMulti) return reject(errMulti);\n        return resolve({ status, result: error || data });\n      });\n    });\n  }\n\n  /**\n   *\n   * @private\n   */\n  _onLocalTickComplete = () => {\n    this.running--;\n    this.queued--;\n    if (!this.options.throttle) return setImmediate(this._queueTick);\n\n    return this.client.throttle(\n      this.toKey('throttle'),\n      this.options.throttle.limit,\n      this.options.throttle.seconds\n    ).then(throttle => {\n      const shouldThrottle = throttle[0] === 1;\n      if (!shouldThrottle) {\n        this.throttled = false;\n        return setImmediate(this._queueTick);\n      }\n\n      this.throttled = true;\n      const timeRemaining = (throttle[2] === 0 ? 1 : throttle[2]);\n      this.log.verbose(`'${this.name}' queue  reached it's throttle limit, resuming in ${timeRemaining} seconds.`);\n      return setTimeout(this._queueTick, timeRemaining * 1000);\n    }).catch(this._queueTick);\n  };\n\n  /**\n   *\n   * @param error\n   * @private\n   */\n  _onLocalTickError = (error) => {\n    this.queued--;\n    this.log.error(error);\n    setImmediate(this._queueTick);\n  };\n\n  _throttleQueue = () => {\n\n  };\n\n  /**\n   *\n   * @returns {*}\n   * @private\n   */\n  _queueTick = () => {\n    if (this.paused || !this.options.enabled) {\n      return void 0;\n    }\n    this.queued++;\n    return this\n      ._getNextJob()\n      .then(job => {\n        this.running++;\n        // queue more jobs if within limit\n        if ((this.running + this.queued) < this.options.concurrency) {\n          // concurrency is a little pointless right now if we're throttling jobs\n          if (!this.options.throttle) setImmediate(this._queueTick);\n        }\n\n        return this._runJob(job).then(this._onLocalTickComplete).catch(this._onLocalTickComplete);\n      }).catch(this._onLocalTickError);\n  };\n\n  /**\n   *\n   * @private\n   */\n  _restartProcessing = () => {\n    this.clients.block.once('ready', this._queueTick);\n  };\n\n  /**\n   * Start the queue.\n   */\n  beginWorking() {\n    if (this.started || !this.options.enabled) {\n      this.log.info(`Queue ${this.name} is currently disabled.`);\n      return void 0;\n    }\n\n    this.queued = 0;\n    this.running = 0;\n    this.started = true;\n\n    this.log.verbose(`Queue '${this.name}' - started with a concurrency of ${this.options.concurrency}.`);\n\n    this.clients.block.once('error', this._restartProcessing);\n    this.clients.block.once('close', this._restartProcessing);\n\n    this.checkStalledJobs().then(() => {\n      this.log.verbose('checkStalledJobs completed');\n    }).catch(() => {\n    });\n\n    return this._queueTick();\n  }\n\n  /**\n   *\n   * @returns {*}\n   */\n  checkStalledJobs() {\n    this.log.verbose('checkStalledJobs');\n    return this.client.checkstalledjobs(\n      this.toKey('stallTime'),\n      this.toKey('stalling'),\n      this.toKey('waiting'),\n      this.toKey('active'),\n      getTimeStamp(),\n      this.options.stallInterval\n    ).then(() => {\n      if (!this.options.enabled || this.paused) return Promise.resolve();\n      return Promise.delay(this.options.stallInterval).then(this.checkStalledJobs);\n    });\n  }\n\n  /**\n   *\n   * @param str\n   * @returns {*}\n   */\n  toKey(str) {\n    if (this.core.cluster.isCluster()) {\n      return `${this.options.keyPrefix}:{${this.name}}:${str}`;\n    }\n    return `${this.options.keyPrefix}:${this.name}:${str}`;\n  }\n\n  /**\n   * Add the eventPrefix to an event name\n   * @param eventName\n   * @returns {string}\n   */\n  toEventName = eventName => `queue:${this.name}:${eventName}`;\n}\n"]}

@@ -142,4 +142,118 @@ "use strict";

`
},
throttle: {
keys: 1,
lua: `
--[[
key 1 -> key name - ip, user id or some unique key to throttle X by
arg 1 -> limit
arg 2 -> seconds
returns {
throttled: -> 1 if should throttle, 0 if still within limit
remaining: -> how many reqs left until throttled,
ttl: -> seconds remaining until limit resets
}
]]
local count = redis.call('INCR', KEYS[1])
local ttl = redis.call('ttl', KEYS[1])
local remaining = tonumber(ARGV[1]) - count
if count == 1 or ttl == -1 then
redis.call('EXPIRE', KEYS[1], ARGV[2])
end
if ttl == -1 then
ttl = tonumber(ARGV[2])
end
if count > tonumber(ARGV[1]) then
return {1, 0, ttl}
end
if remaining == 0 then
return {1, 0, ttl}
end
return {0, remaining, ttl}
`
},
pThrottle: {
keys: 1,
lua: `
--[[
key 1 -> key name - ip, user id or some unique key to throttle X by
arg 1 -> limit
arg 2 -> milliseconds
returns 0 if request is ok
returns 1 if request denied
]]
local count = redis.call('INCR', KEYS[1])
local pttl = redis.call('pttl',KEYS[1])
local remaining = tonumber(ARGV[1]) - count
if count == 1 or pttl == -1 then
redis.call('PEXPIRE', KEYS[1], ARGV[2])
end
if pttl == -1 then
pttl = tonumber(ARGV[2])
end
if count > tonumber(ARGV[1]) then
return {1,0,pttl}
end
if remaining == 0 then
return {1, 0, pttl}
end
return {0, remaining, pttl}
`
},
throttleNoIncr: {
keys: 1,
lua: `
--[[
key 1 -> key name - ip, user id or some unique key to throttle X by
arg 1 -> limit
arg 2 -> seconds
returns 0/1
]]
if redis.call('exists',KEYS[1]) > 0 then
local currentVal = tonumber(redis.call('get',KEYS[1]))
if currentVal >= tonumber(ARGV[1]) then
return 1
else
redis.call('incr',KEYS[1])
return 0
end
else
redis.call('setex',KEYS[1], tonumber(ARGV[2]), 1)
return 0
end
`
},
throttleDecr: {
keys: 1,
lua: `
--[[
key 1 -> key name - ip, user id or some unique key to throttle X by
returns 0/1
]]
if redis.call('exists',KEYS[1]) > 0 then
redis.call('decr',KEYS[1])
return 1
else
return 1
end
`
}
};
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9zY3JpcHRzLmpzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7a0JBMEJlOztBQUViLFVBQVE7QUFDTixVQUFNLENBREE7QUFFTixTQUFLOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7SUFBQTtBQUZDLEdBRks7O0FBcUNiLG9CQUFrQjtBQUNoQixVQUFNLENBRFU7QUFFaEIsU0FBSzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztJQUFBO0FBRlcsR0FyQ0w7O0FBcUZiLGFBQVc7QUFDVCxVQUFNLENBREc7QUFFVCxTQUFLOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0lBQUE7QUFGSTtBQXJGRSxDIiwiZmlsZSI6InNjcmlwdHMuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqXG4gKiBUaGUgTUlUIExpY2Vuc2UgKE1JVClcbiAqXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTYgU2FsYWthclxuICpcbiAqIFBlcm1pc3Npb24gaXMgaGVyZWJ5IGdyYW50ZWQsIGZyZWUgb2YgY2hhcmdlLCB0byBhbnkgcGVyc29uIG9idGFpbmluZyBhIGNvcHlcbiAqIG9mIHRoaXMgc29mdHdhcmUgYW5kIGFzc29jaWF0ZWQgZG9jdW1lbnRhdGlvbiBmaWxlcyAodGhlIFwiU29mdHdhcmVcIiksIHRvIGRlYWxcbiAqIGluIHRoZSBTb2Z0d2FyZSB3aXRob3V0IHJlc3RyaWN0aW9uLCBpbmNsdWRpbmcgd2l0aG91dCBsaW1pdGF0aW9uIHRoZSByaWdodHNcbiAqIHRvIHVzZSwgY29weSwgbW9kaWZ5LCBtZXJnZSwgcHVibGlzaCwgZGlzdHJpYnV0ZSwgc3VibGljZW5zZSwgYW5kL29yIHNlbGxcbiAqIGNvcGllcyBvZiB0aGUgU29mdHdhcmUsIGFuZCB0byBwZXJtaXQgcGVyc29ucyB0byB3aG9tIHRoZSBTb2Z0d2FyZSBpc1xuICogZnVybmlzaGVkIHRvIGRvIHNvLCBzdWJqZWN0IHRvIHRoZSBmb2xsb3dpbmcgY29uZGl0aW9uczpcbiAqXG4gKiBUaGUgYWJvdmUgY29weXJpZ2h0IG5vdGljZSBhbmQgdGhpcyBwZXJtaXNzaW9uIG5vdGljZSBzaGFsbCBiZSBpbmNsdWRlZCBpbiBhbGxcbiAqIGNvcGllcyBvciBzdWJzdGFudGlhbCBwb3J0aW9ucyBvZiB0aGUgU29mdHdhcmUuXG4gKlxuICogVEhFIFNPRlRXQVJFIElTIFBST1ZJREVEIFwiQVMgSVNcIiwgV0lUSE9VVCBXQVJSQU5UWSBPRiBBTlkgS0lORCwgRVhQUkVTUyBPUlxuICogSU1QTElFRCwgSU5DTFVESU5HIEJVVCBOT1QgTElNSVRFRCBUTyBUSEUgV0FSUkFOVElFUyBPRiBNRVJDSEFOVEFCSUxJVFksXG4gKiBGSVRORVNTIEZPUiBBIFBBUlRJQ1VMQVIgUFVSUE9TRSBBTkQgTk9OSU5GUklOR0VNRU5ULiBJTiBOTyBFVkVOVCBTSEFMTCBUSEVcbiAqIEFVVEhPUlMgT1IgQ09QWVJJR0hUIEhPTERFUlMgQkUgTElBQkxFIEZPUiBBTlkgQ0xBSU0sIERBTUFHRVMgT1IgT1RIRVJcbiAqIExJQUJJTElUWSwgV0hFVEhFUiBJTiBBTiBBQ1RJT04gT0YgQ09OVFJBQ1QsIFRPUlQgT1IgT1RIRVJXSVNFLCBBUklTSU5HIEZST00sXG4gKiBPVVQgT0YgT1IgSU4gQ09OTkVDVElPTiBXSVRIIFRIRSBTT0ZUV0FSRSBPUiBUSEUgVVNFIE9SIE9USEVSIERFQUxJTkdTIElOIFRIRVxuICogU09GVFdBUkUuXG4gKlxuICovXG5cbmV4cG9ydCBkZWZhdWx0IHtcblxuICBhZGRKb2I6IHtcbiAgICBrZXlzOiAzLFxuICAgIGx1YTogYFxuICAgICAgICAtLVtbXG4gICAgICAgIGtleSAxIC0+IHJhYjpqb2I6bmFtZTpqb2JzXG4gICAgICAgIGtleSAyIC0+IHJhYjpqb2I6bmFtZTp3YWl0aW5nXG4gICAgICAgIGtleSAzIC0+IHJhYjpqb2I6bmFtZTppZCAoam9iIElEIGNvdW50ZXIpXG4gICAgICAgIGFyZyAxIC0+IGpvYiBkYXRhXG4gICAgICAgIGFyZyAyIC0+IHNob3VsZCBiZSB1bmlxdWU/XG4gICAgICAgIGFyZyAzIC0+IGN1c3RvbUlkXG4gICAgICAgIF1dXG5cbiAgICAgICAgbG9jYWwgam9iSWQgPSBBUkdWWzNdXG5cbiAgICAgICAgaWYgam9iSWQgPT0gXCJcIiB0aGVuXG4gICAgICAgICAgam9iSWQgPSByZWRpcy5jYWxsKFwiaW5jclwiLCBLRVlTWzNdKVxuICAgICAgICBlbmRcblxuICAgICAgICAtLSBpZiB1bmlxdWUgZW5hYmxlZFxuICAgICAgICBpZiBBUkdWWzJdID09IFwidHJ1ZVwiIHRoZW5cbiAgICAgICAgICBsb2NhbCBleGlzdHMgPSByZWRpcy5jYWxsKFwiaHNldG54XCIsIEtFWVNbMV0sIGpvYklkLCBBUkdWWzFdKVxuICAgICAgICAgIGlmIGV4aXN0cyA9PSAxIHRoZW5cbiAgICAgICAgICAgIHJlZGlzLmNhbGwoXCJscHVzaFwiLCBLRVlTWzJdLCBqb2JJZClcbiAgICAgICAgICAgIHJldHVybiBqb2JJZFxuICAgICAgICAgIGVuZFxuICAgICAgICAgIHJldHVybiAwXG4gICAgICAgIGVsc2VcbiAgICAgICAgICAtLSBpZiBub3QgdW5pcXVlIGVuYWJsZWRcbiAgICAgICAgICByZWRpcy5jYWxsKFwiaHNldFwiLCBLRVlTWzFdLCBqb2JJZCwgQVJHVlsxXSlcbiAgICAgICAgICByZWRpcy5jYWxsKFwibHB1c2hcIiwgS0VZU1syXSwgam9iSWQpXG4gICAgICAgICAgcmV0dXJuIGpvYklkXG4gICAgICAgIGVuZFxuICAgIGAsXG4gIH0sXG5cbiAgY2hlY2tTdGFsbGVkSm9iczoge1xuICAgIGtleXM6IDQsXG4gICAgbHVhOiBgXG4gICAgICAgIC0tW1tcbiAgICAgICAga2V5IDEgLT4gcmFiOmpvYjpuYW1lOnN0YWxsVGltZVxuICAgICAgICBrZXkgMiAtPiByYWI6am9iOm5hbWU6c3RhbGxpbmdcbiAgICAgICAga2V5IDMgLT4gcmFiOmpvYjpuYW1lOndhaXRpbmdcbiAgICAgICAga2V5IDQgLT4gcmFiOmpvYjpuYW1lOmFjdGl2ZVxuICAgICAgICBhcmcgMSAtPiBtcyB0aW1lc3RhbXAgKFwibm93XCIpXG4gICAgICAgIGFyZyAyIC0+IG1zIHN0YWxsSW50ZXJ2YWxcblxuICAgICAgICByZXR1cm5zIHtyZXNldEpvYklkMSwgcmVzZXRKb2JJZDIsIC4uLn1cblxuICAgICAgICB3b3JrZXJzIGFyZSByZXNwb25zaWJsZSBmb3IgcmVtb3ZpbmcgdGhlaXIgam9iSWQgZnJvbSB0aGUgc3RhbGxpbmcgc2V0IGV2ZXJ5IHN0YWxsSW50ZXJ2YWwgbXNcbiAgICAgICAgaWYgYSBqb2JJZCBpcyBub3QgcmVtb3ZlZCBmcm9tIHRoZSBzdGFsbGluZyBzZXQgd2l0aGluIGEgc3RhbGxJbnRlcnZhbCB3aW5kb3csXG4gICAgICAgIHdlIGFzc3VtZSB0aGUgam9iIGhhcyBzdGFsbGVkIGFuZCBzaG91bGQgYmUgcmVzZXQgKG1vdmVkIGZyb20gYWN0aXZlIGJhY2sgdG8gd2FpdGluZylcbiAgICAgICAgLS1dXVxuXG4gICAgICAgIGxvY2FsIG5vdyA9IHRvbnVtYmVyKEFSR1ZbMV0pXG4gICAgICAgIGxvY2FsIHN0YWxsVGltZSA9IHRvbnVtYmVyKHJlZGlzLmNhbGwoXCJnZXRcIiwgS0VZU1sxXSkgb3IgMClcblxuICAgICAgICBpZiBub3cgPCBzdGFsbFRpbWUgdGhlblxuICAgICAgICAgIC0tIGhhc24ndCBiZWVuIGxvbmcgZW5vdWdoIChzdGFsbEludGVydmFsKSBzaW5jZSBsYXN0IGNoZWNrXG4gICAgICAgICAgcmV0dXJuIDBcbiAgICAgICAgZW5kXG5cbiAgICAgICAgLS0gcmVzZXQgYW55IHN0YWxsaW5nIGpvYnMgYnkgbW92aW5nIGZyb20gYWN0aXZlIHRvIHdhaXRpbmdcbiAgICAgICAgbG9jYWwgc3RhbGxpbmcgPSByZWRpcy5jYWxsKFwic21lbWJlcnNcIiwgS0VZU1syXSlcbiAgICAgICAgaWYgI3N0YWxsaW5nID4gMCB0aGVuXG4gICAgICAgICAgcmVkaXMuY2FsbChcInJwdXNoXCIsIEtFWVNbM10sIHVucGFjayhzdGFsbGluZykpXG4gICAgICAgICAgZm9yIGkgPSAxLCAjc3RhbGxpbmcgZG9cbiAgICAgICAgICAgIHJlZGlzLmNhbGwoXCJscmVtXCIsIEtFWVNbNF0sIDAsIHN0YWxsaW5nW2ldKVxuICAgICAgICAgIGVuZFxuICAgICAgICAgIHJlZGlzLmNhbGwoXCJkZWxcIiwgS0VZU1syXSlcbiAgICAgICAgZW5kXG5cbiAgICAgICAgLS0gY29weSBjdXJyZW50bHkgYWN0aXZlIGpvYnMgaW50byBzdGFsbGluZyBzZXRcbiAgICAgICAgbG9jYWwgYWN0aXZlcyA9IHJlZGlzLmNhbGwoXCJscmFuZ2VcIiwgS0VZU1s0XSwgMCwgLTEpXG4gICAgICAgIGlmICNhY3RpdmVzID4gMCB0aGVuXG4gICAgICAgICAgcmVkaXMuY2FsbChcInNhZGRcIiwgS0VZU1syXSwgdW5wYWNrKGFjdGl2ZXMpKVxuICAgICAgICBlbmRcblxuICAgICAgICByZWRpcy5jYWxsKFwic2V0XCIsIEtFWVNbMV0sIG5vdyArIEFSR1ZbMl0pXG5cbiAgICAgICAgcmV0dXJuIHN0YWxsaW5nXG4gICAgYCxcbiAgfSxcblxuICByZW1vdmVKb2I6IHtcbiAgICBrZXlzOiA2LFxuICAgIGx1YTogYFxuICAgICAgICAtLVtbXG4gICAgICAgIGtleSAxIC0+IHJhYjpqb2I6dGVzdDpzdWNjZWVkZWRcbiAgICAgICAga2V5IDIgLT4gcmFiOmpvYjp0ZXN0OmZhaWxlZFxuICAgICAgICBrZXkgMyAtPiByYWI6am9iOnRlc3Q6d2FpdGluZ1xuICAgICAgICBrZXkgNCAtPiByYWI6am9iOnRlc3Q6YWN0aXZlXG4gICAgICAgIGtleSA1IC0+IHJhYjpqb2I6dGVzdDpzdGFsbGluZ1xuICAgICAgICBrZXkgNiAtPiByYWI6am9iOnRlc3Q6am9ic1xuICAgICAgICBhcmcgMSAtPiBqb2JJZFxuICAgICAgICBdXVxuXG4gICAgICAgIGxvY2FsIGpvYklkID0gQVJHVlsxXVxuXG4gICAgICAgIGlmIChyZWRpcy5jYWxsKFwic2lzbWVtYmVyXCIsIEtFWVNbMV0sIGpvYklkKSArIHJlZGlzLmNhbGwoXCJzaXNtZW1iZXJcIiwgS0VZU1syXSwgam9iSWQpKSA9PSAwIHRoZW5cbiAgICAgICAgICByZWRpcy5jYWxsKFwibHJlbVwiLCBLRVlTWzNdLCAwLCBqb2JJZClcbiAgICAgICAgICByZWRpcy5jYWxsKFwibHJlbVwiLCBLRVlTWzRdLCAwLCBqb2JJZClcbiAgICAgICAgZW5kXG5cbiAgICAgICAgcmVkaXMuY2FsbChcInNyZW1cIiwgS0VZU1sxXSwgam9iSWQpXG4gICAgICAgIHJlZGlzLmNhbGwoXCJzcmVtXCIsIEtFWVNbMl0sIGpvYklkKVxuICAgICAgICByZWRpcy5jYWxsKFwic3JlbVwiLCBLRVlTWzVdLCBqb2JJZClcbiAgICAgICAgcmVkaXMuY2FsbChcImhkZWxcIiwgS0VZU1s2XSwgam9iSWQpXG4gICAgYCxcbiAgfSxcbn07XG4iXX0=
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/scripts.js"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;kBA0Be;;AAEb,UAAQ;AACN,UAAM,CADA;AAEN,SAAK;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;IAAA;AAFC,GAFK;;AAqCb,oBAAkB;AAChB,UAAM,CADU;AAEhB,SAAK;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;IAAA;AAFW,GArCL;;AAqFb,aAAW;AACT,UAAM,CADG;AAET,SAAK;;;;;;;;;;;;;;;;;;;;;;IAAA;AAFI,GArFE;;AAgHb,YAAU;AACR,UAAM,CADE;AAER,SAAK;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EAAA;AAFG,GAhHG;;AAsJb,aAAW;AACT,UAAM,CADG;AAET,SAAK;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;EAAA;AAFI,GAtJE;;AAyLb,kBAAgB;AACd,UAAM,CADQ;AAEd,SAAK;;;;;;;;;;;;;;;;;;;IAAA;AAFS,GAzLH;;AAiNb,gBAAc;AACZ,UAAM,CADM;AAEZ,SAAK;;;;;;;;;;;;IAAA;AAFO;AAjND,C","file":"scripts.js","sourcesContent":["/**\n *\n * The MIT License (MIT)\n *\n * Copyright (c) 2016 Salakar\n *\n * Permission is hereby granted, free of charge, to any person obtaining a copy\n * of this software and associated documentation files (the \"Software\"), to deal\n * in the Software without restriction, including without limitation the rights\n * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n * copies of the Software, and to permit persons to whom the Software is\n * furnished to do so, subject to the following conditions:\n *\n * The above copyright notice and this permission notice shall be included in all\n * copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\n * SOFTWARE.\n *\n */\n\nexport default {\n\n  addJob: {\n    keys: 3,\n    lua: `\n        --[[\n        key 1 -> rab:job:name:jobs\n        key 2 -> rab:job:name:waiting\n        key 3 -> rab:job:name:id (job ID counter)\n        arg 1 -> job data\n        arg 2 -> should be unique?\n        arg 3 -> customId\n        ]]\n\n        local jobId = ARGV[3]\n\n        if jobId == \"\" then\n          jobId = redis.call(\"incr\", KEYS[3])\n        end\n\n        -- if unique enabled\n        if ARGV[2] == \"true\" then\n          local exists = redis.call(\"hsetnx\", KEYS[1], jobId, ARGV[1])\n          if exists == 1 then\n            redis.call(\"lpush\", KEYS[2], jobId)\n            return jobId\n          end\n          return 0\n        else\n          -- if not unique enabled\n          redis.call(\"hset\", KEYS[1], jobId, ARGV[1])\n          redis.call(\"lpush\", KEYS[2], jobId)\n          return jobId\n        end\n    `,\n  },\n\n  checkStalledJobs: {\n    keys: 4,\n    lua: `\n        --[[\n        key 1 -> rab:job:name:stallTime\n        key 2 -> rab:job:name:stalling\n        key 3 -> rab:job:name:waiting\n        key 4 -> rab:job:name:active\n        arg 1 -> ms timestamp (\"now\")\n        arg 2 -> ms stallInterval\n\n        returns {resetJobId1, resetJobId2, ...}\n\n        workers are responsible for removing their jobId from the stalling set every stallInterval ms\n        if a jobId is not removed from the stalling set within a stallInterval window,\n        we assume the job has stalled and should be reset (moved from active back to waiting)\n        --]]\n\n        local now = tonumber(ARGV[1])\n        local stallTime = tonumber(redis.call(\"get\", KEYS[1]) or 0)\n\n        if now < stallTime then\n          -- hasn't been long enough (stallInterval) since last check\n          return 0\n        end\n\n        -- reset any stalling jobs by moving from active to waiting\n        local stalling = redis.call(\"smembers\", KEYS[2])\n        if #stalling > 0 then\n          redis.call(\"rpush\", KEYS[3], unpack(stalling))\n          for i = 1, #stalling do\n            redis.call(\"lrem\", KEYS[4], 0, stalling[i])\n          end\n          redis.call(\"del\", KEYS[2])\n        end\n\n        -- copy currently active jobs into stalling set\n        local actives = redis.call(\"lrange\", KEYS[4], 0, -1)\n        if #actives > 0 then\n          redis.call(\"sadd\", KEYS[2], unpack(actives))\n        end\n\n        redis.call(\"set\", KEYS[1], now + ARGV[2])\n\n        return stalling\n    `,\n  },\n\n  removeJob: {\n    keys: 6,\n    lua: `\n        --[[\n        key 1 -> rab:job:test:succeeded\n        key 2 -> rab:job:test:failed\n        key 3 -> rab:job:test:waiting\n        key 4 -> rab:job:test:active\n        key 5 -> rab:job:test:stalling\n        key 6 -> rab:job:test:jobs\n        arg 1 -> jobId\n        ]]\n\n        local jobId = ARGV[1]\n\n        if (redis.call(\"sismember\", KEYS[1], jobId) + redis.call(\"sismember\", KEYS[2], jobId)) == 0 then\n          redis.call(\"lrem\", KEYS[3], 0, jobId)\n          redis.call(\"lrem\", KEYS[4], 0, jobId)\n        end\n\n        redis.call(\"srem\", KEYS[1], jobId)\n        redis.call(\"srem\", KEYS[2], jobId)\n        redis.call(\"srem\", KEYS[5], jobId)\n        redis.call(\"hdel\", KEYS[6], jobId)\n    `,\n  },\n\n  throttle: {\n    keys: 1,\n    lua: `\n        --[[\n          key 1 -> key name - ip, user id or some unique key to throttle X by\n          arg 1 -> limit\n          arg 2 -> seconds\n          returns {\n            throttled: ->  1 if should throttle, 0 if still within limit\n            remaining: ->  how many reqs left until throttled,\n            ttl:       ->  seconds remaining until limit resets\n          }\n        ]]\n        \n        local count = redis.call('INCR', KEYS[1])\n        local ttl = redis.call('ttl', KEYS[1])\n        local remaining = tonumber(ARGV[1]) - count\n        \n        if count == 1 or ttl == -1 then\n          redis.call('EXPIRE', KEYS[1], ARGV[2])\n        end\n        \n        if ttl == -1 then\n          ttl = tonumber(ARGV[2])\n        end\n        \n        if count > tonumber(ARGV[1]) then\n          return {1, 0, ttl}\n        end\n        \n        if remaining == 0 then\n          return {1, 0, ttl}\n        end\n        \n        return {0, remaining, ttl}\n  `,\n  },\n\n  pThrottle: {\n    keys: 1,\n    lua: `\n        --[[\n          key 1 -> key name - ip, user id or some unique key to throttle X by\n          arg 1 -> limit\n          arg 2 -> milliseconds\n          returns 0 if request is ok\n          returns 1 if request denied\n        ]]\n        \n        local count = redis.call('INCR', KEYS[1])\n        local pttl = redis.call('pttl',KEYS[1])\n        local remaining = tonumber(ARGV[1]) - count\n\n        if count == 1 or pttl == -1 then\n          redis.call('PEXPIRE', KEYS[1], ARGV[2])\n        end\n        \n        if pttl == -1 then\n          pttl = tonumber(ARGV[2])\n        end\n        \n        if count > tonumber(ARGV[1]) then\n          return {1,0,pttl}\n        end\n        \n        if remaining == 0 then\n          return {1, 0, pttl}\n        end\n        \n        return {0, remaining, pttl}\n  `,\n  },\n\n  throttleNoIncr: {\n    keys: 1,\n    lua: `\n        --[[\n          key 1 -> key name - ip, user id or some unique key to throttle X by\n          arg 1 -> limit\n          arg 2 -> seconds\n          returns 0/1\n        ]]\n        if redis.call('exists',KEYS[1]) > 0 then\n          local currentVal = tonumber(redis.call('get',KEYS[1]))\n          if currentVal >= tonumber(ARGV[1]) then\n             return 1\n          else\n             redis.call('incr',KEYS[1])\n             return 0\n          end\n        else\n          redis.call('setex',KEYS[1], tonumber(ARGV[2]), 1)\n          return 0\n        end\n    `,\n  },\n\n  throttleDecr: {\n    keys: 1,\n    lua: `\n        --[[\n          key 1 -> key name - ip, user id or some unique key to throttle X by\n          returns 0/1\n        ]]\n    \n        if redis.call('exists',KEYS[1]) > 0 then\n          redis.call('decr',KEYS[1])\n          return 1\n        else\n          return 1\n        end\n    `,\n  },\n};\n"]}
{
"name": "redibox-hook-job",
"description": "Advanced redibox powered jobs and queues",
"version": "1.0.3",
"version": "1.1.0",
"main": "lib/index.js",

@@ -43,12 +43,12 @@ "author": "Mike Diarmid",

"devDependencies": {
"babel-cli": "^6.8.0",
"babel-eslint": "^6.0.4",
"babel-istanbul": "^0.8.0",
"babel-plugin-add-module-exports": "^0.1.2",
"babel-plugin-transform-class-properties": "^6.8.0",
"babel-cli": "^6.10.1",
"babel-eslint": "^6.1.0",
"babel-istanbul": "^0.11.0",
"babel-plugin-add-module-exports": "^0.2.1",
"babel-plugin-transform-class-properties": "^6.10.2",
"babel-preset-es2015": "^6.6.0",
"babel-preset-es2015-node-auto": "0.0.4",
"babel-preset-es2015-node-auto": "0.0.5",
"babel-preset-es2015-node4": "^2.1.0",
"babel-preset-es2015-node5": "^1.2.0",
"babel-preset-es2015-node6": "^0.1.3",
"babel-preset-es2015-node6": "^0.2.0",
"babel-preset-stage-0": "^6.5.0",

@@ -58,15 +58,15 @@ "benchmark": "^2.1.0",

"coveralls": "^2.11.9",
"eslint": "^2.9.0",
"eslint-config-airbnb": "^8.0.0",
"eslint-plugin-import": "^1.6.1",
"eslint-plugin-jsx-a11y": "^1.0.4",
"eslint-plugin-react": "^5.0.1",
"eslint": "^2.13.1",
"eslint-config-airbnb": "^9.0.1",
"eslint-plugin-import": "^1.9.2",
"eslint-plugin-jsx-a11y": "^1.5.3",
"eslint-plugin-react": "^5.2.2",
"isparta": "^4.0.0",
"istanbul": "^1.0.0-alpha.2",
"istanbul": "^1.1.0-alpha.1",
"mocha": "^2.4.5",
"redibox": "^2.0.1",
"redibox": "^2.1.0",
"sinon": "^1.17.4"
},
"dependencies": {
"bluebird": "^3.3.5",
"bluebird": "^3.4.1",
"cuid": "^1.3.8",

@@ -73,0 +73,0 @@ "eventemitter3": "^1.2.0"

@@ -1,1 +0,23 @@

## RediBox Job
## RediBox Job Hook
High performance, robust and flexible queue/worker system powered by redis.
**DOCS ARE TODO**🙈
#### Current features
- **Queues**. Jobs can be divided into queues you specify - no pre-determined queues such as 'PriorityQueue', just configure them the way you like it.
- **Concurrency**. Queues can be configured to concurrently run jobs, concurrency can either be per worker or across all workers.
- **Throttling**. Queues can be configured to throttle the number of jobs processed over a time period, e.g. 100 jobs per 10 seconds.
- **Relay/Chained Jobs**. A relay job allows you to automatically follow a specified chain of jobs and relay the result of each job to the next job in the chain, or abort the relay mid chain with the option of sending the final result back to wherever you created the job.
- **Jobs Anywhere**. Flexibility to create jobs anywhere, e.g. create a Job on your public api server and have your separate worker farm process it, automatically return the result right back to were you created the Job, send the result back as your api response and... profit?
- This hook can be configured to run in 'job provider' only mode, no queues, no excess processing, just there to create jobs for somewhere else to consume.
- **Best Practice Implementation**. We use lua scripts for atomic operations and blocking ops such as `BLPOPRPUSH` to get queued items.
- *Surprisingly other libraries such as [Automattic/Kue](https://github.com/Automattic/kue/issues/688#issuecomment-142372665) don't do this correctly and therefore heavily impact on performance / opens them up to race conditions*.
- **Cluster and Sentinel support**. We use `ioredis` and built this implementation with a 'cluster-first' mentality, keys are correctly tagged for slots etc.
- **Easy Redis access inside job runners**. Job runners when called with a job are provided with the full RediBox utility belt and any hooks you configured. A job that creates a job that creates a relay job and creates another job from the relay response - why not? *(insert YO DAWG meme)*
- **Job Uniqueness**. Jobs can be set as 'unique', upon which it's id becomes the sha1sum of the data you provided, any duplicate jobs with the same unique flag will get rejected.
- **Job Timeouts**. Job timeouts can be set on a per job or per queue basis. Parent relay jobs are set to the timeout value provided multiplied by the number of jobs in the chain.
- **Job Retries**. Jobs by default are not retried but this can be changed on a per job basis.
- **Resumes on crash**. Should your node process crash then the queues will pick up and retry any jobs that were running on crash.

@@ -19,4 +19,11 @@ global.HOOK_NAME = 'job';

queues: [
{ name: 'test', concurrency: 5 },
{ name: 'test2', concurrency: 10 },
{
name: 'test2',
concurrency: 5,
// throttle: {
// limit: 2,
// seconds: 10,
// },
},
{ name: 'test', concurrency: 10 },
],

@@ -92,3 +99,5 @@ },

},
}).timeout(2000).unique(true).onSuccess((result) => { console.dir(result)});
}).timeout(2000).unique(true).onSuccess((result) => {
console.dir(result)
});
});

@@ -95,0 +104,0 @@ });