mirror of
https://github.com/zhigang1992/interfake.git
synced 2026-06-10 23:49:19 +08:00
254 lines
8.0 KiB
JavaScript
254 lines
8.0 KiB
JavaScript
var express = require('express');
|
|
var path = require('path');
|
|
var FluentInterface = require('./fluent');
|
|
var corsMiddleware = require('./cors');
|
|
var util = require('core-util-is');
|
|
var url = require('url');
|
|
var merge = require('merge');
|
|
var connectJson = require('connect-json');
|
|
var bodyParser = require('body-parser');
|
|
|
|
function createInvalidDataException(data) {
|
|
return new Error('You have to provide a JSON object with the following structure: \n' + JSON.stringify({ request : { method : '[GET|PUT|POST|DELETE]', url : '(relative URL e.g. /hello)' }, response : { code : '(HTTP Response code e.g. 200/400/500)', body : '(a JSON object)' } }, null, 4) + ' but you provided: \n' + JSON.stringify(data, null, 4));
|
|
}
|
|
|
|
function Interfake(o) {
|
|
o = o || { debug: false };
|
|
var debug = require('./debug')('interfake-server', o.debug);
|
|
var app = express();
|
|
var server;
|
|
var fluentInterface = new FluentInterface(this, o);
|
|
var expectationsLookup = {};
|
|
|
|
// app.use(app.router);
|
|
app.use(connectJson());
|
|
app.use(bodyParser());
|
|
app.use(corsMiddleware);
|
|
|
|
app.post('/_requests?', function(req, res){
|
|
try {
|
|
createRoute(req.body);
|
|
res.send(201, { done : true });
|
|
} catch (e) {
|
|
debug('Error: ', e);
|
|
res.send(400, e);
|
|
}
|
|
});
|
|
|
|
function indexOfRequestRoute(request) {
|
|
var i;
|
|
if (!app.routes[request.method]) return -1;
|
|
|
|
for (i = app.routes[request.method].length - 1; i >= 0; i--) {
|
|
if (app.routes[request.method][i].path === request.url) {
|
|
return i;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
function clearRouteForRequest(request) {
|
|
var i = indexOfRequestRoute(request);
|
|
|
|
if (i === -1) {
|
|
return;
|
|
}
|
|
|
|
app.routes[request.method].splice(i);
|
|
}
|
|
|
|
function determineDelay(delayInput) {
|
|
var result = 0, range, upper, lower;
|
|
if(util.isNumber(delayInput)) {
|
|
result = delayInput;
|
|
} else if(util.isString(delayInput)) {
|
|
range = /([0-9]+)..([0-9]+)/.exec(delayInput);
|
|
upper = +range[2];
|
|
lower = +range[1];
|
|
result = Math.floor( Math.random() * (upper - lower + 1) + lower );
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
function createRouteHash(requestDescriptor) {
|
|
var finalRoute;
|
|
|
|
var path = url.parse(requestDescriptor.url, true);
|
|
requestDescriptor.query = merge(path.query || {}, requestDescriptor.query || {});
|
|
|
|
requestDescriptor.url = path.pathname;
|
|
|
|
var initialRoute = requestDescriptor.method.toUpperCase() + ' ' + requestDescriptor.url;
|
|
if (requestDescriptor.query) {
|
|
var queryKeys = Object.keys(requestDescriptor.query).filter(function (key) {
|
|
return key !== 'callback';
|
|
});
|
|
if (queryKeys.length) {
|
|
debug('Query keys are', queryKeys);
|
|
finalRoute = initialRoute + '?' + queryKeys.sort().map(function (key) {
|
|
return encodeURIComponent(key) + '=' + encodeURIComponent(requestDescriptor.query[key]);
|
|
}).join(';');
|
|
}
|
|
}
|
|
|
|
if (!finalRoute) {
|
|
finalRoute = initialRoute;
|
|
}
|
|
|
|
debug('Lookup hash key will be: ' + finalRoute);
|
|
return finalRoute;
|
|
}
|
|
|
|
function createRoute(data, existingLookupHash) {
|
|
var specifiedRequest, specifiedResponse, afterSpecifiedResponse, lookupHash, existingExpectations;
|
|
if (!data.request || !data.request.method || !data.request.url || !data.response || !data.response.code) {
|
|
throw createInvalidDataException(data);
|
|
}
|
|
|
|
// debug('Setup time', JSON.stringify(data, null, 4));
|
|
|
|
if (data.response.body) {
|
|
debug('Setting up ' + data.request.method + ' ' + data.request.url + ' to return ' + data.response.code + ' with a body of length ' + JSON.stringify(data.response.body).length);
|
|
} else {
|
|
debug('Setting up ' + data.request.method + ' ' + data.request.url + ' to return ' + data.response.code + ' with no body');
|
|
}
|
|
|
|
specifiedRequest = data.request;
|
|
|
|
// clearRouteForRequest(specifiedRequest);
|
|
|
|
if (existingLookupHash) {
|
|
existingExpectations = expectationsLookup[existingLookupHash].pop();
|
|
debug('Looking for existing lookup hash ', existingLookupHash);
|
|
// existingExpectations.request = merge(data.request, existingExpectations.request);
|
|
lookupHash = createRouteHash(specifiedRequest);
|
|
debug('New lookup hash is', lookupHash);
|
|
} else {
|
|
// Register query params/response in lookup hash
|
|
lookupHash = createRouteHash(specifiedRequest);
|
|
}
|
|
|
|
if (expectationsLookup[lookupHash]) {
|
|
debug('Lookup hash', lookupHash, 'already has a route associated with it - there must be more to come.');
|
|
expectationsLookup[lookupHash].push({
|
|
request: data.request,
|
|
response: data.response,
|
|
afterResponse: data.afterResponse
|
|
});
|
|
} else {
|
|
expectationsLookup[lookupHash] = [{
|
|
request: data.request,
|
|
response: data.response,
|
|
afterResponse: data.afterResponse
|
|
}];
|
|
}
|
|
|
|
app[specifiedRequest.method](specifiedRequest.url, function (req, res) {
|
|
var lookupHash = createRouteHash({
|
|
method: req.method,
|
|
url: req.path,
|
|
query: req.query
|
|
});
|
|
var expectDataArray = expectationsLookup[lookupHash];
|
|
|
|
if (!expectDataArray || expectDataArray.length === 0) {
|
|
debug(expectationsLookup);
|
|
debug('Found matching path route, but no matching data for given query params for lookup hash', lookupHash);
|
|
return res.send(404);
|
|
}
|
|
|
|
var expectData = expectDataArray[expectDataArray.length - 1];
|
|
|
|
var specifiedResponse = expectData.response; // req.route.responseData;
|
|
var afterSpecifiedResponse = expectData.afterResponse; //req.route.afterResponseData;
|
|
var responseDelay = determineDelay(specifiedResponse.delay);
|
|
|
|
debug(req.method, 'request to', req.url, 'returning', specifiedResponse.code);
|
|
debug(req.method, 'request to', req.url, 'will be delayed by', responseDelay, 'millis');
|
|
// debug('After response is', afterSpecifiedResponse);
|
|
|
|
var responseBody = specifiedResponse.body;
|
|
|
|
res.setHeader('Content-Type', 'application/json');
|
|
|
|
if (specifiedResponse.headers) {
|
|
Object.keys(specifiedResponse.headers).forEach(function (k) {
|
|
res.setHeader(k, specifiedResponse.headers[k]);
|
|
});
|
|
}
|
|
|
|
if (req.query.callback) {
|
|
debug('Request is asking for jsonp');
|
|
if (typeof responseBody !== 'string') responseBody = JSON.stringify(responseBody);
|
|
responseBody = req.query.callback.trim() + '(' + responseBody + ');';
|
|
}
|
|
setTimeout(function() {
|
|
res.send(specifiedResponse.code, responseBody);
|
|
|
|
if (afterSpecifiedResponse && afterSpecifiedResponse.endpoints) {
|
|
debug('Response sent, setting up', afterSpecifiedResponse.endpoints.length, 'endpoints');
|
|
afterSpecifiedResponse.endpoints.forEach(function (endpoint) {
|
|
createRoute(endpoint);
|
|
});
|
|
}
|
|
}, responseDelay);
|
|
});
|
|
|
|
var numAfterResponse = (data.afterResponse && data.afterResponse.endpoints) ? data.afterResponse.endpoints.length : 0;
|
|
|
|
if (data.response.body) {
|
|
debug('Setup complete: ' + data.request.method + ' ' + data.request.url + ' to return ' + data.response.code + ' with a body of length ' + JSON.stringify(data.response.body).length + ' and ' + numAfterResponse + ' after-responses');
|
|
} else {
|
|
debug('Setup complete: ' + data.request.method + ' ' + data.request.url + ' to return ' + data.response.code + ' with no body and ' + numAfterResponse + ' after-responses');
|
|
}
|
|
return lookupHash;
|
|
}
|
|
|
|
this.createRoute = createRoute;
|
|
|
|
this.get = fluentInterface.forMethod('get');
|
|
this.post = fluentInterface.forMethod('post');
|
|
this.put = fluentInterface.forMethod('put');
|
|
this.delete = fluentInterface.forMethod('delete');
|
|
|
|
this.serveStatic = function (path, directory) {
|
|
path = path || '/_static';
|
|
app.use(path, express.static(directory));
|
|
};
|
|
|
|
this.listen = function (port, callback) {
|
|
port = port || 3000;
|
|
server = app.listen(port, function () {
|
|
debug('Interfake is listening for requests on port ' + port);
|
|
if(util.isFunction(callback)) {
|
|
callback();
|
|
}
|
|
});
|
|
};
|
|
|
|
this.stop = function () {
|
|
if (server) {
|
|
debug('Interfake is stopping');
|
|
server.close(function () {
|
|
debug('Interfake has stopped');
|
|
server = undefined;
|
|
});
|
|
}
|
|
};
|
|
}
|
|
|
|
Interfake.prototype.loadFile = function (filePath) {
|
|
var file;
|
|
|
|
filePath = path.resolve(process.cwd(), filePath);
|
|
|
|
file = require(filePath);
|
|
|
|
file.forEach(function (endpoint) {
|
|
this.createRoute(endpoint);
|
|
}.bind(this));
|
|
};
|
|
|
|
module.exports = Interfake;
|