209 lines
9.2 KiB
JavaScript
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!");
|
|
}
|