This crate allows easy access to your Google Firestore DB via service account or OAuth impersonated Google Firebase Auth credentials.
Features: * Subset of the Firestore v1 API * Optionally handles authentication and token refreshing for you * Support for the downloadable Google service account json file from Google Clound console. (See https://cloud.google.com/storage/docs/reference/libraries#client-libraries-install-cpp)
Usecases: * Strictly typed document read/write/query access * Cloud functions (Google Compute, AWS Lambda) access to Firestore
Limitations: * Listening to document / collection changes is not yet possible
rocket_support
: Enables the rocket guard.
Only Firestore Auth authorized requests can pass this guard.
This feature requires rust nightly, because Rocket itself requires nightly.rustls-tls
: Use rustls instead of native-tls (openssl on Linux).
If you want to compile this crate for musl, this is what you want.
Don't forget to disable the default features with --no-default-features
.This crate operates on DTOs (Data transfer objects) for type-safe operations on your Firestore DB.
```rust use firestoredband_auth::{credentials, sessions};
/// A document structure for demonstration purposes
struct DemoDTO { astring: String, anint: u32, }
let obj = DemoDTO { astring: "abcd".toowned(), an_int: 14, };
/// Write the given object with the document id "servicetest" to the "tests" collection. /// You do not need to provide a document id (use "None" instead) and let Firestore generate one for you. /// /// In either way a document is created or updated (overwritten). /// /// The method will return document metadata (including a possible generated document id) let result = documents::write(&session, "tests", Some("servicetest"), &obj)?;
println!("id: {}, created: {}, updated: {}", result.documentid, result.createtime, result.updated_time); ```
Read the document with the id "service_test" from the Firestore "tests" collection:
rust
let obj : DemoDTO = documents::read(&session, "tests", "service_test")?;
For listing all documents of the "tests" collection you want to use the List
struct which implements the Iterator
trait.
It will hide the complexity of the paging API and fetches new documents when necessary:
rust
let values: documents::List<DemoDTO, _> = documents::list(&session, "tests");
for doc_result in values {
// The document is wrapped in a Result<> because fetching new data could have failed
let doc = doc_result?;
println!("{:?}", doc);
}
Note: The resulting list or list cursor is a snapshot view with a limited lifetime. You cannot keep the iterator for long or expect new documents to appear in an ongoing iteration.
For quering the database you would use the query
method.
In the following example the collection "tests" is queried for document(s) with the "id" field equal to "Sam Weiss".
rust
let objs : Vec<DemoDTO> = documents::query(&session, "tests", "Sam Weiss", dto::FieldOperator::EQUAL, "id")?;
Note: The query method returns a vector, because a query potentially returns multiple matching documents.
The returned Result
will have a FirebaseError
set in any error case.
This custom error type wraps all possible errors (IO, Reqwest, JWT errors etc)
and Google REST API errors. If you want to specifically check for an API error,
you could do so:
rust
let r = documents::delete(&session, "tests/non_existing", true);
if let Err(e) = r.err() {
if let FirebaseError::APIError(code, message, context) = e {
assert_eq!(code, 404);
assert!(message.contains("No document to update"));
assert_eq!(context, "tests/non_existing");
}
}
The code is numeric, the message is what the Google server returned as message. The context string depends on the called method. It may me the collection or document id or any other context information.
"private_key_id": ...
."api_key" : "YOUR_API_KEY"
and replace YOURAPIKEY with your Web API key, to be found in the Google Firebase console in "Project Overview -> Settings - > General".```rust use firestoredband_auth::{credentials, sessions};
/// Create credentials object. You may as well do that programmatically. let cred = credentials::Credentials::from_file("firebase-service-account.json") .expect("Read credentials file");
/// To use any of the Firestore methods, you need a session first. You either want /// an impersonated session bound to a Firebase Auth user or a service account session. let mut session = sessions::service_account::Session::new(&cred) .expect("Create a service account session"); ```
Mutable session variable?: Access (bearer) tokens have a limited lifetime, usually about an hour. They need to be refreshed via a refresh token, which is also part of the session object. When you perform a call to an API, the session will automatically refresh your access token if necessary, and therefore requires the session object to be mutable.
You can create a user session in various ways. If you just have the firebase Auth user_id, you would follow these steps:
```rust use firestoredband_auth::{credentials, sessions};
/// Create credentials object. You may as well do that programmatically. let cred = credentials::Credentials::from_file("firebase-service-account.json") .expect("Read credentials file");
/// To use any of the Firestore methods, you need a session first. /// Create an impersonated session bound to a Firebase Auth user via your service account credentials. let mut session = sessions::user::Session::byuserid(&cred, "theuserid") .expect("Create a user session"); ```
If you have a valid refresh token already and want to generate an access token (and a session object), you do this instead:
rust
let refresh_token = "fkjandsfbajsbfd;asbfdaosa.asduabsifdabsda,fd,a,sdbasfadfasfas.dasdasbfadusbflansf";
let mut session = sessions::user::Session::by_refresh_token(&cred, &refresh_token)?;
The last way to retrieve a session object is by providing a valid access token like so:
rust
let access_token = "fkjandsfbajsbfd;asbfdaosa.asduabsifdabsda,fd,a,sdbasfadfasfas.dasdasbfadusbflansf";
let mut session = sessions::user::Session::by_access_token(&cred, &access_token)?;
The by_access_token
method will fail if the token is not valid anymore.
Please note that a session created this way is not able to automatically refresh its access token.
There is no refresh_token associated with it.
The start up time is crucial for cloud functions. The usual start up procedure includes three IO operations:
Avoid those by embedding the credentials and public key files into your application.
First download the 2 public key files:
securetoken.jwks
service-account.jwks
Create a Credentials
object like so:
rust
use firestore_db_and_auth::credentials::Credentials;
let c = Credentials::new(include_str!("firebase-service-account.json"),
&[include_str!("securetoken.jwks"), include_str!("service-account.jwks")])?;
cargo +nightly doc --features external_doc,rocket_support
To perform a full integration test (cargo test
), you need a valid "firebase-service-account.json" file.
The tests expect a Firebase user with the ID given in tests/test_user_id.txt
to exist.
More Information
Maintenance status: In Development
This library does not have the ambition to mirror the http/gRPC API 1:1. There are auto-generated libraries for this purpose.
This crate uses reqwest as http client. reqwest itself will soon have picked up full support for Rusts async+await features.
All that is left to do here then is to depend on Rust 1.39 and add an "async fn" variant that calls and awaits reqwest for each existing method.