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.
166 lines
5.7 KiB
166 lines
5.7 KiB
7 years ago
|
const peliasQuery = require('pelias-query');
|
||
|
const defaults = require('./search_defaults');
|
||
|
const logger = require('pelias-logger').get('api');
|
||
|
const _ = require('lodash');
|
||
|
const check = require('check-types');
|
||
|
|
||
|
//------------------------------
|
||
|
// general-purpose search query
|
||
|
//------------------------------
|
||
|
const venuesQuery = new peliasQuery.layout.VenuesQuery();
|
||
|
|
||
|
// scoring boost
|
||
|
venuesQuery.score( peliasQuery.view.focus_only_function( peliasQuery.view.phrase ) );
|
||
|
// --------------------------------
|
||
|
|
||
|
// non-scoring hard filters
|
||
|
venuesQuery.filter( peliasQuery.view.boundary_country );
|
||
|
venuesQuery.filter( peliasQuery.view.boundary_circle );
|
||
|
venuesQuery.filter( peliasQuery.view.boundary_rect );
|
||
|
venuesQuery.filter( peliasQuery.view.sources );
|
||
|
// --------------------------------
|
||
|
|
||
|
const adminLayers = ['neighbourhood', 'borough', 'city', 'county', 'state', 'country'];
|
||
|
|
||
|
// This query is a departure from traditional Pelias queries where textual
|
||
|
// names of admin areas were looked up. This query uses the ids returned by
|
||
|
// placeholder for lookups which dramatically reduces the amount of information
|
||
|
// that ES has to store and allows us to have placeholder handle altnames on
|
||
|
// behalf of Pelias.
|
||
|
//
|
||
|
// For the happy path, an input like '30 West 26th Street, Manhattan' would result
|
||
|
// in:
|
||
|
// neighbourhood_id in []
|
||
|
// borough_id in [421205771]
|
||
|
// locality_id in [85945171, 85940551, 85972655]
|
||
|
// localadmin_id in [404502889, 404499147, 404502891, 85972655]
|
||
|
//
|
||
|
// Where the ids are for all the various Manhattans. Each of those could
|
||
|
// conceivably be the Manhattan that the user was referring to so so all must be
|
||
|
// queried for at the same time.
|
||
|
//
|
||
|
// A counter example for this is '1 West Market Street, York, PA' where York, PA
|
||
|
// can be interpreted as a locality OR county. From experience, when there's
|
||
|
// ambiguity between locality and county for an input, the user is, with complete
|
||
|
// metaphysical certitude, referring to the city. If they were referring to the
|
||
|
// county, they would have entered 'York County, PA'. The point is that it's
|
||
|
// insufficient to just query for all ids because, in this case, '1 West Market Street'
|
||
|
// in other cities in York County, PA would be returned and would be both jarring
|
||
|
// to the user and almost certainly leads to incorrect results. For example,
|
||
|
// the following could be returned (all are towns in York County, PA):
|
||
|
// - 1 West Market Street, Dallastown, PA
|
||
|
// - 1 West Market Street, Fawn Grove, PA
|
||
|
// - 1 West Market Street, Shrewsbury, PA
|
||
|
// etc.
|
||
|
//
|
||
|
// To avoid this calamitous response, this query takes the approach of
|
||
|
// "granularity bands". That is, if there are any ids in the first set of any
|
||
|
// of these granularities:
|
||
|
// - neighbourhood
|
||
|
// - borough
|
||
|
// - locality
|
||
|
// - localadmin
|
||
|
// - region
|
||
|
// - macroregion
|
||
|
// - dependency
|
||
|
// - country
|
||
|
//
|
||
|
// then query for all ids in only those layers. Falling back, if there are
|
||
|
// no ids in those layers, query for the county/macrocounty layers.
|
||
|
//
|
||
|
// This methodology ensures that no happened-to-match-on-county results are returned.
|
||
|
//
|
||
|
// The decision was made to include all other layers in one to solve the issue
|
||
|
// where a country and city share a name, such as Mexico, which could be
|
||
|
// interpreted as a country AND city (in Missouri). The data itself will sort
|
||
|
// out which is correct. That is, it's unlikely that "11 Rock Springs Dr" exists
|
||
|
// in Mexico the country due to naming conventions and would be filtered out
|
||
|
// (though it could, but that's good because it's legitimate)
|
||
|
|
||
|
/**
|
||
|
map request variables to query variables for all inputs
|
||
|
provided by this HTTP request. This function operates on res.data which is the
|
||
|
Document-ified placeholder repsonse.
|
||
|
**/
|
||
|
function generateQuery( clean ){
|
||
|
const vs = new peliasQuery.Vars( defaults );
|
||
|
|
||
|
const logParts = ['query:venues', 'parser:libpostal'];
|
||
|
|
||
|
// sources
|
||
|
if( !_.isEmpty(clean.sources) ) {
|
||
|
vs.var( 'sources', clean.sources);
|
||
|
logParts.push('param:sources');
|
||
|
}
|
||
|
|
||
|
// size
|
||
|
if( clean.querySize ) {
|
||
|
vs.var( 'size', clean.querySize );
|
||
|
logParts.push('param:querySize');
|
||
|
}
|
||
|
|
||
|
const mostGranularLayer = adminLayers.find(layer => {
|
||
|
return _.has(clean.parsed_text, layer);
|
||
|
});
|
||
|
|
||
|
if (mostGranularLayer) {
|
||
|
vs.var('input:query', clean.parsed_text[mostGranularLayer]);
|
||
|
}
|
||
|
|
||
|
|
||
|
// focus point
|
||
|
if( check.number(clean['focus.point.lat']) &&
|
||
|
check.number(clean['focus.point.lon']) ){
|
||
|
vs.set({
|
||
|
'focus:point:lat': clean['focus.point.lat'],
|
||
|
'focus:point:lon': clean['focus.point.lon']
|
||
|
});
|
||
|
}
|
||
|
|
||
|
// boundary rect
|
||
|
if( check.number(clean['boundary.rect.min_lat']) &&
|
||
|
check.number(clean['boundary.rect.max_lat']) &&
|
||
|
check.number(clean['boundary.rect.min_lon']) &&
|
||
|
check.number(clean['boundary.rect.max_lon']) ){
|
||
|
vs.set({
|
||
|
'boundary:rect:top': clean['boundary.rect.max_lat'],
|
||
|
'boundary:rect:right': clean['boundary.rect.max_lon'],
|
||
|
'boundary:rect:bottom': clean['boundary.rect.min_lat'],
|
||
|
'boundary:rect:left': clean['boundary.rect.min_lon']
|
||
|
});
|
||
|
}
|
||
|
|
||
|
// boundary circle
|
||
|
if( check.number(clean['boundary.circle.lat']) &&
|
||
|
check.number(clean['boundary.circle.lon']) ){
|
||
|
vs.set({
|
||
|
'boundary:circle:lat': clean['boundary.circle.lat'],
|
||
|
'boundary:circle:lon': clean['boundary.circle.lon']
|
||
|
});
|
||
|
|
||
|
if( check.number(clean['boundary.circle.radius']) ){
|
||
|
vs.set({
|
||
|
'boundary:circle:radius': Math.round( clean['boundary.circle.radius'] ) + 'km'
|
||
|
});
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// boundary country
|
||
|
if( check.string(clean['boundary.country']) ){
|
||
|
vs.set({
|
||
|
'boundary:country': clean['boundary.country']
|
||
|
});
|
||
|
}
|
||
|
|
||
|
// format the log parts into a single coherent string
|
||
|
logger.info(logParts.map(part => `[${part}]`).join(' '));
|
||
|
|
||
|
return {
|
||
|
type: 'fallback',
|
||
|
body: venuesQuery.render(vs)
|
||
|
};
|
||
|
|
||
|
}
|
||
|
|
||
|
module.exports = generateQuery;
|