2017-01-21 03:31:46 +00:00
|
|
|
use std::ops::Deref;
|
|
|
|
|
|
|
|
use request::{self, FromRequest, Request};
|
|
|
|
use outcome::Outcome;
|
|
|
|
use http::Status;
|
|
|
|
|
2017-02-03 02:00:18 +00:00
|
|
|
/// Request guard to retrieve managed state.
|
|
|
|
///
|
|
|
|
/// This type can be used as a request guard to retrieve the state Rocket is
|
|
|
|
/// managing for some type `T`. This allows for the sharing of state across any
|
|
|
|
/// number of handlers. A value for the given type must previously have been
|
|
|
|
/// registered to be managed by Rocket via the
|
|
|
|
/// [manage](/rocket/struct.Rocket.html#method.manage) method. The type being
|
2017-02-04 00:56:29 +00:00
|
|
|
/// managed must be thread safe and sendable across thread boundaries. In other
|
2017-02-03 02:00:18 +00:00
|
|
|
/// words, it must implement `Send + Sync + 'static`.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// Imagine you have some configuration struct of the type `MyConfig` that you'd
|
|
|
|
/// like to initialize at start-up and later access it in several handlers. The
|
|
|
|
/// following example does just this:
|
|
|
|
///
|
|
|
|
/// ```rust
|
2017-08-29 03:14:59 +00:00
|
|
|
/// # #![feature(plugin, decl_macro)]
|
2017-02-03 02:00:18 +00:00
|
|
|
/// # #![plugin(rocket_codegen)]
|
|
|
|
/// # extern crate rocket;
|
|
|
|
/// use rocket::State;
|
|
|
|
///
|
|
|
|
/// // In a real application, this would likely be more complex.
|
2017-07-03 22:13:21 +00:00
|
|
|
/// struct MyConfig {
|
|
|
|
/// user_val: String
|
|
|
|
/// }
|
2017-02-03 02:00:18 +00:00
|
|
|
///
|
|
|
|
/// #[get("/")]
|
|
|
|
/// fn index(state: State<MyConfig>) -> String {
|
2017-07-03 22:13:21 +00:00
|
|
|
/// format!("The config value is: {}", state.user_val)
|
2017-02-03 02:00:18 +00:00
|
|
|
/// }
|
|
|
|
///
|
|
|
|
/// #[get("/raw")]
|
|
|
|
/// fn raw_config_value<'r>(state: State<'r, MyConfig>) -> &'r str {
|
|
|
|
/// // use `inner()` to get a lifetime longer than `deref` gives us
|
2017-07-03 22:13:21 +00:00
|
|
|
/// state.inner().user_val.as_str()
|
2017-02-03 02:00:18 +00:00
|
|
|
/// }
|
|
|
|
///
|
|
|
|
/// fn main() {
|
2017-07-03 22:13:21 +00:00
|
|
|
/// let config = MyConfig {
|
|
|
|
/// user_val: "user input".to_string()
|
|
|
|
/// };
|
|
|
|
///
|
2017-02-03 02:00:18 +00:00
|
|
|
/// # if false { // We don't actually want to launch the server in an example.
|
|
|
|
/// rocket::ignite()
|
|
|
|
/// .mount("/", routes![index, raw_config_value])
|
|
|
|
/// .manage(config)
|
2017-03-16 05:10:09 +00:00
|
|
|
/// .launch();
|
2017-02-03 02:00:18 +00:00
|
|
|
/// # }
|
|
|
|
/// }
|
|
|
|
/// ```
|
2017-07-03 22:13:21 +00:00
|
|
|
///
|
|
|
|
/// # Within Request Guards
|
|
|
|
///
|
|
|
|
/// Because `State` is itself a request guard, managed state can be retrieved
|
|
|
|
/// from another request guard's implementation. In the following code example,
|
|
|
|
/// `Item` retrieves the `MyConfig` managed state in its `FromRequest`
|
|
|
|
/// implementation using the [`Request::guard()`] method.
|
|
|
|
///
|
|
|
|
/// [`Request::guard()`]: /rocket/struct.Request.html#method.guard
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use rocket::State;
|
|
|
|
/// use rocket::request::{self, Request, FromRequest};
|
|
|
|
///
|
|
|
|
/// # struct MyConfig{ user_val: String };
|
|
|
|
/// struct Item(String);
|
|
|
|
///
|
|
|
|
/// impl<'a, 'r> FromRequest<'a, 'r> for Item {
|
|
|
|
/// type Error = ();
|
|
|
|
///
|
|
|
|
/// fn from_request(request: &'a Request<'r>) -> request::Outcome<Item, ()> {
|
|
|
|
/// request.guard::<State<MyConfig>>()
|
|
|
|
/// .map(|my_config| Item(my_config.user_val.clone()))
|
|
|
|
/// }
|
|
|
|
/// }
|
|
|
|
/// ```
|
2017-07-03 22:39:24 +00:00
|
|
|
#[derive(Debug, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
2017-01-21 03:31:46 +00:00
|
|
|
pub struct State<'r, T: Send + Sync + 'static>(&'r T);
|
|
|
|
|
|
|
|
impl<'r, T: Send + Sync + 'static> State<'r, T> {
|
2017-07-03 22:13:21 +00:00
|
|
|
/// Retrieve a borrow to the underyling value with a lifetime of `'r`.
|
2017-01-21 03:31:46 +00:00
|
|
|
///
|
|
|
|
/// Using this method is typically unnecessary as `State` implements `Deref`
|
|
|
|
/// with a `Target` of `T`. This means Rocket will automatically coerce a
|
2017-07-03 22:13:21 +00:00
|
|
|
/// `State<T>` to an `&T` as required. This method should only be used when
|
|
|
|
/// a longer lifetime is required.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```rust
|
|
|
|
/// use rocket::State;
|
|
|
|
///
|
|
|
|
/// struct MyConfig {
|
|
|
|
/// user_val: String
|
|
|
|
/// }
|
|
|
|
///
|
|
|
|
/// // Use `inner()` to get a lifetime of `'r`
|
|
|
|
/// fn handler1<'r>(config: State<'r, MyConfig>) -> &'r str {
|
|
|
|
/// &config.inner().user_val
|
|
|
|
/// }
|
|
|
|
///
|
|
|
|
/// // Use the `Deref` implementation which coerces implicitly
|
|
|
|
/// fn handler2(config: State<MyConfig>) -> String {
|
|
|
|
/// config.user_val.clone()
|
|
|
|
/// }
|
|
|
|
/// ```
|
2017-03-28 10:10:18 +00:00
|
|
|
#[inline(always)]
|
2017-01-21 03:31:46 +00:00
|
|
|
pub fn inner(&self) -> &'r T {
|
|
|
|
self.0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Doc.
|
|
|
|
impl<'a, 'r, T: Send + Sync + 'static> FromRequest<'a, 'r> for State<'r, T> {
|
|
|
|
type Error = ();
|
|
|
|
|
2017-03-28 10:10:18 +00:00
|
|
|
#[inline(always)]
|
2017-01-21 03:31:46 +00:00
|
|
|
fn from_request(req: &'a Request<'r>) -> request::Outcome<State<'r, T>, ()> {
|
2017-03-28 10:10:18 +00:00
|
|
|
match req.get_state::<T>() {
|
|
|
|
Some(state) => Outcome::Success(State(state)),
|
|
|
|
None => {
|
|
|
|
error_!("Attempted to retrieve unmanaged state!");
|
|
|
|
Outcome::Failure((Status::InternalServerError, ()))
|
2017-01-21 03:31:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'r, T: Send + Sync + 'static> Deref for State<'r, T> {
|
|
|
|
type Target = T;
|
|
|
|
|
2017-03-28 10:10:18 +00:00
|
|
|
#[inline(always)]
|
2017-01-21 03:31:46 +00:00
|
|
|
fn deref(&self) -> &T {
|
|
|
|
self.0
|
|
|
|
}
|
|
|
|
}
|