|
|
|
/**
|
|
|
|
* Functions that are used in several benchmark tests
|
|
|
|
*/
|
|
|
|
|
|
|
|
var customUtils = require('../lib/customUtils')
|
|
|
|
, fs = require('fs')
|
|
|
|
, path = require('path')
|
|
|
|
, executeAsap // process.nextTick or setImmediate depending on your Node version
|
|
|
|
;
|
|
|
|
|
|
|
|
try {
|
|
|
|
executeAsap = setImmediate;
|
|
|
|
} catch (e) {
|
|
|
|
executeAsap = process.nextTick;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Ensure the workspace exists and the db is empty
|
|
|
|
*/
|
|
|
|
module.exports.prepareDb = function (filename, cb) {
|
|
|
|
customUtils.ensureDirectoryExists(path.dirname(filename), function () {
|
|
|
|
fs.exists(filename, function (exists) {
|
|
|
|
if (exists) {
|
|
|
|
fs.unlink(filename, cb);
|
|
|
|
} else { return cb(); }
|
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return an array with the numbers from 0 to n-1, in a random order
|
|
|
|
* Uses Fisher Yates algorithm
|
|
|
|
* Useful to get fair tests
|
|
|
|
*/
|
|
|
|
function getRandomArray (n) {
|
|
|
|
var res = []
|
|
|
|
, i, j, temp
|
|
|
|
;
|
|
|
|
|
|
|
|
for (i = 0; i < n; i += 1) { res[i] = i; }
|
|
|
|
|
|
|
|
for (i = n - 1; i >= 1; i -= 1) {
|
|
|
|
j = Math.floor((i + 1) * Math.random());
|
|
|
|
temp = res[i];
|
|
|
|
res[i] = res[j];
|
|
|
|
res[j] = temp;
|
|
|
|
}
|
|
|
|
|
|
|
|
return res;
|
|
|
|
};
|
|
|
|
module.exports.getRandomArray = getRandomArray;
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Insert a certain number of documents for testing
|
|
|
|
*/
|
|
|
|
module.exports.insertDocs = function (d, n, profiler, cb) {
|
|
|
|
var beg = new Date()
|
|
|
|
, order = getRandomArray(n)
|
|
|
|
;
|
|
|
|
|
|
|
|
profiler.step('Begin inserting ' + n + ' docs');
|
|
|
|
|
|
|
|
function runFrom(i) {
|
|
|
|
if (i === n) { // Finished
|
|
|
|
console.log("Average time for one insert: " + (profiler.elapsedSinceLastStep() / n) + "ms");
|
|
|
|
profiler.step('Finished inserting ' + n + ' docs');
|
|
|
|
return cb();
|
|
|
|
}
|
|
|
|
|
|
|
|
d.insert({ docNumber: order[i] }, function (err) {
|
|
|
|
executeAsap(function () {
|
|
|
|
runFrom(i + 1);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
runFrom(0);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find documents with find
|
|
|
|
*/
|
|
|
|
module.exports.findDocs = function (d, n, profiler, cb) {
|
|
|
|
var beg = new Date()
|
|
|
|
, order = getRandomArray(n)
|
|
|
|
;
|
|
|
|
|
|
|
|
profiler.step("Finding " + n + " documents");
|
|
|
|
|
|
|
|
function runFrom(i) {
|
|
|
|
if (i === n) { // Finished
|
|
|
|
console.log("Average time for one find in a collection of " + n + " docs: " + (profiler.elapsedSinceLastStep() / n) + "ms");
|
|
|
|
profiler.step('Finished finding ' + n + ' docs');
|
|
|
|
return cb();
|
|
|
|
}
|
|
|
|
|
|
|
|
d.find({ docNumber: order[i] }, function (err, docs) {
|
|
|
|
if (docs.length !== 1 || docs[0].docNumber !== order[i]) { return cb('One find didnt work'); }
|
|
|
|
executeAsap(function () {
|
|
|
|
runFrom(i + 1);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
runFrom(0);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find documents with findOne
|
|
|
|
*/
|
|
|
|
module.exports.findOneDocs = function (d, n, profiler, cb) {
|
|
|
|
var beg = new Date()
|
|
|
|
, order = getRandomArray(n)
|
|
|
|
;
|
|
|
|
|
|
|
|
profiler.step("FindingOne " + n + " documents");
|
|
|
|
|
|
|
|
function runFrom(i) {
|
|
|
|
if (i === n) { // Finished
|
|
|
|
console.log("Average time for one findOne in a collection of " + n + " docs: " + (profiler.elapsedSinceLastStep() / n) + "ms");
|
|
|
|
profiler.step('Finished finding ' + n + ' docs');
|
|
|
|
return cb();
|
|
|
|
}
|
|
|
|
|
|
|
|
d.findOne({ docNumber: order[i] }, function (err, doc) {
|
|
|
|
if (!doc || doc.docNumber !== order[i]) { return cb('One find didnt work'); }
|
|
|
|
executeAsap(function () {
|
|
|
|
runFrom(i + 1);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
runFrom(0);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update documents
|
|
|
|
* options is the same as the options object for update
|
|
|
|
*/
|
|
|
|
module.exports.updateDocs = function (options, d, n, profiler, cb) {
|
|
|
|
var beg = new Date()
|
|
|
|
, order = getRandomArray(n)
|
|
|
|
;
|
|
|
|
|
|
|
|
profiler.step("Updating " + n + " documents");
|
|
|
|
|
|
|
|
function runFrom(i) {
|
|
|
|
if (i === n) { // Finished
|
|
|
|
console.log("Average time for one update in a collection of " + n + " docs: " + (profiler.elapsedSinceLastStep() / n) + "ms");
|
|
|
|
profiler.step('Finished updating ' + n + ' docs');
|
|
|
|
return cb();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Will not actually modify the document but will take the same time
|
|
|
|
d.update({ docNumber: order[i] }, { docNumber: order[i] }, options, function (err, nr) {
|
|
|
|
if (err) { return cb(err); }
|
|
|
|
if (nr !== 1) { return cb('One update didnt work'); }
|
|
|
|
executeAsap(function () {
|
|
|
|
runFrom(i + 1);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
runFrom(0);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove documents
|
|
|
|
* options is the same as the options object for update
|
|
|
|
*/
|
|
|
|
module.exports.removeDocs = function (options, d, n, profiler, cb) {
|
|
|
|
var beg = new Date()
|
|
|
|
, order = getRandomArray(n)
|
|
|
|
;
|
|
|
|
|
|
|
|
profiler.step("Removing " + n + " documents");
|
|
|
|
|
|
|
|
function runFrom(i) {
|
|
|
|
if (i === n) { // Finished
|
|
|
|
console.log("Average time for one remove and one insert in a collection of " + n + " docs: " + (profiler.elapsedSinceLastStep() / n) + "ms");
|
|
|
|
profiler.step('Finished removing ' + n + ' docs');
|
|
|
|
return cb();
|
|
|
|
}
|
|
|
|
|
|
|
|
d.remove({ docNumber: order[i] }, options, function (err, nr) {
|
|
|
|
if (nr !== 1) { return cb('One remove didnt work'); }
|
|
|
|
d.insert({ docNumber: order[i] }, function (err) { // Reinserting just removed document so that the collection size doesn't change
|
|
|
|
// Time is about 70x smaller for an insert so the impact on the results is minimal
|
|
|
|
executeAsap(function () {
|
|
|
|
runFrom(i + 1);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
runFrom(0);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Load database
|
|
|
|
*/
|
|
|
|
module.exports.loadDatabase = function (d, n, profiler, cb) {
|
|
|
|
var beg = new Date()
|
|
|
|
, order = getRandomArray(n)
|
|
|
|
;
|
|
|
|
|
|
|
|
profiler.step("Loading the database " + n + " times");
|
|
|
|
|
|
|
|
function runFrom(i) {
|
|
|
|
if (i === n) { // Finished
|
|
|
|
console.log("Average time for one loadDatabse for a collection of " + n + " docs: " + (profiler.elapsedSinceLastStep() / n) + "ms");
|
|
|
|
profiler.step('Finished loading a database' + n + ' times');
|
|
|
|
return cb();
|
|
|
|
}
|
|
|
|
|
|
|
|
d.loadDatabase(function (err) {
|
|
|
|
executeAsap(function () {
|
|
|
|
runFrom(i + 1);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
runFrom(0);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|