mirror of https://github.com/pelias/api.git
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
162 lines
5.8 KiB
162 lines
5.8 KiB
const async = require('async'); |
|
const logger = require( 'pelias-logger' ).get( 'api' ); |
|
const source_mapping = require('../helper/type_mapping').source_mapping; |
|
const _ = require('lodash'); |
|
const stable = require('stable'); |
|
const Debug = require('../helper/debug'); |
|
const debugLog = new Debug('middleware:interpolate'); |
|
|
|
/** |
|
example response from interpolation web service: |
|
{ |
|
type: 'Feature', |
|
properties: { |
|
type: 'interpolated', |
|
source: 'mixed', |
|
number: '17', |
|
lat: -41.2887032, |
|
lon: 174.767089 |
|
}, |
|
geometry: { |
|
type: 'Point', |
|
coordinates: [ 174.767089, -41.2887032 ] |
|
} |
|
} |
|
**/ |
|
|
|
// The interpolation middleware layer uses async.map to iterate over the results |
|
// since the interpolation service only operates on single inputs. The problem |
|
// with async.map is that if a single error is returned then the entire batch |
|
// exits early. This function wraps the service call to intercept the error so |
|
// that async.map never returns an error. |
|
function error_intercepting_service(service, req) { |
|
return (street_result, next) => { |
|
service(req, street_result, (err, interpolation_result, metadata) => { |
|
if (err) { |
|
logger.error(`[middleware:interpolation] ${_.defaultTo(err.message, err)}`); |
|
// now that the error has been caught and reported, act as if there was no error |
|
return next(null, null); |
|
} |
|
|
|
// metadata can't be passed as 3rd parameter here, so include in result |
|
if (interpolation_result) { |
|
interpolation_result.metadata = metadata; |
|
} |
|
|
|
// no error occurred, so pass along the result |
|
return next(null, interpolation_result); |
|
}); |
|
}; |
|
} |
|
|
|
function setup(service, should_execute) { |
|
return function controller(req, res, next) { |
|
// bail early if the service shouldn't execute |
|
if (!should_execute(req, res)) { |
|
return next(); |
|
} |
|
|
|
// only interpolate the street-layer results |
|
// save this off into a separate array so that when docs are annotated |
|
// after the interpolate results are returned, no complicated bookkeeping is needed |
|
const street_results = _.get(res, 'data', []).filter(result => result.layer === 'street'); |
|
|
|
// perform interpolations asynchronously for all relevant hits |
|
const start = (new Date()).getTime(); |
|
const initialTime = debugLog.beginTimer(req); |
|
|
|
const startTime = Date.now(); |
|
const logInfo = { |
|
controller: 'interpolation', //technically middleware, but stay consistent with other log lines |
|
street_count: street_results.length, |
|
params: req.clean, |
|
responses: [] |
|
}; |
|
|
|
// call the interpolation service asynchronously on every street result |
|
async.map(street_results, error_intercepting_service(service, req), (err, interpolation_results) => { |
|
// iterate the interpolation results, mapping back into the source results |
|
interpolation_results.forEach((interpolation_result, idx) => { |
|
const source_result = street_results[idx]; |
|
const debugLogInfo = { |
|
outcome: 'hit', //assume hit, update later if not |
|
text: req.clean.parsed_text, |
|
result: interpolation_result |
|
}; |
|
const resultLogInfo = { |
|
outcome: 'hit', //assume hit, update later if not |
|
response_time: _.get(interpolation_result, 'metadata.response_time') |
|
}; |
|
|
|
// invalid / not useful response, debug log for posterity |
|
// note: leave this hit unmodified |
|
if (!_.has(interpolation_result, 'properties')) { |
|
debugLogInfo.outcome = 'miss'; |
|
resultLogInfo.outcome = 'miss'; |
|
logger.debug('interpolation', debugLogInfo); |
|
debugLog.push(req, 'miss'); |
|
return; |
|
} |
|
|
|
// the interpolation service returned a valid result, debug log for posterity |
|
// note: we now merge those values with the existing 'street' record |
|
logger.debug('interpolation', debugLogInfo); |
|
debugLog.push(req, interpolation_result); |
|
|
|
// -- metadata -- |
|
source_result.layer = 'address'; |
|
source_result.match_type = 'interpolated'; |
|
|
|
// -- name -- |
|
source_result.name.default = `${interpolation_result.properties.number} ${source_result.name.default}`; |
|
|
|
// -- source -- |
|
// lookup the lowercased source, defaulting to 'mixed' when not found |
|
// the source mapping is a jagged string->array, so default to 'mixed' as an array |
|
// to ensure that subscript works |
|
source_result.source = _.defaultTo( |
|
source_mapping[_.toLower(interpolation_result.properties.source)], |
|
['mixed'] |
|
)[0]; |
|
|
|
// -- source_id -- |
|
// note: interpolated values have no source_id |
|
delete source_result.source_id; // remove original street source_id |
|
if( interpolation_result.properties.hasOwnProperty( 'source_id' ) ){ |
|
source_result.source_id = interpolation_result.properties.source_id; |
|
} |
|
|
|
// -- address_parts -- |
|
source_result.address_parts.number = interpolation_result.properties.number; |
|
|
|
// -- geo -- |
|
source_result.center_point = { |
|
lat: interpolation_result.properties.lat, |
|
lon: interpolation_result.properties.lon |
|
}; |
|
|
|
// -- bbox -- |
|
delete source_result.bounding_box; |
|
|
|
}); |
|
|
|
// sort the results to ensure that addresses show up higher than street centroids |
|
if (_.has(res, 'data')) { |
|
res.data = stable(res.data, (a, b) => { |
|
if (a.layer === 'address' && b.layer !== 'address') { return -1; } |
|
if (a.layer !== 'address' && b.layer === 'address') { return 1; } |
|
return 0; |
|
}); |
|
} |
|
|
|
|
|
// log and continue |
|
logInfo.total_response_time = Date.now() - startTime; |
|
logger.info('interpolation', logInfo); |
|
debugLog.stopTimer(req, initialTime); |
|
next(); |
|
}); |
|
}; |
|
} |
|
|
|
module.exports = setup;
|
|
|