Files
mongo/jstests/multiVersion/libs/verify_collection_data.js
Kevin Pulo 0c8b0d57c7 SERVER-12746: Fix assert() calls with constant-value args
These calls were mostly mistakes that should have been assert.eq().

assert() will now trip if passed a non-string msg, or too many params.
2015-10-30 20:35:28 +11:00

186 lines
8.1 KiB
JavaScript

// This file contains test helpers to manage and validate collection state. This is useful for
// round trip testing of entire collections.
//
// There are three stages represented in this file:
// 1. Data generation - CollectionDataGenerator class. This contains helpers to generate test data
// for a collection.
// 2. Data persistence - createCollectionWithData function. This class takes a
// CollectionDataGenerator and inserts the generated data into the given
// collection.
// 3. Data validation - CollectionDataValidator class. This class contains functions for saving
// the state of a collection and comparing a collection's state to the
// previously saved state.
//
// Common use case:
// 1. Create a CollectionDataGenerator
// 2. Save collection data using the createCollectionWithData function
// 3. Record collection state in an instance of the CollectionDataValidator class
// 4. Do round trip or other testing
// 5. Validate that collection has not changed using the CollectionDataValidator class
load( './jstests/multiVersion/libs/data_generators.js' )
// Function to actually add the data generated by the given dataGenerator to a collection
createCollectionWithData = function (db, collectionName, dataGenerator) {
// Drop collection if exists
// TODO: add ability to control this
db.getCollection(collectionName).drop();
print("db.createCollection(\"" + collectionName + "\", " +
JSON.stringify(dataGenerator.collectionMetadata.get()) + ");");
assert.eq(db.createCollection(collectionName, dataGenerator.collectionMetadata.get()).ok, 1);
var collection = db.getCollection(collectionName);
var numIndexes = 0;
while (dataGenerator.indexes.hasNext()) {
var nextIndex = dataGenerator.indexes.next();
print("collection.ensureIndex(" + JSON.stringify(nextIndex.spec) + ", " +
JSON.stringify(nextIndex.options) + ");");
var ensureIndexResult = collection.ensureIndex(nextIndex.spec, nextIndex.options);
// XXX: Is this the real way to check for errors?
assert(ensureIndexResult === undefined, tojson(ensureIndexResult));
numIndexes++;
}
// Make sure we actually added all the indexes we think we added. +1 for the _id index.
assert.eq(collection.getIndexes().length, numIndexes + 1);
var numInserted = 0;
while (dataGenerator.data.hasNext()) {
var nextDoc = dataGenerator.data.next();
// Use _id as our ordering field just so we don't have to deal with sorting. This only
// matters here since we can use indexes
nextDoc._id = numInserted;
print("collection.insert(" + JSON.stringify(nextDoc) + ");");
var insertResult = collection.insert(nextDoc);
assert(db.getLastError() == null);
numInserted++;
}
assert.eq(collection.find().count(), numInserted, "counts not equal after inserts");
return db.getCollection(collectionName);
}
// Class to save the state of a collection and later compare the current state of a collection to
// the saved state
function CollectionDataValidator() {
var _initialized = false;
var _collectionInfo = {};
var _indexData = [];
var _collectionData = [];
// Returns the options of the specified collection.
this.getCollectionInfo = function(collection) {
var infoObj = collection.getDB().getCollectionInfos({name: collection.getName()});
assert.eq(1, infoObj.length, "expected collection '" + collection.getName() + "'to exist");
return infoObj[0];
};
// Saves the current state of the collection passed in
this.recordCollectionData = function (collection) {
// Save the metadata for this collection for later comparison.
_collectionInfo = this.getCollectionInfo(collection);
// Save the indexes for this collection for later comparison
_indexData = collection.getIndexes().sort(function(a,b) {
if (a.name > b.name) return 1;
else return -1;
});
// Save the data for this collection for later comparison
_collectionData = collection.find().sort({"_id":1}).toArray();
_initialized = true;
return collection;
}
this.validateCollectionData = function (collection) {
if (!_initialized) {
throw Error("validateCollectionWithAllData called, but data is not initialized");
}
// Get the metadata for this collection
var newCollectionInfo = this.getCollectionInfo(collection);
assert.docEq(_collectionInfo, newCollectionInfo, "collection metadata not equal");
// Get the indexes for this collection
var newIndexData = collection.getIndexes().sort(function(a,b) {
if (a.name > b.name) return 1;
else return -1;
});
for (var i = 0; i < newIndexData.length; i++) {
assert.docEq(_indexData[i], newIndexData[i], "indexes not equal");
}
// Save the data for this collection for later comparison
var newCollectionData = collection.find().sort({"_id":1}).toArray();
for (var i = 0; i < newCollectionData.length; i++) {
assert.docEq(_collectionData[i], newCollectionData[i], "data not equal");
}
return true;
}
}
// Tests of the functions and classes in this file
function collectionDataValidatorTests() {
// TODO: These tests are hackish and depend on implementation details, but they are good enough
// for now to convince us that the CollectionDataValidator is actually checking something
var myValidator;
var myGenerator;
var collection;
myGenerator = new CollectionDataGenerator({ "capped" : true });
collection = createCollectionWithData(db, "test", myGenerator);
myValidator = new CollectionDataValidator();
myValidator.recordCollectionData(collection);
db.test.dropIndex(db.test.getIndexKeys().filter(function(key) { return key.a != null })[0]);
assert.throws(myValidator.validateCollectionData, [collection], "Validation function should have thrown since we modified the collection");
myGenerator = new CollectionDataGenerator({ "capped" : true });
collection = createCollectionWithData(db, "test", myGenerator);
myValidator = new CollectionDataValidator();
myValidator.recordCollectionData(collection);
db.test.update({_id:0}, {dummy:1});
assert.throws(myValidator.validateCollectionData, [collection], "Validation function should have thrown since we modified the collection");
myGenerator = new CollectionDataGenerator({ "capped" : true });
collection = createCollectionWithData(db, "test", myGenerator);
myValidator = new CollectionDataValidator();
myValidator.recordCollectionData(collection);
assert(myValidator.validateCollectionData(collection), "Validation function failed");
myGenerator = new CollectionDataGenerator({ "capped" : false });
collection = createCollectionWithData(db, "test", myGenerator);
myValidator = new CollectionDataValidator();
myValidator.recordCollectionData(collection);
db.test.dropIndex(db.test.getIndexKeys().filter(function(key) { return key.a != null })[0]);
assert.throws(myValidator.validateCollectionData, [collection], "Validation function should have thrown since we modified the collection");
myGenerator = new CollectionDataGenerator({ "capped" : false });
collection = createCollectionWithData(db, "test", myGenerator);
myValidator = new CollectionDataValidator();
myValidator.recordCollectionData(collection);
db.test.update({_id:0}, {dummy:1});
assert.throws(myValidator.validateCollectionData, [collection], "Validation function should have thrown since we modified the collection");
myGenerator = new CollectionDataGenerator({ "capped" : false });
collection = createCollectionWithData(db, "test", myGenerator);
myValidator = new CollectionDataValidator();
myValidator.recordCollectionData(collection);
assert(myValidator.validateCollectionData(collection), "Validation function failed");
print("collection data validator tests passed!");
}