Rename 'Flash' 'name', 'msg' to 'kind', 'message'.

This is now consistent with the serialization of 'Flash' and better
illustrates the purpose of the late 'name' property.

Additionally, take 'Into<String>' instead of 'AsRef<str>' so we only
allocate when necessary.
This commit is contained in:
Sergio Benitez 2021-04-07 19:38:34 -07:00
parent 764ea06e90
commit 8d28e845d9
2 changed files with 50 additions and 47 deletions

View File

@ -1,5 +1,3 @@
use std::convert::AsRef;
use time::Duration; use time::Duration;
use serde::ser::{Serialize, Serializer, SerializeStruct}; use serde::ser::{Serialize, Serializer, SerializeStruct};
@ -27,14 +25,14 @@ const FLASH_COOKIE_DELIM: char = ':';
/// ///
/// # Usage /// # Usage
/// ///
/// Each `Flash` message consists of a `name` and some `msg` contents. A generic /// Each `Flash` message consists of a `kind` and `message`. A generic
/// constructor ([new](#method.new)) can be used to construct a message with any /// constructor ([new](#method.new)) can be used to construct a message of any
/// name, while the [warning](#method.warning), [success](#method.success), and /// kind, while the [warning](#method.warning), [success](#method.success), and
/// [error](#method.error) constructors create messages with the corresponding /// [error](#method.error) constructors create messages with the corresponding
/// names. /// kinds.
/// ///
/// Messages can be retrieved on the request side via the [`FlashMessage`] type /// Messages can be retrieved on the request side via the [`FlashMessage`] type
/// and the [name](#method.name) and [msg](#method.msg) methods. /// and the [kind](#method.kind) and [message](#method.message) methods.
/// ///
/// # Response /// # Response
/// ///
@ -58,13 +56,13 @@ const FLASH_COOKIE_DELIM: char = ':';
/// if name == "special_user" { /// if name == "special_user" {
/// Ok("Hello, special user!") /// Ok("Hello, special user!")
/// } else { /// } else {
/// Err(Flash::error(Redirect::to("/"), "Invalid username.")) /// Err(Flash::error(Redirect::to(uri!(index)), "Invalid username."))
/// } /// }
/// } /// }
/// ///
/// #[get("/")] /// #[get("/")]
/// fn index(flash: Option<FlashMessage>) -> String { /// fn index(flash: Option<FlashMessage<'_>>) -> String {
/// flash.map(|msg| format!("{}: {}", msg.name(), msg.msg())) /// flash.map(|flash| format!("{}: {}", flash.kind(), flash.message()))
/// .unwrap_or_else(|| "Welcome!".to_string()) /// .unwrap_or_else(|| "Welcome!".to_string())
/// } /// }
/// ``` /// ```
@ -77,8 +75,8 @@ const FLASH_COOKIE_DELIM: char = ':';
/// receive the standard welcome message. /// receive the standard welcome message.
#[derive(Debug)] #[derive(Debug)]
pub struct Flash<R> { pub struct Flash<R> {
pub name: String, kind: String,
pub message: String, message: String,
consumed: AtomicBool, consumed: AtomicBool,
inner: R, inner: R,
} }
@ -91,15 +89,15 @@ pub struct Flash<R> {
/// there is a flash cookie present (set by the `Flash` `Responder`), a /// there is a flash cookie present (set by the `Flash` `Responder`), a
/// `FlashMessage` request guard will succeed. /// `FlashMessage` request guard will succeed.
/// ///
/// The flash cookie is cleared if either the [`name()`] or [`msg()`] method is /// The flash cookie is cleared if either the [`kind()`] or [`message()`] method is
/// called. If neither method is called, the flash cookie is not cleared. /// called. If neither method is called, the flash cookie is not cleared.
/// ///
/// [`name()`]: Flash::name() /// [`kind()`]: Flash::kind()
/// [`msg()`]: Flash::msg() /// [`message()`]: Flash::message()
pub type FlashMessage<'a> = crate::response::Flash<&'a CookieJar<'a>>; pub type FlashMessage<'a> = crate::response::Flash<&'a CookieJar<'a>>;
impl<R> Flash<R> { impl<R> Flash<R> {
/// Constructs a new `Flash` message with the given `name`, `msg`, and /// Constructs a new `Flash` message with the given `kind`, `message`, and
/// underlying `responder`. /// underlying `responder`.
/// ///
/// # Examples /// # Examples
@ -111,19 +109,19 @@ impl<R> Flash<R> {
/// use rocket::response::{Redirect, Flash}; /// use rocket::response::{Redirect, Flash};
/// ///
/// # #[allow(unused_variables)] /// # #[allow(unused_variables)]
/// let msg = Flash::new(Redirect::to("/"), "suggestion", "Try this out!"); /// let message = Flash::new(Redirect::to("/"), "suggestion", "Try this out!");
/// ``` /// ```
pub fn new<N: AsRef<str>, M: AsRef<str>>(res: R, name: N, msg: M) -> Flash<R> { pub fn new<K: Into<String>, M: Into<String>>(res: R, kind: K, message: M) -> Flash<R> {
Flash { Flash {
name: name.as_ref().to_string(), kind: kind.into(),
message: msg.as_ref().to_string(), message: message.into(),
consumed: AtomicBool::default(), consumed: AtomicBool::default(),
inner: res, inner: res,
} }
} }
/// Constructs a "success" `Flash` message with the given `responder` and /// Constructs a "success" `Flash` message with the given `responder` and
/// `msg`. /// `message`.
/// ///
/// # Examples /// # Examples
/// ///
@ -134,14 +132,14 @@ impl<R> Flash<R> {
/// use rocket::response::{Redirect, Flash}; /// use rocket::response::{Redirect, Flash};
/// ///
/// # #[allow(unused_variables)] /// # #[allow(unused_variables)]
/// let msg = Flash::success(Redirect::to("/"), "It worked!"); /// let message = Flash::success(Redirect::to("/"), "It worked!");
/// ``` /// ```
pub fn success<S: AsRef<str>>(responder: R, msg: S) -> Flash<R> { pub fn success<S: Into<String>>(responder: R, message: S) -> Flash<R> {
Flash::new(responder, "success", msg) Flash::new(responder, "success", message.into())
} }
/// Constructs a "warning" `Flash` message with the given `responder` and /// Constructs a "warning" `Flash` message with the given `responder` and
/// `msg`. /// `message`.
/// ///
/// # Examples /// # Examples
/// ///
@ -152,14 +150,14 @@ impl<R> Flash<R> {
/// use rocket::response::{Redirect, Flash}; /// use rocket::response::{Redirect, Flash};
/// ///
/// # #[allow(unused_variables)] /// # #[allow(unused_variables)]
/// let msg = Flash::warning(Redirect::to("/"), "Watch out!"); /// let message = Flash::warning(Redirect::to("/"), "Watch out!");
/// ``` /// ```
pub fn warning<S: AsRef<str>>(responder: R, msg: S) -> Flash<R> { pub fn warning<S: Into<String>>(responder: R, message: S) -> Flash<R> {
Flash::new(responder, "warning", msg) Flash::new(responder, "warning", message.into())
} }
/// Constructs an "error" `Flash` message with the given `responder` and /// Constructs an "error" `Flash` message with the given `responder` and
/// `msg`. /// `message`.
/// ///
/// # Examples /// # Examples
/// ///
@ -170,15 +168,15 @@ impl<R> Flash<R> {
/// use rocket::response::{Redirect, Flash}; /// use rocket::response::{Redirect, Flash};
/// ///
/// # #[allow(unused_variables)] /// # #[allow(unused_variables)]
/// let msg = Flash::error(Redirect::to("/"), "Whoops!"); /// let message = Flash::error(Redirect::to("/"), "Whoops!");
/// ``` /// ```
pub fn error<S: AsRef<str>>(responder: R, msg: S) -> Flash<R> { pub fn error<S: Into<String>>(responder: R, message: S) -> Flash<R> {
Flash::new(responder, "error", msg) Flash::new(responder, "error", message.into())
} }
fn cookie(&self) -> Cookie<'static> { fn cookie(&self) -> Cookie<'static> {
let content = format!("{}{}{}{}", let content = format!("{}{}{}{}",
self.name.len(), FLASH_COOKIE_DELIM, self.name, self.message); self.kind.len(), FLASH_COOKIE_DELIM, self.kind, self.message);
Cookie::build(FLASH_COOKIE_NAME, content) Cookie::build(FLASH_COOKIE_NAME, content)
.max_age(Duration::minutes(5)) .max_age(Duration::minutes(5))
@ -192,19 +190,18 @@ impl<R> Flash<R> {
/// the response is the `Outcome` of the wrapped `Responder`. /// the response is the `Outcome` of the wrapped `Responder`.
impl<'r, 'o: 'r, R: Responder<'r, 'o>> Responder<'r, 'o> for Flash<R> { impl<'r, 'o: 'r, R: Responder<'r, 'o>> Responder<'r, 'o> for Flash<R> {
fn respond_to(self, req: &'r Request<'_>) -> response::Result<'o> { fn respond_to(self, req: &'r Request<'_>) -> response::Result<'o> {
trace_!("Flash: setting message: {}:{}", self.name, self.message);
req.cookies().add(self.cookie()); req.cookies().add(self.cookie());
self.inner.respond_to(req) self.inner.respond_to(req)
} }
} }
impl<'a> FlashMessage<'a> { impl<'r> FlashMessage<'r> {
/// Constructs a new message with the given name and message for the given /// Constructs a new message with the given name and message for the given
/// request. /// request.
fn named<'r: 'a>(name: &str, msg: &str, req: &'a Request<'r>) -> FlashMessage<'a> { fn named<S: Into<String>>(kind: S, message: S, req: &'r Request<'_>) -> Self {
Flash { Flash {
name: name.to_string(), kind: kind.into(),
message: msg.to_string(), message: message.into(),
consumed: AtomicBool::new(false), consumed: AtomicBool::new(false),
inner: req.cookies(), inner: req.cookies(),
} }
@ -218,14 +215,20 @@ impl<'a> FlashMessage<'a> {
} }
} }
/// Returns the `name` of this message. /// Returns a tuple of `(kind, message)`, consuming `self`.
pub fn name(&self) -> &str { pub fn into_inner(self) -> (String, String) {
self.clear_cookie_if_needed(); self.clear_cookie_if_needed();
&self.name (self.kind, self.message)
} }
/// Returns the `msg` contents of this message. /// Returns the `kind` of this message.
pub fn msg(&self) -> &str { pub fn kind(&self) -> &str {
self.clear_cookie_if_needed();
&self.kind
}
/// Returns the `message` contents of this message.
pub fn message(&self) -> &str {
self.clear_cookie_if_needed(); self.clear_cookie_if_needed();
&self.message &self.message
} }
@ -260,11 +263,11 @@ impl<'r> FromRequest<'r> for FlashMessage<'r> {
} }
} }
impl<R> Serialize for Flash<R> { impl Serialize for FlashMessage<'_> {
fn serialize<S: Serializer>(&self, ser: S) -> Result<S::Ok, S::Error> { fn serialize<S: Serializer>(&self, ser: S) -> Result<S::Ok, S::Error> {
let mut flash = ser.serialize_struct("Flash", 2)?; let mut flash = ser.serialize_struct("Flash", 2)?;
flash.serialize_field("name", &self.name)?; flash.serialize_field("kind", self.kind())?;
flash.serialize_field("message", &self.message)?; flash.serialize_field("message", self.message())?;
flash.end() flash.end()
} }
} }

View File

@ -17,7 +17,7 @@ fn unused(flash: Option<FlashMessage<'_>>) -> Option<()> {
#[get("/use")] #[get("/use")]
fn used(flash: Option<FlashMessage<'_>>) -> Option<String> { fn used(flash: Option<FlashMessage<'_>>) -> Option<String> {
flash.map(|flash| flash.msg().into()) flash.map(|f| f.message().into())
} }
mod flash_lazy_remove_tests { mod flash_lazy_remove_tests {