Library provides a simple API for Google Firestore based on the official gRPC API:
- Create or update documents using Rust structures and Serde;
- Support for:
- Querying/streaming docs/objects;
- Listing documents/objects (and auto pages scrolling support);
- Listening changes from Firestore;
- Transactions;
- Aggregated Queries;
- Fluent high-level and strongly typed API;
- Full async based on Tokio runtime;
- Macro that helps you use JSON paths as references to your structure fields;
- Implements own Serde serializer to Firestore protobuf values;
- Supports for Firestore timestamp with #[serde(with)]
- Google client based on gcloud-sdk library
that automatically detects GKE environment or application default accounts for local development;
Cargo.toml:
toml
[dependencies]
firestore = "0.12"
All examples available at examples directory.
To run example use it with environment variables: ```
```
To simplify development and developer experience the library provides the Fluent API:
```rust
// Create an instance let db = FirestoreDb::new(&configenvvar("PROJECT_ID")?).await?;
const TESTCOLLECTIONNAME: &'static str = "test";
let mystruct = MyTestStructure { someid: "test-1".tostring(), somestring: "Test".tostring(), onemorestring: "Test2".tostring(), some_num: 42, };
// Create data let objectreturned: MyTestStructure = db.fluent() .insert() .into(TESTCOLLECTIONNAME) .documentid(&mystruct.someid) .object(&my_struct) .execute() .await?;
// Update data let objectupdated: MyTestStructure = db.fluent() .update() .fields(paths!(MyTestStructure::{somenum, onemorestring})) .incol(TESTCOLLECTIONNAME) .documentid(&mystruct.someid) .object(&MyTestStructure { somenum: mystruct.somenum + 1, onemorestring: "updated-value".tostring(), ..my_struct.clone() }) .execute() .await?;
// Get object by id let finditagain: MyTestStructure = db.getobj(TESTCOLLECTIONNAME, &mystruct.some_id).await?;
// Query as a stream our data
let objectstream: BoxStream
let asvec: Vec
// Delete data db.fluent() .delete() .from(TESTCOLLECTIONNAME) .documentid(&mystruct.some_id) .execute() .await?;
```
By default, the types such as DateTime#[serde(with)]
:
```
struct MyTestStructure {
#[serde(with = "firestore::serializeastimestamp")]
created_at: DateTime
In queries you need to use a special wrapping class firestore::FirestoreTimestamp
, for example:
q.field(path!(MyTestStructure::created_at))
.eq(firestore::FirestoreTimestamp(Utc::now()).into())
You can work with nested collection using functions like db.create_object_at
and specifying path/location to documents:
```rust
// Creating a parent doc db.fluent() .insert() .into(TESTPARENTCOLLECTIONNAME) .documentid(&parentstruct.someid) .object(&parent_struct) .execute() .await?;
// The doc path where we store our childs let parentpath = format!( "{}/{}/{}", db.getdocumentspath(), TESTPARENTCOLLECTIONNAME, parentstruct.someid );
// Create a child doc db.fluent() .insert() .into(TESTCHILDCOLLECTIONNAME) .documentid(&childstruct.someid) .parent(&parentpath) .object(&childstruct) .execute() .await?;
// Listing children
let mut objsstream: BoxStream
Looks for credentials in the following places, preferring the first location found:
- A JSON file whose path is specified by the GOOGLEAPPLICATIONCREDENTIALS environment variable.
- A JSON file in a location known to the gcloud command-line tool using gcloud auth application-default login
.
- On Google Compute Engine, it fetches credentials from the metadata server.
Don't confuse gcloud auth login
with gcloud auth application-default login
for local development,
since the first authorize only gcloud
tool to access the Cloud Platform.
The latter obtains user access credentials via a web flow and puts them in the well-known location for Application Default Credentials (ADC).
This command is useful when you are developing code that would normally use a service account but need to run the code in a local development environment where it's easier to provide user credentials.
So to work for local development you need to use gcloud auth application-default login
.
Apache Software License (ASL)
Abdulla Abdurakhmanov