Query Plans
For any given query, the MongoDB query planner chooses and caches the most efficient query plan given the available indexes. To evaluate the efficiency of query plans, the query planner runs all candidate plans during a trial period. In general, the winning plan is the query plan that produces the most results during the trial period while performing the least amount of work.
The associated plan cache entry is used for subsequent queries with the same plan cache query shape.
The following diagram illustrates the query planner logic:
Note
Using explain
ignores all existing plan cache entries and prevents
the MongoDB query planner from creating a new plan cache entry.
Plan Cache Entry State
Starting in MongoDB 4.2, each plan cache query shape is associated with one of three states in the cache:
State | Description |
---|---|
No entry for this shape exists in the cache. For a query, if the cache entry state for a plan cache query shape is Missing:
| |
The entry in the cache is a placeholder entry for this shape. That is, the planner has seen the shape, calculated a value that quantifies the amount of work required by the plan and stored the shape placeholder entry but the plan cache query shape is not used to generate query plans. For a query, if the cache entry state for a shape is Inactive:
| |
The entry in the cache is for the winning plan. The planner can use this entry to generate query plans. For a query, if the cache entry state for a shape is Active: The active entry is used to generate query plans. The planner also evaluates the entry's performance and if its value that quantifies the amount of work required by the plan no longer meets the selection criterion, it will transition to Inactive state. |
See Plan Cache Flushes for additional scenarios that trigger changes to the plan cache.
Query Plan and Cache Information
To view the query plan information for a given query, you can use
db.collection.explain()
or the cursor.explain()
.
To view plan cache information for a collection, you can use the
$planCacheStats
aggregation stage.
Plan Cache Flushes
The query plan cache does not persist if a mongod
restarts or shuts down. In addition:
Catalog operations like index or collection drops clear the plan cache.
Least recently used (LRU) cache replacement mechanism clears the least recently accessed cache entry, regardless of state.
Users can also:
Manually clear the entire plan cache using the
PlanCache.clear()
method.Manually clear specific plan cache entries using the
PlanCache.clearPlansByQuery()
method.
Plan Cache Debug Info Size Limit
Starting in MongoDB 5.0, the
plan cache will save full
plan cache
entries only if the cumulative size of the
plan caches
for all collections is lower than 0.5 GB. When the
cumulative size of the plan caches
for all collections exceeds this
threshold, additional plan cache
entries are stored without the
following debug information:
The estimated size in bytes of a plan cache
entry is available in
the output of $planCacheStats
.
planCacheShapeHash and planCacheKey
planCacheShapeHash
To help identify slow queries with the same plan cache query shape, each plan cache query shape is associated with a query hash. The plan cache query shape hash is a hexadecimal string that represents a hash of the query shape and is dependent only on the query shape.
Note
As with any hash function, two different query shapes may result in the same hash value. However, the occurrence of hash collisions between different query shapes is unlikely.
Starting in MongoDB 8.0, the pre-existing queryHash
field is renamed
to planCacheShapeHash
. If you're using an earlier MongoDB version,
you'll see queryHash
instead of planCacheShapeHash
.
planCacheKey
To provide more insight into the query plan cache, MongoDB offers the planCacheKey
.
planCacheKey
is a hash of the key for the plan cache entry
associated with the query.
Note
Unlike planCacheShapeHash
, planCacheKey
is a function of
both the query shape and the currently available indexes for the
shape. That is, if indexes that can support the query shape are
added/dropped, the planCacheKey
value may change whereas the
planCacheShapeHash
value would not change.
Starting in MongoDB 8.0, the pre-existing queryHash
field is renamed
to planCacheShapeHash
. If you're using an earlier MongoDB version,
you'll see queryHash
instead of planCacheShapeHash
.
For example, consider a collection foo
with the following indexes:
db.foo.createIndex( { x: 1 } ) db.foo.createIndex( { x: 1, y: 1 } ) db.foo.createIndex( { x: 1, z: 1 }, { partialFilterExpression: { x: { $gt: 10 } } } )
The following queries on the collection have the same shape:
db.foo.explain().find( { x: { $gt: 5 } } ) // Query Operation 1 db.foo.explain().find( { x: { $gt: 20 } } ) // Query Operation 2
Given these queries, the index with the partial filter expression can support query operation 2 but not
support query operation 1. Since the indexes available to support query operation 1
differs from query operation 2, the two queries have different
planCacheKey
.
If one of the indexes were dropped, or if a new index { x: 1, a: 1
}
were added, the planCacheKey
for both query operations will
change.
Availability
The planCacheShapeHash
and planCacheKey
are available in:
explain() output fields:
Starting in MongoDB 8.0, the pre-existing
queryHash
field is renamed toplanCacheShapeHash
. If you're using an earlier MongoDB version, you'll seequeryHash
instead ofplanCacheShapeHash
.profiler log messages and diagnostic log messages (i.e. mongod/mongos log messages) when logging slow queries.
$planCacheStats
aggregation stagePlanCache.listQueryShapes()
method/planCacheListQueryShapes
commandPlanCache.getPlansByQuery()
method/planCacheListPlans
command
Index Filters
Index filters are set with the planCacheSetFilter
command
and determine which indexes the planner evaluates for a query shape. A plan cache query shape consists of a combination of query, sort, and
projection specifications. If an index filter exists for a given query
shape, the planner only considers those indexes specified in the
filter.
When an index filter exists for the plan cache query shape, MongoDB ignores the
hint()
. To see whether MongoDB applied an index
filter for a query shape, check the indexFilterSet
field of either the db.collection.explain()
or the
cursor.explain()
method.
Index filters only affect which indexes the planner evaluates; the planner may still select the collection scan as the winning plan for a given plan cache query shape.
Index filters exist for the duration of the server process and do not persist after shutdown. MongoDB also provides a command to manually remove filters.
Because index filters override the expected behavior of the planner
as well as the hint()
method, use index filters
sparingly.
Starting in MongoDB 6.0, an index filter uses the collation previously set using the planCacheSetFilter
command.
Starting in MongoDB 8.0, use query settings instead of adding index filters. Index filters are deprecated starting in MongoDB 8.0.
Query settings have more functionality than index filters. Also, index
filters aren't persistent and you cannot easily create index filters for
all cluster nodes. To add query settings and explore examples, see
setQuerySettings
.