Skip to content

Commit db44a7c

Browse files
committed
introduce HeaderValue type to replace use of String
Signed-off-by: Yaroslav Skopets <[email protected]>
1 parent aa67a50 commit db44a7c

File tree

1 file changed

+261
-0
lines changed

1 file changed

+261
-0
lines changed

src/types.rs

+261
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,9 @@
1212
// See the License for the specific language governing permissions and
1313
// limitations under the License.
1414

15+
use std::fmt;
16+
use std::hash::{Hash, Hasher};
17+
1518
use crate::traits::*;
1619

1720
pub type NewRootContext = fn(context_id: u32) -> Box<dyn RootContext>;
@@ -77,3 +80,261 @@ pub enum PeerType {
7780
}
7881

7982
pub type Bytes = Vec<u8>;
83+
84+
/// Represents an HTTP header value that is not necessarily a UTF-8 encoded string.
85+
#[derive(Eq)]
86+
pub struct HeaderValue {
87+
inner: Result<String, Bytes>,
88+
}
89+
90+
impl HeaderValue {
91+
fn new(inner: Result<String, Bytes>) -> Self {
92+
HeaderValue { inner }
93+
}
94+
95+
pub fn into_vec(self) -> Vec<u8> {
96+
match self.inner {
97+
Ok(string) => string.into_bytes(),
98+
Err(bytes) => bytes,
99+
}
100+
}
101+
102+
pub fn into_string_or_vec(self) -> Result<String, Vec<u8>> {
103+
self.inner
104+
}
105+
}
106+
107+
impl From<Vec<u8>> for HeaderValue {
108+
#[inline]
109+
fn from(data: Vec<u8>) -> Self {
110+
Self::new(match String::from_utf8(data) {
111+
Ok(string) => Ok(string),
112+
Err(err) => Err(err.into_bytes()),
113+
})
114+
}
115+
}
116+
117+
impl From<&[u8]> for HeaderValue {
118+
fn from(data: &[u8]) -> Self {
119+
data.to_owned().into()
120+
}
121+
}
122+
123+
impl From<String> for HeaderValue {
124+
#[inline]
125+
fn from(string: String) -> Self {
126+
Self::new(Ok(string))
127+
}
128+
}
129+
130+
impl From<&str> for HeaderValue {
131+
fn from(data: &str) -> Self {
132+
data.to_owned().into()
133+
}
134+
}
135+
136+
impl fmt::Display for HeaderValue {
137+
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
138+
match self.inner {
139+
Ok(ref string) => fmt::Display::fmt(string, f),
140+
Err(ref bytes) => fmt::Debug::fmt(bytes, f),
141+
}
142+
}
143+
}
144+
145+
impl fmt::Debug for HeaderValue {
146+
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
147+
match self.inner {
148+
Ok(ref string) => fmt::Debug::fmt(string, f),
149+
Err(ref bytes) => fmt::Debug::fmt(bytes, f),
150+
}
151+
}
152+
}
153+
154+
impl AsRef<[u8]> for HeaderValue {
155+
#[inline]
156+
fn as_ref(&self) -> &[u8] {
157+
match self.inner {
158+
Ok(ref string) => string.as_bytes(),
159+
Err(ref bytes) => bytes.as_slice(),
160+
}
161+
}
162+
}
163+
164+
impl PartialEq for HeaderValue {
165+
#[inline]
166+
fn eq(&self, other: &HeaderValue) -> bool {
167+
self.inner == other.inner
168+
}
169+
}
170+
171+
impl PartialEq<String> for HeaderValue {
172+
#[inline]
173+
fn eq(&self, other: &String) -> bool {
174+
self.as_ref() == other.as_bytes()
175+
}
176+
}
177+
178+
impl PartialEq<Vec<u8>> for HeaderValue {
179+
#[inline]
180+
fn eq(&self, other: &Vec<u8>) -> bool {
181+
self.as_ref() == other.as_slice()
182+
}
183+
}
184+
185+
impl PartialEq<&str> for HeaderValue {
186+
#[inline]
187+
fn eq(&self, other: &&str) -> bool {
188+
self.as_ref() == other.as_bytes()
189+
}
190+
}
191+
192+
impl PartialEq<&[u8]> for HeaderValue {
193+
#[inline]
194+
fn eq(&self, other: &&[u8]) -> bool {
195+
self.as_ref() == *other
196+
}
197+
}
198+
199+
impl PartialEq<HeaderValue> for String {
200+
#[inline]
201+
fn eq(&self, other: &HeaderValue) -> bool {
202+
*other == *self
203+
}
204+
}
205+
206+
impl PartialEq<HeaderValue> for Vec<u8> {
207+
#[inline]
208+
fn eq(&self, other: &HeaderValue) -> bool {
209+
*other == *self
210+
}
211+
}
212+
213+
impl PartialEq<HeaderValue> for &str {
214+
#[inline]
215+
fn eq(&self, other: &HeaderValue) -> bool {
216+
*other == *self
217+
}
218+
}
219+
220+
impl PartialEq<HeaderValue> for &[u8] {
221+
#[inline]
222+
fn eq(&self, other: &HeaderValue) -> bool {
223+
*other == *self
224+
}
225+
}
226+
227+
impl Hash for HeaderValue {
228+
#[inline]
229+
fn hash<H: Hasher>(&self, state: &mut H) {
230+
match self.inner {
231+
Ok(ref string) => Hash::hash(string, state),
232+
Err(ref bytes) => Hash::hash(bytes, state),
233+
}
234+
}
235+
}
236+
237+
#[cfg(test)]
238+
mod tests {
239+
use super::*;
240+
use std::collections::hash_map::DefaultHasher;
241+
use std::hash::{Hash, Hasher};
242+
243+
#[test]
244+
fn test_header_value_display_string() {
245+
let string: HeaderValue = String::from("utf-8 encoded string").into();
246+
247+
assert_eq!(format!("{}", string), "utf-8 encoded string");
248+
}
249+
250+
#[test]
251+
fn test_header_value_display_bytes() {
252+
let bytes: HeaderValue = vec![144u8, 145u8, 146u8].into();
253+
254+
assert_eq!(format!("{}", bytes), "[144, 145, 146]");
255+
}
256+
257+
#[test]
258+
fn test_header_value_debug_string() {
259+
let string: HeaderValue = String::from("utf-8 encoded string").into();
260+
261+
assert_eq!(format!("{:?}", string), "\"utf-8 encoded string\"");
262+
}
263+
264+
#[test]
265+
fn test_header_value_debug_bytes() {
266+
let bytes: HeaderValue = vec![144u8, 145u8, 146u8].into();
267+
268+
assert_eq!(format!("{:?}", bytes), "[144, 145, 146]");
269+
}
270+
271+
#[test]
272+
fn test_header_value_as_ref() {
273+
fn receive<T>(value: T)
274+
where
275+
T: AsRef<[u8]>,
276+
{
277+
value.as_ref();
278+
}
279+
280+
let string: HeaderValue = String::from("utf-8 encoded string").into();
281+
receive(string);
282+
283+
let bytes: HeaderValue = vec![144u8, 145u8, 146u8].into();
284+
receive(bytes);
285+
}
286+
287+
#[test]
288+
fn test_header_value_eq_string() {
289+
let string: HeaderValue = String::from("utf-8 encoded string").into();
290+
291+
assert_eq!(string, "utf-8 encoded string");
292+
assert_eq!(string, b"utf-8 encoded string" as &[u8]);
293+
294+
assert_eq!("utf-8 encoded string", string);
295+
assert_eq!(b"utf-8 encoded string" as &[u8], string);
296+
297+
assert_eq!(string, string);
298+
}
299+
300+
#[test]
301+
fn test_header_value_eq_bytes() {
302+
let bytes: HeaderValue = vec![144u8, 145u8, 146u8].into();
303+
304+
assert_eq!(bytes, vec![144u8, 145u8, 146u8]);
305+
assert_eq!(bytes, b"\x90\x91\x92" as &[u8]);
306+
307+
assert_eq!(vec![144u8, 145u8, 146u8], bytes);
308+
assert_eq!(b"\x90\x91\x92" as &[u8], bytes);
309+
310+
assert_eq!(bytes, bytes);
311+
}
312+
313+
fn hash<T: Hash>(t: &T) -> u64 {
314+
let mut h = DefaultHasher::new();
315+
t.hash(&mut h);
316+
h.finish()
317+
}
318+
319+
#[test]
320+
fn test_header_value_hash_string() {
321+
let string: HeaderValue = String::from("utf-8 encoded string").into();
322+
323+
assert_eq!(hash(&string), hash(&"utf-8 encoded string"));
324+
assert_ne!(hash(&string), hash(&b"utf-8 encoded string"));
325+
326+
assert_eq!(hash(&"utf-8 encoded string"), hash(&string));
327+
assert_ne!(hash(&b"utf-8 encoded string"), hash(&string));
328+
}
329+
330+
#[test]
331+
fn test_header_value_hash_bytes() {
332+
let bytes: HeaderValue = vec![144u8, 145u8, 146u8].into();
333+
334+
assert_eq!(hash(&bytes), hash(&vec![144u8, 145u8, 146u8]));
335+
assert_eq!(hash(&bytes), hash(&[144u8, 145u8, 146u8]));
336+
337+
assert_eq!(hash(&vec![144u8, 145u8, 146u8]), hash(&bytes));
338+
assert_eq!(hash(&[144u8, 145u8, 146u8]), hash(&bytes));
339+
}
340+
}

0 commit comments

Comments
 (0)