qdone 2.0.17-alpha → 2.0.19-alpha
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/commonjs/src/consumer.js +8 -5
- package/commonjs/src/defaults.js +3 -0
- package/commonjs/src/idleQueues.js +21 -9
- package/commonjs/src/scheduler/jobExecutor.js +3 -3
- package/package.json +1 -1
- package/src/consumer.js +7 -4
- package/src/defaults.js +3 -0
- package/src/idleQueues.js +18 -9
- package/src/scheduler/jobExecutor.js +1 -1
- package/npm-shrinkwrap.json +0 -16001
package/commonjs/src/consumer.js
CHANGED
|
@@ -118,7 +118,7 @@ exports.getMessages = getMessages;
|
|
|
118
118
|
//
|
|
119
119
|
function processMessages(queues, callback, options) {
|
|
120
120
|
return __awaiter(this, void 0, void 0, function () {
|
|
121
|
-
var opt, lastLatency, systemMonitor, jobExecutor, queueManager, delayTimeout, delay, activeQrls, maxReturnCount, listen, allowedJobs, maxLatency, latency, latencyFactor, freememFactor, targetJobs, jobsLeft, _i, _a, _b, qname, qrl, maxMessages;
|
|
121
|
+
var opt, lastLatency, systemMonitor, jobExecutor, queueManager, delayTimeout, delay, activeQrls, maxReturnCount, listen, allowedJobs, maxLatency, freeMemory, totalMemory, memoryThreshold, latency, latencyFactor, freememFactor, targetJobs, jobsLeft, _i, _a, _b, qname, qrl, maxMessages;
|
|
122
122
|
var _this = this;
|
|
123
123
|
return __generator(this, function (_c) {
|
|
124
124
|
switch (_c.label) {
|
|
@@ -209,16 +209,19 @@ function processMessages(queues, callback, options) {
|
|
|
209
209
|
if (!!shutdownRequested) return [3 /*break*/, 3];
|
|
210
210
|
allowedJobs = Math.max(0, opt.maxConcurrentJobs - jobExecutor.activeJobCount() - maxReturnCount);
|
|
211
211
|
maxLatency = 100;
|
|
212
|
+
freeMemory = (0, os_1.freemem)();
|
|
213
|
+
totalMemory = (0, os_1.totalmem)();
|
|
214
|
+
memoryThreshold = totalMemory * opt.maxMemoryPercent / 100;
|
|
212
215
|
latency = systemMonitor.getLatency() || 10;
|
|
213
216
|
latencyFactor = 1 - Math.abs(Math.min(latency / maxLatency, 1)) // 0 if latency is at max, 1 if latency 0
|
|
214
217
|
;
|
|
215
|
-
freememFactor = Math.min(1, Math.max(0,
|
|
218
|
+
freememFactor = Math.min(1, Math.max(0, freeMemory / memoryThreshold));
|
|
216
219
|
targetJobs = Math.round(allowedJobs * latencyFactor * freememFactor);
|
|
217
220
|
jobsLeft = targetJobs;
|
|
218
|
-
debug({ jobCount: jobExecutor.activeJobCount(), maxReturnCount: maxReturnCount, allowedJobs: allowedJobs, maxLatency: maxLatency, latency: latency, latencyFactor: latencyFactor, freememFactor: freememFactor, targetJobs: targetJobs, activeQrls: activeQrls });
|
|
221
|
+
debug({ jobCount: jobExecutor.activeJobCount(), freeMemory: freeMemory, totalMemory: totalMemory, memoryThreshold: memoryThreshold, maxReturnCount: maxReturnCount, allowedJobs: allowedJobs, maxLatency: maxLatency, latency: latency, latencyFactor: latencyFactor, freememFactor: freememFactor, targetJobs: targetJobs, activeQrls: activeQrls });
|
|
219
222
|
for (_i = 0, _a = queueManager.getPairs(); _i < _a.length; _i++) {
|
|
220
223
|
_b = _a[_i], qname = _b.qname, qrl = _b.qrl;
|
|
221
|
-
debug({ evaluating: { qname
|
|
224
|
+
// debug({ evaluating: { qname, qrl, jobsLeft, activeQrlsHasQrl: activeQrls.has(qrl) } })
|
|
222
225
|
if (jobsLeft <= 0 || activeQrls.has(qrl))
|
|
223
226
|
continue;
|
|
224
227
|
maxMessages = Math.min(10, jobsLeft);
|
|
@@ -227,7 +230,7 @@ function processMessages(queues, callback, options) {
|
|
|
227
230
|
if (opt.verbose) {
|
|
228
231
|
console.error(chalk_1.default.blue('Listening on: '), qname);
|
|
229
232
|
}
|
|
230
|
-
debug({ listenedTo: { qname
|
|
233
|
+
// debug({ listenedTo: { qname, maxMessages, jobsLeft } })
|
|
231
234
|
}
|
|
232
235
|
return [4 /*yield*/, delay(1000)];
|
|
233
236
|
case 2:
|
package/commonjs/src/defaults.js
CHANGED
|
@@ -37,6 +37,7 @@ exports.defaults = Object.freeze({
|
|
|
37
37
|
archive: false,
|
|
38
38
|
activeOnly: false,
|
|
39
39
|
maxConcurrentJobs: 100,
|
|
40
|
+
maxMemoryPercent: 70,
|
|
40
41
|
// Idle Queues
|
|
41
42
|
idleFor: 60,
|
|
42
43
|
delete: false,
|
|
@@ -95,6 +96,7 @@ function getOptionsWithDefaults(options) {
|
|
|
95
96
|
activeOnly: options.activeOnly || options['active-only'] || exports.defaults.activeOnly,
|
|
96
97
|
includeFailed: options.includeFailed || options['include-failed'] || exports.defaults.includeFailed,
|
|
97
98
|
maxConcurrentJobs: options.maxConcurrentJobs || exports.defaults.maxConcurrentJobs,
|
|
99
|
+
maxMemoryPercent: options.maxMemoryPercent || exports.defaults.maxMemoryPercent,
|
|
98
100
|
// Idle Queues
|
|
99
101
|
idleFor: options.idleFor || options['idle-for'] || exports.defaults.idleFor,
|
|
100
102
|
delete: options.delete || exports.defaults.delete,
|
|
@@ -112,6 +114,7 @@ function getOptionsWithDefaults(options) {
|
|
|
112
114
|
opt.waitTime = validateInteger(opt, 'waitTime');
|
|
113
115
|
opt.killAfter = validateInteger(opt, 'killAfter');
|
|
114
116
|
opt.maxConcurrentJobs = validateInteger(opt, 'maxConcurrentJobs');
|
|
117
|
+
opt.maxMemoryPercent = validateInteger(opt, 'maxMemoryPercent');
|
|
115
118
|
opt.idleFor = validateInteger(opt, 'idleFor');
|
|
116
119
|
return opt;
|
|
117
120
|
}
|
|
@@ -85,21 +85,33 @@ var metricNames = [
|
|
|
85
85
|
*/
|
|
86
86
|
function _cheapIdleCheck(qname, qrl, opt) {
|
|
87
87
|
return __awaiter(this, void 0, void 0, function () {
|
|
88
|
-
var client, cmd, data,
|
|
88
|
+
var client, cmd, data, result_1, e_1;
|
|
89
89
|
return __generator(this, function (_a) {
|
|
90
90
|
switch (_a.label) {
|
|
91
91
|
case 0:
|
|
92
|
+
_a.trys.push([0, 2, , 3]);
|
|
92
93
|
client = (0, sqs_js_1.getSQSClient)();
|
|
93
94
|
cmd = new client_sqs_1.GetQueueAttributesCommand({ AttributeNames: exports.attributeNames, QueueUrl: qrl });
|
|
94
95
|
return [4 /*yield*/, client.send(cmd)];
|
|
95
96
|
case 1:
|
|
96
97
|
data = _a.sent();
|
|
97
98
|
debug('data', data);
|
|
98
|
-
|
|
99
|
-
|
|
99
|
+
result_1 = data.Attributes;
|
|
100
|
+
result_1.queue = qname.slice(opt.prefix.length);
|
|
100
101
|
// We are idle if all the messages attributes are zero
|
|
101
|
-
|
|
102
|
-
return [2 /*return*/, { result:
|
|
102
|
+
result_1.idle = exports.attributeNames.filter(function (k) { return result_1[k] === '0'; }).length === exports.attributeNames.length;
|
|
103
|
+
return [2 /*return*/, { result: result_1, SQS: 1 }];
|
|
104
|
+
case 2:
|
|
105
|
+
e_1 = _a.sent();
|
|
106
|
+
if (e_1 instanceof client_sqs_1.QueueDoesNotExist) {
|
|
107
|
+
// Count deleted queues as idle
|
|
108
|
+
return [2 /*return*/, { idle: true, SQS: 1 }];
|
|
109
|
+
}
|
|
110
|
+
else {
|
|
111
|
+
throw e_1;
|
|
112
|
+
}
|
|
113
|
+
return [3 /*break*/, 3];
|
|
114
|
+
case 3: return [2 /*return*/];
|
|
103
115
|
}
|
|
104
116
|
});
|
|
105
117
|
});
|
|
@@ -320,7 +332,7 @@ exports.processQueue = processQueue;
|
|
|
320
332
|
*/
|
|
321
333
|
function processQueuePair(qname, qrl, opt) {
|
|
322
334
|
return __awaiter(this, void 0, void 0, function () {
|
|
323
|
-
var isFifo, normalizeOptions, fqname, fqrl, result, active, fresult, idleCheckResult, factive, _a, dresult, dfresult,
|
|
335
|
+
var isFifo, normalizeOptions, fqname, fqrl, result, active, fresult, idleCheckResult, factive, _a, dresult, dfresult, e_2, deleteResult, resultIncludingDelete;
|
|
324
336
|
return __generator(this, function (_b) {
|
|
325
337
|
switch (_b.label) {
|
|
326
338
|
case 0:
|
|
@@ -385,11 +397,11 @@ function processQueuePair(qname, qrl, opt) {
|
|
|
385
397
|
}
|
|
386
398
|
})];
|
|
387
399
|
case 5:
|
|
388
|
-
|
|
400
|
+
e_2 = _b.sent();
|
|
389
401
|
// Handle the case where the fail queue has been deleted or was never
|
|
390
402
|
// created for some reason
|
|
391
|
-
if (!(
|
|
392
|
-
throw
|
|
403
|
+
if (!(e_2 instanceof client_sqs_1.QueueDoesNotExist))
|
|
404
|
+
throw e_2;
|
|
393
405
|
// Fail queue doesn't exist if we get here
|
|
394
406
|
if (opt.verbose)
|
|
395
407
|
console.error(chalk_1.default.blue('Queue ') + fqname.slice(opt.prefix.length) + chalk_1.default.blue(' does not exist.'));
|
|
@@ -269,7 +269,7 @@ var JobExecutor = /** @class */ (function () {
|
|
|
269
269
|
};
|
|
270
270
|
JobExecutor.prototype.executeJob = function (message, callback, qname, qrl) {
|
|
271
271
|
return __awaiter(this, void 0, void 0, function () {
|
|
272
|
-
var payload, visibilityTimeout, job, oldJob, e,
|
|
272
|
+
var payload, visibilityTimeout, job, oldJob, e, queue, result, err_1;
|
|
273
273
|
return __generator(this, function (_a) {
|
|
274
274
|
switch (_a.label) {
|
|
275
275
|
case 0:
|
|
@@ -316,8 +316,8 @@ var JobExecutor = /** @class */ (function () {
|
|
|
316
316
|
_a.label = 1;
|
|
317
317
|
case 1:
|
|
318
318
|
_a.trys.push([1, 3, , 4]);
|
|
319
|
-
|
|
320
|
-
return [4 /*yield*/,
|
|
319
|
+
queue = qname.slice(this.opt.prefix.length);
|
|
320
|
+
return [4 /*yield*/, callback(queue, payload)];
|
|
321
321
|
case 2:
|
|
322
322
|
result = _a.sent();
|
|
323
323
|
debug('executeJob callback finished', { payload: payload, result: result });
|
package/package.json
CHANGED
package/src/consumer.js
CHANGED
|
@@ -116,14 +116,17 @@ export async function processMessages (queues, callback, options) {
|
|
|
116
116
|
// Figure out how we are running
|
|
117
117
|
const allowedJobs = Math.max(0, opt.maxConcurrentJobs - jobExecutor.activeJobCount() - maxReturnCount)
|
|
118
118
|
const maxLatency = 100
|
|
119
|
+
const freeMemory = freemem()
|
|
120
|
+
const totalMemory = totalmem()
|
|
121
|
+
const memoryThreshold = totalMemory * opt.maxMemoryPercent / 100
|
|
119
122
|
const latency = systemMonitor.getLatency() || 10
|
|
120
123
|
const latencyFactor = 1 - Math.abs(Math.min(latency / maxLatency, 1)) // 0 if latency is at max, 1 if latency 0
|
|
121
|
-
const freememFactor = Math.min(1, Math.max(0,
|
|
124
|
+
const freememFactor = Math.min(1, Math.max(0, freeMemory / memoryThreshold))
|
|
122
125
|
const targetJobs = Math.round(allowedJobs * latencyFactor * freememFactor)
|
|
123
126
|
let jobsLeft = targetJobs
|
|
124
|
-
debug({ jobCount: jobExecutor.activeJobCount(), maxReturnCount, allowedJobs, maxLatency, latency, latencyFactor, freememFactor, targetJobs, activeQrls })
|
|
127
|
+
debug({ jobCount: jobExecutor.activeJobCount(), freeMemory, totalMemory, memoryThreshold, maxReturnCount, allowedJobs, maxLatency, latency, latencyFactor, freememFactor, targetJobs, activeQrls })
|
|
125
128
|
for (const { qname, qrl } of queueManager.getPairs()) {
|
|
126
|
-
debug({ evaluating: { qname, qrl, jobsLeft, activeQrlsHasQrl: activeQrls.has(qrl) } })
|
|
129
|
+
// debug({ evaluating: { qname, qrl, jobsLeft, activeQrlsHasQrl: activeQrls.has(qrl) } })
|
|
127
130
|
if (jobsLeft <= 0 || activeQrls.has(qrl)) continue
|
|
128
131
|
const maxMessages = Math.min(10, jobsLeft)
|
|
129
132
|
listen(qname, qrl, maxMessages)
|
|
@@ -131,7 +134,7 @@ export async function processMessages (queues, callback, options) {
|
|
|
131
134
|
if (opt.verbose) {
|
|
132
135
|
console.error(chalk.blue('Listening on: '), qname)
|
|
133
136
|
}
|
|
134
|
-
debug({ listenedTo: { qname, maxMessages, jobsLeft } })
|
|
137
|
+
// debug({ listenedTo: { qname, maxMessages, jobsLeft } })
|
|
135
138
|
}
|
|
136
139
|
await delay(1000)
|
|
137
140
|
}
|
package/src/defaults.js
CHANGED
|
@@ -37,6 +37,7 @@ export const defaults = Object.freeze({
|
|
|
37
37
|
archive: false,
|
|
38
38
|
activeOnly: false,
|
|
39
39
|
maxConcurrentJobs: 100,
|
|
40
|
+
maxMemoryPercent: 70,
|
|
40
41
|
|
|
41
42
|
// Idle Queues
|
|
42
43
|
idleFor: 60,
|
|
@@ -101,6 +102,7 @@ export function getOptionsWithDefaults (options) {
|
|
|
101
102
|
activeOnly: options.activeOnly || options['active-only'] || defaults.activeOnly,
|
|
102
103
|
includeFailed: options.includeFailed || options['include-failed'] || defaults.includeFailed,
|
|
103
104
|
maxConcurrentJobs: options.maxConcurrentJobs || defaults.maxConcurrentJobs,
|
|
105
|
+
maxMemoryPercent: options.maxMemoryPercent || defaults.maxMemoryPercent,
|
|
104
106
|
|
|
105
107
|
// Idle Queues
|
|
106
108
|
idleFor: options.idleFor || options['idle-for'] || defaults.idleFor,
|
|
@@ -121,6 +123,7 @@ export function getOptionsWithDefaults (options) {
|
|
|
121
123
|
opt.waitTime = validateInteger(opt, 'waitTime')
|
|
122
124
|
opt.killAfter = validateInteger(opt, 'killAfter')
|
|
123
125
|
opt.maxConcurrentJobs = validateInteger(opt, 'maxConcurrentJobs')
|
|
126
|
+
opt.maxMemoryPercent = validateInteger(opt, 'maxMemoryPercent')
|
|
124
127
|
opt.idleFor = validateInteger(opt, 'idleFor')
|
|
125
128
|
|
|
126
129
|
return opt
|
package/src/idleQueues.js
CHANGED
|
@@ -37,15 +37,24 @@ const metricNames = [
|
|
|
37
37
|
* Actual SQS call, used in conjunction with cache.
|
|
38
38
|
*/
|
|
39
39
|
export async function _cheapIdleCheck (qname, qrl, opt) {
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
40
|
+
try {
|
|
41
|
+
const client = getSQSClient()
|
|
42
|
+
const cmd = new GetQueueAttributesCommand({ AttributeNames: attributeNames, QueueUrl: qrl })
|
|
43
|
+
const data = await client.send(cmd)
|
|
44
|
+
debug('data', data)
|
|
45
|
+
const result = data.Attributes
|
|
46
|
+
result.queue = qname.slice(opt.prefix.length)
|
|
47
|
+
// We are idle if all the messages attributes are zero
|
|
48
|
+
result.idle = attributeNames.filter(k => result[k] === '0').length === attributeNames.length
|
|
49
|
+
return { result, SQS: 1 }
|
|
50
|
+
} catch (e) {
|
|
51
|
+
if (e instanceof QueueDoesNotExist) {
|
|
52
|
+
// Count deleted queues as idle
|
|
53
|
+
return { idle: true, SQS: 1 }
|
|
54
|
+
} else {
|
|
55
|
+
throw e
|
|
56
|
+
}
|
|
57
|
+
}
|
|
49
58
|
}
|
|
50
59
|
|
|
51
60
|
/**
|
|
@@ -242,7 +242,7 @@ export class JobExecutor {
|
|
|
242
242
|
// Execute job
|
|
243
243
|
try {
|
|
244
244
|
const queue = qname.slice(this.opt.prefix.length)
|
|
245
|
-
const result = await
|
|
245
|
+
const result = await callback(queue, payload)
|
|
246
246
|
debug('executeJob callback finished', { payload, result })
|
|
247
247
|
if (this.opt.verbose) {
|
|
248
248
|
console.error(chalk.green('SUCCESS'), message.Body)
|