From 13590547738e3d7ce57f2ba85d4ac97ef5fd1703 Mon Sep 17 00:00:00 2001 From: Yoshua Wuyts Date: Sat, 8 Aug 2020 15:40:34 +0200 Subject: [PATCH] Add conditional::LastModified --- src/conditional/if_modified_since.rs | 2 +- src/conditional/if_unmodified_since.rs | 2 +- src/conditional/last_modified.rs | 125 +++++++++++++++++++++++++ src/conditional/mod.rs | 2 + 4 files changed, 129 insertions(+), 2 deletions(-) create mode 100644 src/conditional/last_modified.rs diff --git a/src/conditional/if_modified_since.rs b/src/conditional/if_modified_since.rs index 8e358c57..60a1b373 100644 --- a/src/conditional/if_modified_since.rs +++ b/src/conditional/if_modified_since.rs @@ -45,7 +45,7 @@ impl IfModifiedSince { Self { instant } } - /// Get the expiration time. + /// Returns the last modification time listed. pub fn modified(&self) -> SystemTime { self.instant } diff --git a/src/conditional/if_unmodified_since.rs b/src/conditional/if_unmodified_since.rs index e2fa6a30..dca6cfe7 100644 --- a/src/conditional/if_unmodified_since.rs +++ b/src/conditional/if_unmodified_since.rs @@ -45,7 +45,7 @@ impl IfUnmodifiedSince { Self { instant } } - /// Get the expiration time. + /// Returns the last modification time listed. pub fn modified(&self) -> SystemTime { self.instant } diff --git a/src/conditional/last_modified.rs b/src/conditional/last_modified.rs new file mode 100644 index 00000000..110b7fa3 --- /dev/null +++ b/src/conditional/last_modified.rs @@ -0,0 +1,125 @@ +use crate::headers::{HeaderName, HeaderValue, Headers, ToHeaderValues, LAST_MODIFIED}; +use crate::utils::{fmt_http_date, parse_http_date}; + +use std::fmt::Debug; +use std::option; +use std::time::SystemTime; + +/// HTTP `LastModified` header +/// +/// # Specifications +/// +/// - [RFC 7232, section 2.2: Last-Modified](https://tools.ietf.org/html/rfc7232#section-2.2) +/// +/// # Examples +/// +/// ``` +/// # fn main() -> http_types::Result<()> { +/// # +/// use http_types::Response; +/// use http_types::conditional::LastModified; +/// use std::time::{SystemTime, Duration}; +/// +/// let time = SystemTime::now() + Duration::from_secs(5 * 60); +/// let last_modified = LastModified::new(time); +/// +/// let mut res = Response::new(200); +/// last_modified.apply(&mut res); +/// +/// let last_modified = LastModified::from_headers(res)?.unwrap(); +/// +/// // HTTP dates only have second-precision. +/// let elapsed = time.duration_since(last_modified.modified())?; +/// assert_eq!(elapsed.as_secs(), 0); +/// # +/// # Ok(()) } +/// ``` +#[derive(Debug, Ord, PartialOrd, Eq, PartialEq)] +pub struct LastModified { + instant: SystemTime, +} + +impl LastModified { + /// Create a new instance of `LastModified`. + pub fn new(instant: SystemTime) -> Self { + Self { instant } + } + + /// Returns the last modification time listed. + pub fn modified(&self) -> SystemTime { + self.instant + } + + /// Create an instance of `LastModified` from a `Headers` instance. + pub fn from_headers(headers: impl AsRef) -> crate::Result> { + let headers = match headers.as_ref().get(LAST_MODIFIED) { + Some(headers) => headers, + None => return Ok(None), + }; + + // If we successfully parsed the header then there's always at least one + // entry. We want the last entry. + let header = headers.iter().last().unwrap(); + + let instant = parse_http_date(header.as_str())?; + Ok(Some(Self { instant })) + } + + /// Insert a `HeaderName` + `HeaderValue` pair into a `Headers` instance. + pub fn apply(&self, mut headers: impl AsMut) { + headers.as_mut().insert(LAST_MODIFIED, self.value()); + } + + /// Get the `HeaderName`. + pub fn name(&self) -> HeaderName { + LAST_MODIFIED + } + + /// Get the `HeaderValue`. + pub fn value(&self) -> HeaderValue { + let output = fmt_http_date(self.instant); + + // SAFETY: the internal string is validated to be ASCII. + unsafe { HeaderValue::from_bytes_unchecked(output.into()) } + } +} + +impl ToHeaderValues for LastModified { + type Iter = option::IntoIter; + fn to_header_values(&self) -> crate::Result { + // A HeaderValue will always convert into itself. + Ok(self.value().to_header_values().unwrap()) + } +} + +#[cfg(test)] +mod test { + use super::*; + use crate::headers::Headers; + use std::time::Duration; + + #[test] + fn smoke() -> crate::Result<()> { + let time = SystemTime::now() + Duration::from_secs(5 * 60); + let last_modified = LastModified::new(time); + + let mut headers = Headers::new(); + last_modified.apply(&mut headers); + + let last_modified = LastModified::from_headers(headers)?.unwrap(); + + // HTTP dates only have second-precision + let elapsed = time.duration_since(last_modified.modified())?; + assert_eq!(elapsed.as_secs(), 0); + Ok(()) + } + + #[test] + fn bad_request_on_parse_error() -> crate::Result<()> { + let mut headers = Headers::new(); + headers.insert(LAST_MODIFIED, ""); + let err = LastModified::from_headers(headers).unwrap_err(); + assert_eq!(err.status(), 400); + Ok(()) + } +} diff --git a/src/conditional/mod.rs b/src/conditional/mod.rs index 68dd4fcd..348e307e 100644 --- a/src/conditional/mod.rs +++ b/src/conditional/mod.rs @@ -11,7 +11,9 @@ mod etag; mod if_modified_since; mod if_unmodified_since; +mod last_modified; pub use etag::ETag; pub use if_modified_since::IfModifiedSince; pub use if_unmodified_since::IfUnmodifiedSince; +pub use last_modified::LastModified;