|
|
|
//
|
|
|
|
// Usage:
|
|
|
|
// node benchmark/http_simple_auto.js <args> <target>
|
|
|
|
//
|
|
|
|
// Where:
|
|
|
|
// <args> Arguments to pass to `ab`.
|
|
|
|
// <target> Target to benchmark, e.g. `bytes/1024` or `buffer/8192`.
|
|
|
|
//
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
var http = require('http');
|
|
|
|
var spawn = require('child_process').spawn;
|
|
|
|
|
|
|
|
var port = parseInt(process.env.PORT || 8000);
|
|
|
|
|
|
|
|
var fixed = 'C'.repeat(20 * 1024);
|
|
|
|
|
|
|
|
var stored = {};
|
|
|
|
var storedBuffer = {};
|
|
|
|
|
|
|
|
var server = http.createServer(function(req, res) {
|
|
|
|
var commands = req.url.split('/');
|
|
|
|
var command = commands[1];
|
|
|
|
var body = '';
|
|
|
|
var arg = commands[2];
|
|
|
|
var n_chunks = parseInt(commands[3], 10);
|
|
|
|
var status = 200;
|
|
|
|
var n;
|
|
|
|
var i;
|
|
|
|
|
|
|
|
if (command == 'bytes') {
|
|
|
|
n = parseInt(arg, 10);
|
|
|
|
if (n <= 0)
|
|
|
|
throw new Error('bytes called with n <= 0');
|
|
|
|
if (stored[n] === undefined) {
|
|
|
|
stored[n] = 'C'.repeat(n);
|
|
|
|
}
|
|
|
|
body = stored[n];
|
|
|
|
} else if (command == 'buffer') {
|
|
|
|
n = parseInt(arg, 10);
|
|
|
|
if (n <= 0) throw new Error('bytes called with n <= 0');
|
|
|
|
if (storedBuffer[n] === undefined) {
|
|
|
|
storedBuffer[n] = new Buffer(n);
|
|
|
|
for (i = 0; i < n; i++) {
|
|
|
|
storedBuffer[n][i] = 'C'.charCodeAt(0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
body = storedBuffer[n];
|
|
|
|
|
|
|
|
} else if (command == 'quit') {
|
|
|
|
res.connection.server.close();
|
|
|
|
body = 'quitting';
|
|
|
|
|
|
|
|
} else if (command == 'fixed') {
|
|
|
|
body = fixed;
|
|
|
|
|
|
|
|
} else if (command == 'echo') {
|
|
|
|
res.writeHead(200, { 'Content-Type': 'text/plain',
|
|
|
|
'Transfer-Encoding': 'chunked' });
|
|
|
|
req.pipe(res);
|
|
|
|
return;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
status = 404;
|
|
|
|
body = 'not found\n';
|
|
|
|
}
|
|
|
|
|
|
|
|
// example: http://localhost:port/bytes/512/4
|
|
|
|
// sends a 512 byte body in 4 chunks of 128 bytes
|
|
|
|
if (n_chunks > 0) {
|
|
|
|
res.writeHead(status, { 'Content-Type': 'text/plain',
|
|
|
|
'Transfer-Encoding': 'chunked' });
|
|
|
|
// send body in chunks
|
|
|
|
var len = body.length;
|
|
|
|
var step = Math.floor(len / n_chunks) || 1;
|
|
|
|
|
|
|
|
for (i = 0, n = (n_chunks - 1); i < n; ++i) {
|
|
|
|
res.write(body.slice(i * step, i * step + step));
|
|
|
|
}
|
|
|
|
res.end(body.slice((n_chunks - 1) * step));
|
|
|
|
} else {
|
|
|
|
var content_length = body.length.toString();
|
|
|
|
|
|
|
|
res.writeHead(status, { 'Content-Type': 'text/plain',
|
|
|
|
'Content-Length': content_length });
|
|
|
|
res.end(body);
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
server.listen(port, function() {
|
|
|
|
var url = 'http://127.0.0.1:' + port + '/';
|
|
|
|
|
|
|
|
var n = process.argv.length - 1;
|
|
|
|
process.argv[n] = url + process.argv[n];
|
|
|
|
|
|
|
|
var cp = spawn('ab', process.argv.slice(2));
|
|
|
|
cp.stdout.pipe(process.stdout);
|
|
|
|
cp.stderr.pipe(process.stderr);
|
|
|
|
cp.on('exit', function() {
|
|
|
|
server.close();
|
|
|
|
process.nextTick(dump_mm_stats);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
function dump_mm_stats() {
|
|
|
|
if (typeof global.gc != 'function') return;
|
|
|
|
|
|
|
|
var before = process.memoryUsage();
|
|
|
|
for (var i = 0; i < 10; ++i) global.gc();
|
|
|
|
var after = process.memoryUsage();
|
|
|
|
setTimeout(print_stats, 250); // give GC time to settle
|
|
|
|
|
|
|
|
function print_stats() {
|
|
|
|
console.log('\nBEFORE / AFTER GC');
|
|
|
|
['rss', 'heapTotal', 'heapUsed'].forEach(function(key) {
|
|
|
|
var a = before[key] / (1024 * 1024);
|
|
|
|
var b = after[key] / (1024 * 1024);
|
|
|
|
console.log('%sM / %sM %s', a.toFixed(2), b.toFixed(2), key);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|