3 releases (breaking)
Uses new Rust 2024
| 0.3.0 | Aug 2, 2025 |
|---|---|
| 0.2.0 | Mar 6, 2025 |
| 0.1.0 | Aug 6, 2022 |
| 0.0.0 |
|
#302 in Rust patterns
2,139 downloads per month
Used in serde_map
245KB
4K
SLoC
Typesense (rust library)
Community-maintained Rust client library for Typesense | Work In Progress & Help Wanted!
Collection schema derive
Apply #[derive(Typesense)] to any struct you want to index in Typesense. The macro generates:
- a
collection_schema()definition based on your struct fields and attributes. - a
{struct_name}Partialstruct for partial updates of Typesense documents.
Quick example
#[derive(Typesense, Serialize, Deserialize)]
#[typesense(
collection_name = "mega_products",
default_sorting_field = "price",
symbols_to_index = ["+", "-"]
)]
struct MegaProduct {
id: String,
#[typesense(infix, stem)]
title: String,
#[typesense(facet)]
brand: String,
#[typesense(sort)]
price: f32,
#[typesense(rename = "product_name", sort = true)]
#[serde(rename = "product_name")]
official_name: String,
}
// update a Typesense document using the generated partial struct
let update_payload = MegaProductPartial {
price: Some(25.99),
..Default::default()
};
let result = client
.collection::<MegaProduct>()
.document("product-1")
.update(&update_payload, None)
.await;
Supported collection parameters
| Key | Type | Description / Notes |
|---|---|---|
collection_name |
string | Defaults to the struct name in lowercase |
default_sorting_field |
string | Must match the field name after rename if used |
enable_nested_fields |
bool | Enables Typesense nested field support |
token_separators |
list of strings | List of symbols or special characters to be used for splitting the text into individual words in addition to space and new-line characters. |
symbols_to_index |
list of strings | List of symbols or special characters to be indexed. |
Supported field parameter
| Attribute | Type | Description / Notes |
|---|---|---|
facet |
bool | Enable faceting for the field |
sort |
bool | Marks the field as sortable |
index |
bool | Whether to index the field in memmory |
store |
bool | Whether to store the field on disk |
infix |
bool | Enables infix search |
stem |
bool | Values are stemmed before indexing in-memory. |
range_index |
bool | Enables an index optimized for range filtering on numerical fields |
optional |
bool | Fields with type Option<T> are optional in the generated Typesense schema. Setting this attribute will override it. |
num_dim |
integer | Set this to a non-zero value to treat a field of type float[] as a vector field. |
locale |
string | Locale for text processing |
vec_dist |
string | Distance metric to be used for vector search |
reference |
string | Name of a field in another collection to be used for JOINs |
type |
string | Override the field type in Typesense |
rename |
string | Rename the field in the Typesense schema |
flatten |
-- | Generate Typesense field schemas for a nested struct |
skip |
-- | Skips this field in the Typesense schema |
All boolean attributes can be either set to true using shorthand flags or explicitly set a value =true/false. Example:
#[typesense(facet)]
brand: String,
#[typesense(facet = false)]
weight: f32,
Indexing nested objects
When you have fields that are also structs, you need to mark all structs with #[derive(Typesense)]. The generated Typesense schema for those fields will have type of object (or object[] if the field is a vector).
Applying #[typesense(flatten)] on a field will expand the nested field schemas into the parent.
#[typesense(flatten)]
supplier: SupplierInfo,
If the field has a rename:
#[typesense(flatten, rename = "logistics_data")]
logistics: Logistics,
flattened fields become logistics_data.field_name.
#[typesense(flatten, skip)] produces only the flattened fields and omits the parent object field.
Nested objects example:
#[derive(Typesense, Serialize, Deserialize)]
struct ProductDetails {
#[typesense(facet)]
part_number: String,
#[typesense(skip)]
description: String,
}
#[derive(Typesense, Serialize, Deserialize)]
#[typesense(
collection_name = "mega_products",
)]
struct MegaProduct {
#[typesense(flatten)]
details: ProductDetails,
}
Will generate this schema:
{
"name": "mega_products"
"fields": [
{"name": "details", "type": "object"}, // <-- `#[typesense(flatten, skip)]` will omit this field
{"name": "details.part_number", "type": "string", "facet": true}
// `description` is skipped
],
}
Development
When updating or adding new parameters and endpoints, make changes directly in the Typesense API spec repository.
Once your changes are merged, you can update this project as follows (you can also run tasks individually):
cargo xtask fetch preprocess code-gen
This will:
- Download the latest API spec.
- Write it to our local
openapi.yml. - Preprocess it into
preprocessed_openapi.yml. - Regenerate the
/typesense_codegencrate.
The preprocessing step does two things:
- Flatten the URL params defined as objects into individual URL parameters (in
preprocess_openapi.rs) - Inject OpenAPI vendor attributes (e.g., generic parameters, schema builders) into the spec before code generation (in
add_vendor_attributes.rs)
You can also run code-gen directly through Docker:
docker run --rm \
-v $PWD:/local openapitools/openapi-generator-cli generate \
-i /local/preprocessed_openapi.yml \
-g rust \
-o /local/typesense_codegen \
-t /local/openapi-generator-template \
--additional-properties library=reqwest \
--additional-properties supportMiddleware=true \
--additional-properties useSingleRequestParameter=true
Testing
Make sure you have a Typesense server up and running:
docker compose up
Then run this command in the root folder to run the integration tests:
cargo test-clean -- --all-features
This is an alias command which will run a script to clean up your Typesense server after the tests finish. You can pass any arguments of cargo test after the --.
To run test for wasm (chrome, headless):
cargo test-clean --wasm
If you'd like to contribute, please join our Slack Community and say hello!
lib.rs:
Typesense
Rust client library for Typesense
Examples
#[cfg(any(feature = "tokio_test", target_arch = "wasm32"))]
{
use serde::{Deserialize, Serialize};
use typesense::document::Document;
use typesense::Typesense;
use typesense::apis::collections_api;
use typesense::apis::configuration::{ApiKey, Configuration};
#[derive(Typesense, Serialize, Deserialize)]
#[typesense(collection_name = "companies", default_sorting_field = "num_employees")]
struct Company {
company_name: String,
num_employees: i32,
#[typesense(facet)]
country: String,
}
#[tokio::main]
async fn main() {
let config = Configuration {
base_path: "http://localhost:5000".to_owned(),
api_key: Some(ApiKey {
prefix: None,
key: "VerySecretKey".to_owned(),
}),
..Default::default()
};
let collection = collections_api::create_collection(&config, Company::collection_schema())
.await
.unwrap();
}
}
Dependencies
~6–22MB
~241K SLoC