Files
mongo/jstests/multiVersion/libs/verify_collection_data.js
2014-12-23 16:40:12 -05:00

209 lines
9.2 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, JSON.stringify(ensureIndexResult));
numIndexes++;
}
// Make sure we actually added all the indexes we thing we added. +1 for the _id index.
assert.eq(db.system.indexes.find({"ns" : db.toString() + "." + collection.getName()}).count(),
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 collectionStats = {};
var indexData = [];
var collectionData = [];
// Saves the current state of the collection passed in
this.recordCollectionData = function (collection) {
// Save the indexes for this collection for later comparison
indexData = collection.getDB().system.indexes.find({"ns" : collection.getFullName()}).sort({"name":1}).toArray();
// Save the data for this collection for later comparison
collectionData = collection.find().sort({"_id":1}).toArray();
// Save the metadata for this collection for later comparison.
// NOTE: We do this last since the data and indexes affect this output
collectionStats = collection.stats();
// XXX: in 2.4 avgObjSize was a double, but in 2.6 it is an int
collectionStats['avgObjSize'] = Math.floor(collectionStats['avgObjSize']);
// Delete keys that appear just because we shard
delete collectionStats["primary"];
delete collectionStats["sharded"];
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 newCollectionStats = collection.stats();
// XXX: in 2.4 avgObjSize was a double, but in 2.6 it is an int
newCollectionStats['avgObjSize'] = Math.floor(newCollectionStats['avgObjSize']);
// as of 2.7.1, we no longer use systemFlags
delete collectionStats.systemFlags;
delete newCollectionStats.systemFlags;
// as of 2.7.7, we no longer use paddingFactor and introduced paddingFactorNote
delete collectionStats.paddingFactor;
delete collectionStats.paddingFactorNote;
delete newCollectionStats.paddingFactor;
delete newCollectionStats.paddingFactorNote;
// Delete keys that appear just because we shard
delete newCollectionStats["primary"];
delete newCollectionStats["sharded"];
// as of 2.7.8, we added maxSize
// TODO: when 2.6 is no longer tested, remove following two lines
delete newCollectionStats["maxSize"];
delete collectionStats["maxSize"];
// Delete key added in 2.8-rc3
delete collectionStats["indexDetails"];
delete newCollectionStats["indexDetails"];
assert.docEq(collectionStats, newCollectionStats, "collection metadata not equal");
// Get the indexes for this collection
var newIndexData = collection.getDB().system.indexes.find({"ns" : collection.getFullName()}).sort({"name":1}).toArray();
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.system.indexes.findOne({"key.a": { "$exists" : true } }).key);
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.system.indexes.findOne({"key.a": { "$exists" : true } }).key);
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!");
}