use bytes::BytesMut;
use headers::Error;
use hyper::header::HeaderValue;
use std::cmp::Ordering;
use crate::headers_ext::ContentCoding;
#[derive(Clone, Debug, PartialEq, Eq, Hash)]
pub(crate) struct QualityValue {
value: HeaderValue,
}
#[derive(Clone, Debug, PartialEq, Eq)]
struct QualityMeta<'a> {
pub data: &'a str,
pub quality: u16,
}
impl Ord for QualityMeta<'_> {
fn cmp(&self, other: &Self) -> Ordering {
other.quality.cmp(&self.quality)
}
}
impl PartialOrd for QualityMeta<'_> {
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
Some(self.cmp(other))
}
}
impl<'a> TryFrom<&'a str> for QualityMeta<'a> {
type Error = Error;
fn try_from(val: &'a str) -> Result<Self, Error> {
let parts: Vec<_> = val.split(';').collect();
let data = parts.first().ok_or(Error::invalid())?.trim();
let mut quality = 1000u16;
for part in parts {
let part = part.trim_start();
if let Some(value) = part.strip_prefix("q=") {
let parsed: f32 = match value.trim().parse() {
Ok(parsed) => parsed,
Err(_) => continue,
};
quality = (parsed * 1000_f32) as u16;
}
}
Ok(QualityMeta { data, quality })
}
}
impl QualityValue {
pub(crate) fn iter(&self) -> impl Iterator<Item = &str> {
let mut items: Vec<_> = self
.value
.to_str()
.ok()
.into_iter()
.flat_map(|value_str| value_str.split(','))
.filter_map(|v| QualityMeta::try_from(v).ok())
.collect();
items.sort_by(|a, b| {
let quality_cmp = b.quality.cmp(&a.quality);
if quality_cmp == std::cmp::Ordering::Equal {
let priority_a = ContentCoding::from(a.data).priority();
let priority_b = ContentCoding::from(b.data).priority();
priority_b.cmp(&priority_a)
} else {
quality_cmp
}
});
items.into_iter().map(|pair| pair.data)
}
pub(crate) fn try_from_values<'i, I>(values: &mut I) -> Result<Self, Error>
where
I: Iterator<Item = &'i HeaderValue>,
{
let mut values = values.peekable();
let first = match values.next() {
Some(item) => item,
None => return Ok(HeaderValue::from_static("").into()),
};
if values.peek().is_none() {
return Ok(first.into());
}
let mut buf = BytesMut::from(first.as_bytes());
for value in values {
buf.extend_from_slice(", ".as_bytes());
buf.extend_from_slice(value.as_bytes());
}
Ok(HeaderValue::from_maybe_shared(buf.freeze())
.map_err(|_| Error::invalid())?
.into())
}
}
impl<F: Into<f32>> TryFrom<(&str, F)> for QualityValue {
type Error = Error;
fn try_from(pair: (&str, F)) -> Result<Self, Error> {
let (data, quality) = pair;
let value = HeaderValue::try_from(format!("{data};q={}", quality.into()))
.map_err(|_e| Error::invalid())?;
Ok(Self::from(value))
}
}
impl From<HeaderValue> for QualityValue {
fn from(value: HeaderValue) -> Self {
QualityValue { value }
}
}
impl From<&HeaderValue> for QualityValue {
fn from(value: &HeaderValue) -> Self {
QualityValue {
value: value.clone(),
}
}
}
impl From<&QualityValue> for HeaderValue {
fn from(qual: &QualityValue) -> Self {
qual.value.clone()
}
}
impl From<QualityValue> for HeaderValue {
fn from(qual: QualityValue) -> Self {
qual.value
}
}
#[cfg(test)]
mod tests {
use super::*;
fn without_qualities() {
let val = HeaderValue::from_static("zstd, br, gzip, deflate");
let qual = QualityValue::from(val);
let mut values = qual.iter();
assert_eq!(values.next(), Some("zstd"));
assert_eq!(values.next(), Some("br"));
assert_eq!(values.next(), Some("gzip"));
assert_eq!(values.next(), Some("deflate"));
assert_eq!(values.next(), None);
}
#[test]
fn without_qualities_wrong_order() {
let val = HeaderValue::from_static("gzip, deflate, br, zstd");
let qual = QualityValue::from(val);
let mut values = qual.iter();
assert_eq!(values.next(), Some("zstd"));
assert_eq!(values.next(), Some("br"));
assert_eq!(values.next(), Some("gzip"));
assert_eq!(values.next(), Some("deflate"));
assert_eq!(values.next(), None);
}
#[test]
fn honor_client_preference_order() {
let val = HeaderValue::from_static("gzip, br;q=0.5, zstd;q=0");
let qual = QualityValue::from(val);
let mut values = qual.iter();
assert_eq!(values.next(), Some("gzip"));
assert_eq!(values.next(), Some("br"));
assert_eq!(values.next(), Some("zstd"));
assert_eq!(values.next(), None);
}
#[test]
fn multiple_qualities() {
let val = HeaderValue::from_static("gzip;q=1, br;q=0.8");
let qual = QualityValue::from(val);
let mut values = qual.iter();
assert_eq!(values.next(), Some("gzip"));
assert_eq!(values.next(), Some("br"));
assert_eq!(values.next(), None);
}
#[test]
fn multiple_qualities_wrong_order() {
let val = HeaderValue::from_static("br;q=0.8, gzip;q=1.0");
let qual = QualityValue::from(val);
let mut values = qual.iter();
assert_eq!(values.next(), Some("gzip"));
assert_eq!(values.next(), Some("br"));
assert_eq!(values.next(), None);
}
#[test]
fn multiple_values() {
let val = HeaderValue::from_static("deflate, gzip;q=1, br;q=0.8");
let qual = QualityValue::from(val);
let mut values = qual.iter();
assert_eq!(values.next(), Some("gzip"));
assert_eq!(values.next(), Some("deflate"));
assert_eq!(values.next(), Some("br"));
assert_eq!(values.next(), None);
}
#[test]
fn multiple_values_whitespace() {
let val = HeaderValue::from_static("deflate ;q=0.9, br; q=0.001 ,gzip ; q=0.8");
let qual = QualityValue::from(val);
let mut values = qual.iter();
assert_eq!(values.next(), Some("deflate"));
assert_eq!(values.next(), Some("gzip"));
assert_eq!(values.next(), Some("br"));
assert_eq!(values.next(), None);
}
#[test]
fn multiple_values_wrong_order() {
let val = HeaderValue::from_static("deflate, br;q=0.8, gzip;q=1, *;q=0.1");
let qual = QualityValue::from(val);
let mut values = qual.iter();
assert_eq!(values.next(), Some("gzip"));
assert_eq!(values.next(), Some("deflate"));
assert_eq!(values.next(), Some("br"));
assert_eq!(values.next(), Some("*"));
assert_eq!(values.next(), None);
}
}