> ## Documentation Index
> Fetch the complete documentation index at: https://www.meilisearch.com/docs/llms.txt
> Use this file to discover all available pages before exploring further.

# Configure distinct attribute

> Use the distinct attribute to deduplicate search results by returning only one document per unique value of a given field.

The distinct attribute is a special, user-designated field. It is most commonly used to prevent Meilisearch from returning a set of several similar documents, instead forcing it to return only one.

You may set a distinct attribute in two ways: using the `distinctAttribute` index setting during configuration, or the `distinct` search parameter at search time.

## Setting a distinct attribute during configuration

`distinctAttribute` is an index setting that configures a default distinct attribute Meilisearch applies to all searches and facet retrievals in that index.

<Warning>
  There can be only one `distinctAttribute` per index. Trying to set multiple fields as a `distinctAttribute` will return an error.
</Warning>

<Warning>
  Updating `distinctAttribute` will re-index all documents in the index, which can take some time. When setting up a new index, we recommend updating your index settings first and then adding documents. This reduces RAM consumption compared to adding documents and then changing settings, since Meilisearch would have to re-index the whole dataset.
</Warning>

The value of a field configured as a distinct attribute will always be unique among returned documents. This means **there will never be more than one occurrence of the same value** in the distinct attribute field among the returned documents.

When multiple documents have the same value for the distinct attribute, Meilisearch returns only the highest-ranked result after applying [ranking rules](/capabilities/full_text_search/relevancy/ranking_rules). If two or more documents are equivalent in terms of ranking, Meilisearch returns the first result according to its `internal_id`.

## Example

Suppose you have an e-commerce dataset. For an index that contains information about jackets, you may have several identical items with minor variations such as color or size.

As shown below, this dataset contains three documents representing different versions of a Lee jeans leather jacket. One of the jackets is brown, one is black, and the last one is blue.

<CodeGroup>
  ```json theme={null}
  [
    {
      "id": 1,
      "description": "Leather jacket",
      "brand": "Lee jeans",
      "color": "brown",
      "product_id": "123456"
    },
    {
      "id": 2,
      "description": "Leather jacket",
      "brand": "Lee jeans",
      "color": "black",
      "product_id": "123456"
    },
    {
      "id": 3,
      "description": "Leather jacket",
      "brand": "Lee jeans",
      "color": "blue",
      "product_id": "123456"
    }
  ]
  ```
</CodeGroup>

By default, a search for `lee leather jacket` would return all three documents. This might not be desired, since displaying nearly identical variations of the same item can make results appear cluttered.

In this case, you may want to return only one document with the `product_id` corresponding to this Lee jeans leather jacket. To do so, you could set `product_id` as the `distinctAttribute`.

<CodeGroup>
  ```bash cURL theme={null}
  curl \
    -X PUT 'MEILISEARCH_URL/indexes/jackets/settings/distinct-attribute' \
    -H 'Content-Type: application/json' \
    --data-binary '"product_id"'
  ```

  ```javascript JS theme={null}
  client.index('jackets').updateDistinctAttribute('product_id')
  ```

  ```python Python theme={null}
  client.index('jackets').update_distinct_attribute('product_id')
  ```

  ```php PHP theme={null}
  $client->index('jackets')->updateDistinctAttribute('product_id');
  ```

  ```java Java theme={null}
  client.index("jackets").updateDistinctAttributeSettings("product_id");
  ```

  ```ruby Ruby theme={null}
  client.index('jackets').update_distinct_attribute('product_id')
  ```

  ```go Go theme={null}
  client.Index("jackets").UpdateDistinctAttribute("product_id")
  ```

  ```csharp C# theme={null}
  await client.Index("jackets").UpdateDistinctAttributeAsync("product_id");
  ```

  ```rust Rust theme={null}
  let task: TaskInfo = client
    .index("jackets")
    .set_distinct_attribute("product_id")
    .await
    .unwrap();
  ```

  ```swift Swift theme={null}
  client.index("jackets").updateDistinctAttribute("product_id") { (result) in
      switch result {
      case .success(let task):
          print(task)
      case .failure(let error):
          print(error)
      }
  }
  ```

  ```dart Dart theme={null}
  await client.index('jackets').updateDistinctAttribute('product_id');
  ```
</CodeGroup>

By setting `distinctAttribute` to `product_id`, search requests **will never return more than one document with the same `product_id`**.

After setting the distinct attribute as shown above, querying for `lee leather jacket` would only return the first document found. The response would look like this:

<CodeGroup>
  ```json theme={null}
  {
    "hits": [
      {
        "id": 1,
        "description": "Leather jacket",
        "brand": "Lee jeans",
        "color": "brown",
        "product_id": "123456"
      }
    ],
    "offset": 0,
    "limit": 20,
    "estimatedTotalHits": 1,
    "processingTimeMs": 0,
    "query": "lee leather jacket"
  }
  ```
</CodeGroup>

For more in-depth information on distinct attribute, consult the [API reference](/reference/api/settings/get-distinctattribute).

## Setting a distinct attribute at search time

`distinct` is a search parameter you may add to any search query. It allows you to selectively use distinct attributes depending on the context. `distinct` takes precedence over `distinctAttribute`.

To use an attribute with `distinct`, first add it to the `filterableAttributes` list:

<CodeGroup>
  ```bash cURL theme={null}
  curl \
    -X PUT 'MEILISEARCH_URL/indexes/products/settings/filterable-attributes' \
    -H 'Content-Type: application/json' \
    --data-binary '[
      "product_id",
      "sku",
      "url"
    ]'
  ```

  ```javascript JS theme={null}
  client.index('products').updateFilterableAttributes(['product_id', 'sku', 'url'])
  ```

  ```python Python theme={null}
  client.index('products').update_filterable_attributes(['product_id', 'sku', 'url'])
  ```

  ```php PHP theme={null}
  $client->index('products')->updateFilterableAttributes(['product_id', 'sku', 'url']);
  ```

  ```java Java theme={null}
  Settings settings = new Settings();
  settings.setFilterableAttributes(new String[] { "product_id", "SKU", "url" });
  client.index("products").updateSettings(settings);
  ```

  ```ruby Ruby theme={null}
  client.index('products').update_filterable_attributes([
    'product_id',
    'sku',
    'url'
  ])
  ```

  ```go Go theme={null}
  filterableAttributes := []interface{}{
    "product_id",
    "sku",
    "url",
  }
  client.Index("products").UpdateFilterableAttributes(&filterableAttributes)
  ```

  ```csharp C# theme={null}
  List<string> attributes = new() { "product_id", "sku", "url" };
  TaskInfo result = await client.Index("products").UpdateFilterableAttributesAsync(attributes);
  ```

  ```rust Rust theme={null}
  let task: TaskInfo = client
    .index("products")
    .settings()
    .set_filterable_attributes(["product_id", "sku", "url"])
    .execute()
    .await
    .unwrap();
  ```
</CodeGroup>

Then use `distinct` in a search query, specifying one of the configured attributes:

<CodeGroup>
  ```bash cURL theme={null}
  curl \
    -X POST 'MEILISEARCH_URL/indexes/products/search' \
    -H 'Content-Type: application/json' \
    --data-binary '{
      "q": "white shirt",
      "distinct": "sku"
    }'
  ```

  ```javascript JS theme={null}
  client.index('products').search('white shirt', { distinct: 'sku' })
  ```

  ```python Python theme={null}
  client.index('products').search('white shirt', { distinct: 'sku' })
  ```

  ```php PHP theme={null}
  $client->index('products')->search('white shirt', [
    'distinct' => 'sku'
  ]);
  ```

  ```java Java theme={null}
  SearchRequest searchRequest = SearchRequest.builder().q("white shirt").distinct("sku").build();
  client.index("products").search(searchRequest);
  ```

  ```ruby Ruby theme={null}
  client.index('products').search('white shirt', {
    distinct: 'sku'
  })
  ```

  ```go Go theme={null}
  client.Index("products").Search("white shirt", &meilisearch.SearchRequest{
    Distinct: "sku",
  })
  ```

  ```csharp C# theme={null}
  var params = new SearchQuery()
  {
    Distinct = "sku"
  };
  await client.Index("products").SearchAsync<Product>("white shirt", params);
  ```

  ```rust Rust theme={null}
  let res = client
    .index("products")
    .search()
    .with_query("white shirt")
    .with_distinct("sku")
    .execute()
    .await
    .unwrap();
  ```
</CodeGroup>
