var check = require('check-types'), es = require('elasticsearch'), logger = require( 'pelias-logger' ).get( 'api' ), exceptions = require('elasticsearch-exceptions/lib/exceptions/SupportedExceptions'); // create a list of regular expressions to match against. // note: list created when the server starts up; for performance reasons. var exceptionRegexList = exceptions.map( function( exceptionName ){ return new RegExp( '^' + exceptionName ); }); function sendJSONResponse(req, res, next) { // do nothing if no result data set if (!res || !check.object(res.body) || !check.object(res.body.geocoding)) { return next(); } // default status var statusCode = 200; // 200 OK // vary status code whenever an error was reported var geocoding = res.body.geocoding; if( check.array( geocoding.errors ) && geocoding.errors.length ){ // default status for errors is 400 Bad Request statusCode = 400; // 400 Bad Request // iterate over all reported errors geocoding.errors.forEach( function( err ){ // custom status codes for instances of the Error() object. if( err instanceof Error ){ /* elasticsearch errors see: https://github.com/elastic/elasticsearch-js/blob/master/src/lib/errors.js 408 Request Timeout 500 Internal Server Error 502 Bad Gateway */ if( err instanceof es.errors.RequestTimeout ){ statusCode = Math.max( statusCode, 408 ); } else if( err instanceof es.errors.NoConnections ){ statusCode = Math.max( statusCode, 502 ); } else if( err instanceof es.errors.ConnectionFault ){ statusCode = Math.max( statusCode, 502 ); } else { logger.warn( 'unknown geocoding error object:', err.constructor.name, err ); statusCode = Math.max( statusCode, 500 ); } /* some elasticsearch errors are only returned as strings (not instances of Error). in this case we (unfortunately) need to match the exception at position 0 inside the string. */ } else if( check.string( err ) ){ for( var i=0; i<exceptionRegexList.length; i++ ){ // check error string against a list of known elasticsearch exceptions if( err.match( exceptionRegexList[i] ) ){ statusCode = Math.max( statusCode, 500 ); break; // break on first match } } logger.warn( 'unknown geocoding error string:', err ); } else { logger.warn( 'unknown geocoding error type:', typeof err, err ); } }); } // respond return res.status(statusCode).json(res.body); } module.exports = sendJSONResponse;