|
|
@ -2,7 +2,8 @@ const logger = require('pelias-logger').get('coarse_reverse'); |
|
|
|
const _ = require('lodash'); |
|
|
|
const _ = require('lodash'); |
|
|
|
const Document = require('pelias-model').Document; |
|
|
|
const Document = require('pelias-model').Document; |
|
|
|
|
|
|
|
|
|
|
|
const granularities = [ |
|
|
|
// do not change order, other functionality depends on most-to-least granular order
|
|
|
|
|
|
|
|
const coarse_granularities = [ |
|
|
|
'neighbourhood', |
|
|
|
'neighbourhood', |
|
|
|
'borough', |
|
|
|
'borough', |
|
|
|
'locality', |
|
|
|
'locality', |
|
|
@ -15,24 +16,47 @@ const granularities = [ |
|
|
|
'country' |
|
|
|
'country' |
|
|
|
]; |
|
|
|
]; |
|
|
|
|
|
|
|
|
|
|
|
function getMostGranularLayer(results) { |
|
|
|
// remove non-coarse layers and return what's left (or all if empty)
|
|
|
|
return granularities.find((granularity) => { |
|
|
|
function getEffectiveLayers(requested_layers) { |
|
|
|
return results.hasOwnProperty(granularity); |
|
|
|
const non_coarse_layers_removed = _.without(requested_layers, 'venue', 'address', 'street'); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// if resulting array is empty, use all coarse granularities
|
|
|
|
|
|
|
|
if (_.isEmpty(non_coarse_layers_removed)) { |
|
|
|
|
|
|
|
return coarse_granularities; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// otherwise use requested layers with non-coarse layers removed
|
|
|
|
|
|
|
|
return non_coarse_layers_removed; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// drop from coarse_granularities until there's one that was requested
|
|
|
|
|
|
|
|
// this depends on coarse_granularities being ordered
|
|
|
|
|
|
|
|
function getApplicableRequestedLayers(requested_layers) { |
|
|
|
|
|
|
|
return _.dropWhile(coarse_granularities, (coarse_granularity) => { |
|
|
|
|
|
|
|
return !_.includes(requested_layers, coarse_granularity); |
|
|
|
}); |
|
|
|
}); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// removing non-coarse layers could leave effective_layers empty, so it's
|
|
|
|
// removing non-coarse layers could leave effective_layers empty, so it's
|
|
|
|
// important to check for empty layers here
|
|
|
|
// important to check for empty layers here
|
|
|
|
function hasResultsAtRequestedLayers(results, layers) { |
|
|
|
function hasResultsAtRequestedLayers(results, requested_layers) { |
|
|
|
return _.isEmpty(layers) || !_.isEmpty(_.intersection(layers, Object.keys(results))); |
|
|
|
return !_.isEmpty(_.intersection(_.keys(results), requested_layers)); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
function synthesizeDoc(results) { |
|
|
|
// get the most granular layer from the results by taking the head of the intersection
|
|
|
|
// now create a model.Document from what's level, using the most granular
|
|
|
|
// of coarse_granularities (which are ordered) and the result layers
|
|
|
|
|
|
|
|
// ['neighbourhood', 'borough', 'locality'] - ['locality', 'borough'] = 'borough'
|
|
|
|
|
|
|
|
// this depends on coarse_granularities being ordered
|
|
|
|
|
|
|
|
function getMostGranularLayerOfResult(result_layers) { |
|
|
|
|
|
|
|
return _.head(_.intersection(coarse_granularities, result_layers)); |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// create a model.Document from what's left, using the most granular
|
|
|
|
// result available as the starting point
|
|
|
|
// result available as the starting point
|
|
|
|
// the immediately above cannot be re-used since county may be the most
|
|
|
|
function synthesizeDoc(results) { |
|
|
|
// granular layer requested but the results may start at region (no county found)
|
|
|
|
// find the most granular layer to use as the document layer
|
|
|
|
const most_granular_layer = getMostGranularLayer(results); |
|
|
|
const most_granular_layer = getMostGranularLayerOfResult(_.keys(results)); |
|
|
|
const id = results[most_granular_layer][0].id; |
|
|
|
const id = results[most_granular_layer][0].id; |
|
|
|
|
|
|
|
|
|
|
|
const doc = new Document('whosonfirst', most_granular_layer, id.toString()); |
|
|
|
const doc = new Document('whosonfirst', most_granular_layer, id.toString()); |
|
|
@ -86,8 +110,8 @@ function setup(service, should_execute) { |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// because coarse reverse is called when non-coarse reverse didn't return
|
|
|
|
// because coarse reverse is called when non-coarse reverse didn't return
|
|
|
|
// anything, don't consider non-coarse layers when filtering
|
|
|
|
// anything, treat requested layers as if it didn't contain non-coarse layers
|
|
|
|
const effective_layers = _.without(req.clean.layers, 'venue', 'address', 'street'); |
|
|
|
const effective_layers = getEffectiveLayers(req.clean.layers); |
|
|
|
|
|
|
|
|
|
|
|
const centroid = { |
|
|
|
const centroid = { |
|
|
|
lat: req.clean['point.lat'], |
|
|
|
lat: req.clean['point.lat'], |
|
|
@ -102,27 +126,20 @@ function setup(service, should_execute) { |
|
|
|
return next(); |
|
|
|
return next(); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
// find the finest granularity requested
|
|
|
|
// log how many results there were
|
|
|
|
const finest_granularity_requested = granularities.findIndex((granularity) => { |
|
|
|
logger.info(`[controller:coarse_reverse][queryType:pip][result_count:${_.size(results)}]`); |
|
|
|
return effective_layers.indexOf(granularity) !== -1; |
|
|
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
logger.info(`[controller:coarse_reverse][queryType:pip][result_count:${Object.keys(results).length}]`); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// now remove everything from the response that is more granular than the
|
|
|
|
// now keep everything from the response that is equal to or less granular
|
|
|
|
// most granular layer requested. that is, if effective_layers=['county'],
|
|
|
|
// than the most granular layer requested. that is, if effective_layers=['county'],
|
|
|
|
// remove neighbourhoods, localities, and localadmins
|
|
|
|
// remove neighbourhoods, boroughs, localities, localadmins
|
|
|
|
Object.keys(results).forEach((layer) => { |
|
|
|
const applicable_results = _.pick(results, getApplicableRequestedLayers(effective_layers)); |
|
|
|
if (granularities.indexOf(layer) < finest_granularity_requested) { |
|
|
|
|
|
|
|
delete results[layer]; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
}); |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
res.meta = {}; |
|
|
|
res.meta = {}; |
|
|
|
res.data = []; |
|
|
|
res.data = []; |
|
|
|
// synthesize a doc from results if there's a result at the request layer(s)
|
|
|
|
|
|
|
|
if (hasResultsAtRequestedLayers(results, effective_layers)) { |
|
|
|
// if there's a result at the requested layer(s), synthesize a doc from results
|
|
|
|
res.data.push(synthesizeDoc(results)); |
|
|
|
if (hasResultsAtRequestedLayers(applicable_results, effective_layers)) { |
|
|
|
|
|
|
|
res.data.push(synthesizeDoc(applicable_results)); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
return next(); |
|
|
|
return next(); |
|
|
|