Atlas Search Index Performance
On this page
- Resource Requirements
- Index Size and Configuration
- Considerations
- Search Memory Management
- Creating and Updating an Atlas Search Index
- Eventual Consistency and Indexing Latency
- Document Mapping Explosions
- Storing Source Fields
- Scaling Considerations
- Atlas Search Upgrade
- Scaling Up Indexing Performance
- Atlas Cluster Configuration Change
Resource Requirements
Index Size and Configuration
Important
If you create an Atlas Search index for a collection that has or will soon
have more than 2.1 billion index objects, you must use numPartitions
or shard your cluster. Index partitions
are limited to 2.1 billion objects per partition.
When you create an Atlas Search index, field mapping defaults to dynamic, which means that Atlas Search dynamically indexes all the datatypes that can be dynamically indexed in your collection. Other options such as enabling highlights can also result in your index taking up more disk space. You can reduce the size and performance footprint of your Atlas Search index by:
Specifying a custom index definition to narrow the amount and type of data that is indexed.
Setting the
store
option tofalse
when specifying a string type in an index definition.
Note
Some limitations apply to Atlas Search on M0
, M2
, and M5
clusters only. To learn more, see
Atlas Search Free and Shared Tier Limitations.
Considerations
Some index configuration options can lead to indexes that take up a significant proportion of your disk space. In some cases, your index could be many times larger than the size of your data. Although this is expected behavior, it's important to be aware of the following indexing-intensive features:
Autocomplete
The Atlas Search autocomplete operator can be used to build functionality similar to search-as-you-type in your application. The Atlas Search autocomplete field type can cause large indexes, especially in the following cases:
Using
nGram
tokenization.Setting a wide
minGrams
tomaxGrams
range.Setting a
minGram
value of1
on a collection with millions of documents.
You can reduce the space used by the autocomplete
type index by
doing the following:
Reduce the range of
minGrams
andmaxGrams
to the minimum. Generally, we recommend settingmaxGrams
to the character count of the longest word in the field that you want to query. If you are unsure, for English language fields, we recommend starting withmaxGrams
value of10
.Avoid
nGram
tokenization strategy as, for a given string, Atlas Search creates more tokens fornGram
than foredgeGram
orrightEdgeGram
tokenization.
When indexing a string
field as the autocomplete type, we recommend that you index the
field as the Atlas Search string type also for
the following advantages:
Boost the score of exact matches when using the autocomplete operator. For an example, see the Advanced Example in the How to Use Autocomplete with Atlas Search tutorial.
Query the same field in the same query using the autocomplete operator and another operator that supports string search, such as the text operator. For an example, see the Search Across Multiple Fields example.
Embedded Documents
Atlas Search stops replicating changes for indexes larger than 2,100,000,000
index objects per partition, on a replica set or single shard, where each indexed
embedded document counts as a single object. Using the
embeddedDocuments
field type can result in indexing objects over
this limit, which causes an index to transition to a Stale
queryable state, and may result in stale results.
If you create an Atlas Search index for a collection that has or will soon
have more than 2.1 billion index objects, you must use the
numPartitions
option
or shard your cluster.
The exact number of index objects can vary based on the rate of document changes and deletions. The Search Max Number of Lucene Docs metric provides the upper bound of the current number of index objects across all indexes per replica set or shard. You can approximate the expected number of index objects in a single index by doing the following:
Calculate the number of index objects per document. For every level of nesting, each embedded document counts as a separate index object.
total number of index objects = 1 + number of nested embedded documents Multiply the number of index objects per document by the total number of documents in the collection
total number of index objects x total number of documents in collection
Note that this approximation is a lower bound.
Example
Consider the collection named
schools
, described in this tutorial, and suppose the
collection contains 1000 documents similar to the following:
{ "_id": 0, "name": "Springfield High", "mascot": "Pumas", "teachers": [ { "first": "Jane", "last": "Smith", "classes": [ { "subject": "art of science", "grade": "12th" }, ... // 2 more embedded documents ] }, ... // 1 more embedded document ], "clubs": { "stem": [ { "club_name": "chess", "description": "provides students opportunity to play the board game of chess informally and competitively in tournaments." }, ... // 1 more embedded document ], ... // 1 more embedded document } }
Now consider the index definition for the following fields in the
schools
collection:
The array of documents named teachers
is indexed as the
embeddedDocuments
type with dynamic mappings enabled.
However, the classes
field isn't indexed. Use the
following to calculate the index objects:
Calculate the number of index objects per document.
Number of ``teachers`` embedded documents = up to 2 Total number of index objects per document = 1 + 2 = 3 Multiply by the total number of documents in the collection.
Number of documents in the collection = 1000 Number of index objects per document = 3 Total number of index objects for collection = 1000 x 3 = 3000
The arrays of documents named teachers
and
teachers.classes
are indexed as the embeddedDocuments
type with dynamic mappings enabled. Use the following to
calculate the index objects:
Calculate the number of index objects per document:
Number of documents = 1 Number of ``teachers`` embedded documents = up to 2 Number of ``classes`` embedded documents = up to 3 Number of index objects per document = 1 + ( 2 x 3 ) = 7 Multiply by the total number of documents in the collection.
Number of documents in the collection = 1000 Number of index objects per document = 7 Total number of index objects: 1000 x 7 = 7000
If your collection has large arrays that might generate 2,100,000,000
index objects, you must shard any
clusters that contain indexes with the embeddedDocuments
type.
Faceted Search
If you want to filter and facet your data using the same field, we recommend that you index the field as following Atlas Search types:
stringFacet and token
numberFacet and number
For an example of filtering the data by another field for faceting, see the How to Use Facets with Atlas Search tutorial.
multi
Analyzers
Using a multi
analyzer to analyze the same field multiple
different ways can cause large indexes, especially when analyzing
fields with very long values.
Multilingual Search
You can use the Atlas Search Language Analyzers to index many languages. For the list of languages for which Atlas Search provides built-in analyzers, see the Language Analyzers. For an example, see the How to Run Multilingual Atlas Search Queries tutorial. To index and query languages that are currently not in the list of built-in Language Analyzers, you can create a custom analyzer. For an example, see the Custom Language Analyzer Example.
Suppose you have one document for each language in your collection. Consider the following:
You can index the fields separately in the same index using the Language Analyzers. A single index can support multiple languages in the same query.
Alternatively, you can create an index per language, which is useful in isolating the different language documents. Note that each index is a change stream cursor and so this might be expensive to maintain.
If you have the language documents nested inside a parent document, you can create a single index. However, your index definition payload might be large and your query might be complex.
To learn more about these data models and index definitions, see the MongoDB blog.
Synonym Collections
Inserts and updates to a synonym source collection are fast only if the synonym source collection is small. For best performance, we recommend batching inserts and updates to synonym source collections.
A synonym mapping definition doesn't require additional disk space aside from the disk space utilized by the synonym collection in the database. However, synonym mappings create artifacts in memory and therefore, for synonym collections with many documents, Atlas Search creates artifacts that occupy more memory.
Search Memory Management
Atlas Search uses both filesystem cache and JVM heap memory. It stores various objects like query objects, searcher objects, and other temporary data used during indexing and search operations in the JVM heap to process and track the query. It stores memory mapped files such as segment files, dictionary files, and the like in the filesystem cache to enhance its performance, especially when reading index files.
In deployments where both the mongod
and mongot
processes run on
the same node, the memory allocated to the mongod
varies based on
the cluster tier:
For
M40
and higher tier clusters,mongod
dedicates 50% or more of the physical RAM for the WiredTiger cache and the remaining memory is reserved for in-memory operations, underlying operation systems, and other system services.For
M30
and lower tier clusters,mongod
dedicates 25% of the physical RAM for the WiredTiger cache.
Therefore, the memory allocated to mongot
is a subset of the total
RAM, which can result in resource contention issues between mongod
and mongot
for not only memory, but also for CPU and Disk IO.
In deployments where the mongot
process runs on separate search
nodes, Atlas allocates a part of the available RAM to the JVM heap, and uses a small
amount of memory processes like monitoring and
automation, and the rest of the memory is available for search.
If your search indexes are large and available memory is low, you might
observe performance degradation during indexing and querying due to
insufficient memory. Search indexes might be large if you enable dynamic
mappings in your index for documents with arbitrary keys. This could
cause mapping explosions. The
excessive memory consumption by mongot
can also result in mongot
running OOM, which might also crash mongot
.
You can use the following metrics to determine if mongot
is running
OOM:
An increase in the number of
Search Page Faults
andDisk IOPS
. This happens if the OS keeps retrieving the required pages from disk and reads them into the RAM.Elevated levels of
Normalized Process/System CPU
andIOWait
, which can result in degraded performance.
We recommend migrating to dedicated search nodes for your production-ready applications. It is important to right-size the Search Node because low memory can lead to performance issues.
Creating and Updating an Atlas Search Index
Creating an Atlas Search index is resource-intensive. The performance of your Atlas cluster may be impacted while the index builds.
Atlas replicates all writes on the collection. This means that for each collection with Atlas Search indexes, the writes are amplified to the amount of Atlas Search indexes defined for that collection.
In some instances, your Atlas Search index must be rebuilt. Rebuilding the Atlas Search index also consumes resources and may affect database performance. Atlas Search automatically rebuilds the index only in the event of:
Changes to the index definition
Atlas Search version updates that include breaking changes
Hardware-related problems such as index corruption
Note
Atlas Search supports no-downtime indexing, which means you can continue to run search queries while Atlas Search rebuilds your index. Atlas Search keeps your old index up-to-date while the new index is being built. We recommend allocating free disk space equal to 125% of the disk space used by your old index for this operation. You can view the amount of disk space currently used by your index in the Search Disk Space Used metric.
If your index rebuild fails due to insufficient disk space, we recommend that you temporarily expand your cluster capacity to meet the increased demand. You can make this change manually as described in Fix Storage Issues, even for clusters with autoscaling enabled.
If you deployed separate Search Nodes, for certain changes such as Java 21 upgrade, Atlas automatically deploys additional Search Nodes for the duration of the index rebuild and you don't need to allocate any additional free disk space. Atlas doesn't deploy additional search nodes for an index rebuild that is caused by changes made to that index's definition.
Once Atlas Search rebuilds the index, the old index is automatically replaced without any further action from your side.
Eventual Consistency and Indexing Latency
Atlas Search supports eventual consistency and does not provide any stronger
consistency guarantees. This means that data inserted into a MongoDB
collection and indexed by Atlas Search will not be available immediately for
$search
queries.
Atlas Search reads data from MongoDB change streams and indexes that data in an asynchronous process. This process is typically very fast, but might sometimes be impacted by replication latency, system resource availability, and index definition complexity. A large number of Atlas Search indexes might also contribute to replication lag and latency for Atlas Search indexes.
Document Mapping Explosions
Mapping explosions occur when Atlas Search indexes a document with arbitrary
keys and you have a dynamic mapping.
The mongot
process might consume increasing amounts of memory and
could crash. If you add too many fields to an index, mapping explosions
can occur. To address this issue, you can upgrade your cluster or use a
static mapping that does not index all
fields in your data.
When searching over fields using a wildcard path, design your search to use a tuple-like schema. If you perform a wildcard path search that uses a key-value schema, Atlas Search indexes each key as its own field, which can cause mapping explosions.
Example
An example of a key-value schema is as follows:
ruleBuilder: { ruleName1: <data>, ruleName2: <data>, ..... ruleName1025: <data> }
An example of the same data restructured to use a tuple-like schema is as follows:
{ ruleBuilder: [ {name: ruleName1, data: <data>}, {name: ruleName2, data: <data>}, ... {name: ruleName1025, data: <data>} ] }
Storing Source Fields
You can configure fields to
store on Atlas Search and improve performance of subsequent aggregation
pipeline stages like $sort
, $match
,
$group
, and $skip
. Use this optimization if
your original documents and matched dataset are so large that a full
data lookup is inefficient. To learn more about storing specific fields
on Atlas Search and returning those stored fields only, see
Define Stored Source Fields in Your Atlas Search Index and
Return Stored Source Fields.
We recommend storing only the minimum number of fields required for
subsequent stages. If necessary, you can use $lookup
at
the end of the pipeline stage to retrieve entire documents as shown in
the Examples. Storing unnecessary fields
increases disk utilization and could negatively impact performance
during indexing and querying.
Scaling Considerations
Atlas Search Upgrade
Atlas Search is deployed on your Atlas cluster. When a new version of Atlas Search is deployed, your Atlas cluster might experience brief network failures in returning query results. To mitigate issues during deployment and minimize impact to your application, consider the following:
Implement retry logic in your application.
Configure Atlas maintenance windows.
Note
Atlas Search upgrades start only during the maintenance window and might continue after the maintenance window.
To learn more about the changes in each release, see Atlas Search Changelog.
Scaling Up Indexing Performance
You can scale up your initial sync and steady state indexing for an Atlas Search index by upgrading your cluster to a higher tier with more cores. Atlas Search uses a percentage of all available cores to run both initial sync and steady state indexing and performance improves as new cores are made available by upgrading your cluster.
Atlas Cluster Configuration Change
If you reconfigure your deployment to use the local NVMe storage type
or upscale an NVMe-based cluster, Atlas Search performs an initial sync
of all configured Atlas Search indexes after each node completes its underlying
configuration or upscale action. If the Atlas Search index initial syncs take
longer than the time it took to complete the cluster configuration
change, you can't run $search
queries until the initial
sync completes on all the nodes in your Atlas cluster.
We recommend deploying dedicated Search Nodes to scale your Atlas cluster and
$search
workloads independently. Dedicated Search Nodes
run only the mongot
process and therefore improve the availability,
performance, and workload balancing of the mongot
process.