|
|
|
'use strict';
|
|
|
|
|
|
|
|
const peliasQuery = require('pelias-query');
|
|
|
|
const defaults = require('./search_defaults');
|
|
|
|
const textParser = require('./text_parser');
|
|
|
|
const check = require('check-types');
|
|
|
|
const logger = require('pelias-logger').get('api');
|
|
|
|
const _ = require('lodash');
|
|
|
|
|
|
|
|
//------------------------------
|
|
|
|
// general-purpose search query
|
|
|
|
//------------------------------
|
|
|
|
var fallbackQuery = new peliasQuery.layout.FallbackQuery();
|
|
|
|
var geodisambiguationQuery = new peliasQuery.layout.GeodisambiguationQuery();
|
|
|
|
|
|
|
|
// scoring boost
|
|
|
|
fallbackQuery.score( peliasQuery.view.focus_only_function( peliasQuery.view.phrase ) );
|
|
|
|
fallbackQuery.score( peliasQuery.view.popularity_only_function );
|
|
|
|
fallbackQuery.score( peliasQuery.view.population_only_function );
|
|
|
|
|
|
|
|
geodisambiguationQuery.score( peliasQuery.view.focus_only_function( peliasQuery.view.phrase ) );
|
|
|
|
geodisambiguationQuery.score( peliasQuery.view.popularity_only_function );
|
|
|
|
geodisambiguationQuery.score( peliasQuery.view.population_only_function );
|
|
|
|
// --------------------------------
|
|
|
|
|
|
|
|
// non-scoring hard filters
|
|
|
|
fallbackQuery.filter( peliasQuery.view.boundary_country );
|
|
|
|
fallbackQuery.filter( peliasQuery.view.boundary_circle );
|
|
|
|
fallbackQuery.filter( peliasQuery.view.boundary_rect );
|
|
|
|
fallbackQuery.filter( peliasQuery.view.sources );
|
|
|
|
fallbackQuery.filter( peliasQuery.view.layers );
|
|
|
|
fallbackQuery.filter( peliasQuery.view.categories );
|
|
|
|
|
|
|
|
geodisambiguationQuery.filter( peliasQuery.view.boundary_country );
|
|
|
|
geodisambiguationQuery.filter( peliasQuery.view.boundary_circle );
|
|
|
|
geodisambiguationQuery.filter( peliasQuery.view.boundary_rect );
|
|
|
|
geodisambiguationQuery.filter( peliasQuery.view.sources );
|
|
|
|
geodisambiguationQuery.filter( peliasQuery.view.layers );
|
|
|
|
geodisambiguationQuery.filter( peliasQuery.view.categories );
|
|
|
|
// --------------------------------
|
|
|
|
|
|
|
|
/**
|
|
|
|
map request variables to query variables for all inputs
|
|
|
|
provided by this HTTP request.
|
|
|
|
**/
|
|
|
|
function generateQuery( clean ){
|
|
|
|
|
|
|
|
const vs = new peliasQuery.Vars( defaults );
|
|
|
|
|
|
|
|
let logStr = '[query:search] [parser:libpostal] ';
|
|
|
|
|
|
|
|
// input text
|
|
|
|
vs.var( 'input:name', clean.text );
|
|
|
|
|
|
|
|
// sources
|
|
|
|
if( check.array(clean.sources) && clean.sources.length ) {
|
|
|
|
vs.var( 'sources', clean.sources);
|
|
|
|
logStr += '[param:sources] ';
|
|
|
|
}
|
|
|
|
|
|
|
|
// layers
|
|
|
|
if( check.array(clean.layers) && clean.layers.length ) {
|
|
|
|
vs.var('layers', clean.layers);
|
|
|
|
logStr += '[param:layers] ';
|
|
|
|
}
|
|
|
|
|
|
|
|
// categories
|
|
|
|
if (clean.categories) {
|
|
|
|
vs.var('input:categories', clean.categories);
|
|
|
|
logStr += '[param:categories] ';
|
|
|
|
}
|
|
|
|
|
|
|
|
// size
|
|
|
|
if( clean.querySize ) {
|
|
|
|
vs.var( 'size', clean.querySize );
|
|
|
|
logStr += '[param:querySize] ';
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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']
|
|
|
|
});
|
|
|
|
logStr += '[param:focus_point] ';
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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']
|
|
|
|
});
|
|
|
|
logStr += '[param:boundary_rect] ';
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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'
|
|
|
|
});
|
|
|
|
}
|
|
|
|
logStr += '[param:boundary_circle] ';
|
|
|
|
}
|
|
|
|
|
|
|
|
// boundary country
|
|
|
|
if( check.string(clean['boundary.country']) ){
|
|
|
|
vs.set({
|
|
|
|
'boundary:country': clean['boundary.country']
|
|
|
|
});
|
|
|
|
logStr += '[param:boundary_country] ';
|
|
|
|
}
|
|
|
|
|
|
|
|
// run the address parser
|
|
|
|
if( clean.parsed_text ){
|
|
|
|
textParser( clean.parsed_text, vs );
|
|
|
|
}
|
|
|
|
|
|
|
|
var q = getQuery(clean, vs);
|
|
|
|
|
|
|
|
//console.log(JSON.stringify(q, null, 2));
|
|
|
|
|
|
|
|
if (q !== undefined) {
|
|
|
|
logger.info(logStr);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
logger.info('[parser:libpostal] query type not supported');
|
|
|
|
}
|
|
|
|
|
|
|
|
return q;
|
|
|
|
}
|
|
|
|
|
|
|
|
function getQuery(params, vs) {
|
|
|
|
if (forceLibpostal(params) ||
|
|
|
|
hasStreet(vs) ||
|
|
|
|
isCityStateOnlyWithOptionalCountry(vs) ||
|
|
|
|
isCityCountryOnly(vs) ||
|
|
|
|
isPostalCodeOnly(vs)) {
|
|
|
|
return {
|
|
|
|
type: 'fallback',
|
|
|
|
body: fallbackQuery.render(vs)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
// returning undefined is a signal to a later step that the addressit-parsed
|
|
|
|
// query should be queried for
|
|
|
|
return undefined;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
function forceLibpostal(params) {
|
|
|
|
return _.get(params, 'force_libpostal', false);
|
|
|
|
}
|
|
|
|
|
|
|
|
function hasStreet(vs) {
|
|
|
|
return vs.isset('input:street');
|
|
|
|
}
|
|
|
|
|
|
|
|
function isCityStateOnlyWithOptionalCountry(vs) {
|
|
|
|
var isSet = (layer) => {
|
|
|
|
return vs.isset(`input:${layer}`);
|
|
|
|
};
|
|
|
|
|
|
|
|
var allowedFields = ['locality', 'region'];
|
|
|
|
var disallowedFields = ['query', 'category', 'housenumber', 'street',
|
|
|
|
'neighbourhood', 'borough', 'postcode', 'county'];
|
|
|
|
|
|
|
|
return allowedFields.every(isSet) && !disallowedFields.some(isSet);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
function isCityCountryOnly(vs) {
|
|
|
|
var isSet = (layer) => {
|
|
|
|
return vs.isset(`input:${layer}`);
|
|
|
|
};
|
|
|
|
|
|
|
|
var allowedFields = ['locality', 'country'];
|
|
|
|
var disallowedFields = ['query', 'category', 'housenumber', 'street',
|
|
|
|
'neighbourhood', 'borough', 'postcode', 'county', 'region'];
|
|
|
|
|
|
|
|
return allowedFields.every(isSet) &&
|
|
|
|
!disallowedFields.some(isSet);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
function isPostalCodeOnly(vs) {
|
|
|
|
var isSet = (layer) => {
|
|
|
|
return vs.isset(`input:${layer}`);
|
|
|
|
};
|
|
|
|
|
|
|
|
var allowedFields = ['postcode'];
|
|
|
|
var disallowedFields = ['query', 'category', 'housenumber', 'street',
|
|
|
|
'neighbourhood', 'borough', 'county', 'region', 'country'];
|
|
|
|
|
|
|
|
return allowedFields.every(isSet) &&
|
|
|
|
!disallowedFields.some(isSet);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = generateQuery;
|