mirror of https://github.com/pelias/api.git
missinglink
8 years ago
6 changed files with 599 additions and 2 deletions
@ -0,0 +1,172 @@ |
|||||||
|
|
||||||
|
var logger = require( 'pelias-logger' ).get( 'api' ); |
||||||
|
var service = require('../service/language'); |
||||||
|
|
||||||
|
/** |
||||||
|
example response from language web service: |
||||||
|
{ |
||||||
|
"101748479": { |
||||||
|
"wofid": 101748479, |
||||||
|
"placetype": "locality", |
||||||
|
"iso": "DE", |
||||||
|
"area": 0.031614, |
||||||
|
"lineage": { |
||||||
|
"continent_id": 102191581, |
||||||
|
"country_id": 85633111, |
||||||
|
"county_id": 102063261, |
||||||
|
"locality_id": 101748479, |
||||||
|
"macrocounty_id": 404227567, |
||||||
|
"region_id": 85682571 |
||||||
|
}, |
||||||
|
"rowid": 90425, |
||||||
|
"names": { |
||||||
|
"default": "München", |
||||||
|
"eng": "Munich" |
||||||
|
} |
||||||
|
}, |
||||||
|
} |
||||||
|
**/ |
||||||
|
|
||||||
|
function setup() { |
||||||
|
|
||||||
|
var transport = service.findById(); |
||||||
|
var middleware = function(req, res, next) { |
||||||
|
|
||||||
|
// no-op, request did not require a language change
|
||||||
|
if( !isLanguageChangeRequired( req, res ) ){ |
||||||
|
return next(); |
||||||
|
} |
||||||
|
|
||||||
|
// collect a list of parent ids to fetch translations for
|
||||||
|
var ids = extractIds( res ); |
||||||
|
|
||||||
|
// perform language lookup for all relevant ids
|
||||||
|
var timer = (new Date()).getTime(); |
||||||
|
transport.query( ids, function( err, translations ){ |
||||||
|
|
||||||
|
// update documents using a translation map
|
||||||
|
if( err ){ |
||||||
|
logger.error( '[language] [error]', err ); |
||||||
|
} else { |
||||||
|
updateDocs( req, res, translations ); |
||||||
|
} |
||||||
|
|
||||||
|
logger.info( '[language] [took]', (new Date()).getTime() - timer, 'ms' ); |
||||||
|
next(); |
||||||
|
}); |
||||||
|
}; |
||||||
|
|
||||||
|
middleware.transport = transport; |
||||||
|
return middleware; |
||||||
|
} |
||||||
|
|
||||||
|
// collect a list of parent ids to fetch translations for
|
||||||
|
function extractIds( res ){ |
||||||
|
|
||||||
|
// store ids in an object in order to avoid duplicates
|
||||||
|
var ids = {}; |
||||||
|
|
||||||
|
// convenience function for adding a new id to the object
|
||||||
|
function addId(id) { |
||||||
|
ids[id] = true; |
||||||
|
} |
||||||
|
|
||||||
|
// extract all parent ids from documents
|
||||||
|
res.data.forEach( function( doc ){ |
||||||
|
|
||||||
|
// skip invalid records
|
||||||
|
if( !doc || !doc.parent ){ return; } |
||||||
|
|
||||||
|
// iterate over doc.parent.* attributes
|
||||||
|
for( var attr in doc.parent ){ |
||||||
|
|
||||||
|
// match only attributes ending with '_id'
|
||||||
|
var match = attr.match(/_id$/); |
||||||
|
if( !match ){ continue; } |
||||||
|
|
||||||
|
// skip invalid/empty arrays
|
||||||
|
if( !Array.isArray( doc.parent[attr] ) || !doc.parent[attr].length ){ |
||||||
|
continue; |
||||||
|
} |
||||||
|
|
||||||
|
// add each id as a key in the ids object
|
||||||
|
doc.parent[attr].forEach( addId ); |
||||||
|
} |
||||||
|
}); |
||||||
|
|
||||||
|
// return a deduplicated array of ids
|
||||||
|
return Object.keys( ids ); |
||||||
|
} |
||||||
|
|
||||||
|
// update documents using a translation map
|
||||||
|
function updateDocs( req, res, translations ){ |
||||||
|
|
||||||
|
// sanity check arguments
|
||||||
|
if( !req || !res || !res.data || !translations ){ return; } |
||||||
|
|
||||||
|
// this is the target language we will be translating to
|
||||||
|
var requestLanguage = req.language.iso6393; |
||||||
|
|
||||||
|
// iterate over response documents
|
||||||
|
res.data.forEach( function( doc, p ){ |
||||||
|
|
||||||
|
// skip invalid records
|
||||||
|
if( !doc || !doc.parent ){ return; } |
||||||
|
|
||||||
|
// iterate over doc.parent.* attributes
|
||||||
|
for( var attr in doc.parent ){ |
||||||
|
|
||||||
|
// match only attributes ending with '_id'
|
||||||
|
var match = attr.match(/^(.*)_id$/); |
||||||
|
if( !match ){ continue; } |
||||||
|
|
||||||
|
// adminKey is the property name without the '_id'
|
||||||
|
// eg. for 'country_id', adminKey would be 'country'.
|
||||||
|
var adminKey = match[1]; |
||||||
|
var adminValues = doc.parent[adminKey]; |
||||||
|
|
||||||
|
// skip invalid/empty arrays
|
||||||
|
if( !Array.isArray( adminValues ) || !adminValues.length ){ continue; } |
||||||
|
|
||||||
|
// iterate over adminValues (it's an array and can have more than one value)
|
||||||
|
for( var i in adminValues ){ |
||||||
|
|
||||||
|
// find the corresponding key from the '_id' Array
|
||||||
|
var id = doc.parent[attr][i]; |
||||||
|
if( !id ){ continue; } |
||||||
|
|
||||||
|
// id not found in translation service response
|
||||||
|
if( !translations.hasOwnProperty( id ) ){ |
||||||
|
logger.error( '[language] [error]', 'failed to find translations for', id ); |
||||||
|
continue; |
||||||
|
} |
||||||
|
|
||||||
|
// skip invalid records
|
||||||
|
if( !translations[id].hasOwnProperty( 'names' ) ){ continue; } |
||||||
|
|
||||||
|
// requested language is not available
|
||||||
|
if( !translations[id].names.hasOwnProperty( requestLanguage ) ){ |
||||||
|
logger.info( '[language] [info]', 'missing translation', requestLanguage, id ); |
||||||
|
continue; |
||||||
|
} |
||||||
|
|
||||||
|
// translate 'parent.*' property
|
||||||
|
adminValues[i] = translations[id].names[ requestLanguage ]; |
||||||
|
|
||||||
|
// if the record is an admin record we also translate
|
||||||
|
// the 'name.default' property.
|
||||||
|
if( adminKey === doc.layer ){ |
||||||
|
doc.name.default = translations[id].names[ requestLanguage ]; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
}); |
||||||
|
} |
||||||
|
|
||||||
|
// boolean function to check if changing the language is required
|
||||||
|
function isLanguageChangeRequired( req, res ){ |
||||||
|
return req && res && res.data && res.data.length && |
||||||
|
req.hasOwnProperty('language'); |
||||||
|
} |
||||||
|
|
||||||
|
module.exports = setup; |
@ -0,0 +1,93 @@ |
|||||||
|
|
||||||
|
var logger = require( 'pelias-logger' ).get( 'api' ), |
||||||
|
request = require( 'superagent' ), |
||||||
|
peliasConfig = require( 'pelias-config' ); |
||||||
|
|
||||||
|
/** |
||||||
|
|
||||||
|
language subsitution service client |
||||||
|
|
||||||
|
this file provides a 'transport' which can be used to access the language |
||||||
|
service via a network connnection. |
||||||
|
|
||||||
|
the exported method for this module checks pelias-config for a configuration block such as: |
||||||
|
|
||||||
|
"language": { |
||||||
|
"client": { |
||||||
|
"adapter": "http", |
||||||
|
"host": "http://localhost:6100" |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
for more info on running the service see: https://github.com/pelias/placeholder
|
||||||
|
|
||||||
|
**/ |
||||||
|
|
||||||
|
/** |
||||||
|
NullTransport |
||||||
|
|
||||||
|
disables the service completely |
||||||
|
**/ |
||||||
|
function NullTransport(){} |
||||||
|
NullTransport.prototype.query = function( coord, number, street, cb ){ |
||||||
|
cb(); // no-op
|
||||||
|
}; |
||||||
|
|
||||||
|
/** |
||||||
|
HttpTransport |
||||||
|
|
||||||
|
allows the api to be used via a remote web service |
||||||
|
**/ |
||||||
|
function HttpTransport( host, settings ){ |
||||||
|
this.query = function( ids, cb ){ |
||||||
|
request |
||||||
|
.get( host + '/parser/findbyid' ) |
||||||
|
.set( 'Accept', 'application/json' ) |
||||||
|
.query({ ids: Array.isArray( ids ) ? ids.join(',') : '' }) |
||||||
|
.timeout( settings && settings.timeout || 1000 ) |
||||||
|
.end( function( err, res ){ |
||||||
|
if( err || !res ){ return cb( err ); } |
||||||
|
if( 200 !== res.status ){ return cb( 'non 200 status' ); } |
||||||
|
return cb( null, res.body ); |
||||||
|
}); |
||||||
|
}; |
||||||
|
} |
||||||
|
HttpTransport.prototype.query = function( coord, number, street, cb ){ |
||||||
|
throw new Error( 'language: transport not connected' ); |
||||||
|
}; |
||||||
|
|
||||||
|
/** |
||||||
|
Setup |
||||||
|
|
||||||
|
allows instantiation of transport depending on configuration and preference |
||||||
|
**/ |
||||||
|
module.exports.findById = function setup(){ |
||||||
|
|
||||||
|
// user config
|
||||||
|
var config = peliasConfig.generate(); |
||||||
|
|
||||||
|
// ensure config variables set correctly
|
||||||
|
if( !config.hasOwnProperty('language') || !config.language.hasOwnProperty('client') ){ |
||||||
|
logger.warn( 'language: configuration not found' ); |
||||||
|
} |
||||||
|
|
||||||
|
// valid configuration found
|
||||||
|
else { |
||||||
|
|
||||||
|
// get adapter settings from config
|
||||||
|
var settings = config.language.client; |
||||||
|
|
||||||
|
// http adapter
|
||||||
|
if( 'http' === settings.adapter && settings.hasOwnProperty('host') ){ |
||||||
|
logger.info( 'language: using http transport:', settings.host ); |
||||||
|
if( settings.hasOwnProperty('timeout') ){ |
||||||
|
return new HttpTransport( settings.host, { timeout: parseInt( settings.timeout, 10 ) } ); |
||||||
|
} |
||||||
|
return new HttpTransport( settings.host ); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
// default adapter
|
||||||
|
logger.info( 'language: using null transport' ); |
||||||
|
return new NullTransport(); |
||||||
|
}; |
@ -0,0 +1,107 @@ |
|||||||
|
|
||||||
|
var fs = require('fs'), |
||||||
|
tmp = require('tmp'), |
||||||
|
setup = require('../../../service/language').findById; |
||||||
|
|
||||||
|
module.exports.tests = {}; |
||||||
|
|
||||||
|
module.exports.tests.interface = function(test, common) { |
||||||
|
test('valid interface', function(t) { |
||||||
|
t.equal(typeof setup, 'function', 'setup is a function'); |
||||||
|
t.end(); |
||||||
|
}); |
||||||
|
}; |
||||||
|
|
||||||
|
// adapter factory
|
||||||
|
module.exports.tests.factory = function(test, common) { |
||||||
|
|
||||||
|
test('http adapter', function(t) { |
||||||
|
var config = { language: { client: { |
||||||
|
adapter: 'http', |
||||||
|
host: 'http://example.com' |
||||||
|
}}}; |
||||||
|
|
||||||
|
// adapter is driven by config
|
||||||
|
var tmpfile = tmp.tmpNameSync({ postfix: '.json' }); |
||||||
|
fs.writeFileSync( tmpfile, JSON.stringify( config ), { encoding: 'utf8' } ); |
||||||
|
process.env.PELIAS_CONFIG = tmpfile; |
||||||
|
var adapter = setup(); |
||||||
|
delete process.env.PELIAS_CONFIG; |
||||||
|
|
||||||
|
t.equal(adapter.constructor.name, 'HttpTransport', 'HttpTransport'); |
||||||
|
t.equal(typeof adapter, 'object', 'adapter is an object'); |
||||||
|
t.equal(typeof adapter.query, 'function', 'query is a function'); |
||||||
|
t.equal(adapter.query.length, 2, 'query function signature'); |
||||||
|
t.end(); |
||||||
|
}); |
||||||
|
|
||||||
|
test('null adapter', function(t) { |
||||||
|
var config = { language: { client: { |
||||||
|
adapter: 'null' |
||||||
|
}}}; |
||||||
|
|
||||||
|
// adapter is driven by config
|
||||||
|
var tmpfile = tmp.tmpNameSync({ postfix: '.json' }); |
||||||
|
fs.writeFileSync( tmpfile, JSON.stringify( config ), { encoding: 'utf8' } ); |
||||||
|
process.env.PELIAS_CONFIG = tmpfile; |
||||||
|
var adapter = setup(); |
||||||
|
delete process.env.PELIAS_CONFIG; |
||||||
|
|
||||||
|
t.equal(adapter.constructor.name, 'NullTransport', 'NullTransport'); |
||||||
|
t.equal(typeof adapter, 'object', 'adapter is an object'); |
||||||
|
t.equal(typeof adapter.query, 'function', 'query is a function'); |
||||||
|
t.equal(adapter.query.length, 4, 'query function signature'); |
||||||
|
t.end(); |
||||||
|
}); |
||||||
|
|
||||||
|
test('default adapter', function(t) { |
||||||
|
var config = {}; |
||||||
|
|
||||||
|
// adapter is driven by config
|
||||||
|
var tmpfile = tmp.tmpNameSync({ postfix: '.json' }); |
||||||
|
fs.writeFileSync( tmpfile, JSON.stringify( config ), { encoding: 'utf8' } ); |
||||||
|
process.env.PELIAS_CONFIG = tmpfile; |
||||||
|
var adapter = setup(); |
||||||
|
delete process.env.PELIAS_CONFIG; |
||||||
|
|
||||||
|
t.equal(adapter.constructor.name, 'NullTransport', 'NullTransport'); |
||||||
|
t.equal(typeof adapter, 'object', 'adapter is an object'); |
||||||
|
t.equal(typeof adapter.query, 'function', 'query is a function'); |
||||||
|
t.equal(adapter.query.length, 4, 'query function signature'); |
||||||
|
t.end(); |
||||||
|
}); |
||||||
|
|
||||||
|
}; |
||||||
|
|
||||||
|
// null transport
|
||||||
|
module.exports.tests.NullTransport = function(test, common) { |
||||||
|
|
||||||
|
test('null transport', function(t) { |
||||||
|
|
||||||
|
// adapter is driven by config
|
||||||
|
var tmpfile = tmp.tmpNameSync({ postfix: '.json' }); |
||||||
|
fs.writeFileSync( tmpfile, '{}', { encoding: 'utf8' } ); |
||||||
|
process.env.PELIAS_CONFIG = tmpfile; |
||||||
|
var adapter = setup(); |
||||||
|
delete process.env.PELIAS_CONFIG; |
||||||
|
|
||||||
|
// test null transport performs a no-op
|
||||||
|
adapter.query( null, null, null, function( err, res ){ |
||||||
|
t.equal(err, undefined, 'no-op'); |
||||||
|
t.equal(res, undefined, 'no-op'); |
||||||
|
t.end(); |
||||||
|
}); |
||||||
|
}); |
||||||
|
|
||||||
|
}; |
||||||
|
|
||||||
|
module.exports.all = function (tape, common) { |
||||||
|
|
||||||
|
function test(name, testFunction) { |
||||||
|
return tape('SERVICE language', testFunction); |
||||||
|
} |
||||||
|
|
||||||
|
for( var testCase in module.exports.tests ){ |
||||||
|
module.exports.tests[testCase](test, common); |
||||||
|
} |
||||||
|
}; |
Loading…
Reference in new issue