mirror of https://github.com/pelias/api.git
96 lines
2.8 KiB
96 lines
2.8 KiB
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 ) { |
|
// 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)) { |
|
logger.info(`request timed out on attempt ${currentAttempt}, retrying`); |
|
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 { |
|
// log that a retry was successful |
|
// most requests succeed on first attempt so this declutters log files |
|
if (currentAttempt > 1) { |
|
logger.info(`succeeded on retry ${currentAttempt-1}`); |
|
} |
|
|
|
res.data = docs; |
|
} |
|
logger.debug('[ES response]', docs); |
|
|
|
next(); |
|
}); |
|
debugLog.stopTimer(req, initialTime); |
|
}); |
|
|
|
} |
|
|
|
return controller; |
|
} |
|
|
|
module.exports = setup;
|
|
|