Reactive dependency injection for Rust.
This project is work in progress. APIs will change.
This component was extracted and cleaned up from [OxidizeBot
].
There it served as a central point for handling live reconfiguration of the application, without the need to restart it.
This crate provides a reactive dependency injection system that can reconfigure your application dynamically from changes in your dependencies.
Using an Injector
to deal with this complexity provides a nice option for a
clean architecture.
An injector is also a natural way for structuring an application that gracefully degrades during the absence of configuration: any part of the dependency tree that is not available can simply not be provided.
The following is an example application that receives configuration changes over HTTP.
```rust use anyhow::Error; use asyncinjector::{Provider, Injector, Key, asynctrait};
/// Provider that describes how to construct a database.
struct DatabaseProvider { #[dependency(tag = "\"database/url\"")] url: String, #[dependency(tag = "\"database/connection-limit\"")] connection_limit: u32, }
impl Provider for DatabaseProvider { type Output = Database;
/// Constructor a new database and supply it to the injector.
async fn build(self) -> Option<Self::Output> {
match Database::connect(&self.url, self.connection_limit).await {
Ok(database) => Some(database),
Err(e) => {
log::warn!("failed to connect to database: {}: {}", self.url, e);
None
}
}
}
}
/// A fake webserver handler. /// /// Note: there's no real HTTP framework that looks like this. This is just an /// example. async fn serve(injector: &Injector) -> Result<(), Error> { let server = Server::new()?;
// Fake endpoint to set the database URL.
server.on("POST", "/config/database/url", |url: String| {
injector.update_key(Key::tagged("database/url")?, url);
});
// Fake endpoint to set the database connection limit.
server.on("POST", "/config/database/connection-limit", |limit: u32| {
injector.update_key(Key::tagged("database/connection-limit")?, limit);
});
// Listen for requests.
server.await?;
Ok(())
}
async fn main() -> Result<(), Error> { let injector0 = Injector::new();
/// Setup database provider.
let injector = injector0.clone();
tokio::spawn(async move {
DatabaseProvider::run(&injector0).await;
});
let injector = injector0.clone();
tokio::spawn(async move {
serve(&injector).await.expect("web server errored");
});
let (database_stream, database) = injector0.stream::<Database>();
let application = Application::new(database);
loop {
futures::select! {
// receive new databases when available.
database = database_stream.next() => {
application.database = database;
},
// run the application to completion.
_ = application => {
log::info!("application finished");
},
}
}
} ```