mirror of https://github.com/pelias/api.git
Browse Source
Refactor search and doc controllers to allow for post-processing middleware to handle geojson and response sending. Allows for a much more flexible routing scheme.pull/210/head
Diana Shkolnikov
9 years ago
11 changed files with 159 additions and 185 deletions
@ -0,0 +1,73 @@ |
|||||||
|
var extend = require('extend'); |
||||||
|
var geojsonify = require('../helper/geojsonify').search; |
||||||
|
|
||||||
|
function setup(peliasConfig) { |
||||||
|
|
||||||
|
peliasConfig = peliasConfig || require( 'pelias-config' ).generate().api; |
||||||
|
|
||||||
|
function middleware(req, res, next) { |
||||||
|
return convertToGeocodeJSON(peliasConfig, req, next); |
||||||
|
} |
||||||
|
|
||||||
|
return middleware; |
||||||
|
} |
||||||
|
|
||||||
|
function convertToGeocodeJSON(peliasConfig, req, next) { |
||||||
|
|
||||||
|
req.results.geojson = { geocoding: {} }; |
||||||
|
|
||||||
|
// REQUIRED. A semver.org compliant version number. Describes the version of
|
||||||
|
// the GeocodeJSON spec that is implemented by this instance.
|
||||||
|
req.results.geojson.geocoding.version = '0.1'; |
||||||
|
|
||||||
|
// OPTIONAL. Default: null. The licence of the data. In case of multiple sources,
|
||||||
|
// and then multiple licences, can be an object with one key by source.
|
||||||
|
// Can be a freeform text property describing the licensing details.
|
||||||
|
// Can be a URI on the server, which outlines licensing details.
|
||||||
|
req.results.geojson.geocoding.license = peliasConfig.host + '/license'; // TODO: add to config
|
||||||
|
|
||||||
|
// OPTIONAL. Default: null. The attribution of the data. In case of multiple sources,
|
||||||
|
// and then multiple attributions, can be an object with one key by source.
|
||||||
|
// Can be a URI on the server, which outlines attribution details.
|
||||||
|
req.results.geojson.geocoding.attribution = peliasConfig.host + '/attribution'; // TODO: add to config
|
||||||
|
|
||||||
|
// OPTIONAL. Default: null. The query that has been issued to trigger the
|
||||||
|
// search.
|
||||||
|
// Freeform object.
|
||||||
|
// This is the equivalent of how the engine interpreted the incoming request.
|
||||||
|
// Helpful for debugging and understanding how the input impacts results.
|
||||||
|
req.results.geojson.geocoding.query = req.clean; |
||||||
|
|
||||||
|
// OPTIONAL. Warnings and errors.
|
||||||
|
addMessages(req.results, 'warnings', req.results.geojson.geocoding); |
||||||
|
addMessages(req.results, 'errors', req.results.geojson.geocoding); |
||||||
|
|
||||||
|
// OPTIONAL
|
||||||
|
// Freeform
|
||||||
|
addEngine(peliasConfig, req.results.geojson.geocoding); |
||||||
|
|
||||||
|
// response envelope
|
||||||
|
req.results.geojson.geocoding.timestamp = new Date().getTime(); |
||||||
|
|
||||||
|
// convert docs to geojson and merge with geocoding block
|
||||||
|
extend(req.results.geojson, geojsonify(req.results.data, req.clean)); |
||||||
|
|
||||||
|
next(); |
||||||
|
} |
||||||
|
|
||||||
|
function addMessages(results, msgType, geocoding) { |
||||||
|
if (results.hasOwnProperty(msgType)) { |
||||||
|
geocoding.messages = geocoding.messages || {}; |
||||||
|
geocoding.messages[msgType] = results[msgType]; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
function addEngine(peliasConfig, geocoding) { |
||||||
|
geocoding.engine = { |
||||||
|
name: 'Pelias', |
||||||
|
author: 'Mapzen', |
||||||
|
version: peliasConfig.version // TODO: add to config
|
||||||
|
}; |
||||||
|
} |
||||||
|
|
||||||
|
module.exports = setup; |
@ -0,0 +1,6 @@ |
|||||||
|
function sendJSONResponse(req, res) { |
||||||
|
// respond
|
||||||
|
return res.status(200).json(req.results.geojson); |
||||||
|
} |
||||||
|
|
||||||
|
module.exports = sendJSONResponse; |
@ -1,30 +0,0 @@ |
|||||||
|
|
||||||
var _sanitize = require('../sanitiser/_sanitize'), |
|
||||||
sanitizers = { |
|
||||||
input: require('../sanitiser/_input'), |
|
||||||
size: require('../sanitiser/_size'), |
|
||||||
layers: function( req ) { |
|
||||||
req.query.layers = 'admin'; |
|
||||||
var layers = require('../sanitiser/_layers'); |
|
||||||
return layers(req); |
|
||||||
}, |
|
||||||
latlonzoom: require('../sanitiser/_geo'), |
|
||||||
details: require('../sanitiser/_details') |
|
||||||
}; |
|
||||||
|
|
||||||
var sanitize = function(req, cb) { _sanitize(req, sanitizers, cb); }; |
|
||||||
|
|
||||||
// export sanitize for testing
|
|
||||||
module.exports.sanitize = sanitize; |
|
||||||
|
|
||||||
// middleware
|
|
||||||
module.exports.middleware = function( req, res, next ){ |
|
||||||
sanitize( req, function( err, clean ){ |
|
||||||
if( err ){ |
|
||||||
res.status(400); // 400 Bad Request
|
|
||||||
return next(err); |
|
||||||
} |
|
||||||
req.clean = clean; |
|
||||||
next(); |
|
||||||
}); |
|
||||||
}; |
|
@ -1,29 +0,0 @@ |
|||||||
|
|
||||||
var _sanitize = require('../sanitiser/_sanitize'), |
|
||||||
sanitizers = { |
|
||||||
input: require('../sanitiser/_input'), |
|
||||||
size: require('../sanitiser/_size'), |
|
||||||
layers: require('../sanitiser/_layers'), |
|
||||||
details: require('../sanitiser/_details'), |
|
||||||
latlonzoom: function( req ) { |
|
||||||
var geo = require('../sanitiser/_geo'); |
|
||||||
return geo(req, true); |
|
||||||
} |
|
||||||
}; |
|
||||||
|
|
||||||
var sanitize = function(req, cb) { _sanitize(req, sanitizers, cb); }; |
|
||||||
|
|
||||||
// export sanitize for testing
|
|
||||||
module.exports.sanitize = sanitize; |
|
||||||
|
|
||||||
// middleware
|
|
||||||
module.exports.middleware = function( req, res, next ){ |
|
||||||
sanitize( req, function( err, clean ){ |
|
||||||
if( err ){ |
|
||||||
res.status(400); // 400 Bad Request
|
|
||||||
return next(err); |
|
||||||
} |
|
||||||
req.clean = clean; |
|
||||||
next(); |
|
||||||
}); |
|
||||||
}; |
|
@ -1,72 +0,0 @@ |
|||||||
|
|
||||||
var coarse = require('../../../sanitiser/coarse'), |
|
||||||
_sanitize = coarse.sanitize, |
|
||||||
middleware = coarse.middleware, |
|
||||||
valid_layers = [ 'admin0', 'admin1', 'admin2', 'neighborhood', 'locality', 'local_admin' ], |
|
||||||
defaultClean = require('../sanitiser/_input').defaultClean,
|
|
||||||
sanitize = function(query, cb) { _sanitize({'query':query}, cb); }; |
|
||||||
|
|
||||||
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.layers = function(test, common) { |
|
||||||
test('valid layers', function(t) { |
|
||||||
sanitize({ input: 'test', lat: 0, lon: 0 }, function( err, clean ){ |
|
||||||
t.equal(err, undefined, 'no error'); |
|
||||||
t.deepEqual(clean.layers, valid_layers, 'layers set correctly'); |
|
||||||
}); |
|
||||||
t.end(); |
|
||||||
}); |
|
||||||
}; |
|
||||||
|
|
||||||
module.exports.tests.middleware_failure = function(test, common) { |
|
||||||
test('middleware failure', function(t) { |
|
||||||
var res = { status: function( code ){ |
|
||||||
t.equal(code, 400, 'status set'); |
|
||||||
}}; |
|
||||||
var next = function( message ){ |
|
||||||
var defaultError = 'invalid param \'input\': text length, must be >0'; |
|
||||||
t.equal(message, defaultError); |
|
||||||
t.end(); |
|
||||||
}; |
|
||||||
middleware( {}, res, next ); |
|
||||||
}); |
|
||||||
}; |
|
||||||
|
|
||||||
module.exports.tests.middleware_success = function(test, common) { |
|
||||||
test('middleware success', function(t) { |
|
||||||
var req = { query: { input: 'test', lat: 0, lon: 0 }}; |
|
||||||
var clean = defaultClean; |
|
||||||
clean.layers = valid_layers; |
|
||||||
|
|
||||||
var next = function( message ){ |
|
||||||
t.equal(message, undefined, 'no error message set'); |
|
||||||
t.deepEqual(req.clean, clean); |
|
||||||
t.end(); |
|
||||||
}; |
|
||||||
middleware( req, undefined, next ); |
|
||||||
}); |
|
||||||
}; |
|
||||||
|
|
||||||
module.exports.all = function (tape, common) { |
|
||||||
|
|
||||||
function test(name, testFunction) { |
|
||||||
return tape('SANTIZE /coarse ' + name, testFunction); |
|
||||||
} |
|
||||||
|
|
||||||
for( var testCase in module.exports.tests ){ |
|
||||||
module.exports.tests[testCase](test, common); |
|
||||||
} |
|
||||||
}; |
|
Loading…
Reference in new issue