// In fast count mode the Matcher is bypassed when matching can be performed by a BtreeCursor and // its delegate FieldRangeVector or an IntervalBtreeCursor. The tests below check that fast count // mode is implemented appropriately in specific cases. // // SERVER-1752 t = db.jstests_countc; t.drop(); // Match a subset of inserted values within a $in operator. t.drop(); t.ensureIndex( { a:1 } ); // Save 'a' values 0, 0.5, 1.5, 2.5 ... 97.5, 98.5, 99. t.save( { a:0 } ); t.save( { a:99 } ); for( i = 0; i < 99; ++i ) { t.save( { a:( i + 0.5 ) } ); } // Query 'a' values $in 0, 1, 2, ..., 99. vals = []; for( i = 0; i < 100; ++i ) { vals.push( i ); } // Only values 0 and 99 of the $in set are present in the collection, so the expected count is 2. assert.eq( 2, t.count( { a:{ $in:vals } } ) ); // Match 'a' values within upper and lower limits. t.drop(); t.ensureIndex( { a:1 } ); t.save( { a:[ 1, 2 ] } ); // Will match because 'a' is in range. t.save( { a:9 } ); // Will not match because 'a' is not in range. // Only one document matches. assert.eq( 1, t.count( { a:{ $gt:0, $lt:5 } } ) ); // Match two nested fields within an array. t.drop(); t.ensureIndex( { 'a.b':1, 'a.c':1 } ); t.save( { a:[ { b:2, c:3 }, {} ] } ); // The document does not match because its c value is 3. assert.eq( 0, t.count( { 'a.b':2, 'a.c':2 } ) ); // $gt:string only matches strings. t.drop(); t.ensureIndex( { a:1 } ); t.save( { a:'a' } ); // Will match. t.save( { a:{} } ); // Will not match because {} is not a string. // Only one document matches. assert.eq( 1, t.count( { a:{ $gte:'' } } ) ); // $lte:date only matches dates. t.drop(); t.ensureIndex( { a:1 } ); t.save( { a:new Date( 1 ) } ); // Will match. t.save( { a:true } ); // Will not match because 'true' is not a date. // Only one document matches. assert.eq( 1, t.count( { a:{ $lte:new Date( 1 ) } } ) ); // Querying for 'undefined' triggers an error. t.drop(); t.ensureIndex( { a:1 } ); assert.throws( function() { t.count( { a:undefined } ); } ); // Count using a descending order index. t.drop(); t.ensureIndex( { a:-1 } ); t.save( { a:1 } ); t.save( { a:2 } ); t.save( { a:3 } ); assert.eq( 1, t.count( { a:{ $gt:2 } } ) ); assert.eq( 1, t.count( { a:{ $lt:2 } } ) ); assert.eq( 2, t.count( { a:{ $lte:2 } } ) ); assert.eq( 2, t.count( { a:{ $lt:3 } } ) ); // Count using a compound index. t.drop(); t.ensureIndex( { a:1, b:1 } ); t.save( { a:1, b:2 } ); t.save( { a:2, b:1 } ); t.save( { a:2, b:3 } ); t.save( { a:3, b:4 } ); assert.eq( 1, t.count( { a:1 })); assert.eq( 2, t.count( { a:2 })); assert.eq( 1, t.count( { a:{ $gt:2 } } ) ); assert.eq( 1, t.count( { a:{ $lt:2 } } ) ); assert.eq( 2, t.count( { a:2, b:{ $gt:0 } } ) ); assert.eq( 1, t.count( { a:2, b:{ $lt:3 } } ) ); assert.eq( 1, t.count( { a:1, b:{ $lt:3 } } ) ); // Count using a compound descending order index. t.drop(); t.ensureIndex( { a:1, b:-1 } ); t.save( { a:1, b:2 } ); t.save( { a:2, b:1 } ); t.save( { a:2, b:3 } ); t.save( { a:3, b:4 } ); assert.eq( 1, t.count( { a:{ $gt:2 } } ) ); assert.eq( 1, t.count( { a:{ $lt:2 } } ) ); assert.eq( 2, t.count( { a:2, b:{ $gt:0 } } ) ); assert.eq( 1, t.count( { a:2, b:{ $lt:3 } } ) ); assert.eq( 1, t.count( { a:1, b:{ $lt:3 } } ) ); // Count with a multikey value. t.drop(); t.ensureIndex( { a:1 } ); t.save( { a:[ 1, 2 ] } ); assert.eq( 1, t.count( { a:{ $gt:0, $lte:2 } } ) ); // Count with a match constraint on an unindexed field. t.drop(); t.ensureIndex( { a:1 } ); t.save( { a:1, b:1 } ); t.save( { a:1, b:2 } ); assert.eq( 1, t.count( { a:1, $where:'this.b == 1' } ) );