Expand description
Context passed to and between services as input.
§State
rama
supports two kinds of states:
- type-safe state: this is the
S
generic parameter inContext
and is to be used as much as possible, given its existence and type properties can be validated at compile time - dynamic state: these can be injected as
Extensions
s using methods such asContext::insert
As a rule of thumb one should use the type-safe state (1) in case:
- the state is always expected to exist at the point the middleware/service is called
- the state is specific to the app or middleware
- and the state can be constructed in a default/empty state
The latter is important given the state is often created (or at least reserved) prior to it is actually being populated by the relevant middleware. This is not the case for app-specific state such as Database pools which are created since the start and shared among many different tasks.
The rule could be be simplified to “if you need to .unwrap()
you probably want type-safe state instead”.
It’s however just a guideline and not a hard rule. As maintainers of rama
we’ll do our best to respect it though,
and we recommend you to do the same.
Any state that is optional, and especially optional state injected by middleware, can be inserted using extensions.
It is however important to try as much as possible to then also consume this state in an approach that deals
gracefully with its absence. Good examples of this are header-related inputs. Headers might be set or not,
and so absence of Extensions
s that might be created as a result of these might reasonably not exist.
It might of course still mean the app returns an error response when it is absent, but it should not unwrap/panic.
§State Wraps
rama
was built from the ground up to operate on and between different layers of the network stack.
This has also an impact on state. Because sure, typed state is nice, but state leakage is not. What do I mean with that?
When creating a TcpListener
with state you are essentially creating and injecting state, which will remain
as “read-only” for the enire life cycle of that TcpListener
and to be made available for every incoming tcp connection,
as well as the application requests (Http requests). This is great for stuff that is okay to share, but it is not desired
for state that you wish to have a narrower scope. Examples are state that are tied to a single tcp connection and thus
you do not wish to keep a global cache for this, as it would either be shared or get overly complicated to ensure
you keep things separate and clean.
The solution is to wrap your state.
Example: http_conn_state.rs
The above example shows how can use the [#as_ref(wrap)
] property within an #[derive(AsRef)]
derived “state” struct,
to wrap in a type-safe manner the “app-global” state within the “conn-specific” (tcp) state. This allows you to have
state freshly created for each connection while still having ease of access to the global state.
Note though that you do not need the AsRef
macro or even trait implementation to get this kind of access in your
own app-specific leaf services. It is however useful — and at times even a requirement — in case you want your
middleware stack to also include generic middleware that expect AsRef<T>
trait bounds for type-safe access to
state from within a middleware. E.g. in case your middleware expects a data source for some specific data type,
it is of no use to have that middleware compile without knowing for sure that data source is made available
to that middleware.
§Example
use rama_core::Context;
use std::sync::Arc;
#[derive(Debug)]
struct ServiceState {
value: i32,
}
let state = Arc::new(ServiceState{ value: 5 });
let ctx = Context::with_state(state);
§Example: Extensions
The Context
can be extended with additional data using the Extensions
type.
use rama_core::Context;
let mut ctx = Context::default();
ctx.insert(5i32);
assert_eq!(ctx.get::<i32>(), Some(&5i32));
§Example: State AsRef
The state can be accessed as a reference using the AsRef
trait.
use rama_core::{Context, context};
use std::sync::Arc;
use std::convert::AsRef;
#[derive(Debug)]
struct ProxyDatabase;
#[derive(Debug, context::AsRef)]
struct ServiceState {
db: ProxyDatabase,
}
let state = Arc::new(ServiceState{ db: ProxyDatabase });
let ctx = Context::with_state(state);
let db: &ProxyDatabase = ctx.state().as_ref();
Structs§
- Context passed to and between services as input.
- A type map of protocol extensions.
Derive Macros§
- Derive an implementation of
AsRef
for each field in a struct.