use {HeaderValue};
use super::IterExt;
#[derive(Clone, Debug, Eq, PartialEq)]
pub(crate) struct EntityTag(HeaderValue);
impl EntityTag {
pub fn from_static(bytes: &'static str) -> EntityTag {
let val = HeaderValue::from_static(bytes);
match EntityTag::from_val(&val) {
Some(tag) => tag,
None => {
panic!("invalid static string for EntityTag: {:?}", bytes);
}
}
}
pub fn tag(&self) -> &[u8] {
let bytes = self.0.as_bytes();
let end = bytes.len() - 1;
if bytes[0] == b'W' {
&bytes[3..end]
} else {
&bytes[1..end]
}
}
pub fn is_weak(&self) -> bool {
self.0.as_bytes()[0] == b'W'
}
pub fn strong_eq(&self, other: &EntityTag) -> bool {
!self.is_weak() && !other.is_weak() && self.tag() == other.tag()
}
pub fn weak_eq(&self, other: &EntityTag) -> bool {
self.tag() == other.tag()
}
pub fn strong_ne(&self, other: &EntityTag) -> bool {
!self.strong_eq(other)
}
pub fn weak_ne(&self, other: &EntityTag) -> bool {
!self.weak_eq(other)
}
pub(crate) fn from_val(val: &HeaderValue) -> Option<EntityTag> {
let slice = val.as_bytes();
let length = slice.len();
if length < 2 || slice[length - 1] != b'"' {
return None;
}
let start = match slice[0] {
b'"' => 1,
b'W' => {
if length >= 4 && slice[1] == b'/' && slice[2] == b'"' {
3
} else {
return None;
}
},
_ => return None,
};
if check_slice_validity(&slice[start..length-1]) {
Some(EntityTag(val.clone()))
} else {
None
}
}
}
impl super::TryFromValues for EntityTag {
fn try_from_values<'i, I>(values: &mut I) -> Result<Self, ::Error>
where
I: Iterator<Item = &'i HeaderValue>,
{
values
.just_one()
.and_then(EntityTag::from_val)
.ok_or_else(::Error::invalid)
}
}
impl From<EntityTag> for HeaderValue {
fn from(tag: EntityTag) -> HeaderValue {
tag.0
}
}
impl<'a> From<&'a EntityTag> for HeaderValue {
fn from(tag: &'a EntityTag) -> HeaderValue {
tag.0.clone()
}
}
fn check_slice_validity(slice: &[u8]) -> bool {
slice.iter().all(|&c| {
debug_assert!(
(c >= b'\x21' && c <= b'\x7e') | (c >= b'\x80'),
"EntityTag expects HeaderValue to have check for control characters"
);
c != b'"'
})
}
#[cfg(test)]
mod tests {
use super::*;
fn parse(slice: &[u8]) -> Option<EntityTag> {
let val = HeaderValue::from_bytes(slice).ok()?;
EntityTag::from_val(&val)
}
#[test]
fn test_etag_parse_success() {
let tag = parse(b"\"foobar\"").unwrap();
assert!(!tag.is_weak());
assert_eq!(tag.tag(), b"foobar");
let weak = parse(b"W/\"weaktag\"").unwrap();
assert!(weak.is_weak());
assert_eq!(weak.tag(), b"weaktag");
}
#[test]
fn test_etag_parse_failures() {
macro_rules! fails {
($slice:expr) => {
assert_eq!(parse($slice), None);
}
}
fails!(b"no-dquote");
fails!(b"w/\"the-first-w-is-case sensitive\"");
fails!(b"W/\"");
fails!(b"");
fails!(b"\"unmatched-dquotes1");
fails!(b"unmatched-dquotes2\"");
fails!(b"\"inner\"quotes\"");
}
#[test]
fn test_cmp() {
let mut etag1 = EntityTag::from_static("W/\"1\"");
let mut etag2 = etag1.clone();
assert!(!etag1.strong_eq(&etag2));
assert!(etag1.weak_eq(&etag2));
assert!(etag1.strong_ne(&etag2));
assert!(!etag1.weak_ne(&etag2));
etag2 = EntityTag::from_static("W/\"2\"");
assert!(!etag1.strong_eq(&etag2));
assert!(!etag1.weak_eq(&etag2));
assert!(etag1.strong_ne(&etag2));
assert!(etag1.weak_ne(&etag2));
etag2 = EntityTag::from_static("\"1\"");
assert!(!etag1.strong_eq(&etag2));
assert!(etag1.weak_eq(&etag2));
assert!(etag1.strong_ne(&etag2));
assert!(!etag1.weak_ne(&etag2));
etag1 = EntityTag::from_static("\"1\"");
assert!(etag1.strong_eq(&etag2));
assert!(etag1.weak_eq(&etag2));
assert!(!etag1.strong_ne(&etag2));
assert!(!etag1.weak_ne(&etag2));
}
}