summaryrefslogtreecommitdiff
path: root/jstests/core/existsa.js
blob: 9ef7e9f374c23c4a23561d00104fd89759f06ffc (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
// Sparse indexes are disallowed for $exists:false queries.  SERVER-3918

t = db.jstests_existsa;
t.drop();

t.save( {} );
t.save( { a:1 } );
t.save( { a:{ x:1 }, b:1 } );

/** Configure testing of an index { <indexKeyField>:1 }. */
function setIndex( _indexKeyField ) {
    indexKeyField = _indexKeyField;
    indexKeySpec = {};
    indexKeySpec[ indexKeyField ] = 1;
    t.ensureIndex( indexKeySpec, { sparse:true } );
    indexCursorName = 'BtreeCursor ' + indexKeyField + '_1';
}
setIndex( 'a' );

/** Validate the prefix of 'str'. */
function assertPrefix( prefix, str ) {
    assert.eq( prefix, str.substring( 0, prefix.length ) );
}

/** @return count when hinting the index to use. */
function hintedCount( query ) {
    assertPrefix( indexCursorName, t.find( query ).hint( indexKeySpec ).explain().cursor );
    return t.find( query ).hint( indexKeySpec ).itcount();
}

/** The query field does not exist and the sparse index is not used without a hint. */
function assertMissing( query, expectedMissing, expectedIndexedMissing ) {
    expectedMissing = expectedMissing || 1;
    expectedIndexedMissing = expectedIndexedMissing || 0;
    assert.eq( expectedMissing, t.count( query ) );
    assert.eq( 'BasicCursor', t.find( query ).explain().cursor );
    // We also shouldn't get a different count depending on whether
    // an index is used or not.
    assert.eq( expectedIndexedMissing, hintedCount( query ) );
}

/** The query field exists and the sparse index is used without a hint. */
function assertExists( query, expectedExists ) {
    expectedExists = expectedExists || 2;
    assert.eq( expectedExists, t.count( query ) );
    assert.eq( 0, t.find( query ).explain().cursor.indexOf('BtreeCursor') );
    // An $exists:true predicate generates no index filters.  Add another predicate on the index key
    // to trigger use of the index.
    andClause = {}
    andClause[ indexKeyField ] = { $ne:null };
    Object.extend( query, { $and:[ andClause ] } );
    assert.eq( expectedExists, t.count( query ) );
    assertPrefix( indexCursorName, t.find( query ).explain().cursor );
    assert.eq( expectedExists, hintedCount( query ) );
}

/** The query field exists and the sparse index is not used without a hint. */
function assertExistsUnindexed( query, expectedExists ) {
    expectedExists = expectedExists || 2;
    assert.eq( expectedExists, t.count( query ) );
    assert.eq( 'BasicCursor', t.find( query ).explain().cursor );
    // Even with another predicate on the index key, the sparse index is disallowed.
    andClause = {}
    andClause[ indexKeyField ] = { $ne:null };
    Object.extend( query, { $and:[ andClause ] } );
    assert.eq( expectedExists, t.count( query ) );
    assert.eq( 'BasicCursor', t.find( query ).explain().cursor );
    assert.eq( expectedExists, hintedCount( query ) );
}

// $exists:false queries match the proper number of documents and disallow the sparse index.
assertMissing( { a:{ $exists:false } } );
assertMissing( { a:{ $not:{ $exists:true } } } );
assertMissing( { $and:[ { a:{ $exists:false } } ] } );
assertMissing( { $or:[ { a:{ $exists:false } } ] } );
assertMissing( { $nor:[ { a:{ $exists:true } } ] } );
assertMissing( { 'a.x':{ $exists:false } }, 2, 1 );

// Currently a sparse index is disallowed even if the $exists:false query is on a different field.
assertMissing( { b:{ $exists:false } }, 2, 1 );
assertMissing( { b:{ $exists:false }, a:{ $ne:6 } }, 2, 1 );
assertMissing( { b:{ $not:{ $exists:true } } }, 2, 1 );

// Top level $exists:true queries match the proper number of documents
// and use the sparse index on { a : 1 }.
assertExists( { a:{ $exists:true } } );

// Nested $exists queries match the proper number of documents and disallow the sparse index.
assertExistsUnindexed( { $nor:[ { a:{ $exists:false } } ] } );
assertExistsUnindexed( { $nor:[ { 'a.x':{ $exists:false } } ] }, 1 );
assertExistsUnindexed( { a:{ $not:{ $exists:false } } } );

// Nested $exists queries disallow the sparse index in some cases where it is not strictly
// necessary to do so.  (Descriptive tests.)
assertExistsUnindexed( { $nor:[ { b:{ $exists:false } } ] }, 1 );  // Unindexed field.
assertExists( { $or:[ { a:{ $exists:true } } ] } );  // $exists:true not $exists:false.

// Behavior is similar with $elemMatch.
t.drop();
t.save( { a:[ {} ] } );
t.save( { a:[ { b:1 } ] } );
t.save( { a:[ { b:1 } ] } );
setIndex( 'a.b' );

assertMissing( { a:{ $elemMatch:{ b:{ $exists:false } } } } );
// A $elemMatch predicate is treated as nested, and the index should be used for $exists:true.
assertExists( { a:{ $elemMatch:{ b:{ $exists:true } } } } );

// A non sparse index will not be disallowed.
t.drop();
t.save( {} );
t.ensureIndex( { a:1 } );
assert.eq( 1, t.find( { a:{ $exists:false } } ).itcount() );
assert.eq( 'BtreeCursor a_1', t.find( { a:{ $exists:false } } ).explain().cursor );