2017-06-06 20:41:04 +00:00
|
|
|
use std::fmt;
|
2017-03-07 09:19:06 +00:00
|
|
|
use std::cell::RefMut;
|
|
|
|
|
2018-07-11 00:09:24 +00:00
|
|
|
use cookie::Delta;
|
|
|
|
pub use cookie::{Cookie, Key, CookieJar, SameSite};
|
2017-03-08 11:28:12 +00:00
|
|
|
|
2018-06-07 13:34:47 +00:00
|
|
|
use Header;
|
2017-03-08 11:28:12 +00:00
|
|
|
|
2017-06-24 09:49:16 +00:00
|
|
|
/// Collection of one or more HTTP cookies.
|
|
|
|
///
|
|
|
|
/// The `Cookies` type allows for retrieval of cookies from an incoming request
|
|
|
|
/// as well as modifications to cookies to be reflected by Rocket on outgoing
|
|
|
|
/// responses. `Cookies` is a smart-pointer; it internally borrows and refers to
|
|
|
|
/// the collection of cookies active during a request's life-cycle.
|
|
|
|
///
|
|
|
|
/// # Usage
|
|
|
|
///
|
|
|
|
/// A type of `Cookies` can be retrieved via its `FromRequest` implementation as
|
|
|
|
/// a request guard or via the [`Request::cookies`] method. Individual cookies
|
|
|
|
/// can be retrieved via the [`get`] and [`get_private`] methods. Cookies can be
|
|
|
|
/// added or removed via the [`add`], [`add_private`], [`remove`], and
|
|
|
|
/// [`remove_private`] methods.
|
|
|
|
///
|
|
|
|
/// [`get`]: /rocket/http/enum.Cookies.html#method.get
|
|
|
|
/// [`get_private`]: /rocket/http/enum.Cookies.html#method.get_private
|
|
|
|
/// [`add`]: /rocket/http/enum.Cookies.html#method.add
|
|
|
|
/// [`add_private`]: /rocket/http/enum.Cookies.html#method.add_private
|
|
|
|
/// [`remove`]: /rocket/http/enum.Cookies.html#method.remove
|
|
|
|
/// [`remove_private`]: /rocket/http/enum.Cookies.html#method.remove_private
|
|
|
|
/// [`Request::cookies`]: /rocket/struct.Request.html#method.cookies
|
|
|
|
///
|
|
|
|
/// ## Examples
|
|
|
|
///
|
|
|
|
/// The following short snippet shows `Cookies` being used as a request guard in
|
|
|
|
/// a handler to retrieve the value of a "message" cookie.
|
|
|
|
///
|
|
|
|
/// ```rust
|
2018-09-20 04:14:30 +00:00
|
|
|
/// # #![feature(proc_macro_non_items, proc_macro_gen, decl_macro)]
|
|
|
|
/// # #[macro_use] extern crate rocket;
|
2017-06-24 09:49:16 +00:00
|
|
|
/// use rocket::http::Cookies;
|
|
|
|
///
|
|
|
|
/// #[get("/message")]
|
|
|
|
/// fn message(cookies: Cookies) -> Option<String> {
|
|
|
|
/// cookies.get("message").map(|c| format!("Message: {}", c.value()))
|
|
|
|
/// }
|
|
|
|
/// # fn main() { }
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// The following snippet shows `Cookies` being retrieved from a `Request` in a
|
|
|
|
/// custom request guard implementation for `User`. A [private cookie]
|
|
|
|
/// containing a user's ID is retrieved. If the cookie exists and the ID parses
|
|
|
|
/// as an integer, a `User` structure is validated. Otherwise, the guard
|
|
|
|
/// forwards.
|
|
|
|
///
|
|
|
|
/// [private cookie]: /rocket/http/enum.Cookies.html#private-cookies
|
|
|
|
///
|
|
|
|
/// ```rust
|
2018-09-20 04:14:30 +00:00
|
|
|
/// # #![feature(proc_macro_non_items, proc_macro_gen, decl_macro, never_type)]
|
|
|
|
/// # #[macro_use] extern crate rocket;
|
2017-06-24 09:49:16 +00:00
|
|
|
/// #
|
|
|
|
/// use rocket::http::Status;
|
|
|
|
/// use rocket::outcome::IntoOutcome;
|
|
|
|
/// use rocket::request::{self, Request, FromRequest};
|
|
|
|
///
|
|
|
|
/// // In practice, we'd probably fetch the user from the database.
|
|
|
|
/// struct User(usize);
|
|
|
|
///
|
|
|
|
/// impl<'a, 'r> FromRequest<'a, 'r> for User {
|
2018-08-10 11:42:30 +00:00
|
|
|
/// type Error = !;
|
2017-06-24 09:49:16 +00:00
|
|
|
///
|
2018-08-10 11:42:30 +00:00
|
|
|
/// fn from_request(request: &'a Request<'r>) -> request::Outcome<User, !> {
|
2017-06-24 09:49:16 +00:00
|
|
|
/// request.cookies()
|
|
|
|
/// .get_private("user_id")
|
|
|
|
/// .and_then(|cookie| cookie.value().parse().ok())
|
|
|
|
/// .map(|id| User(id))
|
|
|
|
/// .or_forward(())
|
|
|
|
/// }
|
|
|
|
/// }
|
|
|
|
/// # fn main() { }
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// # Private Cookies
|
|
|
|
///
|
|
|
|
/// _Private_ cookies are just like regular cookies except that they are
|
|
|
|
/// encrypted using authenticated encryption, a form of encryption which
|
|
|
|
/// simultaneously provides confidentiality, integrity, and authenticity. This
|
|
|
|
/// means that private cookies cannot be inspected, tampered with, or
|
|
|
|
/// manufactured by clients. If you prefer, you can think of private cookies as
|
|
|
|
/// being signed and encrypted.
|
|
|
|
///
|
|
|
|
/// Private cookies can be retrieved, added, and removed from a `Cookies`
|
|
|
|
/// collection via the [`get_private`], [`add_private`], and [`remove_private`]
|
|
|
|
/// methods.
|
|
|
|
///
|
|
|
|
/// ## Encryption Key
|
|
|
|
///
|
|
|
|
/// To encrypt private cookies, Rocket uses the 256-bit key specified in the
|
|
|
|
/// `secret_key` configuration parameter. If one is not specified, Rocket will
|
|
|
|
/// automatically generate a fresh key. Note, however, that a private cookie can
|
|
|
|
/// only be decrypted with the same key with which it was encrypted. As such, it
|
|
|
|
/// is important to set a `secret_key` configuration parameter when using
|
|
|
|
/// private cookies so that cookies decrypt properly after an application
|
|
|
|
/// restart. Rocket will emit a warning if an application is run in production
|
|
|
|
/// mode without a configured `secret_key`.
|
|
|
|
///
|
|
|
|
/// Generating a string suitable for use as a `secret_key` configuration value
|
|
|
|
/// is usually done through tools like `openssl`. Using `openssl`, for instance,
|
|
|
|
/// a 256-bit base64 key can be generated with the command `openssl rand -base64
|
|
|
|
/// 32`.
|
2017-03-07 09:19:06 +00:00
|
|
|
pub enum Cookies<'a> {
|
2017-06-24 09:49:16 +00:00
|
|
|
#[doc(hidden)]
|
2017-06-06 20:41:04 +00:00
|
|
|
Jarred(RefMut<'a, CookieJar>, &'a Key),
|
2017-06-24 09:49:16 +00:00
|
|
|
#[doc(hidden)]
|
2017-03-07 09:19:06 +00:00
|
|
|
Empty(CookieJar)
|
|
|
|
}
|
|
|
|
|
2017-03-08 11:28:12 +00:00
|
|
|
impl<'a> Cookies<'a> {
|
2018-06-07 13:34:47 +00:00
|
|
|
/// WARNING: This is unstable! Do not use this method outside of Rocket!
|
2017-06-06 20:41:04 +00:00
|
|
|
#[inline]
|
2018-06-07 13:34:47 +00:00
|
|
|
#[doc(hidden)]
|
|
|
|
pub fn new(jar: RefMut<'a, CookieJar>, key: &'a Key) -> Cookies<'a> {
|
2017-06-06 20:41:04 +00:00
|
|
|
Cookies::Jarred(jar, key)
|
2017-03-07 09:19:06 +00:00
|
|
|
}
|
|
|
|
|
2018-06-07 13:34:47 +00:00
|
|
|
/// WARNING: This is unstable! Do not use this method outside of Rocket!
|
|
|
|
#[doc(hidden)]
|
|
|
|
#[inline(always)]
|
|
|
|
pub fn empty() -> Cookies<'static> {
|
2017-03-07 09:19:06 +00:00
|
|
|
Cookies::Empty(CookieJar::new())
|
|
|
|
}
|
|
|
|
|
2018-06-07 13:34:47 +00:00
|
|
|
/// WARNING: This is unstable! Do not use this method outside of Rocket!
|
|
|
|
#[doc(hidden)]
|
2017-03-08 11:28:12 +00:00
|
|
|
#[inline(always)]
|
2018-06-07 13:34:47 +00:00
|
|
|
pub fn parse_cookie(cookie_str: &str) -> Option<Cookie<'static>> {
|
2017-03-08 11:28:12 +00:00
|
|
|
Cookie::parse_encoded(cookie_str).map(|c| c.into_owned()).ok()
|
|
|
|
}
|
|
|
|
|
2017-07-04 21:51:00 +00:00
|
|
|
/// Adds an original `cookie` to this collection.
|
2018-06-07 13:34:47 +00:00
|
|
|
/// WARNING: This is unstable! Do not use this method outside of Rocket!
|
|
|
|
#[inline]
|
|
|
|
#[doc(hidden)]
|
|
|
|
pub fn add_original(&mut self, cookie: Cookie<'static>) {
|
2017-07-04 21:51:00 +00:00
|
|
|
if let Cookies::Jarred(ref mut jar, _) = *self {
|
|
|
|
jar.add_original(cookie)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-24 09:49:16 +00:00
|
|
|
/// Returns a reference to the `Cookie` inside this container with the name
|
|
|
|
/// `name`. If no such cookie exists, returns `None`.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```rust
|
2018-06-07 13:34:47 +00:00
|
|
|
/// # extern crate rocket;
|
2017-06-24 09:49:16 +00:00
|
|
|
/// use rocket::http::Cookies;
|
|
|
|
///
|
|
|
|
/// fn handler(cookies: Cookies) {
|
|
|
|
/// let cookie = cookies.get("name");
|
|
|
|
/// }
|
|
|
|
/// ```
|
2017-03-07 09:19:06 +00:00
|
|
|
pub fn get(&self, name: &str) -> Option<&Cookie<'static>> {
|
|
|
|
match *self {
|
2017-06-06 20:41:04 +00:00
|
|
|
Cookies::Jarred(ref jar, _) => jar.get(name),
|
|
|
|
Cookies::Empty(_) => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-24 09:49:16 +00:00
|
|
|
/// Returns a reference to the `Cookie` inside this collection with the name
|
|
|
|
/// `name` and authenticates and decrypts the cookie's value, returning a
|
|
|
|
/// `Cookie` with the decrypted value. If the cookie cannot be found, or the
|
|
|
|
/// cookie fails to authenticate or decrypt, `None` is returned.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```rust
|
2018-06-07 13:34:47 +00:00
|
|
|
/// # extern crate rocket;
|
2017-06-24 09:49:16 +00:00
|
|
|
/// use rocket::http::Cookies;
|
|
|
|
///
|
|
|
|
/// fn handler(mut cookies: Cookies) {
|
|
|
|
/// let cookie = cookies.get_private("name");
|
|
|
|
/// }
|
|
|
|
/// ```
|
2017-06-06 20:41:04 +00:00
|
|
|
pub fn get_private(&mut self, name: &str) -> Option<Cookie<'static>> {
|
|
|
|
match *self {
|
|
|
|
Cookies::Jarred(ref mut jar, key) => jar.private(key).get(name),
|
2017-03-07 09:19:06 +00:00
|
|
|
Cookies::Empty(_) => None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-24 09:49:16 +00:00
|
|
|
/// Adds `cookie` to this collection.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```rust
|
2018-06-07 13:34:47 +00:00
|
|
|
/// # extern crate rocket;
|
2017-06-24 09:49:16 +00:00
|
|
|
/// use rocket::http::{Cookie, Cookies};
|
|
|
|
///
|
|
|
|
/// fn handler(mut cookies: Cookies) {
|
|
|
|
/// cookies.add(Cookie::new("name", "value"));
|
|
|
|
///
|
|
|
|
/// let cookie = Cookie::build("name", "value")
|
|
|
|
/// .path("/")
|
|
|
|
/// .secure(true)
|
|
|
|
/// .finish();
|
|
|
|
///
|
|
|
|
/// cookies.add(cookie);
|
|
|
|
/// }
|
|
|
|
/// ```
|
2017-03-07 09:19:06 +00:00
|
|
|
pub fn add(&mut self, cookie: Cookie<'static>) {
|
2017-06-06 20:41:04 +00:00
|
|
|
if let Cookies::Jarred(ref mut jar, _) = *self {
|
2017-03-07 09:19:06 +00:00
|
|
|
jar.add(cookie)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-24 09:49:16 +00:00
|
|
|
/// Adds `cookie` to the collection. The cookie's value is encrypted with
|
|
|
|
/// authenticated encryption assuring confidentiality, integrity, and
|
|
|
|
/// authenticity. The cookie can later be retrieved using
|
|
|
|
/// [`get_private`](#method.get_private) and removed using
|
|
|
|
/// [`remove_private`](#method.remove_private).
|
|
|
|
///
|
2017-10-06 04:58:04 +00:00
|
|
|
/// Unless a value is supplied for the given key, the following defaults are
|
|
|
|
/// set on `cookie` before being added to `self`:
|
|
|
|
///
|
|
|
|
/// * `path`: `"/"`
|
|
|
|
/// * `SameSite`: `Strict`
|
|
|
|
/// * `HttpOnly`: `true`
|
|
|
|
/// * `Expires`: 1 week from now
|
|
|
|
///
|
|
|
|
/// These defaults ensure maximum usability and security. For additional
|
|
|
|
/// security, you may wish to set the `secure` flag.
|
2017-06-24 09:49:16 +00:00
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```rust
|
2018-06-07 13:34:47 +00:00
|
|
|
/// # extern crate rocket;
|
2017-06-24 09:49:16 +00:00
|
|
|
/// use rocket::http::{Cookie, Cookies};
|
|
|
|
///
|
|
|
|
/// fn handler(mut cookies: Cookies) {
|
|
|
|
/// cookies.add_private(Cookie::new("name", "value"));
|
|
|
|
/// }
|
|
|
|
/// ```
|
2017-06-06 20:41:04 +00:00
|
|
|
pub fn add_private(&mut self, mut cookie: Cookie<'static>) {
|
2017-06-24 09:49:16 +00:00
|
|
|
if let Cookies::Jarred(ref mut jar, key) = *self {
|
2017-11-30 17:30:31 +00:00
|
|
|
Cookies::set_private_defaults(&mut cookie);
|
|
|
|
jar.private(key).add(cookie)
|
|
|
|
}
|
|
|
|
}
|
2017-06-06 20:41:04 +00:00
|
|
|
|
2017-11-30 17:30:31 +00:00
|
|
|
/// Adds an original, private `cookie` to the collection.
|
2018-06-07 13:34:47 +00:00
|
|
|
/// WARNING: This is unstable! Do not use this method outside of Rocket!
|
|
|
|
#[doc(hidden)]
|
|
|
|
pub fn add_original_private(&mut self, mut cookie: Cookie<'static>) {
|
2017-11-30 17:30:31 +00:00
|
|
|
if let Cookies::Jarred(ref mut jar, key) = *self {
|
|
|
|
Cookies::set_private_defaults(&mut cookie);
|
|
|
|
jar.private(key).add_original(cookie)
|
|
|
|
}
|
|
|
|
}
|
2017-10-06 04:58:04 +00:00
|
|
|
|
2017-11-30 17:30:31 +00:00
|
|
|
/// For each property mentioned below, this method checks
|
2018-07-12 03:44:09 +00:00
|
|
|
/// if there is a provided value and if there is none, sets
|
|
|
|
/// a default value.
|
2017-11-30 17:30:31 +00:00
|
|
|
/// Default values are:
|
|
|
|
///
|
|
|
|
/// * `path`: `"/"`
|
|
|
|
/// * `SameSite`: `Strict`
|
|
|
|
/// * `HttpOnly`: `true`
|
|
|
|
/// * `Expires`: 1 week from now
|
|
|
|
///
|
|
|
|
fn set_private_defaults(cookie: &mut Cookie<'static>) {
|
|
|
|
if cookie.path().is_none() {
|
|
|
|
cookie.set_path("/");
|
|
|
|
}
|
2017-10-06 04:58:04 +00:00
|
|
|
|
2017-11-30 17:30:31 +00:00
|
|
|
if cookie.http_only().is_none() {
|
|
|
|
cookie.set_http_only(true);
|
|
|
|
}
|
2017-06-06 20:41:04 +00:00
|
|
|
|
2017-11-30 17:30:31 +00:00
|
|
|
if cookie.expires().is_none() {
|
|
|
|
cookie.set_expires(::time::now() + ::time::Duration::weeks(1));
|
|
|
|
}
|
|
|
|
|
|
|
|
if cookie.same_site().is_none() {
|
|
|
|
cookie.set_same_site(SameSite::Strict);
|
2017-06-06 20:41:04 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-24 09:49:16 +00:00
|
|
|
/// Removes `cookie` from this collection and generates a "removal" cookies
|
|
|
|
/// to send to the client on response. For correctness, `cookie` must
|
|
|
|
/// contain the same `path` and `domain` as the cookie that was initially
|
2018-07-12 03:44:09 +00:00
|
|
|
/// set. Failure to provide the initial `path` and `domain` will result in
|
2017-06-24 09:49:16 +00:00
|
|
|
/// cookies that are not properly removed.
|
|
|
|
///
|
|
|
|
/// A "removal" cookie is a cookie that has the same name as the original
|
|
|
|
/// cookie but has an empty value, a max-age of 0, and an expiration date
|
|
|
|
/// far in the past.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```rust
|
2018-06-07 13:34:47 +00:00
|
|
|
/// # extern crate rocket;
|
2017-06-24 09:49:16 +00:00
|
|
|
/// use rocket::http::{Cookie, Cookies};
|
|
|
|
///
|
|
|
|
/// fn handler(mut cookies: Cookies) {
|
|
|
|
/// cookies.remove(Cookie::named("name"));
|
|
|
|
/// }
|
|
|
|
/// ```
|
2017-03-07 09:19:06 +00:00
|
|
|
pub fn remove(&mut self, cookie: Cookie<'static>) {
|
2017-06-06 20:41:04 +00:00
|
|
|
if let Cookies::Jarred(ref mut jar, _) = *self {
|
2017-03-07 09:19:06 +00:00
|
|
|
jar.remove(cookie)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-24 09:49:16 +00:00
|
|
|
/// Removes the private `cookie` from the collection.
|
|
|
|
///
|
|
|
|
/// For correct removal, the passed in `cookie` must contain the same `path`
|
|
|
|
/// and `domain` as the cookie that was initially set. If a path is not set
|
|
|
|
/// on `cookie`, the `"/"` path will automatically be set.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```rust
|
2018-06-07 13:34:47 +00:00
|
|
|
/// # extern crate rocket;
|
2017-06-24 09:49:16 +00:00
|
|
|
/// use rocket::http::{Cookie, Cookies};
|
|
|
|
///
|
|
|
|
/// fn handler(mut cookies: Cookies) {
|
|
|
|
/// cookies.remove_private(Cookie::named("name"));
|
|
|
|
/// }
|
|
|
|
/// ```
|
2017-06-06 20:41:04 +00:00
|
|
|
pub fn remove_private(&mut self, mut cookie: Cookie<'static>) {
|
|
|
|
if let Cookies::Jarred(ref mut jar, key) = *self {
|
2017-06-24 09:49:16 +00:00
|
|
|
if cookie.path().is_none() {
|
|
|
|
cookie.set_path("/");
|
|
|
|
}
|
|
|
|
|
2017-06-06 20:41:04 +00:00
|
|
|
jar.private(key).remove(cookie)
|
2017-03-08 11:28:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-24 09:49:16 +00:00
|
|
|
/// Returns an iterator over all of the cookies present in this collection.
|
|
|
|
///
|
|
|
|
/// # Example
|
|
|
|
///
|
|
|
|
/// ```rust
|
2018-06-07 13:34:47 +00:00
|
|
|
/// # extern crate rocket;
|
2017-06-24 09:49:16 +00:00
|
|
|
/// use rocket::http::Cookies;
|
|
|
|
///
|
|
|
|
/// fn handler(cookies: Cookies) {
|
|
|
|
/// for c in cookies.iter() {
|
|
|
|
/// println!("Name: '{}', Value: '{}'", c.name(), c.value());
|
|
|
|
/// }
|
|
|
|
/// }
|
|
|
|
/// ```
|
2018-07-18 16:45:20 +00:00
|
|
|
pub fn iter(&self) -> impl Iterator<Item=&Cookie<'static>> {
|
2017-03-07 09:19:06 +00:00
|
|
|
match *self {
|
2017-06-06 20:41:04 +00:00
|
|
|
Cookies::Jarred(ref jar, _) => jar.iter(),
|
2017-03-07 09:19:06 +00:00
|
|
|
Cookies::Empty(ref jar) => jar.iter()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-07 13:34:47 +00:00
|
|
|
/// WARNING: This is unstable! Do not use this method outside of Rocket!
|
|
|
|
#[doc(hidden)]
|
|
|
|
#[inline]
|
|
|
|
pub fn delta(&self) -> Delta {
|
2017-03-07 09:19:06 +00:00
|
|
|
match *self {
|
2017-06-06 20:41:04 +00:00
|
|
|
Cookies::Jarred(ref jar, _) => jar.delta(),
|
2017-03-07 09:19:06 +00:00
|
|
|
Cookies::Empty(ref jar) => jar.delta()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-06-06 20:41:04 +00:00
|
|
|
|
|
|
|
impl<'a> fmt::Debug for Cookies<'a> {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
match *self {
|
|
|
|
Cookies::Jarred(ref jar, _) => write!(f, "{:?}", jar),
|
|
|
|
Cookies::Empty(ref jar) => write!(f, "{:?}", jar)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-14 23:24:41 +00:00
|
|
|
impl<'c> From<Cookie<'c>> for Header<'static> {
|
|
|
|
fn from(cookie: Cookie) -> Header<'static> {
|
|
|
|
Header::new("Set-Cookie", cookie.encoded().to_string())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-06 20:41:04 +00:00
|
|
|
impl<'a, 'c> From<&'a Cookie<'c>> for Header<'static> {
|
|
|
|
fn from(cookie: &Cookie) -> Header<'static> {
|
|
|
|
Header::new("Set-Cookie", cookie.encoded().to_string())
|
|
|
|
}
|
|
|
|
}
|