mirror of https://github.com/pelias/api.git
Stephen Hess
7 years ago
2 changed files with 752 additions and 0 deletions
@ -0,0 +1,127 @@
|
||||
'use strict'; |
||||
|
||||
const _ = require('lodash'); |
||||
|
||||
const searchService = require('../service/search'); |
||||
const logger = require('pelias-logger').get('api'); |
||||
const logging = require( '../helper/logging' ); |
||||
const retry = require('retry'); |
||||
const Debug = require('../helper/debug'); |
||||
|
||||
function isRequestTimeout(err) { |
||||
return _.get(err, 'status') === 408; |
||||
} |
||||
|
||||
function setup( apiConfig, esclient, query, should_execute ){ |
||||
function controller( req, res, next ){ |
||||
if (!should_execute(req, res)) { |
||||
return next(); |
||||
} |
||||
|
||||
const debugLog = new Debug('controller:search'); |
||||
|
||||
let cleanOutput = _.cloneDeep(req.clean); |
||||
if (logging.isDNT(req)) { |
||||
cleanOutput = logging.removeFields(cleanOutput); |
||||
} |
||||
// log clean parameters for stats
|
||||
logger.info('[req]', 'endpoint=' + req.path, cleanOutput); |
||||
|
||||
const renderedQuery = query(req.clean); |
||||
|
||||
// if there's no query to call ES with, skip the service
|
||||
if (_.isUndefined(renderedQuery)) { |
||||
debugLog.push(req, 'No query to call ES with. Skipping'); |
||||
return next(); |
||||
} |
||||
|
||||
// options for retry
|
||||
// maxRetries is from the API config with default of 3
|
||||
// factor of 1 means that each retry attempt will esclient requestTimeout
|
||||
const operationOptions = { |
||||
retries: _.get(apiConfig, 'requestRetries', 3), |
||||
factor: 1, |
||||
minTimeout: _.get(esclient, 'transport.requestTimeout') |
||||
}; |
||||
|
||||
// setup a new operation
|
||||
const operation = retry.operation(operationOptions); |
||||
|
||||
// elasticsearch command
|
||||
const cmd = { |
||||
index: apiConfig.indexName, |
||||
searchType: 'dfs_query_then_fetch', |
||||
body: renderedQuery.body |
||||
}; |
||||
|
||||
logger.debug( '[ES req]', cmd ); |
||||
debugLog.push(req, {ES_req: cmd}); |
||||
|
||||
operation.attempt((currentAttempt) => { |
||||
const initialTime = debugLog.beginTimer(req, `Attempt ${currentAttempt}`); |
||||
// query elasticsearch
|
||||
searchService( esclient, cmd, function( err, docs, meta ){ |
||||
// returns true if the operation should be attempted again
|
||||
// (handles bookkeeping of maxRetries)
|
||||
// only consider for status 408 (request timeout)
|
||||
if (isRequestTimeout(err) && operation.retry(err)) { |
||||
logger.info(`request timed out on attempt ${currentAttempt}, retrying`); |
||||
debugLog.stopTimer(req, initialTime, 'request timed out, retrying'); |
||||
return; |
||||
} |
||||
|
||||
// if execution has gotten this far then one of three things happened:
|
||||
// - the request didn't time out
|
||||
// - maxRetries has been hit so we're giving up
|
||||
// - another error occurred
|
||||
// in either case, handle the error or results
|
||||
|
||||
// error handler
|
||||
if( err ){ |
||||
if (_.isObject(err) && err.message) { |
||||
req.errors.push( err.message ); |
||||
} else { |
||||
req.errors.push( err ); |
||||
} |
||||
} |
||||
// set response data
|
||||
else { |
||||
// log that a retry was successful
|
||||
// most requests succeed on first attempt so this declutters log files
|
||||
if (currentAttempt > 1) { |
||||
logger.info(`succeeded on retry ${currentAttempt-1}`); |
||||
} |
||||
|
||||
const messageParts = [ |
||||
'[controller:search]', |
||||
`[queryType:${renderedQuery.type}]`, |
||||
`[es_result_count:${_.defaultTo(docs, []).length}]` |
||||
]; |
||||
|
||||
// if there are docs, concatenate them onto the end of existing results
|
||||
if (docs) { |
||||
res.data = _.concat(_.defaultTo(res.data, []), docs); |
||||
} |
||||
res.meta = meta || {}; |
||||
// store the query_type for subsequent middleware
|
||||
res.meta.query_type = renderedQuery.type; |
||||
|
||||
logger.info(messageParts.join(' ')); |
||||
debugLog.push(req, {queryType: { |
||||
[renderedQuery.type] : { |
||||
es_result_count: parseInt(messageParts[2].slice(17, -1)) |
||||
} |
||||
}}); |
||||
} |
||||
logger.debug('[ES response]', docs); |
||||
next(); |
||||
}); |
||||
debugLog.stopTimer(req, initialTime); |
||||
}); |
||||
|
||||
} |
||||
|
||||
return controller; |
||||
} |
||||
|
||||
module.exports = setup; |
@ -0,0 +1,625 @@
|
||||
'use strict'; |
||||
|
||||
const setup = require('../../../controller/search_with_appending_results'); |
||||
const proxyquire = require('proxyquire').noCallThru(); |
||||
|
||||
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.equal(typeof setup(), 'function', 'setup returns a controller'); |
||||
t.end(); |
||||
}); |
||||
}; |
||||
|
||||
module.exports.tests.success = function(test, common) { |
||||
test('successful request to search service should set data and meta', (t) => { |
||||
const config = { |
||||
indexName: 'indexName value' |
||||
}; |
||||
const esclient = 'this is the esclient'; |
||||
const query = () => { |
||||
return { |
||||
body: 'this is the query body', |
||||
type: 'this is the query type' |
||||
}; |
||||
}; |
||||
|
||||
// request timeout messages willl be written here
|
||||
const infoMesssages = []; |
||||
|
||||
// a controller that validates the esclient and cmd that was passed to the search service
|
||||
const controller = proxyquire('../../../controller/search_with_appending_results', { |
||||
'../service/search': (esclient, cmd, callback) => { |
||||
t.equal(esclient, 'this is the esclient'); |
||||
t.deepEqual(cmd, { |
||||
index: 'indexName value', |
||||
searchType: 'dfs_query_then_fetch', |
||||
body: 'this is the query body' |
||||
}); |
||||
|
||||
const docs = [{}, {}]; |
||||
const meta = { key: 'value' }; |
||||
|
||||
callback(undefined, docs, meta); |
||||
}, |
||||
'pelias-logger': { |
||||
get: (service) => { |
||||
t.equal(service, 'api'); |
||||
return { |
||||
info: (msg) => { |
||||
infoMesssages.push(msg); |
||||
}, |
||||
debug: () => {} |
||||
}; |
||||
} |
||||
} |
||||
})(config, esclient, query, () => { return true; }); |
||||
|
||||
const req = { clean: { }, errors: [], warnings: [] }; |
||||
const res = {}; |
||||
|
||||
const next = () => { |
||||
t.deepEqual(req, { |
||||
clean: {}, |
||||
errors: [], |
||||
warnings: [] |
||||
}); |
||||
t.deepEquals(res.data, [{}, {}]); |
||||
t.deepEquals(res.meta, { key: 'value', query_type: 'this is the query type' }); |
||||
|
||||
t.ok(infoMesssages.find((msg) => { |
||||
return msg === '[controller:search] [queryType:this is the query type] [es_result_count:2]'; |
||||
})); |
||||
t.end(); |
||||
}; |
||||
|
||||
controller(req, res, next); |
||||
|
||||
}); |
||||
|
||||
test('successful request to search service should append to existing results', (t) => { |
||||
const config = { |
||||
indexName: 'indexName value' |
||||
}; |
||||
const esclient = 'this is the esclient'; |
||||
const query = () => { |
||||
return { |
||||
body: 'this is the query body', |
||||
type: 'this is the query type' |
||||
}; |
||||
}; |
||||
|
||||
// request timeout messages willl be written here
|
||||
const infoMesssages = []; |
||||
|
||||
// a controller that validates the esclient and cmd that was passed to the search service
|
||||
const controller = proxyquire('../../../controller/search_with_appending_results', { |
||||
'../service/search': (esclient, cmd, callback) => { |
||||
t.equal(esclient, 'this is the esclient'); |
||||
t.deepEqual(cmd, { |
||||
index: 'indexName value', |
||||
searchType: 'dfs_query_then_fetch', |
||||
body: 'this is the query body' |
||||
}); |
||||
|
||||
const docs = [{ b: 2 }, { c: 3 }]; |
||||
const meta = { key: 'value' }; |
||||
|
||||
callback(undefined, docs, meta); |
||||
}, |
||||
'pelias-logger': { |
||||
get: (service) => { |
||||
t.equal(service, 'api'); |
||||
return { |
||||
info: (msg) => { |
||||
infoMesssages.push(msg); |
||||
}, |
||||
debug: () => {} |
||||
}; |
||||
} |
||||
} |
||||
})(config, esclient, query, () => { return true; }); |
||||
|
||||
const req = { clean: { }, errors: [], warnings: [] }; |
||||
// res.data is prepopulated with 1 result
|
||||
const res = { |
||||
data: [ |
||||
{ a: 1 } |
||||
] |
||||
}; |
||||
|
||||
const next = () => { |
||||
t.deepEqual(req, { |
||||
clean: {}, |
||||
errors: [], |
||||
warnings: [] |
||||
}); |
||||
t.deepEquals(res.data, [{ a: 1 }, { b: 2 }, { c: 3 }], 'results should be concatenated'); |
||||
t.deepEquals(res.meta, { key: 'value', query_type: 'this is the query type' }); |
||||
|
||||
t.ok(infoMesssages.find((msg) => { |
||||
return msg === '[controller:search] [queryType:this is the query type] [es_result_count:2]'; |
||||
})); |
||||
t.end(); |
||||
}; |
||||
|
||||
controller(req, res, next); |
||||
|
||||
}); |
||||
|
||||
test('undefined meta should set empty object into res', (t) => { |
||||
const config = { |
||||
indexName: 'indexName value' |
||||
}; |
||||
const esclient = 'this is the esclient'; |
||||
const query = () => { |
||||
return { |
||||
body: 'this is the query body', |
||||
type: 'this is the query type' |
||||
}; |
||||
}; |
||||
|
||||
// request timeout messages willl be written here
|
||||
const infoMesssages = []; |
||||
|
||||
// a controller that validates the esclient and cmd that was passed to the search service
|
||||
const controller = proxyquire('../../../controller/search_with_appending_results', { |
||||
'../service/search': (esclient, cmd, callback) => { |
||||
t.equal(esclient, 'this is the esclient'); |
||||
t.deepEqual(cmd, { |
||||
index: 'indexName value', |
||||
searchType: 'dfs_query_then_fetch', |
||||
body: 'this is the query body' |
||||
}); |
||||
|
||||
const docs = [{}, {}]; |
||||
|
||||
callback(undefined, docs, undefined); |
||||
}, |
||||
'pelias-logger': { |
||||
get: (service) => { |
||||
t.equal(service, 'api'); |
||||
return { |
||||
info: (msg) => { |
||||
infoMesssages.push(msg); |
||||
}, |
||||
debug: () => {} |
||||
}; |
||||
} |
||||
} |
||||
})(config, esclient, query, () => { return true; }); |
||||
|
||||
const req = { clean: { }, errors: [], warnings: [] }; |
||||
const res = {}; |
||||
|
||||
const next = () => { |
||||
t.deepEqual(req, { |
||||
clean: {}, |
||||
errors: [], |
||||
warnings: [] |
||||
}); |
||||
t.deepEquals(res.data, [{}, {}]); |
||||
t.deepEquals(res.meta, { query_type: 'this is the query type' }); |
||||
|
||||
t.ok(infoMesssages.find((msg) => { |
||||
return msg === '[controller:search] [queryType:this is the query type] [es_result_count:2]'; |
||||
})); |
||||
t.end(); |
||||
}; |
||||
|
||||
controller(req, res, next); |
||||
|
||||
}); |
||||
|
||||
test('undefined docs should log 0 results', (t) => { |
||||
const config = { |
||||
indexName: 'indexName value' |
||||
}; |
||||
const esclient = 'this is the esclient'; |
||||
const query = () => { |
||||
return { |
||||
body: 'this is the query body', |
||||
type: 'this is the query type' |
||||
}; |
||||
}; |
||||
|
||||
// request timeout messages willl be written here
|
||||
const infoMesssages = []; |
||||
|
||||
// a controller that validates the esclient and cmd that was passed to the search service
|
||||
const controller = proxyquire('../../../controller/search_with_appending_results', { |
||||
'../service/search': (esclient, cmd, callback) => { |
||||
t.equal(esclient, 'this is the esclient'); |
||||
t.deepEqual(cmd, { |
||||
index: 'indexName value', |
||||
searchType: 'dfs_query_then_fetch', |
||||
body: 'this is the query body' |
||||
}); |
||||
|
||||
const meta = { key: 'value' }; |
||||
|
||||
callback(undefined, undefined, meta); |
||||
}, |
||||
'pelias-logger': { |
||||
get: (service) => { |
||||
t.equal(service, 'api'); |
||||
return { |
||||
info: (msg) => { |
||||
infoMesssages.push(msg); |
||||
}, |
||||
debug: () => {} |
||||
}; |
||||
} |
||||
} |
||||
})(config, esclient, query, () => { return true; }); |
||||
|
||||
const req = { clean: { }, errors: [], warnings: [] }; |
||||
const res = {}; |
||||
|
||||
const next = () => { |
||||
t.deepEqual(req, { |
||||
clean: {}, |
||||
errors: [], |
||||
warnings: [] |
||||
}); |
||||
t.equals(res.data, undefined); |
||||
t.deepEquals(res.meta, { key: 'value', query_type: 'this is the query type' }); |
||||
|
||||
t.ok(infoMesssages.find((msg) => { |
||||
return msg === '[controller:search] [queryType:this is the query type] [es_result_count:0]'; |
||||
})); |
||||
t.end(); |
||||
}; |
||||
|
||||
controller(req, res, next); |
||||
|
||||
}); |
||||
|
||||
test('successful request on retry to search service should log info message', (t) => { |
||||
const config = { |
||||
indexName: 'indexName value' |
||||
}; |
||||
const esclient = 'this is the esclient'; |
||||
const query = () => { |
||||
return { |
||||
body: 'this is the query body', |
||||
type: 'this is the query type' |
||||
}; |
||||
}; |
||||
|
||||
let searchServiceCallCount = 0; |
||||
|
||||
const timeoutError = { |
||||
status: 408, |
||||
displayName: 'RequestTimeout', |
||||
message: 'Request Timeout after 17ms' |
||||
}; |
||||
|
||||
// request timeout messages willl be written here
|
||||
const infoMesssages = []; |
||||
|
||||
// a controller that validates the esclient and cmd that was passed to the search service
|
||||
const controller = proxyquire('../../../controller/search_with_appending_results', { |
||||
'../service/search': (esclient, cmd, callback) => { |
||||
t.equal(esclient, 'this is the esclient'); |
||||
t.deepEqual(cmd, { |
||||
index: 'indexName value', |
||||
searchType: 'dfs_query_then_fetch', |
||||
body: 'this is the query body' |
||||
}); |
||||
|
||||
if (searchServiceCallCount < 2) { |
||||
// note that the searchService got called
|
||||
searchServiceCallCount++; |
||||
callback(timeoutError); |
||||
} else { |
||||
const docs = [{}, {}]; |
||||
const meta = { key: 'value' }; |
||||
|
||||
callback(undefined, docs, meta); |
||||
} |
||||
|
||||
}, |
||||
'pelias-logger': { |
||||
get: (service) => { |
||||
t.equal(service, 'api'); |
||||
return { |
||||
info: (msg) => { |
||||
infoMesssages.push(msg); |
||||
}, |
||||
debug: () => {} |
||||
}; |
||||
} |
||||
} |
||||
})(config, esclient, query, () => { return true; }); |
||||
|
||||
const req = { clean: { }, errors: [], warnings: [] }; |
||||
const res = {}; |
||||
|
||||
const next = () => { |
||||
t.deepEqual(req, { |
||||
clean: {}, |
||||
errors: [], |
||||
warnings: [] |
||||
}); |
||||
t.deepEquals(res.data, [{}, {}]); |
||||
t.deepEquals(res.meta, { key: 'value', query_type: 'this is the query type' }); |
||||
|
||||
t.ok(infoMesssages.find((msg) => { |
||||
return msg === '[controller:search] [queryType:this is the query type] [es_result_count:2]'; |
||||
})); |
||||
|
||||
t.ok(infoMesssages.find((msg) => { |
||||
return msg === 'succeeded on retry 2'; |
||||
})); |
||||
|
||||
t.end(); |
||||
}; |
||||
|
||||
controller(req, res, next); |
||||
|
||||
}); |
||||
|
||||
}; |
||||
|
||||
module.exports.tests.timeout = function(test, common) { |
||||
test('default # of request timeout retries should be 3', (t) => { |
||||
const config = { |
||||
indexName: 'indexName value' |
||||
}; |
||||
const esclient = 'this is the esclient'; |
||||
const query = () => { |
||||
return { body: 'this is the query body' }; |
||||
}; |
||||
|
||||
let searchServiceCallCount = 0; |
||||
|
||||
const timeoutError = { |
||||
status: 408, |
||||
displayName: 'RequestTimeout', |
||||
message: 'Request Timeout after 17ms' |
||||
}; |
||||
|
||||
// request timeout messages willl be written here
|
||||
const infoMesssages = []; |
||||
|
||||
// a controller that validates the esclient and cmd that was passed to the search service
|
||||
const controller = proxyquire('../../../controller/search_with_appending_results', { |
||||
'../service/search': (esclient, cmd, callback) => { |
||||
t.equal(esclient, 'this is the esclient'); |
||||
t.deepEqual(cmd, { |
||||
index: 'indexName value', |
||||
searchType: 'dfs_query_then_fetch', |
||||
body: 'this is the query body' |
||||
}); |
||||
|
||||
// not that the searchService got called
|
||||
searchServiceCallCount++; |
||||
|
||||
callback(timeoutError); |
||||
}, |
||||
'pelias-logger': { |
||||
get: (service) => { |
||||
t.equal(service, 'api'); |
||||
return { |
||||
info: (msg) => { |
||||
infoMesssages.push(msg); |
||||
}, |
||||
debug: () => {} |
||||
}; |
||||
} |
||||
} |
||||
})(config, esclient, query, () => { return true; }); |
||||
|
||||
const req = { clean: { }, errors: [], warnings: [] }; |
||||
const res = {}; |
||||
|
||||
const next = () => { |
||||
t.equal(searchServiceCallCount, 3+1); |
||||
|
||||
t.ok(infoMesssages.indexOf('request timed out on attempt 1, retrying') !== -1); |
||||
t.ok(infoMesssages.indexOf('request timed out on attempt 2, retrying') !== -1); |
||||
t.ok(infoMesssages.indexOf('request timed out on attempt 3, retrying') !== -1); |
||||
|
||||
t.deepEqual(req, { |
||||
clean: {}, |
||||
errors: [timeoutError.message], |
||||
warnings: [] |
||||
}); |
||||
t.deepEqual(res, {}); |
||||
t.end(); |
||||
}; |
||||
|
||||
controller(req, res, next); |
||||
|
||||
}); |
||||
|
||||
test('explicit apiConfig.requestRetries should retry that many times', (t) => { |
||||
const config = { |
||||
indexName: 'indexName value', |
||||
requestRetries: 17 |
||||
}; |
||||
const esclient = 'this is the esclient'; |
||||
const query = () => { |
||||
return { }; |
||||
}; |
||||
|
||||
let searchServiceCallCount = 0; |
||||
|
||||
const timeoutError = { |
||||
status: 408, |
||||
displayName: 'RequestTimeout', |
||||
message: 'Request Timeout after 17ms' |
||||
}; |
||||
|
||||
// a controller that validates the esclient and cmd that was passed to the search service
|
||||
const controller = proxyquire('../../../controller/search_with_appending_results', { |
||||
'../service/search': (esclient, cmd, callback) => { |
||||
// not that the searchService got called
|
||||
searchServiceCallCount++; |
||||
|
||||
callback(timeoutError); |
||||
} |
||||
})(config, esclient, query, () => { return true; }); |
||||
|
||||
const req = { clean: { }, errors: [], warnings: [] }; |
||||
const res = {}; |
||||
|
||||
const next = () => { |
||||
t.equal(searchServiceCallCount, 17+1); |
||||
t.end(); |
||||
}; |
||||
|
||||
controller(req, res, next); |
||||
|
||||
}); |
||||
|
||||
test('only status code 408 should be considered a retryable request', (t) => { |
||||
const config = { |
||||
indexName: 'indexName value', |
||||
requestRetries: 17 |
||||
}; |
||||
const esclient = 'this is the esclient'; |
||||
const query = () => { |
||||
return { }; |
||||
}; |
||||
|
||||
let searchServiceCallCount = 0; |
||||
|
||||
const nonTimeoutError = { |
||||
status: 500, |
||||
displayName: 'InternalServerError', |
||||
message: 'an internal server error occurred' |
||||
}; |
||||
|
||||
// a controller that validates the esclient and cmd that was passed to the search service
|
||||
const controller = proxyquire('../../../controller/search_with_appending_results', { |
||||
'../service/search': (esclient, cmd, callback) => { |
||||
// not that the searchService got called
|
||||
searchServiceCallCount++; |
||||
|
||||
callback(nonTimeoutError); |
||||
} |
||||
})(config, esclient, query, () => { return true; }); |
||||
|
||||
const req = { clean: { }, errors: [], warnings: [] }; |
||||
const res = {}; |
||||
|
||||
const next = () => { |
||||
t.equal(searchServiceCallCount, 1); |
||||
t.deepEqual(req, { |
||||
clean: {}, |
||||
errors: [nonTimeoutError.message], |
||||
warnings: [] |
||||
}); |
||||
t.end(); |
||||
}; |
||||
|
||||
controller(req, res, next); |
||||
|
||||
}); |
||||
|
||||
test('string error should not retry and be logged as-is', (t) => { |
||||
const config = { |
||||
indexName: 'indexName value', |
||||
requestRetries: 17 |
||||
}; |
||||
const esclient = 'this is the esclient'; |
||||
const query = () => { |
||||
return { }; |
||||
}; |
||||
|
||||
let searchServiceCallCount = 0; |
||||
|
||||
const stringTypeError = 'this is an error string'; |
||||
|
||||
// a controller that validates the esclient and cmd that was passed to the search service
|
||||
const controller = proxyquire('../../../controller/search_with_appending_results', { |
||||
'../service/search': (esclient, cmd, callback) => { |
||||
// not that the searchService got called
|
||||
searchServiceCallCount++; |
||||
|
||||
callback(stringTypeError); |
||||
} |
||||
})(config, esclient, query, () => { return true; }); |
||||
|
||||
const req = { clean: { }, errors: [], warnings: [] }; |
||||
const res = {}; |
||||
|
||||
const next = () => { |
||||
t.equal(searchServiceCallCount, 1); |
||||
t.deepEqual(req, { |
||||
clean: {}, |
||||
errors: [stringTypeError], |
||||
warnings: [] |
||||
}); |
||||
t.end(); |
||||
}; |
||||
|
||||
controller(req, res, next); |
||||
|
||||
}); |
||||
|
||||
}; |
||||
|
||||
module.exports.tests.should_execute = (test, common) => { |
||||
test('should_execute returning false and empty req.errors should call next', (t) => { |
||||
const esclient = () => { |
||||
throw new Error('esclient should not have been called'); |
||||
}; |
||||
const query = () => { |
||||
throw new Error('query should not have been called'); |
||||
}; |
||||
const controller = setup( {}, esclient, query, () => { return false; } ); |
||||
|
||||
const req = { }; |
||||
const res = { }; |
||||
|
||||
const next = () => { |
||||
t.deepEqual(res, { }); |
||||
t.end(); |
||||
}; |
||||
controller(req, res, next); |
||||
|
||||
}); |
||||
|
||||
}; |
||||
|
||||
module.exports.tests.undefined_query = function(test, common) { |
||||
test('query returning undefined should not call service', function(t) { |
||||
// a function that returns undefined
|
||||
const query = () => { |
||||
return undefined; |
||||
}; |
||||
|
||||
let search_service_was_called = false; |
||||
|
||||
const controller = proxyquire('../../../controller/search_with_appending_results', { |
||||
'../service/search': function() { |
||||
search_service_was_called = true; |
||||
throw new Error('search service should not have been called'); |
||||
} |
||||
})(undefined, undefined, query, () => { return true; }); |
||||
|
||||
const next = () => { |
||||
t.notOk(search_service_was_called, 'should have returned before search service was called'); |
||||
t.end(); |
||||
}; |
||||
|
||||
controller({}, {}, next); |
||||
|
||||
}); |
||||
}; |
||||
|
||||
module.exports.all = function (tape, common) { |
||||
|
||||
function test(name, testFunction) { |
||||
return tape('GET /search_with_appending_results ' + name, testFunction); |
||||
} |
||||
|
||||
for( const testCase in module.exports.tests ){ |
||||
module.exports.tests[testCase](test, common); |
||||
} |
||||
}; |
Loading…
Reference in new issue