|
|
|
|
|
|
|
var service = {
|
|
|
|
suggest: require('../service/suggest'),
|
|
|
|
mget: require('../service/mget')
|
|
|
|
};
|
|
|
|
var geojsonify = require('../helper/geojsonify').search;
|
|
|
|
var async = require('async');
|
|
|
|
|
|
|
|
function setup( backend, query ){
|
|
|
|
|
|
|
|
// allow overriding of dependencies
|
|
|
|
backend = backend || require('../src/backend');
|
|
|
|
query = query || require('../query/suggest');
|
|
|
|
|
|
|
|
function controller( req, res, next ){
|
|
|
|
|
|
|
|
var cmd = {
|
|
|
|
index: 'pelias',
|
|
|
|
body: query( req.clean )
|
|
|
|
};
|
|
|
|
|
|
|
|
var SIZE = req.clean.size || 10;
|
|
|
|
|
|
|
|
var query_backend = function(cmd, callback) {
|
|
|
|
// query backend
|
|
|
|
service.suggest( backend, cmd, function( err, docs ){
|
|
|
|
|
|
|
|
// error handler
|
|
|
|
if( err ){ return next( err ); }
|
|
|
|
|
|
|
|
callback(null, docs);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
var dedup = function(combined) {
|
|
|
|
var unique_ids = [];
|
|
|
|
return combined.filter(function(item, pos) {
|
|
|
|
if (unique_ids.indexOf(item.text) === -1) {
|
|
|
|
unique_ids.push(item.text);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
var sort_by_score = function(combined) {
|
|
|
|
return combined.sort(function(a,b) {
|
|
|
|
return b.score - a.score;
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
var reply = function(docs) {
|
|
|
|
|
|
|
|
// convert docs to geojson
|
|
|
|
var geojson = geojsonify( docs );
|
|
|
|
|
|
|
|
// response envelope
|
|
|
|
geojson.date = new Date().getTime();
|
|
|
|
|
|
|
|
// respond
|
|
|
|
return res.status(200).json( geojson );
|
|
|
|
};
|
|
|
|
|
|
|
|
var respond = function(data) {
|
|
|
|
|
|
|
|
// no documents suggested, return empty array to avoid ActionRequestValidationException
|
|
|
|
if( !Array.isArray( data ) || !data.length ){
|
|
|
|
return reply([]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// map suggester output to mget query
|
|
|
|
var query = data.map( function( doc ) {
|
|
|
|
var idParts = doc.text.split(':');
|
|
|
|
return {
|
|
|
|
_index: 'pelias',
|
|
|
|
_type: idParts[0],
|
|
|
|
_id: idParts.slice(1).join(':')
|
|
|
|
};
|
|
|
|
});
|
|
|
|
|
|
|
|
service.mget( backend, query, function( err, docs ){
|
|
|
|
|
|
|
|
// error handler
|
|
|
|
if( err ){ return next( err ); }
|
|
|
|
|
|
|
|
// reply
|
|
|
|
return reply( docs );
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
if (req.clean.input) {
|
|
|
|
var async_query;
|
|
|
|
|
|
|
|
// admin only
|
|
|
|
req.admin = {};
|
|
|
|
for (var k in req.clean) { req.admin[k] = req.clean[k]; }
|
|
|
|
req.admin.layers = ['admin0','admin1','admin2'];
|
|
|
|
|
|
|
|
if (req.clean.input.length < 4 && isNaN(parseInt(req.clean.input, 10))) {
|
|
|
|
async_query = {
|
|
|
|
admin_3p: function(callback){
|
|
|
|
cmd.body = query( req.admin, 3 );
|
|
|
|
query_backend(cmd, callback);
|
|
|
|
},
|
|
|
|
admin_1p: function(callback){
|
|
|
|
cmd.body = query( req.admin, 1 );
|
|
|
|
query_backend(cmd, callback);
|
|
|
|
},
|
|
|
|
all_3p: function(callback) {
|
|
|
|
cmd.body = query( req.clean, 3 );
|
|
|
|
query_backend(cmd, callback);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
} else {
|
|
|
|
async_query = {
|
|
|
|
all_5p: function(callback){
|
|
|
|
cmd.body = query( req.clean, 5);
|
|
|
|
query_backend(cmd, callback);
|
|
|
|
},
|
|
|
|
all_3p: function(callback){
|
|
|
|
cmd.body = query( req.clean, 3);
|
|
|
|
query_backend(cmd, callback);
|
|
|
|
},
|
|
|
|
all_1p: function(callback){
|
|
|
|
cmd.body = query( req.clean, 1 );
|
|
|
|
query_backend(cmd, callback);
|
|
|
|
},
|
|
|
|
admin_1p: function(callback){
|
|
|
|
cmd.body = query( req.admin );
|
|
|
|
query_backend(cmd, callback);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// fuzzy
|
|
|
|
async_query.fuzzy = function(callback){
|
|
|
|
cmd.body = query( req.clean, 3, 'AUTO' );
|
|
|
|
query_backend(cmd, callback);
|
|
|
|
};
|
|
|
|
|
|
|
|
async.parallel(async_query, function(err, results) {
|
|
|
|
// results is equal to: {a: docs, b: docs, c: docs}
|
|
|
|
var splice_length = parseInt((SIZE / Object.keys(results).length), 10);
|
|
|
|
var results_keys = Object.keys(async_query);
|
|
|
|
|
|
|
|
var combined = [];
|
|
|
|
results_keys.forEach(function(key){
|
|
|
|
combined = combined.concat(sort_by_score(results[key]).splice(0,splice_length));
|
|
|
|
});
|
|
|
|
|
|
|
|
combined = dedup(combined);
|
|
|
|
respond(combined);
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
query_backend(cmd, function(err, results) {
|
|
|
|
respond(results);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return controller;
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = setup;
|