Files
elastiq/test/src/worker.js
2016-05-03 09:48:15 -07:00

351 lines
11 KiB
JavaScript

import expect from 'expect.js';
import sinon from 'sinon';
import moment from 'moment';
import { noop, random } from 'lodash';
import Worker from '../../lib/worker';
import elasticsearchMock from '../fixtures/elasticsearch';
import { JOB_STATUS_PROCESSING, JOB_STATUS_COMPLETED, JOB_STATUS_FAILED } from '../../lib/helpers/constants';
const anchor = '2016-04-02T01:02:03.456'; // saturday
const defaults = {
timeout: 10000,
interval: 1500,
size: 10,
};
describe('Worker class', function () {
let anchorMoment;
let clock;
let client;
let mockQueue;
beforeEach(function () {
anchorMoment = moment(anchor);
clock = sinon.useFakeTimers(anchorMoment.valueOf());
client = new elasticsearchMock.Client();
mockQueue = {
client: client
};
});
afterEach(function () {
clock.restore();
});
describe('invalid construction', function () {
it('should throw without a type', function () {
const init = () => new Worker(mockQueue);
expect(init).to.throwException(/type.+string/i);
});
it('should throw without an invalid type', function () {
const init = () => new Worker(mockQueue, { string: false });
expect(init).to.throwException(/type.+string/i);
});
it('should throw without a worker', function () {
const init = () => new Worker(mockQueue, 'test');
expect(init).to.throwException(/worker.+function/i);
});
it('should throw with an invalid worker', function () {
const init = () => new Worker(mockQueue, 'test', { function: false });
expect(init).to.throwException(/worker.+function/i);
});
});
describe('construction', function () {
it('should have a unique ID', function () {
var worker = new Worker(mockQueue, 'test', noop);
expect(worker.id).to.be.a('string');
var worker2 = new Worker(mockQueue, 'test', noop);
expect(worker2.id).to.be.a('string');
expect(worker.id).to.not.equal(worker2.id);
});
});
describe('searching for jobs', function () {
it('should start polling for jobs after interval', function () {
const searchSpy = sinon.spy(mockQueue.client, 'search');
new Worker(mockQueue, 'test', noop);
sinon.assert.notCalled(searchSpy);
clock.tick(defaults.interval);
sinon.assert.calledOnce(searchSpy);
});
it('should use interval option to control polling', function () {
const interval = 567;
const searchSpy = sinon.spy(mockQueue.client, 'search');
new Worker(mockQueue, 'test', noop, { interval });
sinon.assert.notCalled(searchSpy);
clock.tick(interval);
sinon.assert.calledOnce(searchSpy);
});
it('should use default size', function () {
const searchSpy = sinon.spy(mockQueue.client, 'search');
new Worker(mockQueue, 'test', noop);
clock.tick(defaults.interval);
const body = searchSpy.firstCall.args[0].body;
expect(body).to.have.property('size', defaults.size);
});
it('should observe the size option', function () {
const size = 25;
const searchSpy = sinon.spy(mockQueue.client, 'search');
new Worker(mockQueue, 'test', noop, { size });
clock.tick(defaults.interval);
const body = searchSpy.firstCall.args[0].body;
expect(body).to.have.property('size', size);
});
});
describe('claiming a job', function () {
let params;
let job;
let worker;
let updateSpy;
beforeEach(function () {
params = {
index: 'myIndex',
type: 'test',
id: 12345,
version: 3
};
job = mockQueue.client.get(params);
worker = new Worker(mockQueue, 'test', noop);
updateSpy = sinon.spy(mockQueue.client, 'update');
});
it('should use version on update', function () {
worker._claimJob(job);
const query = updateSpy.firstCall.args[0];
expect(query).to.have.property('index', job._index);
expect(query).to.have.property('type', job._type);
expect(query).to.have.property('id', job._id);
expect(query).to.have.property('version', job._version);
});
it('should increment the job attempts', function () {
worker._claimJob(job);
const doc = updateSpy.firstCall.args[0].body.doc;
expect(doc).to.have.property('attempts', job._source.attempts + 1);
});
it('should update the job status', function () {
worker._claimJob(job);
const doc = updateSpy.firstCall.args[0].body.doc;
expect(doc).to.have.property('status', JOB_STATUS_PROCESSING);
});
it('should set job expiration time', function () {
worker._claimJob(job);
const doc = updateSpy.firstCall.args[0].body.doc;
const expiration = anchorMoment.add(defaults.timeout).toISOString();
expect(doc).to.have.property('process_expiration', expiration);
});
it('should fail job if max_attempts are hit', function () {
const failSpy = sinon.spy(worker, '_failJob');
job._source.attempts = job._source.max_attempts;
worker._claimJob(job);
sinon.assert.calledOnce(failSpy);
});
it('should append error message if no existing content', function () {
const failSpy = sinon.spy(worker, '_failJob');
job._source.attempts = job._source.max_attempts;
expect(job._source.output).to.be(undefined);
worker._claimJob(job);
const msg = failSpy.firstCall.args[1];
expect(msg).to.contain('Max attempts reached');
expect(msg).to.contain(job._source.max_attempts);
});
it('should not append message if existing output', function () {
const failSpy = sinon.spy(worker, '_failJob');
job._source.attempts = job._source.max_attempts;
job._source.output = 'i have some output';
worker._claimJob(job);
const msg = failSpy.firstCall.args[1];
expect(msg).to.equal(false);
});
it('should swallow version mismatch errors', function () {
mockQueue.client.update.restore();
sinon.stub(mockQueue.client, 'update').returns(Promise.reject({ statusCode: 409 }));
return worker._claimJob(job);
});
});
describe('failing a job', function () {
let job;
let worker;
let updateSpy;
beforeEach(function () {
job = mockQueue.client.get();
worker = new Worker(mockQueue, 'test', noop);
updateSpy = sinon.spy(mockQueue.client, 'update');
});
it('should use version on update', function () {
worker._failJob(job);
const query = updateSpy.firstCall.args[0];
expect(query).to.have.property('index', job._index);
expect(query).to.have.property('type', job._type);
expect(query).to.have.property('id', job._id);
expect(query).to.have.property('version', job._version);
});
it('should set status to failed', function () {
worker._failJob(job);
const doc = updateSpy.firstCall.args[0].body.doc;
expect(doc).to.have.property('status', JOB_STATUS_FAILED);
});
it('should append error message if supplied', function () {
const msg = 'test message';
worker._failJob(job, msg);
const doc = updateSpy.firstCall.args[0].body.doc;
expect(doc).to.have.property('output');
expect(doc.output).to.have.property('content', msg);
});
it('should swallow version mismatch errors', function () {
mockQueue.client.update.restore();
sinon.stub(mockQueue.client, 'update').returns(Promise.reject({ statusCode: 409 }));
return worker._failJob(job);
});
it('should set completed time and status to failed', function () {
const startTime = moment().valueOf();
const msg = 'test message';
clock.tick(100);
worker._failJob(job, msg);
const doc = updateSpy.firstCall.args[0].body.doc;
expect(doc).to.have.property('output');
expect(doc).to.have.property('status', JOB_STATUS_FAILED);
expect(doc).to.have.property('completed_at');
const completedTimestamp = moment(doc.completed_at).valueOf();
expect(completedTimestamp).to.be.greaterThan(startTime);
});
});
describe('performing a job', function () {
let job;
let payload;
let updateSpy;
beforeEach(function () {
clock.restore();
payload = {
value: random(0, 100, true)
};
job = mockQueue.client.get({}, { payload });
updateSpy = sinon.spy(mockQueue.client, 'update');
});
it('should call the workerFn with the payload', function (done) {
const workerFn = function (jobPayload, cb) {
expect(jobPayload).to.eql(payload);
cb();
};
const worker = new Worker(mockQueue, 'test', workerFn);
worker._performJob(job)
.then(() => done());
});
it('should update the job with the workerFn output', function (done) {
const workerFn = function (jobPayload, cb) {
expect(jobPayload).to.eql(payload);
cb(null, payload);
};
const worker = new Worker(mockQueue, 'test', workerFn);
worker._performJob(job)
.then(() => {
try {
sinon.assert.calledOnce(updateSpy);
const query = updateSpy.firstCall.args[0];
expect(query).to.have.property('index', job._index);
expect(query).to.have.property('type', job._type);
expect(query).to.have.property('id', job._id);
expect(query).to.have.property('version', job._version);
expect(query.body.doc).to.have.property('output');
expect(query.body.doc.output).to.have.property('content_type', false);
expect(query.body.doc.output).to.have.property('content', payload);
done();
} catch (err) {
done(err);
}
});
});
it('should update the job status and completed time', function (done) {
const startTime = moment().valueOf();
const workerFn = function (jobPayload, cb) {
expect(jobPayload).to.eql(payload);
cb(null, payload);
};
const worker = new Worker(mockQueue, 'test', workerFn);
worker._performJob(job)
.then(() => {
try {
sinon.assert.calledOnce(updateSpy);
const doc = updateSpy.firstCall.args[0].body.doc;
expect(doc).to.have.property('status', JOB_STATUS_COMPLETED);
expect(doc).to.have.property('completed_at');
const completedTimestamp = moment(doc.completed_at).valueOf();
expect(completedTimestamp).to.be.greaterThan(startTime);
done();
} catch (err) {
done(err);
}
});
});
it('should reject on job errors', function (done) {
const workerFn = function (jobPayload, cb) {
cb(new Error('test error'));
};
const worker = new Worker(mockQueue, 'test', workerFn);
worker._performJob(job)
.then(() => done(new Error('should not resolve')))
.catch((e) => {
expect(e.message).to.equal('test error');
done();
});
});
it('should append error output to job', function (done) {
const workerFn = function (jobPayload, cb) {
cb(new Error('test error'));
};
const worker = new Worker(mockQueue, 'test', workerFn);
const failStub = sinon.stub(worker, '_failJob');
worker._performJob(job)
.then(() => done(new Error('should not resolve')))
.catch(() => {
try {
sinon.assert.calledOnce(failStub);
sinon.assert.calledWith(failStub, job, 'Error: test error');
done();
} catch (e) {
done(e);
}
});
});
});
});