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 1.2.17 to 1.2.18

benchmark/benchmark-chart.png

40

docs/advanced-usage.md
## Advanced Usage
### Inline vs globally accessible functions
### Generic queue handler vs globally accessible functions
For simplicity reasons, previous examples have mostly shown the functions jobs execute as inline, e.g:
For simplicity reasons, previous examples have mostly shown the functions jobs executing using the default queue handler, e.g:
```javascript
RediBox.hooks.job.create('queue', {
runs: function(job) {
console.log('Job running');
}
// specify a 'handler' function on the 'myqueue' config options
// all jobs without a `runs` will fallback to using this handler
// to process jobs
RediBox.hooks.job.create('myqueue', {
data: {
foo: 'bar',
},
});

@@ -18,3 +22,3 @@ ```

functions to be broken down into file based logic and callable globally. If the job detects a string as a run function,
it'll attempt to execute it, assuming it's a global function.
it'll attempt to deep get the dot notated path and execute the function, assuming it's a global path.

@@ -26,3 +30,3 @@ This allows for much cleaner code, e.g:

RediBox.hooks.job.create('queue', {
runs: 'sails.hooks.api.request',
runs: 'tasks.request',
});

@@ -33,9 +37,10 @@ ```

// request.js
export default function() {
console.log(this.data);
global.tasks = {
request() {
console.log(this.data);
}
}
```
It's also worth noting that with inline functions, the job is bound as the first argument to the function, whilst with
a global function it's bound to it's scope, whereby the job can be accessed with `this`.
It's also worth noting by default the task function is bound to the job instance, the first argument also becomes the job instance. To disable the `bind` you can set the `noBind` option on an individial job (on it's options) or on the queue options to turn it off entirely.

@@ -52,3 +57,3 @@ ### Error handling

export default function() {
return Person.find();
return Person.update(1234, { hasKittens: true });
}

@@ -92,4 +97,4 @@ ```

runs: [
'global.generic.findPerson',
'global.person.updatePerson',
'generic.findPerson',
'person.updatePerson',
],

@@ -194,5 +199,8 @@ data: {

}
// in reality you'd probably want to Promise.map these with a concurrency limit
// but you can just Job.create() without calling the promise - the job hook will queue these for creating at
// the start of the next event loop - with concurrency batching.
return Promise.all(promises);
}
```

102

docs/best-practices.md

@@ -6,19 +6,20 @@ ## Best Practices

With a clustered server environment with high queue concurrency, the speed in which the overall "task" will complete
will be much quicker if task is broken down into many smaller jobs, rather than one large job. Memory/CPU usage will be kept lower
will be much quicker if the task is broken down into many smaller jobs, rather than one large job. Memory/CPU usage will be kept lower
and error trapping will be much more specific to a single job for easier debugging.
**Example**: We've got a very large set of data which needs to be processed. The data needs to be looped over,
manipulated, internal database calls need to be made using the data and finally data needs to be saved.
**Example**: We've got a very large set of data which a user has uploaded which needs to be processed. The data needs to be iterrated over,
manipulated, internal database calls need to be made and finally the data needs to be saved.
A single job could handle this very easily. However we'd need to make use of Node.JS's async compatibility and
the code would become very messy - lots of nested loops, variables etc.
We could handle this very easily in a express request to our api. However this could take a very long time and potentially delay other user requests coming into our api. Consider the below example:
```javascript
// Massive job which finds and updates users
// your api
import each from 'async/each';
export default function() {
const data = require('../my-massive-dataset.json');
// some request controller
export default function(req, res) {
// wherever the users file has been uploaded to
const arrayData = require('../users-massive-dataset.json');
return each(data, function(item, done) {
each(arrayData, function(item, done) {
return User

@@ -32,6 +33,6 @@ .find({ name: item.name })

if (error) {
return Promise.reject(error);
return res.status(500).send(error);
}
return Promise.resolve();
return res.json({ msg: 'Sorry i took a long time, but all done!' });
});

@@ -42,30 +43,64 @@ }

There's two problems here:
There's several problems here:
1. The job would consume lots of memory/cpu on the server performing async requests, storing variables in memory etc.
2. The server running the job is potentially out of action until the job has completed (which might take quite a while).
3. If our job has to update thousands of database records, it's very hard to internally throttle this due to the async nature of Node.
1. The request could consume lots of memory/cpu on the server performing the request.
2. The server running the request is potentially out of action until the processing has completed (which might take several seconds).
3. If our user wants to update thousands of database records, it's very hard to internally throttle this due to the async nature of Node.
Breaking this job down into multiple jobs would be a much better solution:
Instead, let's send this request to our internal node.js worker farm and break it down into multiple jobs as this would be a much better solution:
#### Your API
Your API where the redibox job `enabled` option is set to `false` - which means we're in provider only mode and will not consume jobs on the API. Queues do not need to be specified on your config in this case - only consumers of jobs need the queues specified.
```javascript
// Smaller jobs to update users
export default function() {
const data = require('../my-massive-dataset.json');
for (let i = 0, len = data.length; i < len; i++) {
const item = data[i];
Job.create('my-queue', {
runs: 'updateUser',
// some request controller
export default function(req, res) {
// ...
// lets move the upload into redis - assumes 'uploadData' is the upload.
RediBox.client.set('users:upload:id', uploadData).then(() => {
Job.create('user-uploads', {
runs: 'getDataAndProcessIt',
data: {
name: item.name,
settings: item.settings,
uploadKey: 'users:upload:id',
}
});
}
return res.json({ msg: 'Upload complete - we will notify you when your upload has been processed.'});
});
}
```
#### A server in your worker farm
Your server in your internal worker farm where the redibox job `enabled` option is set to `true`, meaning these servers will be able to consume and also provide new jobs. You'll need to add the config for your queues on these servers, consumers need to know what queues to consume jobs from.
```javascript
// get the upload data and break it into smaller update jobs for load distribution
export function getDataAndProcessIt() {
// 'this' is the current job
// 'this.data' is the current jobs data.
const { dataKey } = this.data;
return Promise.resolve();
return RediBox.client.get(dataKey).then((uploadData) => {
if (!uploadData) return Promise.reject(`Upload not found for key '${dataKey}'`);
// we could just do all the updates in this one job but we have a farm
// so lets distribute the load and spread the updates across all our servers
// by creating individual jobs for each update
for (let i = 0, len = uploadData.length; i < len; i++) {
const item = uploadData[i];
Job.create('user-upload-items', {
runs: 'updateUser',
data: {
name: item.name,
settings: item.settings,
}
});
}
});
}
// your global updateUser job function
// this handles a single item in the users upload
export function updateUser() {

@@ -83,2 +118,7 @@ return User

// TODO
// notify the user that this part of their upload has been processed
// for example they're importing users into their org, show the user as imported
// ...
return Promise.resolve();

@@ -91,3 +131,3 @@ });

1. The first job is fully synchronous and very performant. This will consume little memory and usage.
1. The first job is short, sweet and very performant. This will consume little memory and cpu usage while enabling us to distribute the load for effecient processing.
2. The second job is user specific; we're querying an individual user per job. This can be throttled by using a different

@@ -99,2 +139,4 @@ queue. If a user query fails we can also trap the errors for the specific user.

As you can see this now become distributed and the load is spread across all your workers, each handling jobs of their own concurrently, resulting in faster processing overall.
### Queues

@@ -101,0 +143,0 @@

@@ -67,3 +67,3 @@ ## Queues

There are no limits to how many queues can be created, however from a technical point of view, each queue creates its own
internal Redis connection to perform blocking requests. Therefore bare in mind more queues equals more connections which
internal Redis connection to perform blocking requests. Therefore keep in mind more queues equals more connections which
might need some consideration depending on your Redis environment setup.

@@ -14,4 +14,7 @@ 'use strict';

startupDelay: 750,
stallInterval: 5000
stallInterval: 15000,
autoSave: {
maxJobs: 1000
}
};
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9kZWZhdWx0cy5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7OztBQUFBOzs7a0JBR2U7QUFDYixVQUFRLEVBREs7QUFFYixXQUFTLElBRkk7QUFHYixhQUFXLEtBSEU7QUFJYixnQkFBYyxHQUpEO0FBS2IsaUJBQWU7QUFMRixDIiwiZmlsZSI6ImRlZmF1bHRzLmpzIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiBEZWZhdWx0IENvbmZpZ3VyYXRpb25cbiAqL1xuZXhwb3J0IGRlZmF1bHQge1xuICBxdWV1ZXM6IFtdLFxuICBlbmFibGVkOiB0cnVlLFxuICBrZXlQcmVmaXg6ICdqb2InLFxuICBzdGFydHVwRGVsYXk6IDc1MCxcbiAgc3RhbGxJbnRlcnZhbDogNTAwMCxcbn07XG5cbiJdfQ==
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9kZWZhdWx0cy5qcyJdLCJuYW1lcyI6WyJxdWV1ZXMiLCJlbmFibGVkIiwia2V5UHJlZml4Iiwic3RhcnR1cERlbGF5Iiwic3RhbGxJbnRlcnZhbCIsImF1dG9TYXZlIiwibWF4Sm9icyJdLCJtYXBwaW5ncyI6Ijs7Ozs7QUFBQTs7O2tCQUdlO0FBQ2JBLFVBQVEsRUFESztBQUViQyxXQUFTLElBRkk7QUFHYkMsYUFBVyxLQUhFO0FBSWJDLGdCQUFjLEdBSkQ7QUFLYkMsaUJBQWUsS0FMRjtBQU1iQyxZQUFVO0FBQ1JDLGFBQVM7QUFERDtBQU5HLEMiLCJmaWxlIjoiZGVmYXVsdHMuanMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuIERlZmF1bHQgQ29uZmlndXJhdGlvblxuICovXG5leHBvcnQgZGVmYXVsdCB7XG4gIHF1ZXVlczogW10sXG4gIGVuYWJsZWQ6IHRydWUsXG4gIGtleVByZWZpeDogJ2pvYicsXG4gIHN0YXJ0dXBEZWxheTogNzUwLFxuICBzdGFsbEludGVydmFsOiAxNTAwMCxcbiAgYXV0b1NhdmU6IHtcbiAgICBtYXhKb2JzOiAxMDAwLFxuICB9LFxufTtcblxuIl19

@@ -9,2 +9,8 @@ 'use strict';

var _redibox = require('redibox');
var _bluebird = require('bluebird');
var _bluebird2 = _interopRequireDefault(_bluebird);
var _cuid = require('cuid');

@@ -22,6 +28,2 @@

var _bluebird = require('bluebird');
var _bluebird2 = _interopRequireDefault(_bluebird);
var _defaults = require('./defaults');

@@ -35,4 +37,2 @@

var _redibox = require('redibox');
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }

@@ -91,2 +91,6 @@

/**
* PUBLIC API
*/
/**
* Creates a new job for the specified queue

@@ -105,5 +109,5 @@ * @param queue

if (!this.autoSaveImmediate) {
this.autoSaveImmediate = setImmediate(this.autoSave.bind(this));
} else if (!Object.keys(this.autoCreateQueue).length > 200) {
this.autoSave();
this.autoSaveImmediate = setImmediate(this._autoSave.bind(this));
} else if (!Object.keys(this.autoCreateQueue).length >= this.options.autoSave.maxJobs) {
this._autoSave();
}

@@ -115,5 +119,19 @@

/**
*
* @param queue
* @param jobId
* @returns {Promise.<TResult>}
*/
getJobById(queue, jobId) {
return this.client.hget(this._toQueueKey(queue, 'jobs'), jobId).then(data => _job2.default.fromData(queue, jobId, data));
}
/**
* PRIVATE API
*/
/**
* Auto save any jobs in the queue
*/
autoSave() {
_autoSave() {
clearImmediate(this.autoSaveImmediate);

@@ -136,3 +154,3 @@ this.autoSaveImmediate = null;

if (!jobsToSave.length) return void 0;
if (!jobsToSave.length) return undefined;

@@ -142,7 +160,7 @@ this.log.verbose(`Auto-saving ${ jobsToSave.length } jobs.`);

if (jobsToSave.length === 1) {
return this.autoCreateQueue[jobsToSave[0]].withoutProxy().save(true).then(() => this.cleanupAutoSave.call(this, jobsToSave)).catch(err => this.cleanupAutoSave.call(this, jobsToSave, err));
return this.autoCreateQueue[jobsToSave[0]].withoutProxy().save(true).then(() => this._cleanupAutoSave.call(this, jobsToSave)).catch(err => this._cleanupAutoSave.call(this, jobsToSave, err));
}
/* eslint no-confusing-arrow: 0 */
return _bluebird2.default.map(jobsToSave, ref => this.autoCreateQueue && this.autoCreateQueue[ref] ? this.autoCreateQueue[ref].withoutProxy().save(true) : _bluebird2.default.resolve(), { concurrency: 25 }).then(() => this.cleanupAutoSave.call(this, jobsToSave)).catch(err => this.cleanupAutoSave.call(this, jobsToSave, err));
return _bluebird2.default.map(jobsToSave, ref => this.autoCreateQueue && this.autoCreateQueue[ref] ? this.autoCreateQueue[ref].withoutProxy().save(true) : _bluebird2.default.resolve()).then(() => this._cleanupAutoSave.call(this, jobsToSave)).catch(err => this._cleanupAutoSave.call(this, jobsToSave, err));
}

@@ -155,3 +173,3 @@

*/
cleanupAutoSave(jobsToRemove, possibleError) {
_cleanupAutoSave(jobsToRemove, possibleError) {
if (possibleError) this.log.error(possibleError);

@@ -173,3 +191,3 @@ this.log.verbose(`Removing ${ jobsToRemove.length } saved jobs from the auto-save queue.`);

return void 0;
return undefined;
}

@@ -196,2 +214,16 @@

/**
* Generates a queue prefixed key based on the provided string.
* @param queue
* @param str
* @returns {string}
* @private
*/
_toQueueKey(queue, str) {
if (this.core.cluster.isCluster()) {
return `${ this.options.keyPrefix }:{${ queue }}:${ str }`;
}
return `${ this.options.keyPrefix }:${ queue }:${ str }`;
}
/**
* To enable bypassing of cache for wrap functions

@@ -206,2 +238,2 @@ * Toggles by default or pass in true/false

exports.default = JobHook;
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/hook.js"],"names":[],"mappings":"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","file":"hook.js","sourcesContent":["import cuid from 'cuid';\nimport Job from './job';\nimport Queue from './queue';\nimport Promise from 'bluebird';\nimport defaults from './defaults';\nimport scripts from './scripts';\nimport { BaseHook } from 'redibox';\n\nexport default class JobHook extends BaseHook {\n  constructor() {\n    super('job');\n    this.queues = {};\n    this.autoCreateQueue = null;\n    this.autoSaveImmediate = null;\n  }\n\n  // noinspection JSUnusedGlobalSymbols,JSMethodCanBeStatic\n  /**\n   * Default config for scheduler\n   * @returns {{someDefaultThing: string}}\n   */\n  defaults() {\n    return defaults;\n  }\n\n  /**\n   *\n   * @returns {{addJob, checkStalledJobs, removeJob}}\n   */\n  scripts() {\n    return scripts;\n  }\n\n  /**\n   *\n   */\n  initialize() {\n    if (!this.options.enabled) {\n      return Promise.resolve();\n    }\n\n    for (let i = 0, len = this.options.queues.length; i < len; i++) {\n      const queue = this.options.queues[i];\n      this.queueCreate(queue);\n    }\n\n    this.on('core:ready', () => {\n      setTimeout(() => {\n        for (let i = 0, len = this.options.queues.length; i < len; i++) {\n          const queue = this.options.queues[i];\n          this.queues[queue.name].beginWorking();\n        }\n      }, this.options.startupDelay);\n    });\n\n    return Promise.resolve();\n  }\n\n  /**\n   * Creates a new job for the specified queue\n   * @param queue\n   * @param data\n   * @param options\n   * @returns {*|Job}\n   */\n  create(...args) {\n    const ref = cuid();\n    if (!this.autoCreateQueue) this.autoCreateQueue = {};\n    this.autoCreateQueue[ref] = new Job(this.core, null, args[1], args[2], args[0], true);\n    this.log.verbose(`Creating job for queue ${args[0]} with ref ${ref}`);\n\n    if (!this.autoSaveImmediate) {\n      this.autoSaveImmediate = setImmediate(this.autoSave.bind(this));\n    } else if (!Object.keys(this.autoCreateQueue).length > 200) {\n      this.autoSave();\n    }\n\n    return this.autoCreateQueue[ref];\n  }\n\n  /**\n   * Auto save any jobs in the queue\n   */\n  autoSave() {\n    clearImmediate(this.autoSaveImmediate);\n    this.autoSaveImmediate = null;\n\n    const jobsToSave = [];\n    const refs = Object.keys(this.autoCreateQueue);\n\n    for (let i = 0, iLen = refs.length; i < iLen; i++) {\n      const ref = refs[i];\n      if (!this.autoCreateQueue[ref]._saved) {\n        jobsToSave.push(ref);\n        this.autoCreateQueue[ref]._saved = true;\n      } else {\n        this.autoCreateQueue[ref] = null;\n        delete this.autoCreateQueue[ref];\n      }\n    }\n\n    if (!jobsToSave.length) return void 0;\n\n    this.log.verbose(`Auto-saving ${jobsToSave.length} jobs.`);\n\n    if (jobsToSave.length === 1) {\n      return this.autoCreateQueue[jobsToSave[0]]\n      .withoutProxy()\n      .save(true)\n      .then(() => this.cleanupAutoSave.call(this, jobsToSave))\n      .catch(err => this.cleanupAutoSave.call(this, jobsToSave, err));\n    }\n\n    /* eslint no-confusing-arrow: 0 */\n    return Promise\n    .map(\n      jobsToSave,\n      ref => this.autoCreateQueue && this.autoCreateQueue[ref] ?\n        this.autoCreateQueue[ref].withoutProxy().save(true) :\n        Promise.resolve(),\n      { concurrency: 25 }\n    )\n    .then(() => this.cleanupAutoSave.call(this, jobsToSave))\n    .catch(err => this.cleanupAutoSave.call(this, jobsToSave, err));\n  }\n\n  /**\n   *\n   * @param jobsToRemove\n   * @param possibleError\n   */\n  cleanupAutoSave(jobsToRemove, possibleError) {\n    if (possibleError) this.log.error(possibleError);\n    this.log.verbose(`Removing ${jobsToRemove.length} saved jobs from the auto-save queue.`);\n    if (this.autoCreateQueue) {\n      for (let i = 0, iLen = jobsToRemove.length; i < iLen; i++) {\n        // trying to force garbage collection here\n        this.autoCreateQueue[jobsToRemove[i]] = null;\n        delete this.autoCreateQueue[jobsToRemove[i]];\n      }\n\n      if (!Object.keys(this.autoCreateQueue).length) {\n        // trying to force garbage collection here\n        this.autoCreateQueue = null;\n        delete this.autoCreateQueue;\n      }\n    }\n\n    return void 0;\n  }\n\n  /**\n   *\n   * @param queue\n   */\n  queueCreate(queue) {\n    this.log.verbose(`Queue '${queue.name}' created!`);\n    this.queues[queue.name] = new Queue({ ...this.options, ...queue }, this.core);\n  }\n\n  /**\n   * Converts the users key to the full redis key with module prefix.\n   * @param key\n   * @returns {string}\n   */\n  toKey(key = '') {\n    return `${this.options.keyPrefix}:${key}`;\n  }\n\n  /**\n   * To enable bypassing of cache for wrap functions\n   * Toggles by default or pass in true/false\n   * @param bool\n   */\n  enabled(bool) {\n    this.options.enabled = bool || !this.options.enabled;\n  }\n}\n"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/hook.js"],"names":["JobHook","constructor","queues","autoCreateQueue","autoSaveImmediate","defaults","scripts","initialize","options","enabled","resolve","i","len","length","queue","queueCreate","on","setTimeout","name","beginWorking","startupDelay","create","args","ref","core","log","verbose","setImmediate","_autoSave","bind","Object","keys","autoSave","maxJobs","getJobById","jobId","client","hget","_toQueueKey","then","data","fromData","clearImmediate","jobsToSave","refs","iLen","_saved","push","undefined","withoutProxy","save","_cleanupAutoSave","call","catch","err","map","jobsToRemove","possibleError","error","toKey","key","keyPrefix","str","cluster","isCluster","bool"],"mappings":"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","file":"hook.js","sourcesContent":["import { BaseHook } from 'redibox';\nimport Promise from 'bluebird';\n\nimport cuid from 'cuid';\nimport Job from './job';\nimport Queue from './queue';\nimport defaults from './defaults';\nimport scripts from './scripts';\n\nexport default class JobHook extends BaseHook {\n  constructor() {\n    super('job');\n    this.queues = {};\n    this.autoCreateQueue = null;\n    this.autoSaveImmediate = null;\n  }\n\n  // noinspection JSUnusedGlobalSymbols,JSMethodCanBeStatic\n  /**\n   * Default config for scheduler\n   * @returns {{someDefaultThing: string}}\n   */\n  defaults() {\n    return defaults;\n  }\n\n  /**\n   *\n   * @returns {{addJob, checkStalledJobs, removeJob}}\n   */\n  scripts() {\n    return scripts;\n  }\n\n  /**\n   *\n   */\n  initialize() {\n    if (!this.options.enabled) {\n      return Promise.resolve();\n    }\n\n    for (let i = 0, len = this.options.queues.length; i < len; i++) {\n      const queue = this.options.queues[i];\n      this.queueCreate(queue);\n    }\n\n    this.on('core:ready', () => {\n      setTimeout(() => {\n        for (let i = 0, len = this.options.queues.length; i < len; i++) {\n          const queue = this.options.queues[i];\n          this.queues[queue.name].beginWorking();\n        }\n      }, this.options.startupDelay);\n    });\n\n    return Promise.resolve();\n  }\n\n  /**\n   * PUBLIC API\n   */\n\n  /**\n   * Creates a new job for the specified queue\n   * @param queue\n   * @param data\n   * @param options\n   * @returns {*|Job}\n   */\n  create(...args) {\n    const ref = cuid();\n    if (!this.autoCreateQueue) this.autoCreateQueue = {};\n    this.autoCreateQueue[ref] = new Job(this.core, null, args[1], args[2], args[0], true);\n    this.log.verbose(`Creating job for queue ${args[0]} with ref ${ref}`);\n\n    if (!this.autoSaveImmediate) {\n      this.autoSaveImmediate = setImmediate(this._autoSave.bind(this));\n    } else if (!Object.keys(this.autoCreateQueue).length >= this.options.autoSave.maxJobs) {\n      this._autoSave();\n    }\n\n    return this.autoCreateQueue[ref];\n  }\n\n\n  /**\n   *\n   * @param queue\n   * @param jobId\n   * @returns {Promise.<TResult>}\n   */\n  getJobById(queue, jobId) {\n    return this.client.hget(this._toQueueKey(queue, 'jobs'), jobId).then(data =>\n      Job.fromData(queue, jobId, data)\n    );\n  }\n\n  /**\n   * PRIVATE API\n   */\n\n  /**\n   * Auto save any jobs in the queue\n   */\n  _autoSave() {\n    clearImmediate(this.autoSaveImmediate);\n    this.autoSaveImmediate = null;\n\n    const jobsToSave = [];\n    const refs = Object.keys(this.autoCreateQueue);\n\n    for (let i = 0, iLen = refs.length; i < iLen; i++) {\n      const ref = refs[i];\n      if (!this.autoCreateQueue[ref]._saved) {\n        jobsToSave.push(ref);\n        this.autoCreateQueue[ref]._saved = true;\n      } else {\n        this.autoCreateQueue[ref] = null;\n        delete this.autoCreateQueue[ref];\n      }\n    }\n\n    if (!jobsToSave.length) return undefined;\n\n    this.log.verbose(`Auto-saving ${jobsToSave.length} jobs.`);\n\n    if (jobsToSave.length === 1) {\n      return this.autoCreateQueue[jobsToSave[0]]\n        .withoutProxy()\n        .save(true)\n        .then(() => this._cleanupAutoSave.call(this, jobsToSave))\n        .catch(err => this._cleanupAutoSave.call(this, jobsToSave, err));\n    }\n\n    /* eslint no-confusing-arrow: 0 */\n    return Promise.map(\n      jobsToSave,\n      ref => this.autoCreateQueue && this.autoCreateQueue[ref] ?\n        this.autoCreateQueue[ref].withoutProxy().save(true) :\n        Promise.resolve()\n    )\n      .then(() => this._cleanupAutoSave.call(this, jobsToSave))\n      .catch(err => this._cleanupAutoSave.call(this, jobsToSave, err));\n  }\n\n  /**\n   *\n   * @param jobsToRemove\n   * @param possibleError\n   */\n  _cleanupAutoSave(jobsToRemove, possibleError) {\n    if (possibleError) this.log.error(possibleError);\n    this.log.verbose(`Removing ${jobsToRemove.length} saved jobs from the auto-save queue.`);\n    if (this.autoCreateQueue) {\n      for (let i = 0, iLen = jobsToRemove.length; i < iLen; i++) {\n        // trying to force garbage collection here\n        this.autoCreateQueue[jobsToRemove[i]] = null;\n        delete this.autoCreateQueue[jobsToRemove[i]];\n      }\n\n      if (!Object.keys(this.autoCreateQueue).length) {\n        // trying to force garbage collection here\n        this.autoCreateQueue = null;\n        delete this.autoCreateQueue;\n      }\n    }\n\n    return undefined;\n  }\n\n  /**\n   *\n   * @param queue\n   */\n  queueCreate(queue) {\n    this.log.verbose(`Queue '${queue.name}' created!`);\n    this.queues[queue.name] = new Queue({ ...this.options, ...queue }, this.core);\n  }\n\n  /**\n   * Converts the users key to the full redis key with module prefix.\n   * @param key\n   * @returns {string}\n   */\n  toKey(key = '') {\n    return `${this.options.keyPrefix}:${key}`;\n  }\n\n  /**\n   * Generates a queue prefixed key based on the provided string.\n   * @param queue\n   * @param str\n   * @returns {string}\n   * @private\n   */\n  _toQueueKey(queue, str) {\n    if (this.core.cluster.isCluster()) {\n      return `${this.options.keyPrefix}:{${queue}}:${str}`;\n    }\n    return `${this.options.keyPrefix}:${queue}:${str}`;\n  }\n\n  /**\n   * To enable bypassing of cache for wrap functions\n   * Toggles by default or pass in true/false\n   * @param bool\n   */\n  enabled(bool) {\n    this.options.enabled = bool || !this.options.enabled;\n  }\n}\n"]}

@@ -14,2 +14,2 @@ 'use strict';

exports.default = _hook2.default;
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9pbmRleC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7QUFBQSIsImZpbGUiOiJpbmRleC5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBIb29rIGZyb20gJy4vaG9vayc7XG5leHBvcnQgZGVmYXVsdCBIb29rO1xuIl19
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9pbmRleC5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7Ozs7QUFBQSIsImZpbGUiOiJpbmRleC5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBIb29rIGZyb20gJy4vaG9vayc7XG5cbmV4cG9ydCBkZWZhdWx0IEhvb2s7XG4iXX0=

@@ -36,4 +36,3 @@ 'use strict';

unique: false,
timeout: 60000 }, // 1 minute default timeout
queueName, isNew) {
timeout: 60000 }, queueName, isNew) {
this.id = id;

@@ -100,6 +99,4 @@ this.core = core;

* @static
* @param queue
* @param id
* @param data
* @returns {Job | null}
* @param args
*/

@@ -151,2 +148,3 @@ static fromData(...args) {

// TODO allow hooking into this event rather than us doing this poop here
if (process.env.KUBERNETES_PORT || process.env.KUBERNETES_SERVICE_HOST) {

@@ -183,3 +181,3 @@ /* eslint no-console: 0 */

// lock it so autoSave doesn't pick it up but only deletes it from queue
// lock it so _autoSave doesn't pick it up but only deletes it from queue
this._saved = true;

@@ -216,3 +214,4 @@

return this.onFailureCallback(message.data);
}, this.options.timeout + 1000).then(() => // subscribed callback
}, this.options.timeout + 1000).then(() =>
// now subscribed so save the job
this._save()).catch(error => this.onFailureCallback({

@@ -296,2 +295,23 @@ type: 'job',

remove() {
return this.core.client.removejob(this._toQueueKey('succeeded'), this._toQueueKey('failed'), this._toQueueKey('waiting'), this._toQueueKey('active'), this._toQueueKey('stalling'), this._toQueueKey('jobs'), this.id);
}
/**
* Re-save this job for the purpose of retrying it.
* @param cb
*/
retry() {
return this.core.client.multi().srem(this._toQueueKey('failed'), this.id).lpush(this._toQueueKey('waiting'), this.id);
}
/**
*
* @param set
* @returns {Promise.<boolean>}
*/
inSet(set) {
return this.core.client.sismember(this._toQueueKey(set), this.id).then(result => result === 1);
}
} /**

@@ -324,74 +344,2 @@ *

exports.default = Job;
// 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._saved = false;\n    this.options = options;\n    this.status = 'created';\n    this.subscriptions = [];\n    this.ignoreProxy = false;\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 (!target.ignoreProxy && 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   * Returns job instance with no proxy\n   * @returns {Job}\n   */\n  withoutProxy() {\n    this.ignoreProxy = true;\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(...args) {\n    const obj = tryJSONParse(args[2]);\n    if (!obj) return null;\n    const job = new Job(args[0].core, args[1], obj.data, obj.options, args[0].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\n        if (process.env.KUBERNETES_PORT || process.env.KUBERNETES_SERVICE_HOST) {\n          /* eslint no-console: 0 */\n          const jobData = JSON.stringify(this.data.data || {});\n          console.log(JSON.stringify({\n            level: 'verbose',\n            type: 'redibox_job_created',\n            job: {\n              id: this.id.toString ? this.id.toString() : this.id,\n              runs: this.data.runs,\n              queue: this.queueName,\n              status: 'pending',\n              data: jobData.length > 4000 ? '<! job data too large to display !>' : this.data.data,\n            },\n          }));\n        }\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(auto) {\n    if (!auto && this._saved) return Promise.resolve();\n\n    // lock it so autoSave doesn't pick it up but only deletes it from queue\n    this._saved = true;\n\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.data = {};\n\n          // if there's an error then assume failed.\n          if (message.data.error) return this.onFailureCallback(message.data);\n\n          // is it from the success channel.\n          if (this.subscriptions[0] === message.channel) return this.onSuccessCallback(message.data);\n\n          return this.onFailureCallback(message.data);\n        },\n        this.options.timeout + 1000\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"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/job.js"],"names":["Job","constructor","core","id","data","options","unique","timeout","queueName","isNew","_saved","status","subscriptions","ignoreProxy","type","runs","Array","isArray","length","proxy","Proxy","get","target","name","promise","save","then","bind","undefined","withoutProxy","fromId","queue","client","hget","toKey","fromData","args","obj","job","toData","toObject","excludeData","_save","log","verbose","addjob","_toQueueKey","reject","Error","process","env","KUBERNETES_PORT","KUBERNETES_SERVICE_HOST","jobData","JSON","stringify","console","level","toString","resolve","auto","notifySuccess","push","notifyFailure","pubsub","subscriber","subscribeOnceOf","message","error","onFailureCallback","channel","onSuccessCallback","catch","error_actual","retries","n","onSuccess","notify","onFailure","bool","ms","str","cluster","isCluster","hooks","keyPrefix","remove","removejob","retry","multi","srem","lpush","inSet","set","sismember","result"],"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._saved = false;\n    this.options = options;\n    this.status = 'created';\n    this.subscriptions = [];\n    this.ignoreProxy = false;\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 (!target.ignoreProxy && 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   * Returns job instance with no proxy\n   * @returns {Job}\n   */\n  withoutProxy() {\n    this.ignoreProxy = true;\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   * @returns {Job | null}\n   * @param args\n   */\n  static fromData(...args) {\n    const obj = tryJSONParse(args[2]);\n    if (!obj) return null;\n    const job = new Job(args[0].core, args[1], obj.data, obj.options, args[0].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\n        // TODO allow hooking into this event rather than us doing this poop here\n        if (process.env.KUBERNETES_PORT || process.env.KUBERNETES_SERVICE_HOST) {\n          /* eslint no-console: 0 */\n          const jobData = JSON.stringify(this.data.data || {});\n          console.log(JSON.stringify({\n            level: 'verbose',\n            type: 'redibox_job_created',\n            job: {\n              id: this.id.toString ? this.id.toString() : this.id,\n              runs: this.data.runs,\n              queue: this.queueName,\n              status: 'pending',\n              data: jobData.length > 4000 ? '<! job data too large to display !>' : this.data.data,\n            },\n          }));\n        }\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(auto) {\n    if (!auto && this._saved) return Promise.resolve();\n\n    // lock it so _autoSave doesn't pick it up but only deletes it from queue\n    this._saved = true;\n\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.data = {};\n\n          // if there's an error then assume failed.\n          if (message.data.error) return this.onFailureCallback(message.data);\n\n          // is it from the success channel.\n          if (this.subscriptions[0] === message.channel) return this.onSuccessCallback(message.data);\n\n          return this.onFailureCallback(message.data);\n        },\n        this.options.timeout + 1000\n      ).then(() =>\n        // now subscribed so save the job\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  remove() {\n    return 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);\n  }\n\n  /**\n   * Re-save this job for the purpose of retrying it.\n   * @param cb\n   */\n  retry() {\n    return this.core.client.multi()\n      .srem(this._toQueueKey('failed'), this.id)\n      .lpush(this._toQueueKey('waiting'), this.id);\n  }\n\n  /**\n   *\n   * @param set\n   * @returns {Promise.<boolean>}\n   */\n  inSet(set) {\n    return this.core.client.sismember(this._toQueueKey(set), this.id).then(result => result === 1);\n  }\n\n}\n\nexport default Job;\n"]}

@@ -9,6 +9,4 @@ 'use strict';

var _job = require('./job');
var _redibox = require('redibox');
var _job2 = _interopRequireDefault(_job);
var _bluebird = require('bluebird');

@@ -18,6 +16,2 @@

var _defaults = require('./defaults');
var _defaults2 = _interopRequireDefault(_defaults);
var _eventemitter = require('eventemitter3');

@@ -27,4 +21,10 @@

var _redibox = require('redibox');
var _job = require('./job');
var _job2 = _interopRequireDefault(_job);
var _defaults = require('./defaults');
var _defaults2 = _interopRequireDefault(_defaults);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }

@@ -82,4 +82,5 @@

this._onLocalTickComplete = () => {
this.running--;
this.queued--;
this.running -= 1;
this.queued -= 1;
if (!this.options.throttle) return setImmediate(this._queueTick);

@@ -89,2 +90,3 @@

const shouldThrottle = throttle[0] === 1;
if (!shouldThrottle) {

@@ -103,3 +105,3 @@ this.throttled = false;

this._onLocalTickError = error => {
this.queued--;
this.queued -= 1;
this.log.error(error);

@@ -111,7 +113,11 @@ setImmediate(this._queueTick);

if (this.paused || !this.options.enabled) {
return void 0;
return undefined;
}
this.queued++;
return this._getNextJob().then(job => {
this.running++;
this.queued += 1;
return this._getNextJob((err, job) => {
if (err) return this._onLocalTickError(err);
this.running += 1;
// queue more jobs if within limit

@@ -124,3 +130,3 @@ if (this.running + this.queued < this.options.concurrency) {

return this._runJob(job).then(this._onLocalTickComplete).catch(this._onLocalTickComplete);
}).catch(this._onLocalTickError);
});
};

@@ -135,3 +141,2 @@

this.core = core;
this.client = core.client;
this.paused = false;

@@ -142,7 +147,6 @@ this.started = false;

this.name = options.name;
this.client = core.client;
this.handler = options.handler || null;
this.options = Object.assign({}, _defaults2.default.queue, options || {});
this.core.createClient('block', this).then(() => {
this.log.verbose(`Blocking client for queue '${ this.name }' is ready. Starting queue processor.`);
});
this.core.createClient('block', this);
}

@@ -163,3 +167,3 @@

*/
checkHealth() {
getStatus() {
return this.client.multi().llen(this.toKey('waiting')).llen(this.toKey('active')).scard(this.toKey('succeeded')).scard(this.toKey('failed')).then(results => {

@@ -180,7 +184,5 @@ /* eslint arrow-body-style: 0 */

*/
_getNextJob() {
_getNextJob(cb) {
this.log.verbose(`Getting next job for queue '${ this.name }'.`);
return this.clients.block.brpoplpush(this.toKey('waiting'), this.toKey('active'), 0).then(jobId => _job2.default.fromId(this, jobId).then(job => {
return job;
}));
this.clients.block.brpoplpush(this.toKey('waiting'), this.toKey('active'), 0, (pushError, jobId) => pushError ? cb(pushError) : _job2.default.fromId(this, jobId).then(job => cb(null, job)).catch(err => cb(err)));
}

@@ -198,2 +200,3 @@

// TODO allow hooking into this event rather than us doing this poop here
if (process.env.KUBERNETES_PORT || process.env.KUBERNETES_SERVICE_HOST) {

@@ -241,3 +244,3 @@ /* eslint no-console: 0 */

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

@@ -263,3 +266,3 @@ clearTimeout(preventStallingTimeout);

if (handled) {
return void 0;
return undefined;
}

@@ -283,3 +286,3 @@

if (!handled) {
preventStallingTimeout = setTimeout(preventStalling, this.options.stallInterval / 2);
preventStallingTimeout = setTimeout(preventStalling, this.options.stallInterval / 3);
}

@@ -296,5 +299,7 @@ });

preventStalling(); // start stalling monitor
// start stalling monitoring
preventStalling();
job._internalData = job.data;
job.data = job.data.data || job.data;

@@ -359,3 +364,3 @@

if (job.options.retries > 0) {
job.options.retries = job.options.retries - 1;
job.options.retries -= 1;
job.status = 'retrying';

@@ -367,2 +372,3 @@ multi.hset(this.toKey('jobs'), job.id, job.toData());

multi.hdel(this.toKey('jobs'), job.id);
// TODO track failures and their data somewhere else for reviewing

@@ -373,23 +379,26 @@ // multi.hset(this.toKey('jobs'), job.id, job.toData());

} else {
job.status = 'succeeded';
multi.hdel(this.toKey('jobs'), job.id);
// 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);
if (process.env.KUBERNETES_PORT || process.env.KUBERNETES_SERVICE_HOST) {
/* eslint no-console: 0 */
const jobData = JSON.stringify(job.data.data || {});
console.log(JSON.stringify({
level: 'verbose',
type: 'redibox_job_completed',
job: {
id: job.id.toString ? job.id.toString() : job.id,
runs: job.data.runs,
queue: this.name,
status: job.status,
data: jobData.length > 4000 ? '<! job data too large to display !>' : job.data.data
}
}));
}
job.status = 'succeeded';
multi.hdel(this.toKey('jobs'), job.id);
// 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);
// TODO allow hooking into this event rather than us doing this poop here
if (process.env.KUBERNETES_PORT || process.env.KUBERNETES_SERVICE_HOST) {
/* eslint no-console: 0 */
const jobData = JSON.stringify(job.data.data || {});
console.log(JSON.stringify({
level: 'verbose',
type: 'redibox_job_completed',
job: {
id: job.id.toString ? job.id.toString() : job.id,
runs: job.data.runs,
queue: this.name,
status: job.status,
data: jobData.length > 4000 ? '<! job data too large to display !>' : job.data.data
}
}));
}
}
}

@@ -526,3 +535,3 @@

this.log.info(`Queue ${ this.name } is currently disabled.`);
return void 0;
return;
}

@@ -539,7 +548,4 @@

this.checkStalledJobs().then(() => {
this.log.verbose('checkStalledJobs completed');
}).catch(() => {});
return this._queueTick();
this.checkStalledJobs();
this._queueTick();
}

@@ -553,5 +559,5 @@

this.log.verbose(`${ this.name }: checkStalledJobs`);
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();
return _bluebird2.default.delay(this.options.stallInterval).then(this.checkStalledJobs.bind(this));
this.client.checkstalledjobs(this.toKey('stallTime'), this.toKey('stalling'), this.toKey('waiting'), this.toKey('active'), (0, _redibox.getTimeStamp)(), this.options.stallInterval, () => {
if (!this.options.enabled || this.paused) return;
setTimeout(this.checkStalledJobs.bind(this), this.options.stallInterval);
});

@@ -579,2 +585,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\n/**\n * TODO move to helpers\n * @param errorStack\n * @returns {Array}\n */\nfunction trimStack(errorStack) {\n  const oldStack = errorStack.split('\\n');\n  const stack = [];\n  for (let i = 0, iLen = oldStack.length || stack.length > 19; i < iLen; i++) {\n    const row = oldStack[i];\n\n    // include private modules\n    if (row.includes('@')) {\n      stack.push(row);\n      continue;\n    }\n\n    // exclude job module\n    if (row.includes('redibox-hook-job') || row.includes('redibox/job/lib')) continue;\n\n    // exclude bluebird\n    if (row.includes('bluebird')) continue;\n\n    // exclude waterline\n    if (row.includes('waterline/lib')) continue;\n\n    // exclude waterline\n    if (row.includes('async/lib')) continue;\n\n    // exclude timers.js\n    if (row.includes('timers.js:')) continue;\n\n    stack.push(row);\n  }\n\n  return stack;\n}\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    });\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    const error = typeof jobError === 'string' ? new Error(jobError) : jobError;\n    const stack = trimStack(error.stack);\n\n    if (process.env.KUBERNETES_PORT || process.env.KUBERNETES_SERVICE_HOST) {\n      /* eslint no-console: 0 */\n      const jobData = JSON.stringify(job.data.data || {});\n      console.log(JSON.stringify({\n        level: 'error',\n        type: 'redibox_job_failure',\n        job: {\n          id: job.id.toString ? job.id.toString() : job.id,\n          runs: job.data.runs,\n          queue: this.name,\n          data: jobData.length > 4000 ? '<! job data too large to display !>' : job.data.data,\n          stack,\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      error.stack = stack.join('\\n');\n      this.log.error(error);\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    let promiseOrRes;\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      this._logJobFailure(job, jobError);\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    try {\n      if (job.options.noBind || this.options.noBind) {\n        promiseOrRes = handler(job);\n      } else {\n        promiseOrRes = handler.bind(job, job)(job);\n      }\n    } catch (e) {\n      return handleError(e);\n    }\n\n    if (promiseOrRes && promiseOrRes.then && typeof promiseOrRes.then === 'function') {\n      return promiseOrRes.then(handleOK, handleError).catch(handleError);\n    }\n\n    return handleOK(promiseOrRes);\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      if (process.env.KUBERNETES_PORT || process.env.KUBERNETES_SERVICE_HOST) {\n        /* eslint no-console: 0 */\n        const jobData = JSON.stringify(job.data.data || {});\n        console.log(JSON.stringify({\n          level: 'verbose',\n          type: 'redibox_job_completed',\n          job: {\n            id: job.id.toString ? job.id.toString() : job.id,\n            runs: job.data.runs,\n            queue: this.name,\n            status: job.status,\n            data: jobData.length > 4000 ? '<! job data too large to display !>' : job.data.data,\n          },\n        }));\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    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  /**\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(`${this.name}: 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"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/queue.js"],"names":["trimStack","errorStack","oldStack","split","stack","i","iLen","length","row","includes","push","Queue","constructor","options","core","_onLocalTickComplete","running","queued","throttle","setImmediate","_queueTick","client","toKey","limit","seconds","then","shouldThrottle","throttled","timeRemaining","log","verbose","name","setTimeout","catch","_onLocalTickError","error","paused","enabled","undefined","_getNextJob","err","job","concurrency","_runJob","_restartProcessing","clients","block","once","toEventName","eventName","started","handler","Object","assign","queue","createClient","destroy","keys","map","key","del","getStatus","multi","llen","scard","results","waiting","active","succeeded","failed","cb","brpoplpush","pushError","jobId","fromId","_logJobFailure","jobError","Error","process","env","KUBERNETES_PORT","KUBERNETES_SERVICE_HOST","jobData","JSON","stringify","data","console","level","type","id","toString","runs","join","resolve","Array","isArray","global","preventStallingTimeout","handled","promiseOrRes","handleOK","clearTimeout","_internalData","_finishRelayJob","_finishSingleJob","handleError","preventStalling","srem","stallInterval","timeout","bind","noBind","e","_createJobEvent","worker_id","status","output","_updateJobStatus","lrem","retries","hset","toData","lpush","hdel","notifyFailure","pubsub","publish","notifySuccess","reject","exec","errMulti","result","nextQueue","nextJob","currentJob","shift","initialJob","initialQueue","from_job","from_queue","from_timestamp","hooks","create","beginWorking","info","checkStalledJobs","checkstalledjobs","str","cluster","isCluster","keyPrefix"],"mappings":"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","file":"queue.js","sourcesContent":["import { deepGet, isObject, getTimeStamp, tryJSONParse } from 'redibox';\nimport Promise from 'bluebird';\nimport EventEmitter from 'eventemitter3';\n\nimport Job from './job';\nimport defaults from './defaults';\n\n/**\n * TODO move to helpers\n * @param errorStack\n * @returns {Array}\n */\nfunction trimStack(errorStack) {\n  const oldStack = errorStack.split('\\n');\n  const stack = [];\n  for (let i = 0, iLen = oldStack.length || stack.length > 19; i < iLen; i++) {\n    const row = oldStack[i];\n\n    // include private modules\n    if (row.includes('@')) {\n      stack.push(row);\n      continue;\n    }\n\n    // exclude job module\n    if (row.includes('redibox-hook-job') || row.includes('redibox/job/lib')) continue;\n\n    // exclude bluebird\n    if (row.includes('bluebird')) continue;\n\n    // exclude waterline\n    if (row.includes('waterline/lib')) continue;\n\n    // exclude waterline\n    if (row.includes('async/lib')) continue;\n\n    // exclude timers.js\n    if (row.includes('timers.js:')) continue;\n\n    stack.push(row);\n  }\n\n  return stack;\n}\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.paused = false;\n    this.started = false;\n    this.throttled = false;\n    this.log = this.core.log;\n    this.name = options.name;\n    this.client = core.client;\n    this.handler = options.handler || null;\n    this.options = Object.assign({}, defaults.queue, options || {});\n    this.core.createClient('block', this);\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  getStatus() {\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(cb) {\n    this.log.verbose(`Getting next job for queue '${this.name}'.`);\n    this.clients.block.brpoplpush(\n      this.toKey('waiting'),\n      this.toKey('active'), 0,\n      (pushError, jobId) => pushError ? cb(pushError) : Job.fromId(this, jobId).then(job => cb(null, job)).catch(err => cb(err)));\n  }\n\n  /**\n   *\n   * @param job\n   * @param jobError\n   * @private\n   */\n  _logJobFailure(job, jobError) {\n    const error = typeof jobError === 'string' ? new Error(jobError) : jobError;\n    const stack = trimStack(error.stack);\n\n    // TODO allow hooking into this event rather than us doing this poop here\n    if (process.env.KUBERNETES_PORT || process.env.KUBERNETES_SERVICE_HOST) {\n      /* eslint no-console: 0 */\n      const jobData = JSON.stringify(job.data.data || {});\n      console.log(JSON.stringify({\n        level: 'error',\n        type: 'redibox_job_failure',\n        job: {\n          id: job.id.toString ? job.id.toString() : job.id,\n          runs: job.data.runs,\n          queue: this.name,\n          data: jobData.length > 4000 ? '<! job data too large to display !>' : job.data.data,\n          stack,\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      error.stack = stack.join('\\n');\n      this.log.error(error);\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' ? deepGet(global, this.handler) : this.handler) || deepGet(global, runs);\n\n    let preventStallingTimeout;\n    let handled = false;\n    let promiseOrRes;\n\n    // Handle an \"OK\" response from the promise\n    const handleOK = (data) => {\n      // silently ignore any multiple calls\n      if (handled) return undefined;\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 undefined;\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      this._logJobFailure(job, jobError);\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 / 3);\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    // start stalling monitoring\n    preventStalling();\n\n    job._internalData = job.data;\n\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    try {\n      if (job.options.noBind || this.options.noBind) {\n        promiseOrRes = handler(job);\n      } else {\n        promiseOrRes = handler.bind(job, job)(job);\n      }\n    } catch (e) {\n      return handleError(e);\n    }\n\n    if (promiseOrRes && promiseOrRes.then && typeof promiseOrRes.then === 'function') {\n      return promiseOrRes.then(handleOK, handleError).catch(handleError);\n    }\n\n    return handleOK(promiseOrRes);\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 -= 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\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\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      // TODO allow hooking into this event rather than us doing this poop here\n      if (process.env.KUBERNETES_PORT || process.env.KUBERNETES_SERVICE_HOST) {\n        /* eslint no-console: 0 */\n        const jobData = JSON.stringify(job.data.data || {});\n        console.log(JSON.stringify({\n          level: 'verbose',\n          type: 'redibox_job_completed',\n          job: {\n            id: job.id.toString ? job.id.toString() : job.id,\n            runs: job.data.runs,\n            queue: this.name,\n            status: job.status,\n            data: jobData.length > 4000 ? '<! job data too large to display !>' : job.data.data,\n          },\n        }));\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    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 -= 1;\n    this.queued -= 1;\n\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\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 -= 1;\n    this.log.error(error);\n    setImmediate(this._queueTick);\n  };\n\n  /**\n   *\n   * @returns {*}\n   * @private\n   */\n  _queueTick = () => {\n    if (this.paused || !this.options.enabled) {\n      return undefined;\n    }\n\n    this.queued += 1;\n\n    return this._getNextJob((err, job) => {\n      if (err) return this._onLocalTickError(err);\n      this.running += 1;\n\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    });\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;\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();\n    this._queueTick();\n  }\n\n  /**\n   *\n   * @returns {*}\n   */\n  checkStalledJobs() {\n    this.log.verbose(`${this.name}: checkStalledJobs`);\n    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        if (!this.options.enabled || this.paused) return;\n        setTimeout(::this.checkStalledJobs, this.options.stallInterval);\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"]}

@@ -38,5 +38,5 @@ "use strict";

--[[
key 1 -> rab:job:name:jobs
key 2 -> rab:job:name:waiting
key 3 -> rab:job:name:id (job ID counter)
key 1 -> rdb:job:name:jobs
key 2 -> rdb:job:name:waiting
key 3 -> rdb:job:name:id (job ID counter)
arg 1 -> job data

@@ -74,6 +74,6 @@ arg 2 -> should be unique?

--[[
key 1 -> rab:job:name:stallTime
key 2 -> rab:job:name:stalling
key 3 -> rab:job:name:waiting
key 4 -> rab:job:name:active
key 1 -> rdb:job:queue:stallTime
key 2 -> rdb:job:queue:stalling
key 3 -> rdb:job:queue:waiting
key 4 -> rdb:job:queue:active
arg 1 -> ms timestamp ("now")

@@ -84,5 +84,5 @@ arg 2 -> ms stallInterval

workers are responsible for removing their jobId from the stalling set every stallInterval ms
if a jobId is not removed from the stalling set within a stallInterval window,
we assume the job has stalled and should be reset (moved from active back to waiting)
workers remove the jobs from the stalling set every 'stallInterval' ms
if a job isn't removed from the stall set within the stallInterval time period then
we assume that the job has stalled and we'll reset it (moved from active back to waiting)
--]]

@@ -94,7 +94,6 @@

if now < stallTime then
-- hasn't been long enough (stallInterval) since last check
return 0
end
-- reset any stalling jobs by moving from active to waiting
-- move stalled jobs from active to waiting set
local stalling = redis.call("smembers", KEYS[2])

@@ -109,3 +108,3 @@ if #stalling > 0 then

-- copy currently active jobs into stalling set
-- copy active jobs into stalling set
local actives = redis.call("lrange", KEYS[4], 0, -1)

@@ -126,8 +125,8 @@ if #actives > 0 then

--[[
key 1 -> rab:job:test:succeeded
key 2 -> rab:job:test:failed
key 3 -> rab:job:test:waiting
key 4 -> rab:job:test:active
key 5 -> rab:job:test:stalling
key 6 -> rab:job:test:jobs
key 1 -> rdb:job:queue:succeeded
key 2 -> rdb:job:queue:failed
key 3 -> rdb:job:queue:waiting
key 4 -> rdb:job:queue:active
key 5 -> rdb:job:queue:stalling
key 6 -> rdb:job:queue:jobs
arg 1 -> jobId

@@ -264,2 +263,2 @@ ]]

};
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/scripts.js"],"names":[],"mappings":";;;;;AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;kBA0Be;;AAEb,UAAQ;AACN,UAAM,CADA;AAEN,SAAM;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAFA,GAFK;;AAqCb,oBAAkB;AAChB,UAAM,CADU;AAEhB,SAAM;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAFU,GArCL;;AAqFb,aAAW;AACT,UAAM,CADG;AAET,SAAM;;;;;;;;;;;;;;;;;;;;;;;AAFG,GArFE;;AAgHb,YAAU;AACR,UAAM,CADE;AAER,SAAM;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAFE,GAhHG;;AAsJb,aAAW;AACT,UAAM,CADG;AAET,SAAM;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAFG,GAtJE;;AAyLb,kBAAgB;AACd,UAAM,CADQ;AAEd,SAAM;;;;;;;;;;;;;;;;;;;;AAFQ,GAzLH;;AAiNb,gBAAc;AACZ,UAAM,CADM;AAEZ,SAAM;;;;;;;;;;;;;AAFM;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"]}
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/scripts.js"],"names":["addJob","keys","lua","checkStalledJobs","removeJob","throttle","pThrottle","throttleNoIncr","throttleDecr"],"mappings":";;;;;AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;kBA0Be;;AAEbA,UAAQ;AACNC,UAAM,CADA;AAENC,SAAM;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAFA,GAFK;;AAqCbC,oBAAkB;AAChBF,UAAM,CADU;AAEhBC,SAAM;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAFU,GArCL;;AAoFbE,aAAW;AACTH,UAAM,CADG;AAETC,SAAM;;;;;;;;;;;;;;;;;;;;;;;AAFG,GApFE;;AA+GbG,YAAU;AACRJ,UAAM,CADE;AAERC,SAAM;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAFE,GA/GG;;AAqJbI,aAAW;AACTL,UAAM,CADG;AAETC,SAAM;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAFG,GArJE;;AAwLbK,kBAAgB;AACdN,UAAM,CADQ;AAEdC,SAAM;;;;;;;;;;;;;;;;;;;;AAFQ,GAxLH;;AAgNbM,gBAAc;AACZP,UAAM,CADM;AAEZC,SAAM;;;;;;;;;;;;;AAFM;AAhND,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 -> rdb:job:name:jobs\n        key 2 -> rdb:job:name:waiting\n        key 3 -> rdb: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 -> rdb:job:queue:stallTime\n        key 2 -> rdb:job:queue:stalling\n        key 3 -> rdb:job:queue:waiting\n        key 4 -> rdb:job:queue:active\n        arg 1 -> ms timestamp (\"now\")\n        arg 2 -> ms stallInterval\n\n        returns {resetJobId1, resetJobId2, ...}\n\n        workers remove the jobs from the stalling set every 'stallInterval' ms\n        if a job isn't removed from the stall set within the stallInterval time period then\n        we assume that the job has stalled and we'll reset it (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          return 0\n        end\n\n        -- move stalled jobs from active to waiting set\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 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 -> rdb:job:queue:succeeded\n        key 2 -> rdb:job:queue:failed\n        key 3 -> rdb:job:queue:waiting\n        key 4 -> rdb:job:queue:active\n        key 5 -> rdb:job:queue:stalling\n        key 6 -> rdb:job:queue: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.2.17",
"description": "High performance, robust and flexible queue/worker system powered by redis.",
"version": "1.2.18",
"main": "lib/index.js",

@@ -17,2 +17,8 @@ "author": "Mike Diarmid",

"scheduler",
"job",
"jobs",
"queue",
"queues",
"worker",
"workers",
"scheduling",

@@ -44,3 +50,3 @@ "waterline",

"devDependencies": {
"babel-cli": "^6.11.4",
"babel-cli": "^6.14.0",
"babel-eslint": "^6.1.2",

@@ -50,27 +56,30 @@ "babel-istanbul": "^0.11.0",

"babel-plugin-transform-class-properties": "^6.11.5",
"babel-preset-es2015": "^6.6.0",
"babel-preset-es2015": "^6.14.0",
"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.2.0",
"babel-preset-es2015-node6": "^0.3.0",
"babel-preset-stage-0": "^6.5.0",
"bee-queue": "^0.3.0",
"benchmark": "^2.1.1",
"bull": "^1.0.0",
"chai": "^3.5.0",
"coveralls": "^2.11.11",
"eslint": "^3.1.1",
"eslint-config-airbnb": "^9.0.1",
"eslint-plugin-import": "^1.11.1",
"eslint-plugin-jsx-a11y": "^2.0.1",
"eslint-plugin-react": "^5.2.2",
"coveralls": "^2.11.14",
"eslint": "^3.6.0",
"eslint-config-airbnb": "^11.2.0",
"eslint-plugin-import": "^1.16.0",
"eslint-plugin-jsx-a11y": "^2.2.2",
"eslint-plugin-react": "^6.3.0",
"isparta": "^4.0.0",
"istanbul": "^1.1.0-alpha.1",
"mocha": "^2.4.5",
"redibox": "^2.1.0",
"sinon": "^1.17.4"
"kue": "^0.11.1",
"mocha": "^3.0.2",
"redibox": "^2.3.0",
"sinon": "^1.17.6"
},
"dependencies": {
"bluebird": "^3.4.1",
"bluebird": "^3.4.6",
"cuid": "^1.3.8",
"eventemitter3": "^1.2.0"
"eventemitter3": "^2.0.0"
}
}

@@ -15,7 +15,5 @@ ## RediBox Job

```
```javascript
RediBox.hooks.job.create('my-queue', {
runs: function(job) {
console.log('The value of foo is ' + job.data.foo);
},
runs: 'sails.hooks.myhook.runJobFoo', // dot notated path to a globally accessible function
data: {

@@ -28,5 +26,3 @@ foo: 'bar',

Job.create('my-queue', {
runs: function(job) {
console.log('The value of foo is ' + job.data.foo);
},
runs: 'sails.hooks.myhook.runJobBar', // dot notated path to a globally accessible function
data: {

@@ -33,0 +29,0 @@ foo: 'bar',

global.HOOK_NAME = 'job';
var Redibox = require('redibox').default;
var UserHook = require('./lib/hook').default;
const Redibox = require('redibox').default;
const UserHook = require('./lib/hook').default;
console.dir(UserHook)
const config = {

@@ -75,3 +73,2 @@ hooks: {},

console.log('FOO BAR');
console.log(foborh);
return Promise.resolve({ hello: 123 });

@@ -82,87 +79,19 @@ };

console.log('FOO BAR END');
// assert.equal(this.data.hello, 123);
return Promise.resolve();
};
const tester = new Promise((resolve) => {
return resolve();
});
global.RediBox = new Redibox(config, () => {
global.Hook = RediBox.hooks[global.HOOK_NAME];
global.RediBoxCluster = new Redibox(clusterConfig, () => {
global.HookCluster = global.RediBoxCluster.hooks[global.HOOK_NAME];
tester.then(() => {
Hook
.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'bar',
},
})
.onSuccess((result) => {
console.log('s'.repeat(80));
console.dir(result);
console.log('s'.repeat(80));
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
})
.onFailure((result) => {
console.log('e'.repeat(80));
console.error(result.error);
console.log('e'.repeat(80));
});
console.log('HOOK READY');
Hook
.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'bar',
},
})
.onSuccess((result) => {
console.log('s'.repeat(80));
console.dir(result);
console.log('s'.repeat(80));
Hook.create('test2', {

@@ -174,586 +103,10 @@ runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],

});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
Hook.create('test2', {
runs: ['fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBar', 'fooBarEnd'],
data: {
foo: 'barz',
},
});
})
.onFailure((result) => {
console.log('e'.repeat(80));
console.error(result.error);
console.log('e'.repeat(80));
});
});
});
clusterConfig.hooks[global.HOOK_NAME] = UserHook;

Sorry, the diff of this file is not supported yet

Sorry, the diff of this file is not supported yet

SocketSocket SOC 2 Logo

Product

  • Package Alerts
  • Integrations
  • Docs
  • Pricing
  • FAQ
  • Roadmap
  • Changelog

Packages

npm

Stay in touch

Get open source security insights delivered straight into your inbox.


  • Terms
  • Privacy
  • Security

Made with ⚡️ by Socket Inc