headers/common/etag.rs
1use std::str::FromStr;
2
3use crate::util::EntityTag;
4
5/// `ETag` header, defined in [RFC7232](https://datatracker.ietf.org/doc/html/rfc7232#section-2.3)
6///
7/// The `ETag` header field in a response provides the current entity-tag
8/// for the selected representation, as determined at the conclusion of
9/// handling the request. An entity-tag is an opaque validator for
10/// differentiating between multiple representations of the same
11/// resource, regardless of whether those multiple representations are
12/// due to resource state changes over time, content negotiation
13/// resulting in multiple representations being valid at the same time,
14/// or both. An entity-tag consists of an opaque quoted string, possibly
15/// prefixed by a weakness indicator.
16///
17/// # ABNF
18///
19/// ```text
20/// ETag = entity-tag
21/// ```
22///
23/// # Example values
24///
25/// * `"xyzzy"`
26/// * `W/"xyzzy"`
27/// * `""`
28///
29/// # Examples
30///
31/// ```
32/// let etag = "\"xyzzy\"".parse::<headers::ETag>().unwrap();
33/// ```
34#[derive(Clone, Debug, PartialEq, Eq)]
35pub struct ETag(pub(super) EntityTag);
36
37derive_header! {
38 ETag(_),
39 name: ETAG
40}
41
42impl ETag {
43 #[cfg(test)]
44 pub(crate) fn from_static(src: &'static str) -> ETag {
45 ETag(EntityTag::from_static(src))
46 }
47}
48
49error_type!(InvalidETag);
50
51impl FromStr for ETag {
52 type Err = InvalidETag;
53 fn from_str(src: &str) -> Result<Self, Self::Err> {
54 let val = src.parse().map_err(|_| InvalidETag { _inner: () })?;
55
56 EntityTag::from_owned(val)
57 .map(ETag)
58 .ok_or(InvalidETag { _inner: () })
59 }
60}
61
62/*
63test_etag {
64 // From the RFC
65 test_header!(test1,
66 vec![b"\"xyzzy\""],
67 Some(ETag(EntityTag::new(false, "xyzzy".to_owned()))));
68 test_header!(test2,
69 vec![b"W/\"xyzzy\""],
70 Some(ETag(EntityTag::new(true, "xyzzy".to_owned()))));
71 test_header!(test3,
72 vec![b"\"\""],
73 Some(ETag(EntityTag::new(false, "".to_owned()))));
74 // Own tests
75 test_header!(test4,
76 vec![b"\"foobar\""],
77 Some(ETag(EntityTag::new(false, "foobar".to_owned()))));
78 test_header!(test5,
79 vec![b"\"\""],
80 Some(ETag(EntityTag::new(false, "".to_owned()))));
81 test_header!(test6,
82 vec![b"W/\"weak-etag\""],
83 Some(ETag(EntityTag::new(true, "weak-etag".to_owned()))));
84 test_header!(test7,
85 vec![b"W/\"\x65\x62\""],
86 Some(ETag(EntityTag::new(true, "\u{0065}\u{0062}".to_owned()))));
87 test_header!(test8,
88 vec![b"W/\"\""],
89 Some(ETag(EntityTag::new(true, "".to_owned()))));
90 test_header!(test9,
91 vec![b"no-dquotes"],
92 None::<ETag>);
93 test_header!(test10,
94 vec![b"w/\"the-first-w-is-case-sensitive\""],
95 None::<ETag>);
96 test_header!(test11,
97 vec![b""],
98 None::<ETag>);
99 test_header!(test12,
100 vec![b"\"unmatched-dquotes1"],
101 None::<ETag>);
102 test_header!(test13,
103 vec![b"unmatched-dquotes2\""],
104 None::<ETag>);
105 test_header!(test14,
106 vec![b"matched-\"dquotes\""],
107 None::<ETag>);
108 test_header!(test15,
109 vec![b"\""],
110 None::<ETag>);
111}
112*/