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
10 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