mirror of https://github.com/pelias/api.git
Diana Shkolnikov
9 years ago
7 changed files with 494 additions and 175 deletions
@ -0,0 +1,42 @@
|
||||
var Document = require('pelias-model').Document; |
||||
|
||||
/** |
||||
* Determine and set place id, type, and source |
||||
* |
||||
* @param {object} src |
||||
* @param {object} dst |
||||
*/ |
||||
function addMetaData(src, dst) { |
||||
dst.id = src._id; |
||||
dst.gid = makeGid(src); |
||||
dst.layer = lookupLayer(src); |
||||
dst.source = lookupSource(src); |
||||
dst.source_id = lookupSourceId(src); |
||||
if (src.hasOwnProperty('bounding_box')) { |
||||
dst.bounding_box = src.bounding_box; |
||||
} |
||||
} |
||||
|
||||
/** |
||||
* Create a gid from a document |
||||
* |
||||
* @param {object} src |
||||
*/ |
||||
function makeGid(src) { |
||||
var doc = new Document(lookupSource(src), lookupLayer(src), src._id); |
||||
return doc.getGid(); |
||||
} |
||||
|
||||
function lookupSource(src) { |
||||
return src.source; |
||||
} |
||||
|
||||
function lookupSourceId(src) { |
||||
return src.source_id; |
||||
} |
||||
|
||||
function lookupLayer(src) { |
||||
return src.layer; |
||||
} |
||||
|
||||
module.exports = addMetaData; |
@ -0,0 +1,133 @@
|
||||
var _ = require('lodash'); |
||||
|
||||
// Properties to be copied
|
||||
// If a property is identified as a single string, assume it should be presented as a string in response
|
||||
// If something other than string is desired, use the following structure: { name: 'category', type: 'array' }
|
||||
var DETAILS_PROPS = [ |
||||
{ name: 'housenumber', type: 'string' }, |
||||
{ name: 'street', type: 'string' }, |
||||
{ name: 'postalcode', type: 'string' }, |
||||
{ name: 'confidence', type: 'default' }, |
||||
{ name: 'distance', type: 'default' }, |
||||
{ name: 'country', type: 'string' }, |
||||
{ name: 'country_gid', type: 'string' }, |
||||
{ name: 'country_a', type: 'string' }, |
||||
{ name: 'macroregion', type: 'string' }, |
||||
{ name: 'macroregion_gid', type: 'string' }, |
||||
{ name: 'macroregion_a', type: 'string' }, |
||||
{ name: 'region', type: 'string' }, |
||||
{ name: 'region_gid', type: 'string' }, |
||||
{ name: 'region_a', type: 'string' }, |
||||
{ name: 'macrocounty', type: 'string' }, |
||||
{ name: 'macrocounty_gid', type: 'string' }, |
||||
{ name: 'macrocounty_a', type: 'string' }, |
||||
{ name: 'county', type: 'string' }, |
||||
{ name: 'county_gid', type: 'string' }, |
||||
{ name: 'county_a', type: 'string' }, |
||||
{ name: 'localadmin', type: 'string' }, |
||||
{ name: 'localadmin_gid', type: 'string' }, |
||||
{ name: 'localadmin_a', type: 'string' }, |
||||
{ name: 'locality', type: 'string' }, |
||||
{ name: 'locality_gid', type: 'string' }, |
||||
{ name: 'locality_a', type: 'string' }, |
||||
{ name: 'borough', type: 'string' }, |
||||
{ name: 'borough_gid', type: 'string' }, |
||||
{ name: 'borough_a', type: 'string' }, |
||||
{ name: 'neighbourhood', type: 'string' }, |
||||
{ name: 'neighbourhood_gid', type: 'string' }, |
||||
{ name: 'bounding_box', type: 'default' }, |
||||
{ name: 'category', type: 'array', condition: checkCategoryParam } |
||||
]; |
||||
|
||||
function checkCategoryParam(params) { |
||||
return _.isObject(params) && params.hasOwnProperty('categories'); |
||||
} |
||||
|
||||
/** |
||||
* Add details properties |
||||
* |
||||
* @param {object} params clean query params |
||||
* @param {object} src |
||||
* @param {object} dst |
||||
*/ |
||||
function addDetails(params, src, dst) { |
||||
copyProperties(params, src, DETAILS_PROPS, dst); |
||||
} |
||||
|
||||
/** |
||||
* Copy specified properties from source to dest. |
||||
* Ignore missing properties. |
||||
* |
||||
* @param {object} params clean query params |
||||
* @param {object} source |
||||
* @param {[]} props |
||||
* @param {object} dst |
||||
*/ |
||||
function copyProperties( params, source, props, dst ) { |
||||
props.forEach( function ( prop ) { |
||||
|
||||
// if condition isn't met, just return without setting the property
|
||||
if (_.isFunction(prop.condition) && !prop.condition(params)) { |
||||
return; |
||||
} |
||||
|
||||
var property = { |
||||
name: prop.name || prop, |
||||
type: prop.type || 'default' |
||||
}; |
||||
|
||||
var value = null; |
||||
if ( source.hasOwnProperty( property.name ) ) { |
||||
|
||||
switch (property.type) { |
||||
case 'string': |
||||
value = getStringValue(source[property.name]); |
||||
break; |
||||
case 'array': |
||||
value = getArrayValue(source[property.name]); |
||||
break; |
||||
// default behavior is to copy property exactly as is
|
||||
default: |
||||
value = source[property.name]; |
||||
} |
||||
|
||||
if (_.isNumber(value) || (value && !_.isEmpty(value))) { |
||||
dst[property.name] = value; |
||||
} |
||||
} |
||||
}); |
||||
} |
||||
|
||||
function getStringValue(property) { |
||||
// isEmpty check works for all types of values: strings, arrays, objects
|
||||
if (_.isEmpty(property)) { |
||||
return ''; |
||||
} |
||||
|
||||
if (_.isString(property)) { |
||||
return property; |
||||
} |
||||
|
||||
// array value, take first item in array (at this time only used for admin values)
|
||||
if (_.isArray(property)) { |
||||
return property[0]; |
||||
} |
||||
|
||||
return _.toString(property); |
||||
} |
||||
|
||||
|
||||
function getArrayValue(property) { |
||||
// isEmpty check works for all types of values: strings, arrays, objects
|
||||
if (_.isEmpty(property)) { |
||||
return ''; |
||||
} |
||||
|
||||
if (_.isArray(property)) { |
||||
return property; |
||||
} |
||||
|
||||
return [property]; |
||||
} |
||||
|
||||
module.exports = addDetails; |
@ -0,0 +1,29 @@
|
||||
|
||||
var type_mapping = require('../helper/type_mapping'); |
||||
var sanitizeAll = require('../sanitiser/sanitizeAll'), |
||||
sanitizers = { |
||||
quattroshapes_deprecation: require('../sanitiser/_deprecate_quattroshapes'), |
||||
singleScalarParameters: require('../sanitiser/_single_scalar_parameters'), |
||||
layers: require('../sanitiser/_targets')('layers', type_mapping.layer_mapping), |
||||
sources: require('../sanitiser/_targets')('sources', type_mapping.source_mapping), |
||||
// depends on the layers and sources sanitisers, must be run after them
|
||||
sources_and_layers: require('../sanitiser/_sources_and_layers'), |
||||
size: require('../sanitiser/_size')(/* use defaults*/), |
||||
private: require('../sanitiser/_flag_bool')('private', false), |
||||
geo_reverse: require('../sanitiser/_geo_reverse'), |
||||
boundary_country: require('../sanitiser/_boundary_country'), |
||||
categories: require('../sanitiser/_categories') |
||||
}; |
||||
|
||||
var sanitize = function(req, cb) { sanitizeAll(req, sanitizers, cb); }; |
||||
|
||||
// export sanitize for testing
|
||||
module.exports.sanitize = sanitize; |
||||
module.exports.sanitiser_list = sanitizers; |
||||
|
||||
// middleware
|
||||
module.exports.middleware = function( req, res, next ){ |
||||
sanitize( req, function( err, clean ){ |
||||
next(); |
||||
}); |
||||
}; |
@ -0,0 +1,204 @@
|
||||
|
||||
// @todo: refactor this test, it's pretty messy, brittle and hard to follow
|
||||
|
||||
var reverse = require('../../../sanitiser/reverse'), |
||||
sanitize = reverse.sanitize, |
||||
middleware = reverse.middleware, |
||||
defaults = require('../../../query/reverse_defaults'), |
||||
defaultError = 'missing param \'lat\'', |
||||
defaultClean = { 'point.lat': 0, |
||||
'point.lon': 0, |
||||
'boundary.circle.lat': 0, |
||||
'boundary.circle.lon': 0, |
||||
'boundary.circle.radius': parseFloat(defaults['boundary:circle:radius']), |
||||
size: 10, |
||||
private: false |
||||
}; |
||||
|
||||
// these are the default values you would expect when no input params are specified.
|
||||
// @todo: why is this different from $defaultClean?
|
||||
var emptyClean = { private: false, size: 10 }; |
||||
|
||||
module.exports.tests = {}; |
||||
|
||||
module.exports.tests.interface = function(test, common) { |
||||
test('sanitize interface', function(t) { |
||||
t.equal(typeof sanitize, 'function', 'sanitize is a function'); |
||||
t.equal(sanitize.length, 2, 'sanitize interface'); |
||||
t.end(); |
||||
}); |
||||
test('middleware interface', function(t) { |
||||
t.equal(typeof middleware, 'function', 'middleware is a function'); |
||||
t.equal(middleware.length, 3, 'sanitizee has a valid middleware'); |
||||
t.end(); |
||||
}); |
||||
}; |
||||
|
||||
module.exports.tests.sanitisers = function(test, common) { |
||||
test('check sanitiser list', function (t) { |
||||
var expected = ['quattroshapes_deprecation', 'singleScalarParameters', 'layers', |
||||
'sources', 'sources_and_layers', 'size', 'private', 'geo_reverse', 'boundary_country', 'categories']; |
||||
t.deepEqual(Object.keys(reverse.sanitiser_list), expected); |
||||
t.end(); |
||||
}); |
||||
}; |
||||
|
||||
module.exports.tests.sanitize_lat = function(test, common) { |
||||
var lats = { |
||||
invalid: [], |
||||
valid: [ 0, 45, 90, -0, '0', '45', '90', -181, -120, -91, 91, 120, 181 ], |
||||
missing: ['', undefined, null] |
||||
}; |
||||
test('invalid lat', function(t) { |
||||
lats.invalid.forEach( function( lat ){ |
||||
var req = { query: { 'point.lat': lat, 'point.lon': 0 } }; |
||||
sanitize(req, function(){ |
||||
t.equal(req.errors[0], 'invalid param \'point.lat\': must be >-90 and <90', lat + ' is an invalid latitude'); |
||||
t.deepEqual(req.clean, emptyClean, 'clean only has default values set'); |
||||
}); |
||||
}); |
||||
t.end(); |
||||
}); |
||||
test('valid lat', function(t) { |
||||
lats.valid.forEach( function( lat ){ |
||||
var req = { query: { 'point.lat': lat, 'point.lon': 0 } }; |
||||
sanitize(req, function(){ |
||||
var expected_lat = parseFloat( lat ); |
||||
t.deepEqual(req.errors, [], 'no errors'); |
||||
t.equal(req.clean['point.lat'], expected_lat, 'clean set correctly (' + lat + ')'); |
||||
}); |
||||
}); |
||||
t.end(); |
||||
}); |
||||
test('missing lat', function(t) { |
||||
lats.missing.forEach( function( lat ){ |
||||
var req = { query: { 'point.lat': lat, 'point.lon': 0 } }; |
||||
sanitize(req, function(){ |
||||
t.equal(req.errors[0], 'missing param \'point.lat\'', 'latitude is a required field'); |
||||
t.deepEqual(req.clean, emptyClean, 'clean only has default values set'); |
||||
}); |
||||
}); |
||||
t.end(); |
||||
}); |
||||
}; |
||||
|
||||
module.exports.tests.sanitize_lon = function(test, common) { |
||||
var lons = { |
||||
valid: [ -360, -181, 181, -180, -1, -0, 0, 45, 90, '-180', '0', '180' ], |
||||
missing: ['', undefined, null] |
||||
}; |
||||
test('valid lon', function(t) { |
||||
lons.valid.forEach( function( lon ){ |
||||
var req = { query: { 'point.lat': 0, 'point.lon': lon } }; |
||||
sanitize(req, function(){ |
||||
var expected_lon = parseFloat( lon ); |
||||
t.deepEqual(req.errors, [], 'no errors'); |
||||
t.equal(req.clean['point.lon'], expected_lon, 'clean set correctly (' + lon + ')'); |
||||
}); |
||||
}); |
||||
t.end(); |
||||
}); |
||||
test('missing lon', function(t) { |
||||
lons.missing.forEach( function( lon ){ |
||||
var req = { query: { 'point.lat': 0, 'point.lon': lon } }; |
||||
|
||||
// @todo: why is lat set?
|
||||
var expected = { 'point.lat': 0, private: false, size: 10 }; |
||||
sanitize(req, function(){ |
||||
t.equal(req.errors[0], 'missing param \'point.lon\'', 'longitude is a required field'); |
||||
t.deepEqual(req.clean, expected, 'clean only has default values set'); |
||||
}); |
||||
}); |
||||
t.end(); |
||||
}); |
||||
}; |
||||
|
||||
module.exports.tests.sanitize_size = function(test, common) { |
||||
test('invalid size value', function(t) { |
||||
var req = { query: { size: 'a', 'point.lat': 0, 'point.lon': 0 } }; |
||||
sanitize(req, function(){ |
||||
t.equal(req.clean.size, 10, 'default size set'); |
||||
t.end(); |
||||
}); |
||||
}); |
||||
test('below min size value', function(t) { |
||||
var req = { query: { size: -100, 'point.lat': 0, 'point.lon': 0 } }; |
||||
sanitize(req, function(){ |
||||
t.equal(req.clean.size, 1, 'min size set'); |
||||
t.end(); |
||||
}); |
||||
}); |
||||
test('above max size value', function(t) { |
||||
var req = { query: { size: 9999, 'point.lat': 0, 'point.lon': 0 } }; |
||||
sanitize(req, function(){ |
||||
t.equal(req.clean.size, 40, 'max size set'); |
||||
t.end(); |
||||
}); |
||||
}); |
||||
}; |
||||
|
||||
module.exports.tests.sanitize_private = function(test, common) { |
||||
var invalid_values = [null, -1, 123, NaN, 'abc']; |
||||
invalid_values.forEach(function(value) { |
||||
test('invalid private param ' + value, function(t) { |
||||
var req = { query: { 'point.lat': 0, 'point.lon': 0, 'private': value } }; |
||||
sanitize(req, function(){ |
||||
t.equal(req.clean.private, false, 'default private set (to false)'); |
||||
t.end(); |
||||
}); |
||||
}); |
||||
}); |
||||
|
||||
var valid_values = ['true', true, 1, '1']; |
||||
valid_values.forEach(function(value) { |
||||
test('valid private param ' + value, function(t) { |
||||
var req = { query: { 'point.lat': 0, 'point.lon': 0, 'private': value } }; |
||||
sanitize(req, function(){ |
||||
t.equal(req.clean.private, true, 'private set to true'); |
||||
t.end(); |
||||
}); |
||||
}); |
||||
}); |
||||
|
||||
var valid_false_values = ['false', false, 0]; |
||||
valid_false_values.forEach(function(value) { |
||||
test('test setting false explicitly ' + value, function(t) { |
||||
var req = { query: { 'point.lat': 0, 'point.lon': 0, 'private': value } }; |
||||
sanitize(req, function(){ |
||||
t.equal(req.clean.private, false, 'private set to false'); |
||||
t.end(); |
||||
}); |
||||
}); |
||||
}); |
||||
|
||||
test('test default behavior', function(t) { |
||||
var req = { query: { 'point.lat': 0, 'point.lon': 0 } }; |
||||
sanitize(req, function(){ |
||||
t.equal(req.clean.private, false, 'private set to false'); |
||||
t.end(); |
||||
}); |
||||
}); |
||||
}; |
||||
|
||||
module.exports.tests.middleware_success = function(test, common) { |
||||
test('middleware success', function(t) { |
||||
var req = { query: { 'point.lat': 0, 'point.lon': 0 }}; |
||||
var next = function(){ |
||||
t.deepEqual(req.errors, [], 'no error message set'); |
||||
t.deepEqual(req.clean, defaultClean); |
||||
t.end(); |
||||
}; |
||||
middleware( req, undefined, next ); |
||||
}); |
||||
}; |
||||
|
||||
module.exports.all = function (tape, common) { |
||||
|
||||
function test(name, testFunction) { |
||||
return tape('SANTIZE /reverse ' + name, testFunction); |
||||
} |
||||
|
||||
for( var testCase in module.exports.tests ){ |
||||
module.exports.tests[testCase](test, common); |
||||
} |
||||
}; |
Loading…
Reference in new issue