k-nearest neighbor (kNN) search

A k-nearest neighbor (kNN) search finds the k nearest vectors to a query vector, as measured by a similarity metric.

Common use cases for kNN include:

  • Relevance ranking based on natural language processing (NLP) algorithms
  • Product recommendations and recommendation engines
  • Similarity search for images or videos

Prerequisites

  • To run a kNN search, you must be able to convert your data into meaningful vector values. You can create these vectors using a natural language processing (NLP) model in Elasticsearch, or generate them outside Elasticsearch. Vectors can be added to documents as dense_vector field values. Queries are represented as vectors with the same dimension.

    Design your vectors so that the closer a document’s vector is to a query vector, based on a similarity metric, the better its match.

  • To complete the steps in this guide, you must have the following index privileges:

    • create_index or manage to create an index with a dense_vector field
    • create, index, or write to add data to the index you created
    • read to search the index

kNN methods

Elasticsearch supports two methods for kNN search:

In most cases, you’ll want to use approximate kNN. Approximate kNN offers lower latency at the cost of slower indexing and imperfect accuracy.

Exact, brute-force kNN guarantees accurate results but doesn’t scale well with large datasets. With this approach, a script_score query must scan each matching document to compute the vector function, which can result in slow search speeds. However, you can improve latency by using a query to limit the number of matching documents passed to the function. If you filter your data to a small subset of documents, you can get good search performance using this approach.

Approximate kNN

Compared to other types of search, approximate kNN search has specific resource requirements. In particular, all vector data must fit in the node’s page cache for it to be efficient. Please consult the approximate kNN search tuning guide for important notes on configuration and sizing.

To run an approximate kNN search, use the knn option to search one or more dense_vector fields with indexing enabled.

  1. Explicitly map one or more dense_vector fields. Approximate kNN search requires the following mapping options:

    • A similarity value. This value determines the similarity metric used to score documents based on similarity between the query and document vector. For a list of available metrics, see the similarity parameter documentation. The similarity setting defaults to cosine.
    1. resp = client.indices.create(
    2. index="image-index",
    3. mappings={
    4. "properties": {
    5. "image-vector": {
    6. "type": "dense_vector",
    7. "dims": 3,
    8. "similarity": "l2_norm"
    9. },
    10. "title-vector": {
    11. "type": "dense_vector",
    12. "dims": 5,
    13. "similarity": "l2_norm"
    14. },
    15. "title": {
    16. "type": "text"
    17. },
    18. "file-type": {
    19. "type": "keyword"
    20. }
    21. }
    22. },
    23. )
    24. print(resp)
    1. response = client.indices.create(
    2. index: 'image-index',
    3. body: {
    4. mappings: {
    5. properties: {
    6. "image-vector": {
    7. type: 'dense_vector',
    8. dims: 3,
    9. similarity: 'l2_norm'
    10. },
    11. "title-vector": {
    12. type: 'dense_vector',
    13. dims: 5,
    14. similarity: 'l2_norm'
    15. },
    16. title: {
    17. type: 'text'
    18. },
    19. "file-type": {
    20. type: 'keyword'
    21. }
    22. }
    23. }
    24. }
    25. )
    26. puts response
    1. const response = await client.indices.create({
    2. index: "image-index",
    3. mappings: {
    4. properties: {
    5. "image-vector": {
    6. type: "dense_vector",
    7. dims: 3,
    8. similarity: "l2_norm",
    9. },
    10. "title-vector": {
    11. type: "dense_vector",
    12. dims: 5,
    13. similarity: "l2_norm",
    14. },
    15. title: {
    16. type: "text",
    17. },
    18. "file-type": {
    19. type: "keyword",
    20. },
    21. },
    22. },
    23. });
    24. console.log(response);
    1. PUT image-index
    2. {
    3. "mappings": {
    4. "properties": {
    5. "image-vector": {
    6. "type": "dense_vector",
    7. "dims": 3,
    8. "similarity": "l2_norm"
    9. },
    10. "title-vector": {
    11. "type": "dense_vector",
    12. "dims": 5,
    13. "similarity": "l2_norm"
    14. },
    15. "title": {
    16. "type": "text"
    17. },
    18. "file-type": {
    19. "type": "keyword"
    20. }
    21. }
    22. }
    23. }
  2. Index your data.

    1. POST image-index/_bulk?refresh=true
    2. { "index": { "_id": "1" } }
    3. { "image-vector": [1, 5, -20], "title-vector": [12, 50, -10, 0, 1], "title": "moose family", "file-type": "jpg" }
    4. { "index": { "_id": "2" } }
    5. { "image-vector": [42, 8, -15], "title-vector": [25, 1, 4, -12, 2], "title": "alpine lake", "file-type": "png" }
    6. { "index": { "_id": "3" } }
    7. { "image-vector": [15, 11, 23], "title-vector": [1, 5, 25, 50, 20], "title": "full moon", "file-type": "jpg" }
    8. ...
  3. Run the search using the knn option or the knn query (expert case).

    1. resp = client.search(
    2. index="image-index",
    3. knn={
    4. "field": "image-vector",
    5. "query_vector": [
    6. -5,
    7. 9,
    8. -12
    9. ],
    10. "k": 10,
    11. "num_candidates": 100
    12. },
    13. fields=[
    14. "title",
    15. "file-type"
    16. ],
    17. )
    18. print(resp)
    1. response = client.search(
    2. index: 'image-index',
    3. body: {
    4. knn: {
    5. field: 'image-vector',
    6. query_vector: [
    7. -5,
    8. 9,
    9. -12
    10. ],
    11. k: 10,
    12. num_candidates: 100
    13. },
    14. fields: [
    15. 'title',
    16. 'file-type'
    17. ]
    18. }
    19. )
    20. puts response
    1. const response = await client.search({
    2. index: "image-index",
    3. knn: {
    4. field: "image-vector",
    5. query_vector: [-5, 9, -12],
    6. k: 10,
    7. num_candidates: 100,
    8. },
    9. fields: ["title", "file-type"],
    10. });
    11. console.log(response);
    1. POST image-index/_search
    2. {
    3. "knn": {
    4. "field": "image-vector",
    5. "query_vector": [-5, 9, -12],
    6. "k": 10,
    7. "num_candidates": 100
    8. },
    9. "fields": [ "title", "file-type" ]
    10. }

The document _score is determined by the similarity between the query and document vector. See similarity for more information on how kNN search scores are computed.

Support for approximate kNN search was added in version 8.0. Before this, dense_vector fields did not support enabling index in the mapping. If you created an index prior to 8.0 containing dense_vector fields, then to support approximate kNN search the data must be reindexed using a new field mapping that sets index: true which is the default option.

Tune approximate kNN for speed or accuracy

To gather results, the kNN search API finds a num_candidates number of approximate nearest neighbor candidates on each shard. The search computes the similarity of these candidate vectors to the query vector, selecting the k most similar results from each shard. The search then merges the results from each shard to return the global top k nearest neighbors.

You can increase num_candidates for more accurate results at the cost of slower search speeds. A search with a high value for num_candidates considers more candidates from each shard. This takes more time, but the search has a higher probability of finding the true k top nearest neighbors.

Similarly, you can decrease num_candidates for faster searches with potentially less accurate results.

Approximate kNN using byte vectors

The approximate kNN search API supports byte value vectors in addition to float value vectors. Use the knn option to search a dense_vector field with element_type set to byte and indexing enabled.

  1. Explicitly map one or more dense_vector fields with element_type set to byte and indexing enabled.

    1. resp = client.indices.create(
    2. index="byte-image-index",
    3. mappings={
    4. "properties": {
    5. "byte-image-vector": {
    6. "type": "dense_vector",
    7. "element_type": "byte",
    8. "dims": 2
    9. },
    10. "title": {
    11. "type": "text"
    12. }
    13. }
    14. },
    15. )
    16. print(resp)
    1. response = client.indices.create(
    2. index: 'byte-image-index',
    3. body: {
    4. mappings: {
    5. properties: {
    6. "byte-image-vector": {
    7. type: 'dense_vector',
    8. element_type: 'byte',
    9. dims: 2
    10. },
    11. title: {
    12. type: 'text'
    13. }
    14. }
    15. }
    16. }
    17. )
    18. puts response
    1. const response = await client.indices.create({
    2. index: "byte-image-index",
    3. mappings: {
    4. properties: {
    5. "byte-image-vector": {
    6. type: "dense_vector",
    7. element_type: "byte",
    8. dims: 2,
    9. },
    10. title: {
    11. type: "text",
    12. },
    13. },
    14. },
    15. });
    16. console.log(response);
    1. PUT byte-image-index
    2. {
    3. "mappings": {
    4. "properties": {
    5. "byte-image-vector": {
    6. "type": "dense_vector",
    7. "element_type": "byte",
    8. "dims": 2
    9. },
    10. "title": {
    11. "type": "text"
    12. }
    13. }
    14. }
    15. }
  2. Index your data ensuring all vector values are integers within the range [-128, 127].

    1. resp = client.bulk(
    2. index="byte-image-index",
    3. refresh=True,
    4. operations=[
    5. {
    6. "index": {
    7. "_id": "1"
    8. }
    9. },
    10. {
    11. "byte-image-vector": [
    12. 5,
    13. -20
    14. ],
    15. "title": "moose family"
    16. },
    17. {
    18. "index": {
    19. "_id": "2"
    20. }
    21. },
    22. {
    23. "byte-image-vector": [
    24. 8,
    25. -15
    26. ],
    27. "title": "alpine lake"
    28. },
    29. {
    30. "index": {
    31. "_id": "3"
    32. }
    33. },
    34. {
    35. "byte-image-vector": [
    36. 11,
    37. 23
    38. ],
    39. "title": "full moon"
    40. }
    41. ],
    42. )
    43. print(resp)
    1. response = client.bulk(
    2. index: 'byte-image-index',
    3. refresh: true,
    4. body: [
    5. {
    6. index: {
    7. _id: '1'
    8. }
    9. },
    10. {
    11. "byte-image-vector": [
    12. 5,
    13. -20
    14. ],
    15. title: 'moose family'
    16. },
    17. {
    18. index: {
    19. _id: '2'
    20. }
    21. },
    22. {
    23. "byte-image-vector": [
    24. 8,
    25. -15
    26. ],
    27. title: 'alpine lake'
    28. },
    29. {
    30. index: {
    31. _id: '3'
    32. }
    33. },
    34. {
    35. "byte-image-vector": [
    36. 11,
    37. 23
    38. ],
    39. title: 'full moon'
    40. }
    41. ]
    42. )
    43. puts response
    1. const response = await client.bulk({
    2. index: "byte-image-index",
    3. refresh: "true",
    4. operations: [
    5. {
    6. index: {
    7. _id: "1",
    8. },
    9. },
    10. {
    11. "byte-image-vector": [5, -20],
    12. title: "moose family",
    13. },
    14. {
    15. index: {
    16. _id: "2",
    17. },
    18. },
    19. {
    20. "byte-image-vector": [8, -15],
    21. title: "alpine lake",
    22. },
    23. {
    24. index: {
    25. _id: "3",
    26. },
    27. },
    28. {
    29. "byte-image-vector": [11, 23],
    30. title: "full moon",
    31. },
    32. ],
    33. });
    34. console.log(response);
    1. POST byte-image-index/_bulk?refresh=true
    2. { "index": { "_id": "1" } }
    3. { "byte-image-vector": [5, -20], "title": "moose family" }
    4. { "index": { "_id": "2" } }
    5. { "byte-image-vector": [8, -15], "title": "alpine lake" }
    6. { "index": { "_id": "3" } }
    7. { "byte-image-vector": [11, 23], "title": "full moon" }
  3. Run the search using the knn option ensuring the query_vector values are integers within the range [-128, 127].

    1. resp = client.search(
    2. index="byte-image-index",
    3. knn={
    4. "field": "byte-image-vector",
    5. "query_vector": [
    6. -5,
    7. 9
    8. ],
    9. "k": 10,
    10. "num_candidates": 100
    11. },
    12. fields=[
    13. "title"
    14. ],
    15. )
    16. print(resp)
    1. response = client.search(
    2. index: 'byte-image-index',
    3. body: {
    4. knn: {
    5. field: 'byte-image-vector',
    6. query_vector: [
    7. -5,
    8. 9
    9. ],
    10. k: 10,
    11. num_candidates: 100
    12. },
    13. fields: [
    14. 'title'
    15. ]
    16. }
    17. )
    18. puts response
    1. const response = await client.search({
    2. index: "byte-image-index",
    3. knn: {
    4. field: "byte-image-vector",
    5. query_vector: [-5, 9],
    6. k: 10,
    7. num_candidates: 100,
    8. },
    9. fields: ["title"],
    10. });
    11. console.log(response);
    1. POST byte-image-index/_search
    2. {
    3. "knn": {
    4. "field": "byte-image-vector",
    5. "query_vector": [-5, 9],
    6. "k": 10,
    7. "num_candidates": 100
    8. },
    9. "fields": [ "title" ]
    10. }

Note: In addition to the standard byte array, one can also provide a hex-encoded string value for the query_vector param. As an example, the search request above can also be expressed as follows, which would yield the same results

  1. resp = client.search(
  2. index="byte-image-index",
  3. knn={
  4. "field": "byte-image-vector",
  5. "query_vector": "fb09",
  6. "k": 10,
  7. "num_candidates": 100
  8. },
  9. fields=[
  10. "title"
  11. ],
  12. )
  13. print(resp)
  1. response = client.search(
  2. index: 'byte-image-index',
  3. body: {
  4. knn: {
  5. field: 'byte-image-vector',
  6. query_vector: 'fb09',
  7. k: 10,
  8. num_candidates: 100
  9. },
  10. fields: [
  11. 'title'
  12. ]
  13. }
  14. )
  15. puts response
  1. const response = await client.search({
  2. index: "byte-image-index",
  3. knn: {
  4. field: "byte-image-vector",
  5. query_vector: "fb09",
  6. k: 10,
  7. num_candidates: 100,
  8. },
  9. fields: ["title"],
  10. });
  11. console.log(response);
  1. POST byte-image-index/_search
  2. {
  3. "knn": {
  4. "field": "byte-image-vector",
  5. "query_vector": "fb09",
  6. "k": 10,
  7. "num_candidates": 100
  8. },
  9. "fields": [ "title" ]
  10. }

If you want to provide float vectors, but want the memory savings of byte vectors, you can use the quantization feature. Quantization allows you to provide float vectors, but internally they are indexed as byte vectors. Additionally, the original float vectors are still retained in the index.

The default index type for dense_vector is int8_hnsw.

To use quantization, you can use the index type int8_hnsw or int4_hnsw object in the dense_vector mapping.

  1. resp = client.indices.create(
  2. index="quantized-image-index",
  3. mappings={
  4. "properties": {
  5. "image-vector": {
  6. "type": "dense_vector",
  7. "element_type": "float",
  8. "dims": 2,
  9. "index": True,
  10. "index_options": {
  11. "type": "int8_hnsw"
  12. }
  13. },
  14. "title": {
  15. "type": "text"
  16. }
  17. }
  18. },
  19. )
  20. print(resp)
  1. response = client.indices.create(
  2. index: 'quantized-image-index',
  3. body: {
  4. mappings: {
  5. properties: {
  6. "image-vector": {
  7. type: 'dense_vector',
  8. element_type: 'float',
  9. dims: 2,
  10. index: true,
  11. index_options: {
  12. type: 'int8_hnsw'
  13. }
  14. },
  15. title: {
  16. type: 'text'
  17. }
  18. }
  19. }
  20. }
  21. )
  22. puts response
  1. const response = await client.indices.create({
  2. index: "quantized-image-index",
  3. mappings: {
  4. properties: {
  5. "image-vector": {
  6. type: "dense_vector",
  7. element_type: "float",
  8. dims: 2,
  9. index: true,
  10. index_options: {
  11. type: "int8_hnsw",
  12. },
  13. },
  14. title: {
  15. type: "text",
  16. },
  17. },
  18. },
  19. });
  20. console.log(response);
  1. PUT quantized-image-index
  2. {
  3. "mappings": {
  4. "properties": {
  5. "image-vector": {
  6. "type": "dense_vector",
  7. "element_type": "float",
  8. "dims": 2,
  9. "index": true,
  10. "index_options": {
  11. "type": "int8_hnsw"
  12. }
  13. },
  14. "title": {
  15. "type": "text"
  16. }
  17. }
  18. }
  19. }
  1. Index your float vectors.

    1. resp = client.bulk(
    2. index="quantized-image-index",
    3. refresh=True,
    4. operations=[
    5. {
    6. "index": {
    7. "_id": "1"
    8. }
    9. },
    10. {
    11. "image-vector": [
    12. 0.1,
    13. -2
    14. ],
    15. "title": "moose family"
    16. },
    17. {
    18. "index": {
    19. "_id": "2"
    20. }
    21. },
    22. {
    23. "image-vector": [
    24. 0.75,
    25. -1
    26. ],
    27. "title": "alpine lake"
    28. },
    29. {
    30. "index": {
    31. "_id": "3"
    32. }
    33. },
    34. {
    35. "image-vector": [
    36. 1.2,
    37. 0.1
    38. ],
    39. "title": "full moon"
    40. }
    41. ],
    42. )
    43. print(resp)
    1. response = client.bulk(
    2. index: 'quantized-image-index',
    3. refresh: true,
    4. body: [
    5. {
    6. index: {
    7. _id: '1'
    8. }
    9. },
    10. {
    11. "image-vector": [
    12. 0.1,
    13. -2
    14. ],
    15. title: 'moose family'
    16. },
    17. {
    18. index: {
    19. _id: '2'
    20. }
    21. },
    22. {
    23. "image-vector": [
    24. 0.75,
    25. -1
    26. ],
    27. title: 'alpine lake'
    28. },
    29. {
    30. index: {
    31. _id: '3'
    32. }
    33. },
    34. {
    35. "image-vector": [
    36. 1.2,
    37. 0.1
    38. ],
    39. title: 'full moon'
    40. }
    41. ]
    42. )
    43. puts response
    1. const response = await client.bulk({
    2. index: "quantized-image-index",
    3. refresh: "true",
    4. operations: [
    5. {
    6. index: {
    7. _id: "1",
    8. },
    9. },
    10. {
    11. "image-vector": [0.1, -2],
    12. title: "moose family",
    13. },
    14. {
    15. index: {
    16. _id: "2",
    17. },
    18. },
    19. {
    20. "image-vector": [0.75, -1],
    21. title: "alpine lake",
    22. },
    23. {
    24. index: {
    25. _id: "3",
    26. },
    27. },
    28. {
    29. "image-vector": [1.2, 0.1],
    30. title: "full moon",
    31. },
    32. ],
    33. });
    34. console.log(response);
    1. POST quantized-image-index/_bulk?refresh=true
    2. { "index": { "_id": "1" } }
    3. { "image-vector": [0.1, -2], "title": "moose family" }
    4. { "index": { "_id": "2" } }
    5. { "image-vector": [0.75, -1], "title": "alpine lake" }
    6. { "index": { "_id": "3" } }
    7. { "image-vector": [1.2, 0.1], "title": "full moon" }
  2. Run the search using the knn option. When searching, the float vector is automatically quantized to a byte vector.

    1. resp = client.search(
    2. index="quantized-image-index",
    3. knn={
    4. "field": "image-vector",
    5. "query_vector": [
    6. 0.1,
    7. -2
    8. ],
    9. "k": 10,
    10. "num_candidates": 100
    11. },
    12. fields=[
    13. "title"
    14. ],
    15. )
    16. print(resp)
    1. response = client.search(
    2. index: 'quantized-image-index',
    3. body: {
    4. knn: {
    5. field: 'image-vector',
    6. query_vector: [
    7. 0.1,
    8. -2
    9. ],
    10. k: 10,
    11. num_candidates: 100
    12. },
    13. fields: [
    14. 'title'
    15. ]
    16. }
    17. )
    18. puts response
    1. const response = await client.search({
    2. index: "quantized-image-index",
    3. knn: {
    4. field: "image-vector",
    5. query_vector: [0.1, -2],
    6. k: 10,
    7. num_candidates: 100,
    8. },
    9. fields: ["title"],
    10. });
    11. console.log(response);
    1. POST quantized-image-index/_search
    2. {
    3. "knn": {
    4. "field": "image-vector",
    5. "query_vector": [0.1, -2],
    6. "k": 10,
    7. "num_candidates": 100
    8. },
    9. "fields": [ "title" ]
    10. }

Since the original float vectors are still retained in the index, you can optionally use them for re-scoring. Meaning, you can search over all the vectors quickly using the int8_hnsw index and then rescore only the top k results. This provides the best of both worlds, fast search and accurate scoring.

  1. resp = client.search(
  2. index="quantized-image-index",
  3. knn={
  4. "field": "image-vector",
  5. "query_vector": [
  6. 0.1,
  7. -2
  8. ],
  9. "k": 15,
  10. "num_candidates": 100
  11. },
  12. fields=[
  13. "title"
  14. ],
  15. rescore={
  16. "window_size": 10,
  17. "query": {
  18. "rescore_query": {
  19. "script_score": {
  20. "query": {
  21. "match_all": {}
  22. },
  23. "script": {
  24. "source": "cosineSimilarity(params.query_vector, 'image-vector') + 1.0",
  25. "params": {
  26. "query_vector": [
  27. 0.1,
  28. -2
  29. ]
  30. }
  31. }
  32. }
  33. }
  34. }
  35. },
  36. )
  37. print(resp)
  1. response = client.search(
  2. index: 'quantized-image-index',
  3. body: {
  4. knn: {
  5. field: 'image-vector',
  6. query_vector: [
  7. 0.1,
  8. -2
  9. ],
  10. k: 15,
  11. num_candidates: 100
  12. },
  13. fields: [
  14. 'title'
  15. ],
  16. rescore: {
  17. window_size: 10,
  18. query: {
  19. rescore_query: {
  20. script_score: {
  21. query: {
  22. match_all: {}
  23. },
  24. script: {
  25. source: "cosineSimilarity(params.query_vector, 'image-vector') + 1.0",
  26. params: {
  27. query_vector: [
  28. 0.1,
  29. -2
  30. ]
  31. }
  32. }
  33. }
  34. }
  35. }
  36. }
  37. }
  38. )
  39. puts response
  1. const response = await client.search({
  2. index: "quantized-image-index",
  3. knn: {
  4. field: "image-vector",
  5. query_vector: [0.1, -2],
  6. k: 15,
  7. num_candidates: 100,
  8. },
  9. fields: ["title"],
  10. rescore: {
  11. window_size: 10,
  12. query: {
  13. rescore_query: {
  14. script_score: {
  15. query: {
  16. match_all: {},
  17. },
  18. script: {
  19. source:
  20. "cosineSimilarity(params.query_vector, 'image-vector') + 1.0",
  21. params: {
  22. query_vector: [0.1, -2],
  23. },
  24. },
  25. },
  26. },
  27. },
  28. },
  29. });
  30. console.log(response);
  1. POST quantized-image-index/_search
  2. {
  3. "knn": {
  4. "field": "image-vector",
  5. "query_vector": [0.1, -2],
  6. "k": 15,
  7. "num_candidates": 100
  8. },
  9. "fields": [ "title" ],
  10. "rescore": {
  11. "window_size": 10,
  12. "query": {
  13. "rescore_query": {
  14. "script_score": {
  15. "query": {
  16. "match_all": {}
  17. },
  18. "script": {
  19. "source": "cosineSimilarity(params.query_vector, 'image-vector') + 1.0",
  20. "params": {
  21. "query_vector": [0.1, -2]
  22. }
  23. }
  24. }
  25. }
  26. }
  27. }
  28. }

The kNN search API supports restricting the search using a filter. The search will return the top k documents that also match the filter query.

The following request performs an approximate kNN search filtered by the file-type field:

  1. resp = client.search(
  2. index="image-index",
  3. knn={
  4. "field": "image-vector",
  5. "query_vector": [
  6. 54,
  7. 10,
  8. -2
  9. ],
  10. "k": 5,
  11. "num_candidates": 50,
  12. "filter": {
  13. "term": {
  14. "file-type": "png"
  15. }
  16. }
  17. },
  18. fields=[
  19. "title"
  20. ],
  21. source=False,
  22. )
  23. print(resp)
  1. response = client.search(
  2. index: 'image-index',
  3. body: {
  4. knn: {
  5. field: 'image-vector',
  6. query_vector: [
  7. 54,
  8. 10,
  9. -2
  10. ],
  11. k: 5,
  12. num_candidates: 50,
  13. filter: {
  14. term: {
  15. "file-type": 'png'
  16. }
  17. }
  18. },
  19. fields: [
  20. 'title'
  21. ],
  22. _source: false
  23. }
  24. )
  25. puts response
  1. const response = await client.search({
  2. index: "image-index",
  3. knn: {
  4. field: "image-vector",
  5. query_vector: [54, 10, -2],
  6. k: 5,
  7. num_candidates: 50,
  8. filter: {
  9. term: {
  10. "file-type": "png",
  11. },
  12. },
  13. },
  14. fields: ["title"],
  15. _source: false,
  16. });
  17. console.log(response);
  1. POST image-index/_search
  2. {
  3. "knn": {
  4. "field": "image-vector",
  5. "query_vector": [54, 10, -2],
  6. "k": 5,
  7. "num_candidates": 50,
  8. "filter": {
  9. "term": {
  10. "file-type": "png"
  11. }
  12. }
  13. },
  14. "fields": ["title"],
  15. "_source": false
  16. }

The filter is applied during the approximate kNN search to ensure that k matching documents are returned. This contrasts with a post-filtering approach, where the filter is applied after the approximate kNN search completes. Post-filtering has the downside that it sometimes returns fewer than k results, even when there are enough matching documents.

Approximate kNN search and filtering

Unlike conventional query filtering, where more restrictive filters typically lead to faster queries, applying filters in an approximate kNN search with an HNSW index can decrease performance. This is because searching the HNSW graph requires additional exploration to obtain the num_candidates that meet the filter criteria.

To avoid significant performance drawbacks, Lucene implements the following strategies per segment:

  • If the filtered document count is less than or equal to num_candidates, the search bypasses the HNSW graph and uses a brute force search on the filtered documents.
  • While exploring the HNSW graph, if the number of nodes explored exceeds the number of documents that satisfy the filter, the search will stop exploring the graph and switch to a brute force search over the filtered documents.

Combine approximate kNN with other features

You can perform hybrid retrieval by providing both the knn option and a query:

  1. resp = client.search(
  2. index="image-index",
  3. query={
  4. "match": {
  5. "title": {
  6. "query": "mountain lake",
  7. "boost": 0.9
  8. }
  9. }
  10. },
  11. knn={
  12. "field": "image-vector",
  13. "query_vector": [
  14. 54,
  15. 10,
  16. -2
  17. ],
  18. "k": 5,
  19. "num_candidates": 50,
  20. "boost": 0.1
  21. },
  22. size=10,
  23. )
  24. print(resp)
  1. response = client.search(
  2. index: 'image-index',
  3. body: {
  4. query: {
  5. match: {
  6. title: {
  7. query: 'mountain lake',
  8. boost: 0.9
  9. }
  10. }
  11. },
  12. knn: {
  13. field: 'image-vector',
  14. query_vector: [
  15. 54,
  16. 10,
  17. -2
  18. ],
  19. k: 5,
  20. num_candidates: 50,
  21. boost: 0.1
  22. },
  23. size: 10
  24. }
  25. )
  26. puts response
  1. const response = await client.search({
  2. index: "image-index",
  3. query: {
  4. match: {
  5. title: {
  6. query: "mountain lake",
  7. boost: 0.9,
  8. },
  9. },
  10. },
  11. knn: {
  12. field: "image-vector",
  13. query_vector: [54, 10, -2],
  14. k: 5,
  15. num_candidates: 50,
  16. boost: 0.1,
  17. },
  18. size: 10,
  19. });
  20. console.log(response);
  1. POST image-index/_search
  2. {
  3. "query": {
  4. "match": {
  5. "title": {
  6. "query": "mountain lake",
  7. "boost": 0.9
  8. }
  9. }
  10. },
  11. "knn": {
  12. "field": "image-vector",
  13. "query_vector": [54, 10, -2],
  14. "k": 5,
  15. "num_candidates": 50,
  16. "boost": 0.1
  17. },
  18. "size": 10
  19. }

This search finds the global top k = 5 vector matches, combines them with the matches from the match query, and finally returns the 10 top-scoring results. The knn and query matches are combined through a disjunction, as if you took a boolean or between them. The top k vector results represent the global nearest neighbors across all index shards.

The score of each hit is the sum of the knn and query scores. You can specify a boost value to give a weight to each score in the sum. In the example above, the scores will be calculated as

  1. score = 0.9 * match_score + 0.1 * knn_score

The knn option can also be used with aggregations. In general, Elasticsearch computes aggregations over all documents that match the search. So for approximate kNN search, aggregations are calculated on the top k nearest documents. If the search also includes a query, then aggregations are calculated on the combined set of knn and query matches.

kNN search enables you to perform semantic search by using a previously deployed text embedding model. Instead of literal matching on search terms, semantic search retrieves results based on the intent and the contextual meaning of a search query.

Under the hood, the text embedding NLP model generates a dense vector from the input query string called model_text you provide. Then, it is searched against an index containing dense vectors created with the same text embedding machine learning model. The search results are semantically similar as learned by the model.

To perform semantic search:

  • you need an index that contains the dense vector representation of the input data to search against,
  • you must use the same text embedding model for search that you used to create the dense vectors from the input data,
  • the text embedding NLP model deployment must be started.

Reference the deployed text embedding model or the model deployment in the query_vector_builder object and provide the search query as model_text:

  1. (...)
  2. {
  3. "knn": {
  4. "field": "dense-vector-field",
  5. "k": 10,
  6. "num_candidates": 100,
  7. "query_vector_builder": {
  8. "text_embedding": {
  9. "model_id": "my-text-embedding-model",
  10. "model_text": "The opposite of blue"
  11. }
  12. }
  13. }
  14. }
  15. (...)

The natural language processing task to perform. It must be text_embedding.

The ID of the text embedding model to use to generate the dense vectors from the query string. Use the same model that generated the embeddings from the input text in the index you search against. You can use the value of the deployment_id instead in the model_id argument.

The query string from which the model generates the dense vector representation.

For more information on how to deploy a trained model and use it to create text embeddings, refer to this end-to-end example.

Search multiple kNN fields

In addition to hybrid retrieval, you can search more than one kNN vector field at a time:

  1. resp = client.search(
  2. index="image-index",
  3. query={
  4. "match": {
  5. "title": {
  6. "query": "mountain lake",
  7. "boost": 0.9
  8. }
  9. }
  10. },
  11. knn=[
  12. {
  13. "field": "image-vector",
  14. "query_vector": [
  15. 54,
  16. 10,
  17. -2
  18. ],
  19. "k": 5,
  20. "num_candidates": 50,
  21. "boost": 0.1
  22. },
  23. {
  24. "field": "title-vector",
  25. "query_vector": [
  26. 1,
  27. 20,
  28. -52,
  29. 23,
  30. 10
  31. ],
  32. "k": 10,
  33. "num_candidates": 10,
  34. "boost": 0.5
  35. }
  36. ],
  37. size=10,
  38. )
  39. print(resp)
  1. response = client.search(
  2. index: 'image-index',
  3. body: {
  4. query: {
  5. match: {
  6. title: {
  7. query: 'mountain lake',
  8. boost: 0.9
  9. }
  10. }
  11. },
  12. knn: [
  13. {
  14. field: 'image-vector',
  15. query_vector: [
  16. 54,
  17. 10,
  18. -2
  19. ],
  20. k: 5,
  21. num_candidates: 50,
  22. boost: 0.1
  23. },
  24. {
  25. field: 'title-vector',
  26. query_vector: [
  27. 1,
  28. 20,
  29. -52,
  30. 23,
  31. 10
  32. ],
  33. k: 10,
  34. num_candidates: 10,
  35. boost: 0.5
  36. }
  37. ],
  38. size: 10
  39. }
  40. )
  41. puts response
  1. const response = await client.search({
  2. index: "image-index",
  3. query: {
  4. match: {
  5. title: {
  6. query: "mountain lake",
  7. boost: 0.9,
  8. },
  9. },
  10. },
  11. knn: [
  12. {
  13. field: "image-vector",
  14. query_vector: [54, 10, -2],
  15. k: 5,
  16. num_candidates: 50,
  17. boost: 0.1,
  18. },
  19. {
  20. field: "title-vector",
  21. query_vector: [1, 20, -52, 23, 10],
  22. k: 10,
  23. num_candidates: 10,
  24. boost: 0.5,
  25. },
  26. ],
  27. size: 10,
  28. });
  29. console.log(response);
  1. POST image-index/_search
  2. {
  3. "query": {
  4. "match": {
  5. "title": {
  6. "query": "mountain lake",
  7. "boost": 0.9
  8. }
  9. }
  10. },
  11. "knn": [ {
  12. "field": "image-vector",
  13. "query_vector": [54, 10, -2],
  14. "k": 5,
  15. "num_candidates": 50,
  16. "boost": 0.1
  17. },
  18. {
  19. "field": "title-vector",
  20. "query_vector": [1, 20, -52, 23, 10],
  21. "k": 10,
  22. "num_candidates": 10,
  23. "boost": 0.5
  24. }],
  25. "size": 10
  26. }

This search finds the global top k = 5 vector matches for image-vector and the global k = 10 for the title-vector. These top values are then combined with the matches from the match query and the top-10 documents are returned. The multiple knn entries and the query matches are combined through a disjunction, as if you took a boolean or between them. The top k vector results represent the global nearest neighbors across all index shards.

The scoring for a doc with the above configured boosts would be:

  1. score = 0.9 * match_score + 0.1 * knn_score_image-vector + 0.5 * knn_score_title-vector

Search kNN with expected similarity

While kNN is a powerful tool, it always tries to return k nearest neighbors. Consequently, when using knn with a filter, you could filter out all relevant documents and only have irrelevant ones left to search. In that situation, knn will still do its best to return k nearest neighbors, even though those neighbors could be far away in the vector space.

To alleviate this worry, there is a similarity parameter available in the knn clause. This value is the required minimum similarity for a vector to be considered a match. The knn search flow with this parameter is as follows:

  • Apply any user provided filter queries
  • Explore the vector space to get k vectors
  • Do not return any vectors that are further away than the configured similarity

similarity is the true similarity before it has been transformed into _score and boost applied.

For each configured similarity, here is the corresponding inverted _score function. This is so if you are wanting to filter from a _score perspective, you can do this minor transformation to correctly reject irrelevant results.

  • l2_norm: sqrt((1 / _score) - 1)
  • cosine: (2 * _score) - 1
  • dot_product: (2 * _score) - 1
  • max_inner_product:

    • _score < 1: 1 - (1 / _score)
    • _score >= 1: _score - 1

Here is an example. In this example we search for the given query_vector for k nearest neighbors. However, with filter applied and requiring that the found vectors have at least the provided similarity between them.

  1. resp = client.search(
  2. index="image-index",
  3. knn={
  4. "field": "image-vector",
  5. "query_vector": [
  6. 1,
  7. 5,
  8. -20
  9. ],
  10. "k": 5,
  11. "num_candidates": 50,
  12. "similarity": 36,
  13. "filter": {
  14. "term": {
  15. "file-type": "png"
  16. }
  17. }
  18. },
  19. fields=[
  20. "title"
  21. ],
  22. source=False,
  23. )
  24. print(resp)
  1. response = client.search(
  2. index: 'image-index',
  3. body: {
  4. knn: {
  5. field: 'image-vector',
  6. query_vector: [
  7. 1,
  8. 5,
  9. -20
  10. ],
  11. k: 5,
  12. num_candidates: 50,
  13. similarity: 36,
  14. filter: {
  15. term: {
  16. "file-type": 'png'
  17. }
  18. }
  19. },
  20. fields: [
  21. 'title'
  22. ],
  23. _source: false
  24. }
  25. )
  26. puts response
  1. const response = await client.search({
  2. index: "image-index",
  3. knn: {
  4. field: "image-vector",
  5. query_vector: [1, 5, -20],
  6. k: 5,
  7. num_candidates: 50,
  8. similarity: 36,
  9. filter: {
  10. term: {
  11. "file-type": "png",
  12. },
  13. },
  14. },
  15. fields: ["title"],
  16. _source: false,
  17. });
  18. console.log(response);
  1. POST image-index/_search
  2. {
  3. "knn": {
  4. "field": "image-vector",
  5. "query_vector": [1, 5, -20],
  6. "k": 5,
  7. "num_candidates": 50,
  8. "similarity": 36,
  9. "filter": {
  10. "term": {
  11. "file-type": "png"
  12. }
  13. }
  14. },
  15. "fields": ["title"],
  16. "_source": false
  17. }

In our data set, the only document with the file type of png has a vector of [42, 8, -15]. The l2_norm distance between [42, 8, -15] and [1, 5, -20] is 41.412, which is greater than the configured similarity of 36. Meaning, this search will return no hits.

It is common for text to exceed a particular model’s token limit and requires chunking before building the embeddings for individual chunks. When using nested with dense_vector, you can achieve nearest passage retrieval without copying top-level document metadata.

Here is a simple passage vectors index that stores vectors and some top-level metadata for filtering.

  1. resp = client.indices.create(
  2. index="passage_vectors",
  3. mappings={
  4. "properties": {
  5. "full_text": {
  6. "type": "text"
  7. },
  8. "creation_time": {
  9. "type": "date"
  10. },
  11. "paragraph": {
  12. "type": "nested",
  13. "properties": {
  14. "vector": {
  15. "type": "dense_vector",
  16. "dims": 2,
  17. "index_options": {
  18. "type": "hnsw"
  19. }
  20. },
  21. "text": {
  22. "type": "text",
  23. "index": False
  24. }
  25. }
  26. }
  27. }
  28. },
  29. )
  30. print(resp)
  1. response = client.indices.create(
  2. index: 'passage_vectors',
  3. body: {
  4. mappings: {
  5. properties: {
  6. full_text: {
  7. type: 'text'
  8. },
  9. creation_time: {
  10. type: 'date'
  11. },
  12. paragraph: {
  13. type: 'nested',
  14. properties: {
  15. vector: {
  16. type: 'dense_vector',
  17. dims: 2,
  18. index_options: {
  19. type: 'hnsw'
  20. }
  21. },
  22. text: {
  23. type: 'text',
  24. index: false
  25. }
  26. }
  27. }
  28. }
  29. }
  30. }
  31. )
  32. puts response
  1. const response = await client.indices.create({
  2. index: "passage_vectors",
  3. mappings: {
  4. properties: {
  5. full_text: {
  6. type: "text",
  7. },
  8. creation_time: {
  9. type: "date",
  10. },
  11. paragraph: {
  12. type: "nested",
  13. properties: {
  14. vector: {
  15. type: "dense_vector",
  16. dims: 2,
  17. index_options: {
  18. type: "hnsw",
  19. },
  20. },
  21. text: {
  22. type: "text",
  23. index: false,
  24. },
  25. },
  26. },
  27. },
  28. },
  29. });
  30. console.log(response);
  1. PUT passage_vectors
  2. {
  3. "mappings": {
  4. "properties": {
  5. "full_text": {
  6. "type": "text"
  7. },
  8. "creation_time": {
  9. "type": "date"
  10. },
  11. "paragraph": {
  12. "type": "nested",
  13. "properties": {
  14. "vector": {
  15. "type": "dense_vector",
  16. "dims": 2,
  17. "index_options": {
  18. "type": "hnsw"
  19. }
  20. },
  21. "text": {
  22. "type": "text",
  23. "index": false
  24. }
  25. }
  26. }
  27. }
  28. }
  29. }

With the above mapping, we can index multiple passage vectors along with storing the individual passage text.

  1. resp = client.bulk(
  2. index="passage_vectors",
  3. refresh=True,
  4. operations=[
  5. {
  6. "index": {
  7. "_id": "1"
  8. }
  9. },
  10. {
  11. "full_text": "first paragraph another paragraph",
  12. "creation_time": "2019-05-04",
  13. "paragraph": [
  14. {
  15. "vector": [
  16. 0.45,
  17. 45
  18. ],
  19. "text": "first paragraph",
  20. "paragraph_id": "1"
  21. },
  22. {
  23. "vector": [
  24. 0.8,
  25. 0.6
  26. ],
  27. "text": "another paragraph",
  28. "paragraph_id": "2"
  29. }
  30. ]
  31. },
  32. {
  33. "index": {
  34. "_id": "2"
  35. }
  36. },
  37. {
  38. "full_text": "number one paragraph number two paragraph",
  39. "creation_time": "2020-05-04",
  40. "paragraph": [
  41. {
  42. "vector": [
  43. 1.2,
  44. 4.5
  45. ],
  46. "text": "number one paragraph",
  47. "paragraph_id": "1"
  48. },
  49. {
  50. "vector": [
  51. -1,
  52. 42
  53. ],
  54. "text": "number two paragraph",
  55. "paragraph_id": "2"
  56. }
  57. ]
  58. }
  59. ],
  60. )
  61. print(resp)
  1. response = client.bulk(
  2. index: 'passage_vectors',
  3. refresh: true,
  4. body: [
  5. {
  6. index: {
  7. _id: '1'
  8. }
  9. },
  10. {
  11. full_text: 'first paragraph another paragraph',
  12. creation_time: '2019-05-04',
  13. paragraph: [
  14. {
  15. vector: [
  16. 0.45,
  17. 45
  18. ],
  19. text: 'first paragraph',
  20. paragraph_id: '1'
  21. },
  22. {
  23. vector: [
  24. 0.8,
  25. 0.6
  26. ],
  27. text: 'another paragraph',
  28. paragraph_id: '2'
  29. }
  30. ]
  31. },
  32. {
  33. index: {
  34. _id: '2'
  35. }
  36. },
  37. {
  38. full_text: 'number one paragraph number two paragraph',
  39. creation_time: '2020-05-04',
  40. paragraph: [
  41. {
  42. vector: [
  43. 1.2,
  44. 4.5
  45. ],
  46. text: 'number one paragraph',
  47. paragraph_id: '1'
  48. },
  49. {
  50. vector: [
  51. -1,
  52. 42
  53. ],
  54. text: 'number two paragraph',
  55. paragraph_id: '2'
  56. }
  57. ]
  58. }
  59. ]
  60. )
  61. puts response
  1. const response = await client.bulk({
  2. index: "passage_vectors",
  3. refresh: "true",
  4. operations: [
  5. {
  6. index: {
  7. _id: "1",
  8. },
  9. },
  10. {
  11. full_text: "first paragraph another paragraph",
  12. creation_time: "2019-05-04",
  13. paragraph: [
  14. {
  15. vector: [0.45, 45],
  16. text: "first paragraph",
  17. paragraph_id: "1",
  18. },
  19. {
  20. vector: [0.8, 0.6],
  21. text: "another paragraph",
  22. paragraph_id: "2",
  23. },
  24. ],
  25. },
  26. {
  27. index: {
  28. _id: "2",
  29. },
  30. },
  31. {
  32. full_text: "number one paragraph number two paragraph",
  33. creation_time: "2020-05-04",
  34. paragraph: [
  35. {
  36. vector: [1.2, 4.5],
  37. text: "number one paragraph",
  38. paragraph_id: "1",
  39. },
  40. {
  41. vector: [-1, 42],
  42. text: "number two paragraph",
  43. paragraph_id: "2",
  44. },
  45. ],
  46. },
  47. ],
  48. });
  49. console.log(response);
  1. POST passage_vectors/_bulk?refresh=true
  2. { "index": { "_id": "1" } }
  3. { "full_text": "first paragraph another paragraph", "creation_time": "2019-05-04", "paragraph": [ { "vector": [ 0.45, 45 ], "text": "first paragraph", "paragraph_id": "1" }, { "vector": [ 0.8, 0.6 ], "text": "another paragraph", "paragraph_id": "2" } ] }
  4. { "index": { "_id": "2" } }
  5. { "full_text": "number one paragraph number two paragraph", "creation_time": "2020-05-04", "paragraph": [ { "vector": [ 1.2, 4.5 ], "text": "number one paragraph", "paragraph_id": "1" }, { "vector": [ -1, 42 ], "text": "number two paragraph", "paragraph_id": "2" } ] }

The query will seem very similar to a typical kNN search:

  1. resp = client.search(
  2. index="passage_vectors",
  3. fields=[
  4. "full_text",
  5. "creation_time"
  6. ],
  7. source=False,
  8. knn={
  9. "query_vector": [
  10. 0.45,
  11. 45
  12. ],
  13. "field": "paragraph.vector",
  14. "k": 2,
  15. "num_candidates": 2
  16. },
  17. )
  18. print(resp)
  1. response = client.search(
  2. index: 'passage_vectors',
  3. body: {
  4. fields: [
  5. 'full_text',
  6. 'creation_time'
  7. ],
  8. _source: false,
  9. knn: {
  10. query_vector: [
  11. 0.45,
  12. 45
  13. ],
  14. field: 'paragraph.vector',
  15. k: 2,
  16. num_candidates: 2
  17. }
  18. }
  19. )
  20. puts response
  1. const response = await client.search({
  2. index: "passage_vectors",
  3. fields: ["full_text", "creation_time"],
  4. _source: false,
  5. knn: {
  6. query_vector: [0.45, 45],
  7. field: "paragraph.vector",
  8. k: 2,
  9. num_candidates: 2,
  10. },
  11. });
  12. console.log(response);
  1. POST passage_vectors/_search
  2. {
  3. "fields": ["full_text", "creation_time"],
  4. "_source": false,
  5. "knn": {
  6. "query_vector": [
  7. 0.45,
  8. 45
  9. ],
  10. "field": "paragraph.vector",
  11. "k": 2,
  12. "num_candidates": 2
  13. }
  14. }

Note below that even though we have 4 total vectors, we still return two documents. kNN search over nested dense_vectors will always diversify the top results over the top-level document. Meaning, "k" top-level documents will be returned, scored by their nearest passage vector (e.g. "paragraph.vector").

  1. {
  2. "took": 4,
  3. "timed_out": false,
  4. "_shards": {
  5. "total": 1,
  6. "successful": 1,
  7. "skipped": 0,
  8. "failed": 0
  9. },
  10. "hits": {
  11. "total": {
  12. "value": 2,
  13. "relation": "eq"
  14. },
  15. "max_score": 1.0,
  16. "hits": [
  17. {
  18. "_index": "passage_vectors",
  19. "_id": "1",
  20. "_score": 1.0,
  21. "fields": {
  22. "creation_time": [
  23. "2019-05-04T00:00:00.000Z"
  24. ],
  25. "full_text": [
  26. "first paragraph another paragraph"
  27. ]
  28. }
  29. },
  30. {
  31. "_index": "passage_vectors",
  32. "_id": "2",
  33. "_score": 0.9997144,
  34. "fields": {
  35. "creation_time": [
  36. "2020-05-04T00:00:00.000Z"
  37. ],
  38. "full_text": [
  39. "number one paragraph number two paragraph"
  40. ]
  41. }
  42. }
  43. ]
  44. }
  45. }

What if you wanted to filter by some top-level document metadata? You can do this by adding filter to your knn clause.

filter will always be over the top-level document metadata. This means you cannot filter based on nested field metadata.

  1. resp = client.search(
  2. index="passage_vectors",
  3. fields=[
  4. "creation_time",
  5. "full_text"
  6. ],
  7. source=False,
  8. knn={
  9. "query_vector": [
  10. 0.45,
  11. 45
  12. ],
  13. "field": "paragraph.vector",
  14. "k": 2,
  15. "num_candidates": 2,
  16. "filter": {
  17. "bool": {
  18. "filter": [
  19. {
  20. "range": {
  21. "creation_time": {
  22. "gte": "2019-05-01",
  23. "lte": "2019-05-05"
  24. }
  25. }
  26. }
  27. ]
  28. }
  29. }
  30. },
  31. )
  32. print(resp)
  1. response = client.search(
  2. index: 'passage_vectors',
  3. body: {
  4. fields: [
  5. 'creation_time',
  6. 'full_text'
  7. ],
  8. _source: false,
  9. knn: {
  10. query_vector: [
  11. 0.45,
  12. 45
  13. ],
  14. field: 'paragraph.vector',
  15. k: 2,
  16. num_candidates: 2,
  17. filter: {
  18. bool: {
  19. filter: [
  20. {
  21. range: {
  22. creation_time: {
  23. gte: '2019-05-01',
  24. lte: '2019-05-05'
  25. }
  26. }
  27. }
  28. ]
  29. }
  30. }
  31. }
  32. }
  33. )
  34. puts response
  1. const response = await client.search({
  2. index: "passage_vectors",
  3. fields: ["creation_time", "full_text"],
  4. _source: false,
  5. knn: {
  6. query_vector: [0.45, 45],
  7. field: "paragraph.vector",
  8. k: 2,
  9. num_candidates: 2,
  10. filter: {
  11. bool: {
  12. filter: [
  13. {
  14. range: {
  15. creation_time: {
  16. gte: "2019-05-01",
  17. lte: "2019-05-05",
  18. },
  19. },
  20. },
  21. ],
  22. },
  23. },
  24. },
  25. });
  26. console.log(response);
  1. POST passage_vectors/_search
  2. {
  3. "fields": [
  4. "creation_time",
  5. "full_text"
  6. ],
  7. "_source": false,
  8. "knn": {
  9. "query_vector": [
  10. 0.45,
  11. 45
  12. ],
  13. "field": "paragraph.vector",
  14. "k": 2,
  15. "num_candidates": 2,
  16. "filter": {
  17. "bool": {
  18. "filter": [
  19. {
  20. "range": {
  21. "creation_time": {
  22. "gte": "2019-05-01",
  23. "lte": "2019-05-05"
  24. }
  25. }
  26. }
  27. ]
  28. }
  29. }
  30. }
  31. }

Now we have filtered based on the top level "creation_time" and only one document falls within that range.

  1. {
  2. "took": 4,
  3. "timed_out": false,
  4. "_shards": {
  5. "total": 1,
  6. "successful": 1,
  7. "skipped": 0,
  8. "failed": 0
  9. },
  10. "hits": {
  11. "total": {
  12. "value": 1,
  13. "relation": "eq"
  14. },
  15. "max_score": 1.0,
  16. "hits": [
  17. {
  18. "_index": "passage_vectors",
  19. "_id": "1",
  20. "_score": 1.0,
  21. "fields": {
  22. "creation_time": [
  23. "2019-05-04T00:00:00.000Z"
  24. ],
  25. "full_text": [
  26. "first paragraph another paragraph"
  27. ]
  28. }
  29. }
  30. ]
  31. }
  32. }

Nested kNN Search with Inner hits

Additionally, if you wanted to extract the nearest passage for a matched document, you can supply inner_hits to the knn clause.

When using inner_hits and multiple knn clauses, be sure to specify the inner_hits.name field. Otherwise, a naming clash can occur and fail the search request.

  1. resp = client.search(
  2. index="passage_vectors",
  3. fields=[
  4. "creation_time",
  5. "full_text"
  6. ],
  7. source=False,
  8. knn={
  9. "query_vector": [
  10. 0.45,
  11. 45
  12. ],
  13. "field": "paragraph.vector",
  14. "k": 2,
  15. "num_candidates": 2,
  16. "inner_hits": {
  17. "_source": False,
  18. "fields": [
  19. "paragraph.text"
  20. ],
  21. "size": 1
  22. }
  23. },
  24. )
  25. print(resp)
  1. const response = await client.search({
  2. index: "passage_vectors",
  3. fields: ["creation_time", "full_text"],
  4. _source: false,
  5. knn: {
  6. query_vector: [0.45, 45],
  7. field: "paragraph.vector",
  8. k: 2,
  9. num_candidates: 2,
  10. inner_hits: {
  11. _source: false,
  12. fields: ["paragraph.text"],
  13. size: 1,
  14. },
  15. },
  16. });
  17. console.log(response);
  1. POST passage_vectors/_search
  2. {
  3. "fields": [
  4. "creation_time",
  5. "full_text"
  6. ],
  7. "_source": false,
  8. "knn": {
  9. "query_vector": [
  10. 0.45,
  11. 45
  12. ],
  13. "field": "paragraph.vector",
  14. "k": 2,
  15. "num_candidates": 2,
  16. "inner_hits": {
  17. "_source": false,
  18. "fields": [
  19. "paragraph.text"
  20. ],
  21. "size": 1
  22. }
  23. }
  24. }

Now the result will contain the nearest found paragraph when searching.

  1. {
  2. "took": 4,
  3. "timed_out": false,
  4. "_shards": {
  5. "total": 1,
  6. "successful": 1,
  7. "skipped": 0,
  8. "failed": 0
  9. },
  10. "hits": {
  11. "total": {
  12. "value": 2,
  13. "relation": "eq"
  14. },
  15. "max_score": 1.0,
  16. "hits": [
  17. {
  18. "_index": "passage_vectors",
  19. "_id": "1",
  20. "_score": 1.0,
  21. "fields": {
  22. "creation_time": [
  23. "2019-05-04T00:00:00.000Z"
  24. ],
  25. "full_text": [
  26. "first paragraph another paragraph"
  27. ]
  28. },
  29. "inner_hits": {
  30. "paragraph": {
  31. "hits": {
  32. "total": {
  33. "value": 2,
  34. "relation": "eq"
  35. },
  36. "max_score": 1.0,
  37. "hits": [
  38. {
  39. "_index": "passage_vectors",
  40. "_id": "1",
  41. "_nested": {
  42. "field": "paragraph",
  43. "offset": 0
  44. },
  45. "_score": 1.0,
  46. "fields": {
  47. "paragraph": [
  48. {
  49. "text": [
  50. "first paragraph"
  51. ]
  52. }
  53. ]
  54. }
  55. }
  56. ]
  57. }
  58. }
  59. }
  60. },
  61. {
  62. "_index": "passage_vectors",
  63. "_id": "2",
  64. "_score": 0.9997144,
  65. "fields": {
  66. "creation_time": [
  67. "2020-05-04T00:00:00.000Z"
  68. ],
  69. "full_text": [
  70. "number one paragraph number two paragraph"
  71. ]
  72. },
  73. "inner_hits": {
  74. "paragraph": {
  75. "hits": {
  76. "total": {
  77. "value": 2,
  78. "relation": "eq"
  79. },
  80. "max_score": 0.9997144,
  81. "hits": [
  82. {
  83. "_index": "passage_vectors",
  84. "_id": "2",
  85. "_nested": {
  86. "field": "paragraph",
  87. "offset": 1
  88. },
  89. "_score": 0.9997144,
  90. "fields": {
  91. "paragraph": [
  92. {
  93. "text": [
  94. "number two paragraph"
  95. ]
  96. }
  97. ]
  98. }
  99. }
  100. ]
  101. }
  102. }
  103. }
  104. }
  105. ]
  106. }
  107. }

Indexing considerations

For approximate kNN search, Elasticsearch stores the dense vector values of each segment as an HNSW graph. Indexing vectors for approximate kNN search can take substantial time because of how expensive it is to build these graphs. You may need to increase the client request timeout for index and bulk requests. The approximate kNN tuning guide contains important guidance around indexing performance, and how the index configuration can affect search performance.

In addition to its search-time tuning parameters, the HNSW algorithm has index-time parameters that trade off between the cost of building the graph, search speed, and accuracy. When setting up the dense_vector mapping, you can use the index_options argument to adjust these parameters:

  1. resp = client.indices.create(
  2. index="image-index",
  3. mappings={
  4. "properties": {
  5. "image-vector": {
  6. "type": "dense_vector",
  7. "dims": 3,
  8. "similarity": "l2_norm",
  9. "index_options": {
  10. "type": "hnsw",
  11. "m": 32,
  12. "ef_construction": 100
  13. }
  14. }
  15. }
  16. },
  17. )
  18. print(resp)
  1. response = client.indices.create(
  2. index: 'image-index',
  3. body: {
  4. mappings: {
  5. properties: {
  6. "image-vector": {
  7. type: 'dense_vector',
  8. dims: 3,
  9. similarity: 'l2_norm',
  10. index_options: {
  11. type: 'hnsw',
  12. m: 32,
  13. ef_construction: 100
  14. }
  15. }
  16. }
  17. }
  18. }
  19. )
  20. puts response
  1. const response = await client.indices.create({
  2. index: "image-index",
  3. mappings: {
  4. properties: {
  5. "image-vector": {
  6. type: "dense_vector",
  7. dims: 3,
  8. similarity: "l2_norm",
  9. index_options: {
  10. type: "hnsw",
  11. m: 32,
  12. ef_construction: 100,
  13. },
  14. },
  15. },
  16. },
  17. });
  18. console.log(response);
  1. PUT image-index
  2. {
  3. "mappings": {
  4. "properties": {
  5. "image-vector": {
  6. "type": "dense_vector",
  7. "dims": 3,
  8. "similarity": "l2_norm",
  9. "index_options": {
  10. "type": "hnsw",
  11. "m": 32,
  12. "ef_construction": 100
  13. }
  14. }
  15. }
  16. }
  17. }
  • When using kNN search in cross-cluster search, the ccs_minimize_roundtrips option is not supported.
  • Elasticsearch uses the HNSW algorithm to support efficient kNN search. Like most kNN algorithms, HNSW is an approximate method that sacrifices result accuracy for improved search speed. This means the results returned are not always the true k closest neighbors.

Approximate kNN search always uses the dfs_query_then_fetch search type in order to gather the global top k matches across shards. You cannot set the search_type explicitly when running kNN search.

Exact kNN

To run an exact kNN search, use a script_score query with a vector function.

  1. Explicitly map one or more dense_vector fields. If you don’t intend to use the field for approximate kNN, set the index mapping option to false. This can significantly improve indexing speed.

    1. resp = client.indices.create(
    2. index="product-index",
    3. mappings={
    4. "properties": {
    5. "product-vector": {
    6. "type": "dense_vector",
    7. "dims": 5,
    8. "index": False
    9. },
    10. "price": {
    11. "type": "long"
    12. }
    13. }
    14. },
    15. )
    16. print(resp)
    1. response = client.indices.create(
    2. index: 'product-index',
    3. body: {
    4. mappings: {
    5. properties: {
    6. "product-vector": {
    7. type: 'dense_vector',
    8. dims: 5,
    9. index: false
    10. },
    11. price: {
    12. type: 'long'
    13. }
    14. }
    15. }
    16. }
    17. )
    18. puts response
    1. const response = await client.indices.create({
    2. index: "product-index",
    3. mappings: {
    4. properties: {
    5. "product-vector": {
    6. type: "dense_vector",
    7. dims: 5,
    8. index: false,
    9. },
    10. price: {
    11. type: "long",
    12. },
    13. },
    14. },
    15. });
    16. console.log(response);
    1. PUT product-index
    2. {
    3. "mappings": {
    4. "properties": {
    5. "product-vector": {
    6. "type": "dense_vector",
    7. "dims": 5,
    8. "index": false
    9. },
    10. "price": {
    11. "type": "long"
    12. }
    13. }
    14. }
    15. }
  2. Index your data.

    1. POST product-index/_bulk?refresh=true
    2. { "index": { "_id": "1" } }
    3. { "product-vector": [230.0, 300.33, -34.8988, 15.555, -200.0], "price": 1599 }
    4. { "index": { "_id": "2" } }
    5. { "product-vector": [-0.5, 100.0, -13.0, 14.8, -156.0], "price": 799 }
    6. { "index": { "_id": "3" } }
    7. { "product-vector": [0.5, 111.3, -13.0, 14.8, -156.0], "price": 1099 }
    8. ...
  3. Use the search API to run a script_score query containing a vector function.

    To limit the number of matched documents passed to the vector function, we recommend you specify a filter query in the script_score.query parameter. If needed, you can use a match_all query in this parameter to match all documents. However, matching all documents can significantly increase search latency.

    1. resp = client.search(
    2. index="product-index",
    3. query={
    4. "script_score": {
    5. "query": {
    6. "bool": {
    7. "filter": {
    8. "range": {
    9. "price": {
    10. "gte": 1000
    11. }
    12. }
    13. }
    14. }
    15. },
    16. "script": {
    17. "source": "cosineSimilarity(params.queryVector, 'product-vector') + 1.0",
    18. "params": {
    19. "queryVector": [
    20. -0.5,
    21. 90,
    22. -10,
    23. 14.8,
    24. -156
    25. ]
    26. }
    27. }
    28. }
    29. },
    30. )
    31. print(resp)
    1. response = client.search(
    2. index: 'product-index',
    3. body: {
    4. query: {
    5. script_score: {
    6. query: {
    7. bool: {
    8. filter: {
    9. range: {
    10. price: {
    11. gte: 1000
    12. }
    13. }
    14. }
    15. }
    16. },
    17. script: {
    18. source: "cosineSimilarity(params.queryVector, 'product-vector') + 1.0",
    19. params: {
    20. "queryVector": [
    21. -0.5,
    22. 90,
    23. -10,
    24. 14.8,
    25. -156
    26. ]
    27. }
    28. }
    29. }
    30. }
    31. }
    32. )
    33. puts response
    1. const response = await client.search({
    2. index: "product-index",
    3. query: {
    4. script_score: {
    5. query: {
    6. bool: {
    7. filter: {
    8. range: {
    9. price: {
    10. gte: 1000,
    11. },
    12. },
    13. },
    14. },
    15. },
    16. script: {
    17. source: "cosineSimilarity(params.queryVector, 'product-vector') + 1.0",
    18. params: {
    19. queryVector: [-0.5, 90, -10, 14.8, -156],
    20. },
    21. },
    22. },
    23. },
    24. });
    25. console.log(response);
    1. POST product-index/_search
    2. {
    3. "query": {
    4. "script_score": {
    5. "query" : {
    6. "bool" : {
    7. "filter" : {
    8. "range" : {
    9. "price" : {
    10. "gte": 1000
    11. }
    12. }
    13. }
    14. }
    15. },
    16. "script": {
    17. "source": "cosineSimilarity(params.queryVector, 'product-vector') + 1.0",
    18. "params": {
    19. "queryVector": [-0.5, 90.0, -10, 14.8, -156.0]
    20. }
    21. }
    22. }
    23. }
    24. }

Oversampling and rescoring for quantized vectors

All forms of quantization will result in some accuracy loss and as the quantization level increases the accuracy loss will also increase. Generally, we have found that: - int8 requires minimal if any rescoring - int4 requires some rescoring for higher accuracy and larger recall scenarios. Generally, oversampling by 1.5x-2x recovers most of the accuracy loss. - bbq requires rescoring except on exceptionally large indices or models specifically designed for quantization. We have found that between 3x-5x oversampling is generally sufficient. But for fewer dimensions or vectors that do not quantize well, higher oversampling may be required.

There are two main ways to oversample and rescore. The first is to utilize the rescore section in the _search request.

Here is an example using the top level knn search with oversampling and using rescore to rerank the results:

  1. resp = client.search(
  2. index="my-index",
  3. size=10,
  4. knn={
  5. "query_vector": [
  6. 0.04283529,
  7. 0.85670587,
  8. -0.51402352,
  9. 0
  10. ],
  11. "field": "my_int4_vector",
  12. "k": 20,
  13. "num_candidates": 50
  14. },
  15. rescore={
  16. "window_size": 20,
  17. "query": {
  18. "rescore_query": {
  19. "script_score": {
  20. "query": {
  21. "match_all": {}
  22. },
  23. "script": {
  24. "source": "(dotProduct(params.queryVector, 'my_int4_vector') + 1.0)",
  25. "params": {
  26. "queryVector": [
  27. 0.04283529,
  28. 0.85670587,
  29. -0.51402352,
  30. 0
  31. ]
  32. }
  33. }
  34. }
  35. },
  36. "query_weight": 0,
  37. "rescore_query_weight": 1
  38. }
  39. },
  40. )
  41. print(resp)
  1. const response = await client.search({
  2. index: "my-index",
  3. size: 10,
  4. knn: {
  5. query_vector: [0.04283529, 0.85670587, -0.51402352, 0],
  6. field: "my_int4_vector",
  7. k: 20,
  8. num_candidates: 50,
  9. },
  10. rescore: {
  11. window_size: 20,
  12. query: {
  13. rescore_query: {
  14. script_score: {
  15. query: {
  16. match_all: {},
  17. },
  18. script: {
  19. source: "(dotProduct(params.queryVector, 'my_int4_vector') + 1.0)",
  20. params: {
  21. queryVector: [0.04283529, 0.85670587, -0.51402352, 0],
  22. },
  23. },
  24. },
  25. },
  26. query_weight: 0,
  27. rescore_query_weight: 1,
  28. },
  29. },
  30. });
  31. console.log(response);
  1. POST /my-index/_search
  2. {
  3. "size": 10,
  4. "knn": {
  5. "query_vector": [0.04283529, 0.85670587, -0.51402352, 0],
  6. "field": "my_int4_vector",
  7. "k": 20,
  8. "num_candidates": 50
  9. },
  10. "rescore": {
  11. "window_size": 20,
  12. "query": {
  13. "rescore_query": {
  14. "script_score": {
  15. "query": {
  16. "match_all": {}
  17. },
  18. "script": {
  19. "source": "(dotProduct(params.queryVector, 'my_int4_vector') + 1.0)",
  20. "params": {
  21. "queryVector": [0.04283529, 0.85670587, -0.51402352, 0]
  22. }
  23. }
  24. }
  25. },
  26. "query_weight": 0,
  27. "rescore_query_weight": 1
  28. }
  29. }
  30. }

The number of results to return, note its only 10 and we will oversample by 2x, gathering 20 nearest neighbors.

The number of results to return from the KNN search. This will do an approximate KNN search with 50 candidates per HNSW graph and use the quantized vectors, returning the 20 most similar vectors according to the quantized score. Additionally, since this is the top-level knn object, the global top 20 results will from all shards will be gathered before rescoring. Combining with rescore, this is oversampling by 2x, meaning gathering 20 nearest neighbors according to quantized scoring and rescoring with higher fidelity float vectors.

The number of results to rescore, if you want to rescore all results, set this to the same value as k

The script to rescore the results. Script score will interact directly with the originally provided float32 vector.

The weight of the original query, here we simply throw away the original score

The weight of the rescore query, here we only use the rescore query

The second way is to score per shard with the knn query and script_score query. Generally, this means that there will be more rescoring per shard, but this can increase overall recall at the cost of compute.

  1. resp = client.search(
  2. index="my-index",
  3. size=10,
  4. query={
  5. "script_score": {
  6. "query": {
  7. "knn": {
  8. "query_vector": [
  9. 0.04283529,
  10. 0.85670587,
  11. -0.51402352,
  12. 0
  13. ],
  14. "field": "my_int4_vector",
  15. "num_candidates": 20
  16. }
  17. },
  18. "script": {
  19. "source": "(dotProduct(params.queryVector, 'my_int4_vector') + 1.0)",
  20. "params": {
  21. "queryVector": [
  22. 0.04283529,
  23. 0.85670587,
  24. -0.51402352,
  25. 0
  26. ]
  27. }
  28. }
  29. }
  30. },
  31. )
  32. print(resp)
  1. const response = await client.search({
  2. index: "my-index",
  3. size: 10,
  4. query: {
  5. script_score: {
  6. query: {
  7. knn: {
  8. query_vector: [0.04283529, 0.85670587, -0.51402352, 0],
  9. field: "my_int4_vector",
  10. num_candidates: 20,
  11. },
  12. },
  13. script: {
  14. source: "(dotProduct(params.queryVector, 'my_int4_vector') + 1.0)",
  15. params: {
  16. queryVector: [0.04283529, 0.85670587, -0.51402352, 0],
  17. },
  18. },
  19. },
  20. },
  21. });
  22. console.log(response);
  1. POST /my-index/_search
  2. {
  3. "size": 10,
  4. "query": {
  5. "script_score": {
  6. "query": {
  7. "knn": {
  8. "query_vector": [0.04283529, 0.85670587, -0.51402352, 0],
  9. "field": "my_int4_vector",
  10. "num_candidates": 20
  11. }
  12. },
  13. "script": {
  14. "source": "(dotProduct(params.queryVector, 'my_int4_vector') + 1.0)",
  15. "params": {
  16. "queryVector": [0.04283529, 0.85670587, -0.51402352, 0]
  17. }
  18. }
  19. }
  20. }
  21. }

The number of results to return

The knn query to perform the initial search, this is executed per-shard

The number of candidates to use for the initial approximate knn search. This will search using the quantized vectors and return the top 20 candidates per shard to then be scored

The script to score the results. Script score will interact directly with the originally provided float32 vector.