use std::fmt;
use std::ops::{Bound, RangeBounds};
use http::{HeaderName, HeaderValue};
use crate::{util, Error, Header};
#[derive(Clone, Debug, PartialEq)]
pub struct ContentRange {
range: Option<(u64, u64)>,
complete_length: Option<u64>,
}
error_type!(InvalidContentRange);
impl ContentRange {
pub fn bytes(
range: impl RangeBounds<u64>,
complete_length: impl Into<Option<u64>>,
) -> Result<ContentRange, InvalidContentRange> {
let complete_length = complete_length.into();
let start = match range.start_bound() {
Bound::Included(&s) => s,
Bound::Excluded(&s) => s + 1,
Bound::Unbounded => 0,
};
let end = match range.end_bound() {
Bound::Included(&e) => e,
Bound::Excluded(&e) => e - 1,
Bound::Unbounded => match complete_length {
Some(max) => max - 1,
None => return Err(InvalidContentRange { _inner: () }),
},
};
Ok(ContentRange {
range: Some((start, end)),
complete_length,
})
}
pub fn unsatisfied_bytes(complete_length: u64) -> Self {
ContentRange {
range: None,
complete_length: Some(complete_length),
}
}
pub fn bytes_range(&self) -> Option<(u64, u64)> {
self.range
}
pub fn bytes_len(&self) -> Option<u64> {
self.complete_length
}
}
impl Header for ContentRange {
fn name() -> &'static HeaderName {
&::http::header::CONTENT_RANGE
}
fn decode<'i, I: Iterator<Item = &'i HeaderValue>>(values: &mut I) -> Result<Self, Error> {
values
.next()
.and_then(|v| v.to_str().ok())
.and_then(|s| split_in_two(s, ' '))
.and_then(|(unit, spec)| {
if unit != "bytes" {
return None;
}
let (range, complete_length) = split_in_two(spec, '/')?;
let complete_length = if complete_length == "*" {
None
} else {
Some(complete_length.parse().ok()?)
};
let range = if range == "*" {
None
} else {
let (first_byte, last_byte) = split_in_two(range, '-')?;
let first_byte = first_byte.parse().ok()?;
let last_byte = last_byte.parse().ok()?;
if last_byte < first_byte {
return None;
}
Some((first_byte, last_byte))
};
Some(ContentRange {
range,
complete_length,
})
})
.ok_or_else(Error::invalid)
}
fn encode<E: Extend<HeaderValue>>(&self, values: &mut E) {
struct Adapter<'a>(&'a ContentRange);
impl fmt::Display for Adapter<'_> {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.write_str("bytes ")?;
if let Some((first_byte, last_byte)) = self.0.range {
write!(f, "{}-{}", first_byte, last_byte)?;
} else {
f.write_str("*")?;
}
f.write_str("/")?;
if let Some(v) = self.0.complete_length {
write!(f, "{}", v)
} else {
f.write_str("*")
}
}
}
values.extend(::std::iter::once(util::fmt(Adapter(self))));
}
}
fn split_in_two(s: &str, separator: char) -> Option<(&str, &str)> {
let mut iter = s.splitn(2, separator);
match (iter.next(), iter.next()) {
(Some(a), Some(b)) => Some((a, b)),
_ => None,
}
}