Latency
Percentile
Latency
Exact filters for attributes to refine search results for. Think of it as a SQL WHERE clause.
See Filtering Parameters below for details.
When combined with a vector, the query planner will automatically combine the attribute index and the approximate nearest neighbor index for best performance and recall.
For the best performance, separate documents into namespaces instead of filtering where possible. See also Performance.
Example: ["And", [["id", "Gte", 1000], ["permissions", "In", ["3d7a7296-3d6a-4796-8fb0-f90406b1f621", "92ef7c95-a212-43a4-ae4e-0ebc96a65764"]]]]
Optionally the vector to search for.
It must have the same number of dimensions as the vectors in the namespace.
Example: [0.1, 0.2, 0.3, ..., 76.8]
Used for BM25 full-text search or ordering by attributes.
Currently, you can pass a rank_by
parameter or a vector
parameter, but not both. If neither is passed, results are sorted by ID.
For hybrid search, you must do multiple queries (e.g. BM25 + vector) and combine the results client-side with e.g. reciprocal-rank fusion. We encourage users to write a strong query layer abstraction, as it's not uncommon to do 6 turbopuffer queries per user query.
Order by attribute example: ["timestamp", "desc"]
BM25: ["text", "BM25", "fox jumping"]
BM25 with multiple fields: ["Sum", [["text", "BM25", "fox jumping"], ["name", "BM25", "fox jumping"]]]
The function used to calculate vector similarity. Possible values are cosine_distance
or euclidean_squared
.
cosine_distance
is defined as 1 - cosine_similarity
and ranges from 0 to 2.
Lower is better.
euclidean_squared
is defined as sum((x - y)^2)
. Lower is better.
Number of results to return.
Return vectors for the search results. Vectors are large and slow to deserialize on the client, so use this option only if you need them.
List of attribute names to return in the response. Can be set to true
to
return all attributes. Return only the ones you need for best performance.
{ // Request payload
"vector": [0.1, 0.1],
"distance_metric": "distance_metric",
"top_k": 10
}
[ // Response payload
{ "dist": 0.0, "id": 1 },
{ "dist": 2.0, "id": 2 }
]
When you need to filter documents, you can combine filters with vector search or use them alone. Here's an example of finding recent public documents:
{ // Request payload
"filters": ["And", [
["timestamp", "Gte", 1709251200], // Documents after March 1, 2024
["public", "Eq", true]
]],
"vector": [0.1, 0.2, 0.3], // Optional: include vector to combine with filters
"distance_metric": "cosine_distance", // Required if vector is set
"top_k": 10,
"include_attributes": ["title", "timestamp"]
}
[ // Response payload
{
"id": 1,
"dist": 0.15, // Only present when vector search is used
"attributes": {
"title": "Getting Started Guide",
"timestamp": 1709337600 // March 2, 2024
}
},
{
"id": 2,
"dist": 0.28,
"attributes": {
"title": "Advanced Features",
"timestamp": 1709424000 // March 3, 2024
}
}
]
Filter-only (no vector or FTS/BM25) queries can specify a rank_by
parameter to order results by a specific attribute (i.e. SQL ORDER BY
). For example, to order by timestamp in descending order:
{ // Request payload
"filters": ["timestamp", "Lt", 1709251200], // Documents before March 1, 2024
"rank_by": ["timestamp", "desc"], // Order by timestamp in descending order
"top_k": 1000,
"include_attributes": ["title", "timestamp"]
}
[ // Response payload
{
"id": 6,
"attributes": {
"title": "Roadmap",
"timestamp": 1709020800 // Feb 27th, 2024
}
},
{
"id": 4,
"attributes": {
"title": "Performance Guide",
"timestamp": 1708761600 // Feb 24th, 2024
}
}
]
Ordering by multiple attributes isn't yet implemented.
Similar to SQL, the ordering of results is not guaranteed when multiple documents have the same attribute value for the rank_by
parameter. Array attributes aren't supported.
The FTS attribute must be configured with full_text_search
set in the schema
when upserting documents. See Schema documentation and
the Full-Text Search guide for more details.
You can combine BM25 full-text search with filters to search within specific document subsets. For hybrid search combining both vector and BM25 results, see Hybrid Search.
{ // Request payload
"rank_by": ["content", "BM25", "quick fox"],
"filters": ["And", [
["timestamp", "Gte", 1709251200], // Documents after March 1, 2024
["public", "Eq", true]
]],
"top_k": 10,
"include_attributes": ["title", "content", "timestamp"]
}
[ // Response payload
{
"id": 1,
"dist": 1.28, // BM25 relevance score (higher is better)
"attributes": {
"title": "Animal Stories",
"content": "The quick brown fox jumps over the lazy dog",
"timestamp": 1709337600 // March 2, 2024
}
},
{
"id": 2,
"dist": 0.85,
"attributes": {
"title": "Forest Tales",
"content": "A quick red fox darted through the forest",
"timestamp": 1709424000 // March 3, 2024
}
}
]
Filters allow you to narrow down results by applying exact conditions to attributes. Conditions are arrays with an attribute name, operation, and value, for example:
["attr_name", "Eq", 42]
["page_id", "In", ["page1", "page2"]]
["user_migrated_at", "NotEq", null]
Values must have the same type as the attribute's value, or an array of that type for operators like In
.
Conditions can be combined using {And,Or}
operations:
// basic And condition
"filters": ["And", [
["attr_name", "Eq", 42],
["page_id", "In", ["page1", "page2"]]
]]
// conditions can be nested
"filters": ["And", [
["page_id", "In", ["page1", "page2"]],
["Or", [
["public", "Eq", 1],
["permission_id", "In", ["3iQK2VC4", "wzw8zpnQ"]]
]]
]]
// legacy API: an object may provided instead (implicitly And)
"filters": {
"attr_name": ["Eq", 42],
"page_id": ["In", ["page1", "page2"]],
}
Filters can also be applied to the id
field, which refers to the document ID. `
Exact match for id
or attributes
values. If value is null
, matches documents missing the attribute.
Inverse of Eq
, for attributes
values. If value is null
, matches documents with the attribute.
Matches any id
or attributes
values contained in the provided list. If both the provided value and the target document field are arrays, then this checks if any elements of the two sets intersect.
Inverse of In
, matches any attributes
values not contained in the provided list.
For ints, this is a numeric less-than on attributes
values. For strings, lexicographic less-than.
For ints, this is a numeric less-than-or-equal on attributes
values. For strings, lexicographic less-than-or-equal.
For ints, this is a numeric greater-than on attributes
values. For strings, lexicographic greater-than.
For ints, this is a numeric greater-than-or-equal on attributes
values. For strings, lexicographic greater-than-or-equal.
Unix-style glob match against attributes
values. The full syntax is described in the globset documentation. Glob patterns with a concrete prefix like "foo*" internally compile to efficient range queries
Inverse of Glob
, Unix-style glob filters against attributes
values. The full syntax is described in the globset documentation.
Case insensitive version of Glob
.
Case insensitive version of NotGlob
.
Matches if all of the filters match.
Matches if at least one of the filters matches.
Using nested And
and Or
filters:
{
"vector": [0.1, 0.1],
"distance_metric": "euclidean_squared",
"top_k": 10,
"include_attributes": ["key1"],
"filters": [
"And",
[
["id", "In", [1, 2, 3]],
["key1", "Eq", "one"],
["filename", "NotGlob", "/vendor/**"],
[
"Or",
[
["filename", "Glob", "**.tsx"],
["filename", "Glob", "**.js"]
]
]
]
]
}
If you need to paginate the entire namespace, use the more performant Export endpoint.
By default, results are sorted in ascending ID order, which allows pagination with an ID greater-than filter for filter-only queries. You can specify a rank_by
parameter to order results by attributes, see Ordering by Attributes.
ns = tpuf.Namespace("pagination")
last_id = None
while True:
results = ns.query(
top_k=1000,
filters=["And", [
["timestamp", "Gte", 1],
["id", "Gte" if last_id is None else "Gt", last_id or 0]
]]
)
print(results)
if not results or len(results) < 1000:
break
last_id = results[-1].id
Currently paginating beyond the first page for full-text search and vector
search is not supported. Pass a larger top_k
value to get more results and
paginate client-side. If you need a higher limit, please contact us.