|
|
|
const _ = require('lodash');
|
|
|
|
const retry = require('retry');
|
|
|
|
|
|
|
|
const mgetService = require('../service/mget');
|
|
|
|
const logger = require('pelias-logger').get('api');
|
|
|
|
const Debug = require('../helper/debug');
|
|
|
|
const debugLog = new Debug('controller:place');
|
|
|
|
|
|
|
|
function requestHasErrors(request) {
|
|
|
|
return _.get(request, 'errors', []).length > 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
function isRequestTimeout(err) {
|
|
|
|
return _.get(err, 'status') === 408;
|
|
|
|
}
|
|
|
|
|
|
|
|
function setup( apiConfig, esclient ){
|
|
|
|
function controller( req, res, next ){
|
|
|
|
// do not run controller when a request validation error has occurred.
|
|
|
|
if (requestHasErrors(req)){
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
// options for retry
|
|
|
|
// maxRetries is from the API config with default of 3
|
|
|
|
// factor of 1 means that each retry attempt will esclient requestTimeout
|
|
|
|
const operationOptions = {
|
|
|
|
retries: _.get(apiConfig, 'requestRetries', 3),
|
|
|
|
factor: 1,
|
|
|
|
minTimeout: _.get(esclient, 'transport.requestTimeout')
|
|
|
|
};
|
|
|
|
|
|
|
|
// setup a new operation
|
|
|
|
const operation = retry.operation(operationOptions);
|
|
|
|
|
|
|
|
const cmd = req.clean.ids.map( function(id) {
|
|
|
|
return {
|
|
|
|
_index: apiConfig.indexName,
|
|
|
|
_type: id.layer,
|
|
|
|
_id: id.id
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
logger.debug( '[ES req]', cmd );
|
|
|
|
debugLog.push(req, {ES_req: cmd});
|
|
|
|
|
|
|
|
operation.attempt((currentAttempt) => {
|
|
|
|
const initialTime = debugLog.beginTimer(req);
|
|
|
|
|
|
|
|
mgetService( esclient, cmd, function( err, docs, data) {
|
|
|
|
const message = {
|
|
|
|
controller: 'place',
|
|
|
|
queryType: 'place',
|
|
|
|
result_count: _.get(data, 'docs.length'),
|
|
|
|
response_time: _.get(data, 'response_time', undefined),
|
|
|
|
params: req.clean,
|
|
|
|
retries: currentAttempt - 1
|
|
|
|
};
|
|
|
|
logger.info('elasticsearch', message);
|
|
|
|
|
|
|
|
// returns true if the operation should be attempted again
|
|
|
|
// (handles bookkeeping of maxRetries)
|
|
|
|
// only consider for status 408 (request timeout)
|
|
|
|
if (isRequestTimeout(err) && operation.retry(err)) {
|
|
|
|
debugLog.stopTimer(req, initialTime, `request timed out on attempt ${currentAttempt}, retrying`);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if execution has gotten this far then one of three things happened:
|
|
|
|
// - the request didn't time out
|
|
|
|
// - maxRetries has been hit so we're giving up
|
|
|
|
// - another error occurred
|
|
|
|
// in either case, handle the error or results
|
|
|
|
|
|
|
|
// error handler
|
|
|
|
if( err ){
|
|
|
|
if (_.isObject(err) && err.message) {
|
|
|
|
req.errors.push( err.message );
|
|
|
|
} else {
|
|
|
|
req.errors.push( err );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// set response data
|
|
|
|
else {
|
|
|
|
res.data = docs;
|
|
|
|
}
|
|
|
|
logger.debug('[ES response]', docs);
|
|
|
|
|
|
|
|
next();
|
|
|
|
});
|
|
|
|
debugLog.stopTimer(req, initialTime);
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return controller;
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = setup;
|