_routing field

_routing field

A document is routed to a particular shard in an index using the following formulas:

  1. routing_factor = num_routing_shards / num_primary_shards
  2. shard_num = (hash(_routing) % num_routing_shards) / routing_factor

num_routing_shards is the value of the index.number_of_routing_shards index setting. num_primary_shards is the value of the index.number_of_shards index setting.

The default _routing value is the document’s _id. Custom routing patterns can be implemented by specifying a custom routing value per document. For instance:

  1. resp = client.index(
  2. index="my-index-000001",
  3. id="1",
  4. routing="user1",
  5. refresh=True,
  6. document={
  7. "title": "This is a document"
  8. },
  9. )
  10. print(resp)
  11. resp1 = client.get(
  12. index="my-index-000001",
  13. id="1",
  14. routing="user1",
  15. )
  16. print(resp1)
  1. response = client.index(
  2. index: 'my-index-000001',
  3. id: 1,
  4. routing: 'user1',
  5. refresh: true,
  6. body: {
  7. title: 'This is a document'
  8. }
  9. )
  10. puts response
  11. response = client.get(
  12. index: 'my-index-000001',
  13. id: 1,
  14. routing: 'user1'
  15. )
  16. puts response
  1. const response = await client.index({
  2. index: "my-index-000001",
  3. id: 1,
  4. routing: "user1",
  5. refresh: "true",
  6. document: {
  7. title: "This is a document",
  8. },
  9. });
  10. console.log(response);
  11. const response1 = await client.get({
  12. index: "my-index-000001",
  13. id: 1,
  14. routing: "user1",
  15. });
  16. console.log(response1);
  1. PUT my-index-000001/_doc/1?routing=user1&refresh=true
  2. {
  3. "title": "This is a document"
  4. }
  5. GET my-index-000001/_doc/1?routing=user1

This document uses user1 as its routing value, instead of its ID.

The same routing value needs to be provided when getting, deleting, or updating the document.

The value of the _routing field is accessible in queries:

  1. resp = client.search(
  2. index="my-index-000001",
  3. query={
  4. "terms": {
  5. "_routing": [
  6. "user1"
  7. ]
  8. }
  9. },
  10. )
  11. print(resp)
  1. response = client.search(
  2. index: 'my-index-000001',
  3. body: {
  4. query: {
  5. terms: {
  6. _routing: [
  7. 'user1'
  8. ]
  9. }
  10. }
  11. }
  12. )
  13. puts response
  1. const response = await client.search({
  2. index: "my-index-000001",
  3. query: {
  4. terms: {
  5. _routing: ["user1"],
  6. },
  7. },
  8. });
  9. console.log(response);
  1. GET my-index-000001/_search
  2. {
  3. "query": {
  4. "terms": {
  5. "_routing": [ "user1" ]
  6. }
  7. }
  8. }

Querying on the _routing field (also see the ids query)

Data streams do not support custom routing unless they were created with the allow_custom_routing setting enabled in the template.

Searching with custom routing

Custom routing can reduce the impact of searches. Instead of having to fan out a search request to all the shards in an index, the request can be sent to just the shard that matches the specific routing value (or values):

  1. resp = client.search(
  2. index="my-index-000001",
  3. routing="user1,user2",
  4. query={
  5. "match": {
  6. "title": "document"
  7. }
  8. },
  9. )
  10. print(resp)
  1. response = client.search(
  2. index: 'my-index-000001',
  3. routing: 'user1,user2',
  4. body: {
  5. query: {
  6. match: {
  7. title: 'document'
  8. }
  9. }
  10. }
  11. )
  12. puts response
  1. const response = await client.search({
  2. index: "my-index-000001",
  3. routing: "user1,user2",
  4. query: {
  5. match: {
  6. title: "document",
  7. },
  8. },
  9. });
  10. console.log(response);
  1. GET my-index-000001/_search?routing=user1,user2
  2. {
  3. "query": {
  4. "match": {
  5. "title": "document"
  6. }
  7. }
  8. }

This search request will only be executed on the shards associated with the user1 and user2 routing values.

Making a routing value required

When using custom routing, it is important to provide the routing value whenever indexing, getting, deleting, or updating a document.

Forgetting the routing value can lead to a document being indexed on more than one shard. As a safeguard, the _routing field can be configured to make a custom routing value required for all CRUD operations:

  1. resp = client.indices.create(
  2. index="my-index-000002",
  3. mappings={
  4. "_routing": {
  5. "required": True
  6. }
  7. },
  8. )
  9. print(resp)
  10. resp1 = client.index(
  11. index="my-index-000002",
  12. id="1",
  13. document={
  14. "text": "No routing value provided"
  15. },
  16. )
  17. print(resp1)
  1. response = client.indices.create(
  2. index: 'my-index-000002',
  3. body: {
  4. mappings: {
  5. _routing: {
  6. required: true
  7. }
  8. }
  9. }
  10. )
  11. puts response
  12. response = client.index(
  13. index: 'my-index-000002',
  14. id: 1,
  15. body: {
  16. text: 'No routing value provided'
  17. }
  18. )
  19. puts response
  1. const response = await client.indices.create({
  2. index: "my-index-000002",
  3. mappings: {
  4. _routing: {
  5. required: true,
  6. },
  7. },
  8. });
  9. console.log(response);
  10. const response1 = await client.index({
  11. index: "my-index-000002",
  12. id: 1,
  13. document: {
  14. text: "No routing value provided",
  15. },
  16. });
  17. console.log(response1);
  1. PUT my-index-000002
  2. {
  3. "mappings": {
  4. "_routing": {
  5. "required": true
  6. }
  7. }
  8. }
  9. PUT my-index-000002/_doc/1
  10. {
  11. "text": "No routing value provided"
  12. }

Routing is required for all documents.

This index request throws a routing_missing_exception.

Unique IDs with custom routing

When indexing documents specifying a custom _routing, the uniqueness of the _id is not guaranteed across all of the shards in the index. In fact, documents with the same _id might end up on different shards if indexed with different _routing values.

It is up to the user to ensure that IDs are unique across the index.

Routing to an index partition

An index can be configured such that custom routing values will go to a subset of the shards rather than a single shard. This helps mitigate the risk of ending up with an imbalanced cluster while still reducing the impact of searches.

This is done by providing the index level setting index.routing_partition_size at index creation. As the partition size increases, the more evenly distributed the data will become at the expense of having to search more shards per request.

When this setting is present, the formulas for calculating the shard become:

  1. routing_value = hash(_routing) + hash(_id) % routing_partition_size
  2. shard_num = (routing_value % num_routing_shards) / routing_factor

That is, the _routing field is used to calculate a set of shards within the index and then the _id is used to pick a shard within that set.

To enable this feature, the index.routing_partition_size should have a value greater than 1 and less than index.number_of_shards.

Once enabled, the partitioned index will have the following limitations:

  • Mappings with join field relationships cannot be created within it.
  • All mappings within the index must have the _routing field marked as required.