All Products
Search
Document Center

ApsaraDB for MongoDB:Query plans and query replanning

Last Updated:Apr 22, 2024

This topic describes the definition of query plans. This topic also explains the possible causes of query replanning and provides solutions.

Query planner

For a query, the MongoDB query planner chooses and caches the most efficient query plan based on available indexes. The following figure shows how the query planner works.

image

The evaluation of the most efficient query plan is based on the number of work units (works) performed by the query execution plan when the query planner evaluates candidate plans. The query plan entry cached in the query planner can be used for queries with the same query shape.

The query plan entry cached in the query planner can be in one of the following states:

  • Missing: No entry exists in the query plan cache.

  • Inactive: The entry exists in the query plan cache and a works value is generated after evaluation. The entry can be converted to the Active state.

  • Active: The entry exists in the query plan cache. A winning query plan can be converted to the Inactive state.

The query plan cache is stored entirely in memory and does not persist. The cache is cleared each time the MongoDB database is restarted. The cache is also cleared if tables or indexes are deleted. The query plan cache has a size limit and follows the least recently used (LRU) cache replacement mechanism. Therefore, entries that are less frequently accessed are evicted from the cache from time to time.

In specific cases, you can run the following commands to manage query plans:

  • Clear the query plan cache from a specified collection.

    db.<collection>.getPlanCache().clear()
  • Query all query shapes in a specified collection.

    db.<collection>.getPlanCache().listQueryShapes()
  • Query the query plan for a specified query.

    db.<collection>.getPlanCache().getPlansByQuery({"query": {"name": "testname"}, "sort": { "name": 1 })

queryHash and planCacheKey

MongoDB 4.2 or later introduces queryHash to define query shapes. Each query shape is associated with a queryHash value. MongoDB 4.2 or later also introduces planCacheKey. Unlike queryHash, planCacheKey is a function of both a query shape and the available indexes for the shape. If indexes that can support the query shape are created or deleted, the planCacheKey value may change but the queryHash value remains unchanged.

For example, a table contains the following indexes and query shapes:

  • Index

    db.foo.createIndex( { x: 1 } )
    db.foo.createIndex( { x: 1, y: 1 } )
    db.foo.createIndex( { x: 1, z: 1 }, { partialFilterExpression: { x: { $gt: 10 } } } )
  • Query shapes

    db.foo.explain().find( { x: { $gt: 5 } } ) // Query operation 1
    db.foo.explain().find( { x: { $gt: 20 } } ) // Query operation 2 

The third index can support only query operation 2 but cannot support query operation 1. This way, the two queries have different planCacheKey values. If the {x:1, a:1} index is created, the planCacheKey values for both query operations change.

Query replanning

If data change occurs in a collection, the cached query plan is no longer suitable for the collection. The query plan must be replaced to ensure that the plan is synchronized with the change to the data.

If you want to execute a query plan that has the same query shape as a cached query plan, the query planner does not calculate the plan but directly uses the cached query plan. The query planner continuously evaluates the execution efficiency of the cached query plan. If the query planner determines that the cached query plan is more than 10 times less efficient than another query plan, the query planner stops and evicts the cached query plan and reevaluates the query plan. This process is known as query replanning.

Impacts and solutions

You may come across the "replanned":true keyword in slow query logs. This keyword indicates that the query planner cannot provide a consistently efficient query plan for the query conditions of a specific query shape.

Example of slow query logs:

"replanned":true,"replanReason":"cached plan was less efficient than expected: expected trial execution to take X works but it took at least 10X works"

Impacts

  • Frequent query replanning operations may degrade query performance.

  • Excessive query replanning can potentially lead to contention for mutex locks, resulting in high CPU utilization.

Solutions

  • Temporarily upgrade the instance specifications to alleviate database loads. For more information, see Overview.

  • We recommend that you clear query plans, and then check whether the query analyzer can select a more suitable query plan.

  • Use the hint() function to specify indexes for query conditions that are replanned in the business code. Example:

    db.<collection>.find({a:"ABC"},{b:1,_id:0}).sort({c:1}).hint({ a:1, c:1, b:1} )
    Note

    If hints are used for a query, the query plan selected by the query planner does not take effect.

  • Use index filtering to restrict indexes that you want to use for query conditions that are replanned in the business code. Example:

    // Configure index filtering.
    db.runCommand(
       {
          planCacheSetFilter: "<collection>",
          query: { a: "ABC" },
          projection: { b: 1, _id: 0 },
          sort: { c: 1 },
          indexes: [
             { a: 1, c: 1 , b: 1 }
          ]
       }
    )
    // Delete existing configurations.
    db.runCommand(
       {
          planCacheClearFilters: "<collection>"
       }
    )
    Note

    Index filters override the expected behavior of the query planner in selecting query plans.

  • (Recommended) Optimize queries and create effective indexes for the queries to prevent query replanning.

    Note

    We recommend that you check and modify query statements, available indexes, and document patterns instead of using hints and index filters.

  • (Recommended) If the major version of your ApsaraDB for MongoDB instance is 4.2 or 4.4, we recommend that you update the minor version of the instance to the latest version. This can effectively reduce the use of mutex locks. You can also upgrade the major version of the instance to 5.0 or later. For more information about the associated kernel JIRA ticket, see SERVER-40805. For more information about how to update the minor version of an instance and upgrade the major version of the instance, see Update the minor version of an ApsaraDB for MongoDB instance and Upgrade the major version of an ApsaraDB for MongoDB instance.

If the issues persist after you use the preceding methods, you can submit a ticket to contact technical support.

References