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.
128 lines
4.0 KiB
128 lines
4.0 KiB
7 years ago
|
'use strict';
|
||
|
|
||
|
const _ = require('lodash');
|
||
|
|
||
|
const searchService = require('../service/search');
|
||
|
const logger = require('pelias-logger').get('api');
|
||
|
const logging = require( '../helper/logging' );
|
||
|
const retry = require('retry');
|
||
|
const Debug = require('../helper/debug');
|
||
|
|
||
|
function isRequestTimeout(err) {
|
||
|
return _.get(err, 'status') === 408;
|
||
|
}
|
||
|
|
||
|
function setup( apiConfig, esclient, query, should_execute ){
|
||
|
function controller( req, res, next ){
|
||
|
if (!should_execute(req, res)) {
|
||
|
return next();
|
||
|
}
|
||
|
|
||
|
const debugLog = new Debug('controller:search');
|
||
|
|
||
|
let cleanOutput = _.cloneDeep(req.clean);
|
||
|
if (logging.isDNT(req)) {
|
||
|
cleanOutput = logging.removeFields(cleanOutput);
|
||
|
}
|
||
|
// log clean parameters for stats
|
||
|
logger.info('[req]', 'endpoint=' + req.path, cleanOutput);
|
||
|
|
||
|
const renderedQuery = query(req.clean);
|
||
|
|
||
|
// if there's no query to call ES with, skip the service
|
||
|
if (_.isUndefined(renderedQuery)) {
|
||
|
debugLog.push(req, 'No query to call ES with. Skipping');
|
||
|
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);
|
||
|
|
||
|
// elasticsearch command
|
||
|
const cmd = {
|
||
|
index: apiConfig.indexName,
|
||
|
searchType: 'dfs_query_then_fetch',
|
||
|
body: renderedQuery.body
|
||
|
};
|
||
|
|
||
|
logger.debug( '[ES req]', cmd );
|
||
|
debugLog.push(req, {ES_req: cmd});
|
||
|
|
||
|
operation.attempt((currentAttempt) => {
|
||
|
const initialTime = debugLog.beginTimer(req, `Attempt ${currentAttempt}`);
|
||
|
// query elasticsearch
|
||
|
searchService( esclient, cmd, function( err, docs, meta ){
|
||
|
// 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, 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}`);
|
||
|
}
|
||
|
|
||
|
const messageParts = [
|
||
|
'[controller:search]',
|
||
|
`[queryType:${renderedQuery.type}]`,
|
||
|
`[es_result_count:${_.defaultTo(docs, []).length}]`
|
||
|
];
|
||
|
|
||
|
// if there are docs, concatenate them onto the end of existing results
|
||
|
if (docs) {
|
||
|
res.data = _.concat(_.defaultTo(res.data, []), docs);
|
||
|
}
|
||
|
res.meta = meta || {};
|
||
|
// store the query_type for subsequent middleware
|
||
|
res.meta.query_type = renderedQuery.type;
|
||
|
|
||
|
logger.info(messageParts.join(' '));
|
||
|
debugLog.push(req, {queryType: {
|
||
|
[renderedQuery.type] : {
|
||
|
es_result_count: parseInt(messageParts[2].slice(17, -1))
|
||
|
}
|
||
|
}});
|
||
|
}
|
||
|
logger.debug('[ES response]', docs);
|
||
|
next();
|
||
|
});
|
||
|
debugLog.stopTimer(req, initialTime);
|
||
|
});
|
||
|
|
||
|
}
|
||
|
|
||
|
return controller;
|
||
|
}
|
||
|
|
||
|
module.exports = setup;
|