|
|
|
var peliasQuery = require('pelias-query'),
|
|
|
|
defaults = require('./defaults'),
|
|
|
|
textParser = require('./text_parser'),
|
|
|
|
check = require('check-types');
|
|
|
|
|
|
|
|
//------------------------------
|
|
|
|
// general-purpose search query
|
|
|
|
//------------------------------
|
|
|
|
var query = new peliasQuery.layout.FilteredBooleanQuery();
|
|
|
|
|
|
|
|
// mandatory matches
|
|
|
|
query.score( peliasQuery.view.boundary_country, 'must' );
|
|
|
|
query.score( peliasQuery.view.ngrams, 'must' );
|
|
|
|
|
|
|
|
// scoring boost
|
|
|
|
query.score( peliasQuery.view.phrase );
|
|
|
|
query.score( peliasQuery.view.focus( peliasQuery.view.phrase ) );
|
|
|
|
query.score( peliasQuery.view.popularity(['admin0','admin1','admin2']) );
|
|
|
|
|
|
|
|
// address components
|
|
|
|
query.score( peliasQuery.view.address('housenumber') );
|
|
|
|
query.score( peliasQuery.view.address('street') );
|
|
|
|
query.score( peliasQuery.view.address('postcode') );
|
|
|
|
|
|
|
|
// admin components
|
|
|
|
query.score( peliasQuery.view.admin('alpha3') );
|
|
|
|
query.score( peliasQuery.view.admin('admin0') );
|
|
|
|
query.score( peliasQuery.view.admin('admin1') );
|
|
|
|
query.score( peliasQuery.view.admin('admin1_abbr') );
|
|
|
|
query.score( peliasQuery.view.admin('admin2') );
|
|
|
|
query.score( peliasQuery.view.admin('local_admin') );
|
|
|
|
query.score( peliasQuery.view.admin('locality') );
|
|
|
|
query.score( peliasQuery.view.admin('neighborhood') );
|
|
|
|
|
|
|
|
// non-scoring hard filters
|
|
|
|
query.filter( peliasQuery.view.boundary_circle );
|
|
|
|
query.filter( peliasQuery.view.boundary_rect );
|
|
|
|
|
|
|
|
// groovy scripts used to handle tie-breaking
|
|
|
|
query.sort( peliasQuery.view.sort_numeric_script('admin_boost') );
|
|
|
|
query.sort( peliasQuery.view.sort_numeric_script('popularity') );
|
|
|
|
query.sort( peliasQuery.view.sort_numeric_script('population') );
|
|
|
|
|
|
|
|
// --------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
map request variables to query variables for all inputs
|
|
|
|
provided by this HTTP request.
|
|
|
|
**/
|
|
|
|
function generateQuery( clean ){
|
|
|
|
|
|
|
|
var vs = new peliasQuery.Vars( defaults );
|
|
|
|
|
|
|
|
// input text
|
|
|
|
vs.var( 'input:name', clean.text );
|
|
|
|
|
|
|
|
// size
|
|
|
|
if( clean.size ){
|
|
|
|
vs.var( 'size', clean.size );
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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']
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// focus viewport
|
|
|
|
// @todo: change these to the correct request variable names
|
|
|
|
// @todo: calculate the centroid from the viewport box
|
|
|
|
if( clean.focus && clean.focus.viewport ){
|
|
|
|
var vp = clean.focus.viewport;
|
|
|
|
vs.set({
|
|
|
|
'focus:point:lat': vp.min_lat + ( vp.max_lat - vp.min_lat ) / 2,
|
|
|
|
'focus:point:lon': vp.min_lon + ( vp.max_lon - vp.min_lon ) / 2
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
|
|
|
// @todo: change these to the correct request variable names
|
|
|
|
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']
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// run the address parser
|
|
|
|
if( clean.parsed_text ){
|
|
|
|
textParser( clean.parsed_text, vs );
|
|
|
|
}
|
|
|
|
|
|
|
|
return query.render( vs );
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = generateQuery;
|