﻿---
title: Query across Serverless projects with ES|QL
description: Learn how to use the ES|QL language in Elasticsearch to query across multiple Serverless projects. Learn about index resolution, project routing, and accessing project metadata.
url: https://docs-v3-preview.elastic.dev/elastic/elasticsearch/pull/144300/reference/query-languages/esql/esql-cross-serverless-projects
products:
  - Elasticsearch
applies_to:
  - Elastic Cloud Serverless: Preview
  - Elastic Stack: Unavailable
---

# Query across Serverless projects with ES|QL
[Cross-project search](https://docs-v3-preview.elastic.dev/elastic/docs-content/tree/main/explore-analyze/cross-project-search) (CPS) enables you to run queries across multiple [linked Serverless projects](https://docs-v3-preview.elastic.dev/elastic/docs-content/tree/main/explore-analyze/cross-project-search#project-linking) from a single request.
There are several ways to control which projects a query runs against:
- **[Query all projects](#query-all-projects-default)**: If you just want to query across all linked projects, no special syntax is required. Queries automatically run against the origin and all linked projects by default.
- **[Use project routing](#use-project-routing)**: Use project routing to limit the scope of your search to specific projects before query execution. Excluded projects are not queried.
- **[Use index expressions](#use-index-expressions)**: Use index expressions for fine-grained control over which projects and indices are queried, by qualifying index names with a project alias. Search expressions can be used independently or combined with project routing.


## Before you begin

This page covers ES|QL-specific CPS behavior. Before continuing, make sure you are familiar with the following:
- [Cross-project search](https://docs-v3-preview.elastic.dev/elastic/docs-content/tree/main/explore-analyze/cross-project-search)
- [Linked projects](https://docs-v3-preview.elastic.dev/elastic/docs-content/tree/main/explore-analyze/cross-project-search/cross-project-search-link-projects)
- [How search works in CPS](https://docs-v3-preview.elastic.dev/elastic/docs-content/tree/main/explore-analyze/cross-project-search/cross-project-search-search)
- [Project routing in CPS](https://docs-v3-preview.elastic.dev/elastic/docs-content/tree/main/explore-analyze/cross-project-search/cross-project-search-project-routing)
- [Tags in CPS](https://docs-v3-preview.elastic.dev/elastic/docs-content/tree/main/explore-analyze/cross-project-search/cross-project-search-tags)


## Query all projects (default)

The default behavior is to query across the origin project and all linked projects automatically.
The following example queries the `data` index and includes the `_index` metadata field to identify which project each result came from:
```json

{
  "query": "FROM data METADATA _index", <1>
  "include_execution_metadata": true     <2>
}
```

The response includes:
- a `_clusters` object showing the status of each participating project
- a `values` array where each row includes the qualified index name identifying which project the document came from

<dropdown title="Example response">
  ```json
  {
    "took": 329,
    "is_partial": false,
    "columns": [
      { "name": "_index", "type": "keyword" }
    ],
    "values": [
      ["data"],                      
      ["linked-project-1:data"]      
    ],
    "_clusters": {
      "total": 2,
      "successful": 2,
      "running": 0,
      "skipped": 0,
      "partial": 0,
      "failed": 0,
      "details": {
        "_origin": {                 
          "status": "successful",
          "indices": "data",
          "took": 328,
          "_shards": { "total": 1, "successful": 1, "skipped": 0, "failed": 0 }
        },
        "linked-project-1": {        
          "status": "successful",
          "indices": "data",
          "took": 256,
          "_shards": { "total": 1, "successful": 1, "skipped": 0, "failed": 0 }
        }
      }
    }
  }
  ```
</dropdown>


## Use project routing

[Project routing](https://docs-v3-preview.elastic.dev/elastic/docs-content/tree/main/explore-analyze/cross-project-search/cross-project-search-project-routing) limits the scope of a query to specific projects, based on tag values.
Project routing happens before query execution, so excluded projects are never queried. This can help reduce cost and latency.
<note>
  Project routing expressions use Lucene query syntax. The `:` operator matches a tag value, equivalent to `=` in other query languages. For example, `_alias:my-project` matches projects whose alias is `my-project`.
</note>

You can specify project routing in two ways:
- [Embed project routing in the query with `SET`](#option-1-use-the-set-source-command): This approach works wherever you can write an ES|QL query.
- [Pass project routing in the `_query` API request body](#option-2-pass-project_routing-in-the-api-request-body): You can pass a `project_routing` field to keep project routing logic separate from the query string.

<important>
  If both options are combined, `SET project_routing` takes precedence.
</important>


### Option 1: Use the `SET` source command

`SET project_routing` embeds project routing directly within the ES|QL query. You can use this approach wherever you write ES|QL. [`SET`](https://docs-v3-preview.elastic.dev/elastic/elasticsearch/pull/144300/reference/query-languages/esql/commands/set) must appear before other ES|QL commands. The semicolon after the last parameter separates it from the rest of the query. The order of parameters within `SET` does not matter.
```esql
SET project_routing="_alias:my-project";    
FROM data
| STATS COUNT(*)
```


### Option 2: Pass `project_routing` in the API request body

If you are constructing the full `_query` request, you can pass the `project_routing` field in the request body. This keeps project routing logic separate from the query string:
```json

{
  "query": "FROM data | STATS COUNT(*)",
  "project_routing": "_alias:my-project"    <1>
}
```


### Reference a named project routing expression

Both options support referencing a named project routing expression using the `@` prefix.
Before you can reference a named expression, you must create it using the `_project_routing` API.
For instructions, refer to [Using named project routing expressions](https://docs-v3-preview.elastic.dev/elastic/docs-content/tree/main/explore-analyze/cross-project-search/cross-project-search-project-routing#creating-and-managing-named-project-routing-expressions).
<tab-set>
  <tab-item title="Request body">
    ```json

    {
      "query": "FROM logs | STATS COUNT(*)",
      "project_routing": "@custom-expression"
    }
    ```
  </tab-item>

  <tab-item title="SET directive">
    ```esql
    SET project_routing="@custom-expression";
    FROM logs
    | STATS COUNT(*)
    ```
  </tab-item>
</tab-set>


## Use index expressions

ES|QL supports two types of [index expressions](https://docs-v3-preview.elastic.dev/elastic/docs-content/tree/main/explore-analyze/cross-project-search/cross-project-search-search#search-expressions):
- **Unqualified expressions** have no project prefix and search across all projects. Example: `logs*`.
- **Qualified expressions** include a project alias prefix to target a specific project or set of projects. Example: `project1:logs*`.


### Restrict to the origin project

Use `_origin:` to target only the project from which the query is run:
```esql
FROM _origin:data    
| STATS COUNT(*)
```


### Restrict to a specific linked project

Prefix the index name with the linked project's alias:
```esql
FROM linked-project-1:data    
| STATS COUNT(*)
```


### Exclude specific projects

Prefix an index expression with `-` to exclude it from the resolved set.
The following example uses `-_origin:*` to exclude all indices from the origin project:
```esql
FROM data,-_origin:*    
| STATS COUNT(*)
```

<note>
  `*:` in CPS does not behave like `*:` in [cross-cluster search (CCS)](https://docs-v3-preview.elastic.dev/elastic/elasticsearch/pull/144300/reference/query-languages/esql/esql-cross-clusters) (which is used to query across clusters in non-serverless deployments):
  - In CCS, `*:` targets all remote clusters and excludes the local cluster.
  - In CPS, `*:` resolves against all projects including the origin, the same as an unqualified expression.
</note>


### Combine qualified and unqualified expressions

You can mix unqualified and qualified expressions in the same query:
```esql
FROM data, _origin:logs    
| LIMIT 100
```

<tip>
  Error handling differs between expression types. Unqualified expressions fail only if the index exists in none of the searched projects. Qualified expressions fail if the index is missing from the targeted project, regardless of whether it exists elsewhere.
  For a detailed explanation, refer to [Unqualified expression behavior](https://docs-v3-preview.elastic.dev/elastic/docs-content/tree/main/explore-analyze/cross-project-search/cross-project-search-search#behavior-unqualified).
</tip>


## Include project metadata in results

Use the `METADATA` keyword in a `FROM` command to include project-level information alongside query results.
Project metadata fields use the `_project.` prefix to distinguish them from document fields.
You can use project metadata fields in two ways:
- As columns in returned result rows, to identify which project each document came from.
- In downstream commands such as `WHERE`, `STATS`, and `KEEP`, to filter, aggregate, or sort results by project. Note: `WHERE` [filters results after all projects are queried](#filter-results-by-project-tag) and does not limit query scope.

Available fields include all predefined tags and any custom tags you have defined.
You can also use wildcard patterns such as `_project.my-prefix*` or `_project.*`.
For a full list of predefined tags, refer to [Tags in CPS](https://docs-v3-preview.elastic.dev/elastic/docs-content/tree/main/explore-analyze/cross-project-search/cross-project-search-tags).
<important>
  You must declare a project metadata field in the `METADATA` clause to use it anywhere in the query, including in `WHERE`, `STATS`, `KEEP`, and other downstream commands.
</important>


### Return project alias alongside results

Include `_project._alias` in `METADATA` to add the project alias as a column on each result row:
```esql
FROM logs* METADATA _project._alias    
| KEEP @timestamp, message, _project._alias
```

<dropdown title="Example response">
  ```json
  {
    "took": 47,
    "is_partial": false,
    "columns": [
      { "name": "@timestamp", "type": "date" },
      { "name": "message", "type": "keyword" },
      { "name": "_project._alias", "type": "keyword" }
    ],
    "values": [
      ["2025-01-15T10:23:00.000Z", "connection established", "origin-project"],    
      ["2025-01-15T10:24:00.000Z", "request timeout", "linked-project-1"],         
      ["2025-01-15T10:25:00.000Z", "disk full", "linked-project-1"]
    ]
  }
  ```
</dropdown>


### Aggregate results by project

Include `_project._alias` in `METADATA` to group and count results by project:
```esql
FROM logs* METADATA _project._alias    
| STATS doc_count = COUNT(*) BY _project._alias
```


### Filter results by project tag

A project tag in a `WHERE` clause filters the result set after the query runs across all projects. It does not limit which projects are queried.
The following examples show the difference between filtering with `WHERE` and restricting the query scope with project routing.

#### Filter with `WHERE` (post-query)

```esql
FROM logs* METADATA _project._csp    
| WHERE _project._csp == "aws"       
```

<important>
  Filtering with `WHERE` on a project tag happens after all projects are queried. To optimize a query, use [project routing](#use-project-routing) to select projects before execution.
</important>


#### Restrict with project routing (pre-query)

```esql
SET project_routing="_alias:aws-project";    
FROM logs*
| STATS COUNT(*)
```


### Use project routing and METADATA together

Project routing and project metadata serve different purposes and are independent of each other.
Project routing determines which projects are queried, before execution.
METADATA makes tag values available in query results and downstream commands, at query time.
Using a tag in `METADATA` does not route the query. Using project routing does not populate `METADATA` fields.
To both restrict queried projects and include tag values in results, specify both:
```esql
SET project_routing="_alias:*linked*";    
FROM logs METADATA _project._alias        
| STATS COUNT(*) BY _project._alias
```


## Limitations


### Project routing supports alias only

Initially, project routing only supports the `_alias` tag.
Other predefined tags (`_csp`, `_region`, and so on) and custom tags are not yet supported as project routing criteria.

### LOOKUP JOIN across projects

ES|QL `LOOKUP JOIN` follows the same constraints as [ES|QL cross-cluster `LOOKUP JOIN`](/elastic/elasticsearch/pull/144300/reference/query-languages/esql/esql-lookup-join#cross-cluster-support).
The lookup index must exist on every project being queried, because each project uses its own local copy of the lookup index data.

## Related pages

- [ES|QL cross-cluster search](https://docs-v3-preview.elastic.dev/elastic/elasticsearch/pull/144300/reference/query-languages/esql/esql-cross-clusters): the equivalent feature for non-serverless deployments.
- [`FROM` command](https://docs-v3-preview.elastic.dev/elastic/elasticsearch/pull/144300/reference/query-languages/esql/commands/from): full reference for index expressions and `METADATA` syntax.
- [`SET` directive](https://docs-v3-preview.elastic.dev/elastic/elasticsearch/pull/144300/reference/query-languages/esql/commands/set): full reference for the `SET` directive in ES|QL.
- [ES|QL metadata fields](https://docs-v3-preview.elastic.dev/elastic/elasticsearch/pull/144300/reference/query-languages/esql/esql-metadata-fields): full reference for metadata fields available in ES|QL queries.
- [ES|QL `LOOKUP JOIN`](https://docs-v3-preview.elastic.dev/elastic/elasticsearch/pull/144300/reference/query-languages/esql/esql-lookup-join): details on `LOOKUP JOIN` constraints, including cross-cluster and cross-project support.