Cross-cluster search in Amazon OpenSearch Service
Cross-cluster search in Amazon OpenSearch Service lets you perform queries and aggregations across multiple connected domains. It often makes more sense to use multiple smaller domains instead of a single large domain, especially when you're running different types of workloads.
Workload-specific domains enable you to perform the following tasks:
-
Optimize each domain by choosing instance types for specific workloads.
-
Establish fault-isolation boundaries across workloads. This means that if one of your workloads fails, the fault is contained within that specific domain and doesn't impact your other workloads.
-
Scale more easily across domains.
Cross-cluster search supports OpenSearch Dashboards, so you can create visualizations and
dashboards across all your domains. You pay standard AWS data transfer
charges
Note
Open source OpenSearch also has documentation
Topics
Limitations
Cross-cluster search has several important limitations:
-
You can't connect an Elasticsearch domain to an OpenSearch domain.
-
You can't connect to self-managed OpenSearch/Elasticsearch clusters.
-
To connect domains across Regions, both domains must be on Elasticsearch 7.10 or later or OpenSearch.
-
A domain can have a maximum of 20 outgoing connections. Similarly, a domain can have a maximum of 20 incoming connections. In other words, one domain can connect to a maximum of 20 other domains.
-
The source domain must be on the same or a higher version than the destination domain. If you set up a bidirectional connection between two domains and you want to upgrade one or both of them, you must first delete one of the connections.
-
You can't use custom dictionaries or SQL with cross-cluster search.
-
You can't use AWS CloudFormation to connect domains.
-
You can't use cross-cluster search on M3 or burstable (T2 and T3) instances.
Cross-cluster search prerequisites
Before you set up cross-cluster search, make sure that your domains meet the following requirements:
-
Two OpenSearch domains, or Elasticsearch domains on version 6.7 or later
-
Fine-grained access control enabled
-
Node-to-node encryption enabled
Cross-cluster search pricing
There is no additional charge for searching across domains.
Setting up a connection
The “source” domain refers to the domain that a cross-cluster search request originates from. In other words, the source domain is the one that you send the initial search request to.
The “destination” domain is the domain that the source domain queries.
A cross-cluster connection is unidirectional from the source to the destination domain. This means that the destination domain can’t query the source domain. However, you can set up another connection in the opposite direction.
The source domain creates an "outbound" connection to the destination domain. The destination domain receives an "inbound" connection request from the source domain.
To set up a connection
-
On your domain dashboard, choose a domain and go to the Connections tab.
-
In the Outbound connections section, choose Request.
-
For Connection alias, enter a name for your connection.
-
Choose between connecting to a domain in your AWS account and Region or in another account or Region.
-
To connect to a cluster in your AWS account and Region, select the domain from the dropdown menu and choose Request.
-
To connect to a cluster in another AWS account or Region, select the ARN of the remote domain and choose Request. To connect domains across Regions, both domains must be running Elasticsearch version 7.10 or later or OpenSearch.
-
-
To skip unavailable clusters for cluster queries, select Skip unavailable. This setting ensures that your cross-cluster queries return partial results despite failures on one or more remote clusters.
-
Cross-cluster search first validates the connection request to make sure the prerequisites are met. If the domains are found to be incompatible, the connection request enters the
Validation failed
state. -
After the connection request is validated successfully, it is sent to the destination domain, where it needs to be approved. Until this approval happens, the connection remains in a
Pending acceptance
state. When the connection request is accepted at the destination domain, the state changes toActive
and the destination domain becomes available for queries.-
The domain page shows you the overall domain health and instance health details of your destination domain. Only domain owners have the flexibility to create, view, remove, and monitor connections to or from their domains.
-
After the connection is established, any traffic that flows between the nodes of the connected domains is encrypted. If you connect a VPC domain to a non-VPC domain and the non-VPC domain is a public endpoint that can receive traffic from the internet, the cross-cluster traffic between the domains is still encrypted and secure.
Removing a connection
Removing a connection stops any cross-cluster operation on its indexes.
-
On your domain dashboard, go to the Connections tab.
-
Select the domain connections that you want to remove and choose Delete, then confirm deletion.
You can perform these steps on either the source or destination domain to remove the
connection. After you remove the connection, it's still visible with a Deleted
status for a period of 15 days.
You can't delete a domain with active cross-cluster connections. To delete a domain, first remove all incoming and outgoing connections from that domain. This ensures you take into account the cross-cluster domain users before deleting the domain.
Setting up security and sample walkthrough
-
You send a cross-cluster search request to the source domain.
-
The source domain evaluates that request against its domain access policy. Because cross-cluster search requires fine-grained access control, we recommend an open access policy on the source domain.
{ "Version": "2012-10-17", "Statement": [ { "Effect": "Allow", "Principal": { "AWS": [ "*" ] }, "Action": [ "es:ESHttp*" ], "Resource": "arn:aws:es:region:account:domain/src-domain/*" } ] }
Note
If you include remote indexes in the path, you must URL-encode the URI in the domain ARN. For example, use
arn:aws:es:us-east-1:123456789012:domain/my-domain/local_index,dst%3Aremote_index
rather thanarn:aws:es:us-east-1:123456789012:domain/my-domain/local_index,dst:remote_index
.If you choose to use a restrictive access policy in addition to fine-grained access control, your policy must allow access to
es:ESHttpGet
at a minimum.{ "Version": "2012-10-17", "Statement": [ { "Effect": "Allow", "Principal": { "AWS": [ "arn:aws:iam::123456789012:user/test-user" ] }, "Action": "es:ESHttpGet", "Resource": "arn:aws:es:region:account:domain/src-domain/*" } ] }
-
Fine-grained access control on the source domain evaluates the request:
-
Is the request signed with valid IAM or HTTP basic credentials?
-
If so, does the user have permission to perform the search and access the data?
If the request only searches data on the destination domain (for example,
dest-alias:dest-index/_search
), you only need permissions on the destination domain.If the request searches data on both domains (for example,
source-index,dest-alias:dest-index/_search
), you need permissions on both domains.In fine-grained access control, users must have the
indices:admin/shards/search_shards
permission in addition to standardread
orsearch
permissions for the relevant indexes. -
-
The source domain passes the request to the destination domain. The destination domain evaluates this request against its domain access policy. You must include the
es:ESCrossClusterGet
permission on the destination domain:{ "Version": "2012-10-17", "Statement": [ { "Effect": "Allow", "Principal": { "AWS": "*" }, "Action": "es:ESCrossClusterGet", "Resource": "arn:aws:es:region:account:domain/dst-domain" } ] }
Make sure that the
es:ESCrossClusterGet
permission is applied for/dst-domain
and not/dst-domain/*
.However, this minimum policy only allows cross-cluster searches. To perform other operations, such as indexing documents and performing standard searches, you need additional permissions. We recommend the following policy on the destination domain:
{ "Version": "2012-10-17", "Statement": [ { "Effect": "Allow", "Principal": { "AWS": [ "*" ] }, "Action": [ "es:ESHttp*" ], "Resource": "arn:aws:es:region:account:domain/dst-domain/*" }, { "Effect": "Allow", "Principal": { "AWS": "*" }, "Action": "es:ESCrossClusterGet", "Resource": "arn:aws:es:region:account:domain/dst-domain" } ] }
Note
All cross-cluster search requests between domains are encrypted in transit by default as part of node-to-node encryption.
-
The destination domain performs the search and returns the results to the source domain.
-
The source domain combines its own results (if any) with the results from the destination domain and returns them to you.
-
We recommend Postman
for testing requests: -
On the destination domain, index a document:
POST https://dst-domain.us-east-1.es.amazonaws.com/books/_doc/1 { "Dracula": "Bram Stoker" }
-
To query this index from the source domain, include the connection alias of the destination domain within the query.
GET https://src-domain.us-east-1.es.amazonaws.com/<connection_alias>:books/_search { ... "hits": [ { "_index": "source-destination:books", "_type": "_doc", "_id": "1", "_score": 1, "_source": { "Dracula": "Bram Stoker" } } ] }
You can find the connection alias on the Connections tab on your domain dashboard.
-
If you set up a connection between
domain-a -> domain-b
with connection aliascluster_b
anddomain-a -> domain-c
with connection aliascluster_c
, searchdomain-a
,domain-b
, anddomain-c
as follows:GET https://src-domain.us-east-1.es.amazonaws.com/local_index,cluster_b:b_index,cluster_c:c_index/_search { "query": { "match": { "user": "domino" } } }
Response
{ "took": 150, "timed_out": false, "_shards": { "total": 3, "successful": 3, "failed": 0, "skipped": 0 }, "_clusters": { "total": 3, "successful": 3, "skipped": 0 }, "hits": { "total": 3, "max_score": 1, "hits": [ { "_index": "local_index", "_type": "_doc", "_id": "0", "_score": 1, "_source": { "user": "domino", "message": "Lets unite the new mutants", "likes": 0 } }, { "_index": "cluster_b:b_index", "_type": "_doc", "_id": "0", "_score": 2, "_source": { "user": "domino", "message": "I'm different", "likes": 0 } }, { "_index": "cluster_c:c_index", "_type": "_doc", "_id": "0", "_score": 3, "_source": { "user": "domino", "message": "So am I", "likes": 0 } } ] } }
If you did not choose to skip unavailable clusters in your connection setup, all destination clusters that you search must be available for your search request to run successfully. Otherwise, the whole request fails—even if one of the domains is not available, no search results are returned.
-
OpenSearch Dashboards
You can visualize data from multiple connected domains in the same way as from a single
domain, except that you must access the remote indexes using
connection-alias:index
. So, your index pattern must match
connection-alias:index
.