mirror of https://github.com/pelias/api.git
111 lines
3.0 KiB
111 lines
3.0 KiB
'use strict'; |
|
|
|
const peliasQuery = require('pelias-query'); |
|
const defaults = require('./reverse_defaults'); |
|
const check = require('check-types'); |
|
const _ = require('lodash'); |
|
const logger = require('pelias-logger').get('api'); |
|
|
|
//------------------------------ |
|
// reverse geocode query |
|
//------------------------------ |
|
var query = new peliasQuery.layout.FilteredBooleanQuery(); |
|
|
|
// mandatory matches |
|
query.score( peliasQuery.view.boundary_country, 'must' ); |
|
|
|
// scoring boost |
|
query.sort( peliasQuery.view.sort_distance ); |
|
|
|
// non-scoring hard filters |
|
query.filter( peliasQuery.view.boundary_circle ); |
|
query.filter( peliasQuery.view.sources ); |
|
query.filter( peliasQuery.view.layers ); |
|
query.filter( peliasQuery.view.categories ); |
|
|
|
// -------------------------------- |
|
|
|
function generateQuery( clean ){ |
|
|
|
const vs = new peliasQuery.Vars( defaults ); |
|
|
|
let logStr = '[query:reverse] '; |
|
|
|
// set size |
|
if( clean.querySize ){ |
|
vs.var( 'size', clean.querySize); |
|
logStr += '[param:querySize] '; |
|
} |
|
|
|
// 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 ) { |
|
// only include non-coarse layers |
|
vs.var( 'layers', _.intersection(clean.layers, ['address', 'street', 'venue'])); |
|
logStr += '[param:layers] '; |
|
} |
|
|
|
// focus point to score by distance |
|
if( check.number(clean['point.lat']) && |
|
check.number(clean['point.lon']) ){ |
|
vs.set({ |
|
'focus:point:lat': clean['point.lat'], |
|
'focus:point:lon': clean['point.lon'] |
|
}); |
|
logStr += '[param:focus_point] '; |
|
} |
|
|
|
// bounding circle |
|
// note: the sanitizers will take care of the case |
|
// where point.lan/point.lon are provided in the |
|
// absense of boundary.circle.lat/boundary.circle.lon |
|
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.undefined(clean['boundary.circle.radius'])){ |
|
// for coarse reverse when boundary circle radius is undefined |
|
vs.set({ |
|
'boundary:circle:radius': defaults['boundary:circle:radius'] |
|
}); |
|
} else if (check.number(clean['boundary.circle.radius'])){ |
|
// plain reverse where boundary circle is a valid number |
|
vs.set({ |
|
'boundary:circle:radius': 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] '; |
|
} |
|
|
|
// categories |
|
if (clean.categories) { |
|
vs.var('input:categories', clean.categories); |
|
logStr += '[param:categories] '; |
|
} |
|
|
|
logger.info(logStr); |
|
|
|
return { |
|
type: 'reverse', |
|
body: query.render(vs) |
|
}; |
|
} |
|
|
|
module.exports = generateQuery;
|
|
|