|
|
|
'use strict';
|
|
|
|
|
|
|
|
var _ = require('lodash');
|
|
|
|
|
|
|
|
var service = { search: require('../service/search') };
|
|
|
|
var logger = require('pelias-logger').get('api');
|
|
|
|
var logging = require( '../helper/logging' );
|
|
|
|
const retry = require('retry');
|
|
|
|
|
|
|
|
function setup( config, esclient, query ){
|
|
|
|
function controller( req, res, next ){
|
|
|
|
// do not run controller when a request
|
|
|
|
// validation error has occurred.
|
|
|
|
if( req.errors && req.errors.length ){
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
// do not run controller if there are already results
|
|
|
|
// this was added during libpostal integration. if the libpostal parse/query
|
|
|
|
// doesn't return anything then fallback to old search-engine-y behavior
|
|
|
|
if (res && res.hasOwnProperty('data') && res.data.length > 0) {
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
var cleanOutput = _.cloneDeep(req.clean);
|
|
|
|
if (logging.isDNT(req)) {
|
|
|
|
cleanOutput = logging.removeFields(cleanOutput);
|
|
|
|
}
|
|
|
|
// log clean parameters for stats
|
|
|
|
logger.info('[req]', 'endpoint=' + req.path, cleanOutput);
|
|
|
|
|
|
|
|
var renderedQuery = query(req.clean);
|
|
|
|
|
|
|
|
// if there's no query to call ES with, skip the service
|
|
|
|
if (_.isUndefined(renderedQuery)) {
|
|
|
|
return next();
|
|
|
|
}
|
|
|
|
|
|
|
|
logger.debug( '[ES req]', cmd );
|
|
|
|
|
|
|
|
// options for retry
|
|
|
|
// maxRetries is from the ES client which defaults to 3
|
|
|
|
// factor of 1 means that each retry attempt will esclient requestTimeout
|
|
|
|
const operationOptions = {
|
|
|
|
retries: _.get(esclient, 'transport.maxRetries'),
|
|
|
|
factor: 1,
|
|
|
|
minTimeout: _.get(esclient, 'transport.requestTimeout')
|
|
|
|
};
|
|
|
|
|
|
|
|
// prepend the config timeout since the total number of attempts is maxRetries+1
|
|
|
|
const timeouts = [operationOptions.minTimeout].concat(retry.timeouts(operationOptions));
|
|
|
|
|
|
|
|
// setup a new operation
|
|
|
|
const operation = retry.operation(operationOptions);
|
|
|
|
|
|
|
|
// elasticsearch command
|
|
|
|
var cmd = {
|
|
|
|
index: config.indexName,
|
|
|
|
searchType: 'dfs_query_then_fetch',
|
|
|
|
body: renderedQuery.body
|
|
|
|
};
|
|
|
|
|
|
|
|
operation.attempt((currentAttempt) => {
|
|
|
|
// query elasticsearch
|
|
|
|
service.search( esclient, cmd, function( err, docs, meta ){
|
|
|
|
// returns true if the operation should be attempted again
|
|
|
|
// (handles bookkeeping of maxRetries)
|
|
|
|
if (operation.retry(err)) {
|
|
|
|
logger.info('request timed out, retrying');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
res.meta = meta || {};
|
|
|
|
// store the query_type for subsequent middleware
|
|
|
|
res.meta.query_type = renderedQuery.type;
|
|
|
|
|
|
|
|
logger.info(`[controller:search] [queryType:${renderedQuery.type}] [es_result_count:` +
|
|
|
|
(res.data && res.data.length ? res.data.length : 0));
|
|
|
|
}
|
|
|
|
logger.debug('[ES response]', docs);
|
|
|
|
next();
|
|
|
|
});
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return controller;
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = setup;
|