Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Bulk Optimization with Auto-generated Doc IDs #14260

Open
khushbr opened this issue Jun 13, 2024 · 3 comments
Open

Bulk Optimization with Auto-generated Doc IDs #14260

khushbr opened this issue Jun 13, 2024 · 3 comments
Labels
enhancement Enhancement or improvement to existing feature or request Indexing:Performance RFC Issues requesting major changes

Comments

@khushbr
Copy link

khushbr commented Jun 13, 2024

Is your feature request related to a problem? Please describe

OpenSearch Bulk API executes multiple indexing/update/delete operations in a single call. For each of these operation, the name of index/stream or the alias is required and user can additionally also provide a custom doc ID. In case the doc ID is not provided, OpenSearch auto-generates a docID, a 128 bit UUID, which for practical purposes is unique. In absence of custom routing, the doc ID value is used to determine the shard routing info for the document through a function of Mod murmur3 hash on doc ID. Post this,TransportBulkAction on co-ordinator node generates per-Shard TransportShardBulkAction and sends them to the corresponding primaries. The co-ordinator node waits for response from all shards before sending the response back to the client.

In case of a slow shard/node scenario (say, shard in INITIALIZING state or node undergoing Garbage Collection) - the shard ends up becoming a bottleneck in bulk flow, increasing the tail latencies and holding up the resources, queue on the co-ordinator, potentially causing rejections.

The goal of the project is to tweak the document routing logic for auto-generated Doc IDs to better handle slow shard/node, provide better latencies by saving on network roundtrip and reduce the chatter b/w the co-ordinator and nodes.

  • [Part-1] Within a bulk request, index all documents belonging to an index to the same primary (selected in random order). Do not fan out and send multiple per-shard transport requests.
  • [Part-2] Within a bulk request, index all documents to the same node (if possible) or minimum set of nodes.
  • [Part-3] Adaptively Select the candidate shard and node based on the most recent latencies seen(sufficient?) and resource usage. This is to deterministically improve the indexing latency and also reduces the risk of hotspot.

The above optimizations should work within following constraints:

  • Get/Update by _id should work
  • Does not break (Online) Index Split operations
  • Does not disturb Custom routing
  • Introduces no deviation in behavior between indexing the same document with index API or bulk API
  • Minimal (< 1%) impact on the CPU usage and disk usage.

Describe the solution you'd like

In this section, we discuss the approache to solve the [Part-1].

Pre-generated DocIDs Cache/Store: Maintain a store with doc IDs tagged per-shard. In the TransportShardBulkAction.doRun() execution, instead of computing the docIDs on the fly, the pre-computed values are assigned. A background thread periodically refills the cache store (we can also explore using async futures to execute the cache refill), the per key (ShardID) refill count is a function of shard throughput. In case the store doesn’t have IDs for a shard, the algorithm falls back to the brute force approach. In the bulk request, for ‘m’ DocWriteRequests and ‘n’ Routing Shards, generate (n * m) docIDs and reject the docIDs not mapping to our randomly selected Target Shard. Minimal locking is used to get/refill/evict the store in a thread-safe manner using ConcurrentLinkedQueue and semaphores.

The above approaches need to be Benchmark for one shard is slow vs many shards are slow. Measure for indexing speed, cpu and memory usage (coordinator node), storage efficiency and lookup speed. Another dimension, measure for cluster throughput and rejects.

Related component

Indexing:Performance

Describe alternatives you've considered

Biased Hash Function: Currently in OpenSearch, the routing Shard is closely coupled with docID. The docID generated is a UUID with requirement of no collision for practical purposed. This docID is ran through a Murmur3 Hash (non-cryptographic) and then a mod function to generate the shard ID integer value to uniformly distribute the documents across the shards. We want to explore if it is possible to maintain these 2 primitives.

One of the simplest approach is to encode the shardId (randomly selected) information in the docID, along with the UUID and forgo the Murmur3 Hash Mod for routing shard calculation <encode_version>:<base36_shard_id>:<document_id>.

The update by doc _id or get by ids query can be handled at the client communication layer by returning the doc ID as the concatenated string value. At co-ordinator node on transport, a decoder extracts the version, shard ID and routes the doc ID to the specific shard ID. Since there is no hash calculation, the document routing is fast, with no footprint on the CPU cycles and JVM. However, there are drawbacks to this approach:

  1. Backward Incompatible - For the indices on older version code, the document routing falls back to the older request routing algorithm. However, for version upgrade scenario, this behavior starts failing.
  2. Shard Split - The shard split(Online) need to be handled separately. In case of Shard Split on an existing Index, the search _id requests start failing as unable to locate the resident shard. A workaround here can be to maintain the split history in a durable table (inside Index Metadata), consistent across cluster restarts. Say a Shard (1) splits into 3 shards to give Shard (11), Shard (12) and Shard (13) - we maintain this information in the table and route the requests based on the docID (identify the source shard) and shard split history (identify the split shard). This approach however adds additional map lookup and thus has impact on the latency.

The below table provides the performance trade-off for the various approaches mentioned:

Brute Force Pre-generated DocIDs Store Biased Hash Function
Memory No Impact For slow moving indices, the docIDs will lie unused in the store untill evicted. This will waste memory. On-par with current implementation
CPU CPU cycles wasted for generating the rejected docIDs Minor overhead in store maintenance Depends on complexity of Hash Function; will consume more CPU
Latency More than other 2 approaches but expected to perform better than current implementation for worse case - slow shard scenario With doc ID generation running in parallel, latency regression is not expected. Fast
Throughput No Impact Segment locking when consuming the store can reduce the throughput No Impact

Additional context

No response

@khushbr khushbr added enhancement Enhancement or improvement to existing feature or request untriaged labels Jun 13, 2024
@khushbr
Copy link
Author

khushbr commented Jun 13, 2024

Benchmark Results

#### Summary:
  • Improvement in the tail latencies(100th percentile) but the average latency has increased.
  • The Cumulative Indexing time is less with cache but the percentile latency is high (50th, 90th and 99.99th) with the cache batch size of 10000.
  • The min, max and median throughput value is less with cache vs without. Given the max throughput hits 5K docs/s, updating the batch size to 50000 can increase the throughput.

1. DocID Generation Latency

50th percentile latency (ms) 90th percentile latency (ms) 99th percentile latency (ms) 99.99th percentile latency (ms) 100th percentile latency (ms)
Baseline 0.00145 0.00168 0.00234 0.04285 36.12002
Brute Force 0.067 0.14999 0.19022 0.3226 28.77094
Pre-Generated DocID Store (Background thread) 0.00019 0.00027 0.0006 0.01702 6.5273
Pre-Generated DocID Store (CompletableFuture) 0.07115 0.18344 0.24779 1.11933 105.91981
Pre-Generated DocID Store (Sync) 0.00539 0.01137 0.09186 0.22667 20.83127

2. Uber level Benchmarks

Min Throughput (docs/s) Mean Throughput (docs/s) Median Throughput (docs/s) Max Throughput (docs/s) error rate (%) Store size (GB) 50th percentile latency (ms) 90th percentile latency (ms) 99th percentile latency (ms) 99.99th percentile latency (ms) 100th percentile latency (ms) Cumulative indexing time of primary shards (min) Max cumulative indexing time across primary shards(min) Total Young Gen GC time (s) Total Young Gen GC count
Default 53276.7 63182.9 62268.1 76517.7 0.11 39.7833 315.284 1457.56 5548.24 10108.3 10802.4 105.244 21.3867 3.251 124
With Cache - Synchronized Block 42141.2 48362.4 47869 53992.4 0.03 35.5269 621.462 1018.8 6042.76 8983.25 10197 73.0625 15.1691 2.318 100
With Non-blocking Queue 43178.7 47667.3 47371.7 50017.2 0.02 36.3795 617.615 1027.93 4956.33 9716.62 10361.4 72.6281 16.1898 4.471 171
With CompletableFuture Refill 26981.1 29596.6 29531.3 30921.2 0.32 35.612 1132.23 1637.44 5979.19 10556.1 10770.7 72.0051 14.7818 43.547 1430
Sync flow Refill 40375.2 43389.9 43513 45694.8 0.11 35.7327 677.158 1118.33 6275.53 10183 10869.5 73.4668 16.064 3.356 134

Comments:

  1. Possible Locking at Doc ID generation/consumption as indicated by high error rate for ‘network connection timed out’.
  2. High Total Young GC Count indicate lot of objects created in Young Gen. Can we take heap samples to analyze the top consumers ?
  3. The refill Batch size is 10000 and refill frequency is 5ms, the code falls back to brute force approach of generating doc ID for given shard on fly if the docIDs are not available in the cache. This is happening frequently (2226 times in 1 minute). Tune the refill frequency to 1ms and the batch size to 200000 and accquire locks together instead of 1 at a time.

@khushbr khushbr removed the untriaged label Jun 13, 2024
@shwetathareja
Copy link
Member

@khushbr please link the POC branch code for reference to co-relate with nos.

@vikasvb90 vikasvb90 added the RFC Issues requesting major changes label Jul 8, 2024
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement Enhancement or improvement to existing feature or request Indexing:Performance RFC Issues requesting major changes
Projects
None yet
Development

No branches or pull requests

3 participants