const logger = require('pelias-logger').get('coarse_reverse'); const _ = require('lodash'); const Document = require('pelias-model').Document; // do not change order, other functionality depends on most-to-least granular order const coarse_granularities = [ 'neighbourhood', 'borough', 'locality', 'localadmin', 'county', 'macrocounty', 'region', 'macroregion', 'dependency', 'country' ]; // remove non-coarse layers and return what's left (or all if empty) function getEffectiveLayers(requested_layers) { 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 // important to check for empty layers here function hasResultsAtRequestedLayers(results, requested_layers) { return !_.isEmpty(_.intersection(_.keys(results), requested_layers)); } // get the most granular layer from the results by taking the head of the intersection // 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 function synthesizeDoc(results) { // find the most granular layer to use as the document layer const most_granular_layer = getMostGranularLayerOfResult(_.keys(results)); const id = results[most_granular_layer][0].id; const doc = new Document('whosonfirst', most_granular_layer, id.toString()); doc.setName('default', results[most_granular_layer][0].name); if (results[most_granular_layer][0].hasOwnProperty('centroid')) { doc.setCentroid( results[most_granular_layer][0].centroid ); } if (results[most_granular_layer][0].hasOwnProperty('bounding_box')) { const parsedBoundingBox = results[most_granular_layer][0].bounding_box.split(',').map(parseFloat); doc.setBoundingBox({ upperLeft: { lat: parsedBoundingBox[3], lon: parsedBoundingBox[0] }, lowerRight: { lat: parsedBoundingBox[1], lon: parsedBoundingBox[2] } }); } if (_.has(results, 'country[0].abbr')) { doc.setAlpha3(results.country[0].abbr); } // assign the administrative hierarchy Object.keys(results).forEach((layer) => { if (results[layer][0].hasOwnProperty('abbr')) { doc.addParent(layer, results[layer][0].name, results[layer][0].id.toString(), results[layer][0].abbr); } else { doc.addParent(layer, results[layer][0].name, results[layer][0].id.toString()); } }); const esDoc = doc.toESDocument(); esDoc.data._id = esDoc._id; esDoc.data._type = esDoc._type; return esDoc.data; } function setup(service, should_execute) { function controller(req, res, next) { // do not run controller when a request validation error has occurred if (!should_execute(req, res)) { return next(); } // because coarse reverse is called when non-coarse reverse didn't return // anything, treat requested layers as if it didn't contain non-coarse layers const effective_layers = getEffectiveLayers(req.clean.layers); const centroid = { lat: req.clean['point.lat'], lon: req.clean['point.lon'] }; service(req, (err, results) => { // if there's an error, log it and bail if (err) { logger.info(`[controller:coarse_reverse][error]`); logger.error(err); return next(); } // log how many results there were logger.info(`[controller:coarse_reverse][queryType:pip][result_count:${_.size(results)}]`); // now keep everything from the response that is equal to or less granular // than the most granular layer requested. that is, if effective_layers=['county'], // remove neighbourhoods, boroughs, localities, localadmins const applicable_results = _.pick(results, getApplicableRequestedLayers(effective_layers)); res.meta = {}; res.data = []; // if there's a result at the requested layer(s), synthesize a doc from results if (hasResultsAtRequestedLayers(applicable_results, effective_layers)) { res.data.push(synthesizeDoc(applicable_results)); } return next(); }); } return controller; } module.exports = setup;