use std::str::FromStr;
use time::{Date, format_description::well_known::Rfc2822};
use crate::reset_time::{ResetTime, ResetTimeKind};
use crate::error::{Error, Result};
#[derive(Copy, Clone, Debug, PartialEq)]
pub struct RateLimit {
pub reset: ResetTime,
}
impl RateLimit {
#[cfg(feature = "http")]
pub fn new(headers: &http::HeaderMap) -> std::result::Result<Self, Error> {
Self::extract(crate::convert::header_map_str_pairs(headers))
}
pub fn extract<'a, I>(headers: I) -> std::result::Result<Self, Error>
where
I: IntoIterator<Item = (&'a str, &'a str)>,
{
let mut retry_after_val = None;
for (k, v) in headers {
if k.eq_ignore_ascii_case("retry-after") {
retry_after_val = Some(v);
break;
}
}
let reset = match retry_after_val {
Some(retry_after_str) => {
if Date::parse(retry_after_str, &Rfc2822).is_ok() {
ResetTime::new(retry_after_str, ResetTimeKind::ImfFixdate)?
} else {
ResetTime::new(retry_after_str, ResetTimeKind::Seconds)?
}
}
None => return Err(Error::MissingRetryAfter),
};
Ok(RateLimit { reset })
}
#[must_use]
pub const fn reset(&self) -> ResetTime {
self.reset
}
}
impl FromStr for RateLimit {
type Err = Error;
fn from_str(map: &str) -> Result<Self> {
RateLimit::extract(crate::convert::parse_header_lines(map))
}
}
#[cfg(test)]
mod tests {
use super::*;
use indoc::indoc;
use time::macros::datetime;
#[test]
fn retry_after_seconds() {
let headers = indoc! {"
Retry-After: 19
"};
let rate = RateLimit::from_str(headers).unwrap();
assert_eq!(rate.reset(), ResetTime::Seconds(19));
}
#[test]
fn retry_after_seconds_case_sensitive() {
let headers = indoc! {"
retry-after: 19
"};
let rate = RateLimit::from_str(headers).unwrap();
assert_eq!(rate.reset(), ResetTime::Seconds(19));
}
#[test]
fn retry_after_imf_fixdate() {
let headers = indoc! {"
Retry-After: Fri, 31 Dec 1999 23:59:59 GMT
"};
let rate = RateLimit::from_str(headers).unwrap();
assert_eq!(
rate.reset(),
ResetTime::DateTime(datetime!(1999-12-31 23:59:59 UTC))
);
}
}