|
|
|
const setup = require('../../../controller/place');
|
|
|
|
const proxyquire = require('proxyquire').noCallThru();
|
|
|
|
|
|
|
|
module.exports.tests = {};
|
|
|
|
|
|
|
|
module.exports.tests.interface = (test, common) => {
|
|
|
|
test('valid interface', (t) => {
|
|
|
|
t.equal(typeof setup, 'function', 'setup is a function');
|
|
|
|
t.equal(typeof setup(), 'function', 'setup returns a controller');
|
|
|
|
t.end();
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports.tests.success = (test, common) => {
|
|
|
|
test('successful request to search service should set data and meta', (t) => {
|
|
|
|
const config = {
|
|
|
|
indexName: 'indexName value'
|
|
|
|
};
|
|
|
|
const esclient = 'this is the esclient';
|
|
|
|
|
|
|
|
// request timeout messages willl be written here
|
|
|
|
const infoMesssages = [];
|
|
|
|
|
|
|
|
// a controller that validates the esclient and cmd that was passed to the search service
|
|
|
|
const controller = proxyquire('../../../controller/place', {
|
|
|
|
'../service/mget': (esclient, query, callback) => {
|
|
|
|
t.equal(esclient, 'this is the esclient');
|
|
|
|
t.deepEqual(query, [
|
|
|
|
{
|
|
|
|
_index: 'indexName value',
|
|
|
|
_type: 'layer1',
|
|
|
|
_id: 'id1'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
_index: 'indexName value',
|
|
|
|
_type: 'layer2',
|
|
|
|
_id: 'id2'
|
|
|
|
}
|
|
|
|
]);
|
|
|
|
|
|
|
|
const docs = [{}, {}];
|
|
|
|
|
|
|
|
callback(undefined, docs);
|
|
|
|
}
|
|
|
|
})(config, esclient);
|
|
|
|
|
|
|
|
const req = {
|
|
|
|
clean: {
|
|
|
|
ids: [
|
|
|
|
{
|
|
|
|
id: 'id1',
|
|
|
|
layer: 'layer1'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
id: 'id2',
|
|
|
|
layer: 'layer2'
|
|
|
|
}
|
|
|
|
]
|
|
|
|
},
|
|
|
|
errors: [],
|
|
|
|
warnings: []
|
|
|
|
};
|
|
|
|
const res = {};
|
|
|
|
|
|
|
|
const next = () => {
|
|
|
|
t.deepEqual(req.errors, []);
|
|
|
|
t.deepEqual(req.warnings, []);
|
|
|
|
t.deepEquals(res.data, [{}, {}]);
|
|
|
|
t.end();
|
|
|
|
};
|
|
|
|
|
|
|
|
controller(req, res, next);
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports.tests.error_conditions = (test, common) => {
|
|
|
|
test('non-empty req.errors should ', (t) => {
|
|
|
|
const esclient = () => {
|
|
|
|
throw new Error('esclient should not have been called');
|
|
|
|
};
|
|
|
|
const controller = setup( {}, esclient );
|
|
|
|
|
|
|
|
// the existence of `errors` means that a sanitizer detected an error,
|
|
|
|
// so don't call the esclient
|
|
|
|
const req = {
|
|
|
|
errors: ['error']
|
|
|
|
};
|
|
|
|
const res = { };
|
|
|
|
|
|
|
|
t.doesNotThrow(() => {
|
|
|
|
controller(req, res, () => {});
|
|
|
|
});
|
|
|
|
t.end();
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
test('mgetService returning error should add to req.errors and ignore docs', (t) => {
|
|
|
|
const config = {
|
|
|
|
indexName: 'indexName value'
|
|
|
|
};
|
|
|
|
const esclient = 'this is the esclient';
|
|
|
|
|
|
|
|
// request timeout messages willl be written here
|
|
|
|
const infoMesssages = [];
|
|
|
|
|
|
|
|
const nonTimeoutError = {
|
|
|
|
status: 500,
|
|
|
|
displayName: 'InternalServerError',
|
|
|
|
message: 'an internal server error occurred'
|
|
|
|
};
|
|
|
|
|
|
|
|
// a controller that validates the esclient and cmd that was passed to the search service
|
|
|
|
const controller = proxyquire('../../../controller/place', {
|
|
|
|
'../service/mget': (esclient, query, callback) => {
|
|
|
|
const docs = [{}, {}];
|
|
|
|
|
|
|
|
callback(nonTimeoutError, docs);
|
|
|
|
}
|
|
|
|
})(config, esclient);
|
|
|
|
|
|
|
|
const req = {
|
|
|
|
clean: {
|
|
|
|
ids: [
|
|
|
|
{
|
|
|
|
id: 'id1',
|
|
|
|
layers: 'layer1'
|
|
|
|
}
|
|
|
|
]
|
|
|
|
},
|
|
|
|
errors: [],
|
|
|
|
warnings: []
|
|
|
|
};
|
|
|
|
const res = {};
|
|
|
|
|
|
|
|
const next = () => {
|
|
|
|
t.deepEqual(req.errors, [nonTimeoutError.message]);
|
|
|
|
t.deepEqual(req.warnings, []);
|
|
|
|
t.deepEquals(res.data, undefined);
|
|
|
|
t.end();
|
|
|
|
};
|
|
|
|
|
|
|
|
controller(req, res, next);
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports.tests.timeout = function(test, common) {
|
|
|
|
test('default # of request timeout retries should be 3', (t) => {
|
|
|
|
const config = {
|
|
|
|
indexName: 'indexName value'
|
|
|
|
};
|
|
|
|
const esclient = 'this is the esclient';
|
|
|
|
|
|
|
|
let searchServiceCallCount = 0;
|
|
|
|
|
|
|
|
const timeoutError = {
|
|
|
|
status: 408,
|
|
|
|
displayName: 'RequestTimeout',
|
|
|
|
message: 'Request Timeout after 17ms'
|
|
|
|
};
|
|
|
|
|
|
|
|
// request timeout messages willl be written here
|
|
|
|
const infoMesssages = [];
|
|
|
|
|
|
|
|
// a controller that validates the esclient and cmd that was passed to the search service
|
|
|
|
const controller = proxyquire('../../../controller/place', {
|
|
|
|
'../service/mget': (esclient, cmd, callback) => {
|
|
|
|
// not that the searchService got called
|
|
|
|
searchServiceCallCount++;
|
|
|
|
|
|
|
|
callback(timeoutError);
|
|
|
|
},
|
|
|
|
'pelias-logger': {
|
|
|
|
get: (service) => {
|
|
|
|
t.equal(service, 'api');
|
|
|
|
return {
|
|
|
|
info: (msg) => {
|
|
|
|
infoMesssages.push(msg);
|
|
|
|
},
|
|
|
|
debug: () => {}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})(config, esclient);
|
|
|
|
|
|
|
|
const req = {
|
|
|
|
clean: {
|
|
|
|
ids: [
|
|
|
|
{
|
|
|
|
id: 'id1',
|
|
|
|
layers: 'layer1'
|
|
|
|
}
|
|
|
|
]
|
|
|
|
},
|
|
|
|
errors: [],
|
|
|
|
warnings: []
|
|
|
|
};
|
|
|
|
const res = {};
|
|
|
|
|
|
|
|
const next = () => {
|
|
|
|
t.equal(searchServiceCallCount, 3+1);
|
|
|
|
|
|
|
|
t.ok(infoMesssages.indexOf('request timed out on attempt 1, retrying') !== -1);
|
|
|
|
t.ok(infoMesssages.indexOf('request timed out on attempt 2, retrying') !== -1);
|
|
|
|
t.ok(infoMesssages.indexOf('request timed out on attempt 3, retrying') !== -1);
|
|
|
|
|
|
|
|
t.deepEqual(req.errors, [timeoutError.message]);
|
|
|
|
t.deepEqual(res, {});
|
|
|
|
t.end();
|
|
|
|
};
|
|
|
|
|
|
|
|
controller(req, res, next);
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
test('explicit apiConfig.requestRetries should retry that many times', (t) => {
|
|
|
|
const config = {
|
|
|
|
indexName: 'indexName value',
|
|
|
|
requestRetries: 17
|
|
|
|
};
|
|
|
|
const esclient = 'this is the esclient';
|
|
|
|
|
|
|
|
let searchServiceCallCount = 0;
|
|
|
|
|
|
|
|
const timeoutError = {
|
|
|
|
status: 408,
|
|
|
|
displayName: 'RequestTimeout',
|
|
|
|
message: 'Request Timeout after 17ms'
|
|
|
|
};
|
|
|
|
|
|
|
|
// a controller that validates the esclient and cmd that was passed to the search service
|
|
|
|
const controller = proxyquire('../../../controller/place', {
|
|
|
|
'../service/mget': (esclient, cmd, callback) => {
|
|
|
|
// not that the searchService got called
|
|
|
|
searchServiceCallCount++;
|
|
|
|
|
|
|
|
callback(timeoutError);
|
|
|
|
}
|
|
|
|
})(config, esclient);
|
|
|
|
|
|
|
|
const req = {
|
|
|
|
clean: {
|
|
|
|
ids: [
|
|
|
|
{
|
|
|
|
id: 'id1',
|
|
|
|
layers: 'layer1'
|
|
|
|
}
|
|
|
|
]
|
|
|
|
},
|
|
|
|
errors: [],
|
|
|
|
warnings: []
|
|
|
|
};
|
|
|
|
const res = {};
|
|
|
|
|
|
|
|
const next = () => {
|
|
|
|
t.equal(searchServiceCallCount, 17+1);
|
|
|
|
t.end();
|
|
|
|
};
|
|
|
|
|
|
|
|
controller(req, res, next);
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
test('only status code 408 should be considered a retryable request', (t) => {
|
|
|
|
const config = {
|
|
|
|
indexName: 'indexName value'
|
|
|
|
};
|
|
|
|
const esclient = 'this is the esclient';
|
|
|
|
|
|
|
|
let searchServiceCallCount = 0;
|
|
|
|
|
|
|
|
const nonTimeoutError = {
|
|
|
|
status: 500,
|
|
|
|
displayName: 'InternalServerError',
|
|
|
|
message: 'an internal server error occurred'
|
|
|
|
};
|
|
|
|
|
|
|
|
// a controller that validates the esclient and cmd that was passed to the search service
|
|
|
|
const controller = proxyquire('../../../controller/place', {
|
|
|
|
'../service/mget': (esclient, cmd, callback) => {
|
|
|
|
// not that the searchService got called
|
|
|
|
searchServiceCallCount++;
|
|
|
|
|
|
|
|
callback(nonTimeoutError);
|
|
|
|
}
|
|
|
|
})(config, esclient);
|
|
|
|
|
|
|
|
const req = {
|
|
|
|
clean: {
|
|
|
|
ids: [
|
|
|
|
{
|
|
|
|
id: 'id1',
|
|
|
|
layers: 'layer1'
|
|
|
|
}
|
|
|
|
]
|
|
|
|
},
|
|
|
|
errors: [],
|
|
|
|
warnings: []
|
|
|
|
};
|
|
|
|
const res = {};
|
|
|
|
|
|
|
|
const next = () => {
|
|
|
|
t.equal(searchServiceCallCount, 1);
|
|
|
|
t.deepEqual(req.errors, [nonTimeoutError.message]);
|
|
|
|
t.end();
|
|
|
|
};
|
|
|
|
|
|
|
|
controller(req, res, next);
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
test('string error should not retry and be logged as-is', (t) => {
|
|
|
|
const config = {
|
|
|
|
indexName: 'indexName value'
|
|
|
|
};
|
|
|
|
const esclient = 'this is the esclient';
|
|
|
|
|
|
|
|
let searchServiceCallCount = 0;
|
|
|
|
|
|
|
|
const stringTypeError = 'this is an error string';
|
|
|
|
|
|
|
|
// a controller that validates the esclient and cmd that was passed to the search service
|
|
|
|
const controller = proxyquire('../../../controller/place', {
|
|
|
|
'../service/mget': (esclient, cmd, callback) => {
|
|
|
|
// not that the searchService got called
|
|
|
|
searchServiceCallCount++;
|
|
|
|
|
|
|
|
callback(stringTypeError);
|
|
|
|
}
|
|
|
|
})(config, esclient);
|
|
|
|
|
|
|
|
const req = {
|
|
|
|
clean: {
|
|
|
|
ids: [
|
|
|
|
{
|
|
|
|
id: 'id1',
|
|
|
|
layers: 'layer1'
|
|
|
|
}
|
|
|
|
]
|
|
|
|
},
|
|
|
|
errors: [],
|
|
|
|
warnings: []
|
|
|
|
};
|
|
|
|
const res = {};
|
|
|
|
|
|
|
|
const next = () => {
|
|
|
|
t.equal(searchServiceCallCount, 1);
|
|
|
|
t.deepEqual(req.errors, [stringTypeError]);
|
|
|
|
t.end();
|
|
|
|
};
|
|
|
|
|
|
|
|
controller(req, res, next);
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
module.exports.all = (tape, common) => {
|
|
|
|
|
|
|
|
function test(name, testFunction) {
|
|
|
|
return tape('GET /place ' + name, testFunction);
|
|
|
|
}
|
|
|
|
|
|
|
|
for( const testCase in module.exports.tests ){
|
|
|
|
module.exports.tests[testCase](test, common);
|
|
|
|
}
|
|
|
|
};
|