mirror of https://github.com/pelias/api.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
371 lines
8.7 KiB
371 lines
8.7 KiB
|
|
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); |
|
} |
|
};
|
|
|