rama_http_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::<rama_http_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
49rama_utils::macros::error::static_str_error! {
50 #[doc = "etag is not valid"]
51 pub struct InvalidETag;
52}
53
54impl FromStr for ETag {
55 type Err = InvalidETag;
56 fn from_str(src: &str) -> Result<Self, Self::Err> {
57 let val = src.parse().map_err(|_| InvalidETag)?;
58
59 EntityTag::from_owned(val).map(ETag).ok_or(InvalidETag)
60 }
61}
62
63/*
64test_etag {
65 // From the RFC
66 test_header!(test1,
67 vec![b"\"xyzzy\""],
68 Some(ETag(EntityTag::new(false, "xyzzy".to_owned()))));
69 test_header!(test2,
70 vec![b"W/\"xyzzy\""],
71 Some(ETag(EntityTag::new(true, "xyzzy".to_owned()))));
72 test_header!(test3,
73 vec![b"\"\""],
74 Some(ETag(EntityTag::new(false, "".to_owned()))));
75 // Own tests
76 test_header!(test4,
77 vec![b"\"foobar\""],
78 Some(ETag(EntityTag::new(false, "foobar".to_owned()))));
79 test_header!(test5,
80 vec![b"\"\""],
81 Some(ETag(EntityTag::new(false, "".to_owned()))));
82 test_header!(test6,
83 vec![b"W/\"weak-etag\""],
84 Some(ETag(EntityTag::new(true, "weak-etag".to_owned()))));
85 test_header!(test7,
86 vec![b"W/\"\x65\x62\""],
87 Some(ETag(EntityTag::new(true, "\u{0065}\u{0062}".to_owned()))));
88 test_header!(test8,
89 vec![b"W/\"\""],
90 Some(ETag(EntityTag::new(true, "".to_owned()))));
91 test_header!(test9,
92 vec![b"no-dquotes"],
93 None::<ETag>);
94 test_header!(test10,
95 vec![b"w/\"the-first-w-is-case-sensitive\""],
96 None::<ETag>);
97 test_header!(test11,
98 vec![b""],
99 None::<ETag>);
100 test_header!(test12,
101 vec![b"\"unmatched-dquotes1"],
102 None::<ETag>);
103 test_header!(test13,
104 vec![b"unmatched-dquotes2\""],
105 None::<ETag>);
106 test_header!(test14,
107 vec![b"matched-\"dquotes\""],
108 None::<ETag>);
109 test_header!(test15,
110 vec![b"\""],
111 None::<ETag>);
112}
113*/