|
1 |
| -header! { |
2 |
| - #[doc="`Content-Length` header, defined in"] |
3 |
| - #[doc="[RFC7230](http://tools.ietf.org/html/rfc7230#section-3.3.2)"] |
4 |
| - #[doc=""] |
5 |
| - #[doc="When a message does not have a `Transfer-Encoding` header field, a"] |
6 |
| - #[doc="Content-Length header field can provide the anticipated size, as a"] |
7 |
| - #[doc="decimal number of octets, for a potential payload body. For messages"] |
8 |
| - #[doc="that do include a payload body, the Content-Length field-value"] |
9 |
| - #[doc="provides the framing information necessary for determining where the"] |
10 |
| - #[doc="body (and message) ends. For messages that do not include a payload"] |
11 |
| - #[doc="body, the Content-Length indicates the size of the selected"] |
12 |
| - #[doc="representation."] |
13 |
| - #[doc=""] |
14 |
| - #[doc="# ABNF"] |
15 |
| - #[doc="```plain"] |
16 |
| - #[doc="Content-Length = 1*DIGIT"] |
17 |
| - #[doc="```"] |
18 |
| - #[doc=""] |
19 |
| - #[doc="# Example values"] |
20 |
| - #[doc="* `3495`"] |
21 |
| - #[doc=""] |
22 |
| - #[doc="# Example"] |
23 |
| - #[doc="```"] |
24 |
| - #[doc="use hyper::header::{Headers, ContentLength};"] |
25 |
| - #[doc=""] |
26 |
| - #[doc="let mut headers = Headers::new();"] |
27 |
| - #[doc="headers.set(ContentLength(1024u64));"] |
28 |
| - #[doc="```"] |
29 |
| - (ContentLength, "Content-Length") => [u64] |
30 |
| - |
31 |
| - test_content_length { |
32 |
| - // Testcase from RFC |
33 |
| - test_header!(test1, vec![b"3495"], Some(HeaderField(3495))); |
| 1 | +use std::fmt; |
| 2 | + |
| 3 | +use header::{HeaderFormat, Header, parsing}; |
| 4 | + |
| 5 | +#[doc="`Content-Length` header, defined in"] |
| 6 | +#[doc="[RFC7230](http://tools.ietf.org/html/rfc7230#section-3.3.2)"] |
| 7 | +#[doc=""] |
| 8 | +#[doc="When a message does not have a `Transfer-Encoding` header field, a"] |
| 9 | +#[doc="Content-Length header field can provide the anticipated size, as a"] |
| 10 | +#[doc="decimal number of octets, for a potential payload body. For messages"] |
| 11 | +#[doc="that do include a payload body, the Content-Length field-value"] |
| 12 | +#[doc="provides the framing information necessary for determining where the"] |
| 13 | +#[doc="body (and message) ends. For messages that do not include a payload"] |
| 14 | +#[doc="body, the Content-Length indicates the size of the selected"] |
| 15 | +#[doc="representation."] |
| 16 | +#[doc=""] |
| 17 | +#[doc="# ABNF"] |
| 18 | +#[doc="```plain"] |
| 19 | +#[doc="Content-Length = 1*DIGIT"] |
| 20 | +#[doc="```"] |
| 21 | +#[doc=""] |
| 22 | +#[doc="# Example values"] |
| 23 | +#[doc="* `3495`"] |
| 24 | +#[doc=""] |
| 25 | +#[doc="# Example"] |
| 26 | +#[doc="```"] |
| 27 | +#[doc="use hyper::header::{Headers, ContentLength};"] |
| 28 | +#[doc=""] |
| 29 | +#[doc="let mut headers = Headers::new();"] |
| 30 | +#[doc="headers.set(ContentLength(1024u64));"] |
| 31 | +#[doc="```"] |
| 32 | +#[derive(Clone, Copy, Debug, PartialEq)] |
| 33 | +pub struct ContentLength(pub u64); |
| 34 | + |
| 35 | +impl Header for ContentLength { |
| 36 | + #[inline] |
| 37 | + fn header_name() -> &'static str { |
| 38 | + "Content-Length" |
| 39 | + } |
| 40 | + fn parse_header(raw: &[Vec<u8>]) -> ::Result<ContentLength> { |
| 41 | + // If multiple Content-Length headers were sent, everything can still |
| 42 | + // be alright if they all contain the same value, and all parse |
| 43 | + // correctly. If not, then it's an error. |
| 44 | + raw.iter() |
| 45 | + .map(::std::ops::Deref::deref) |
| 46 | + .map(parsing::from_raw_str) |
| 47 | + .fold(None, |prev, x| { |
| 48 | + match (prev, x) { |
| 49 | + (None, x) => Some(x), |
| 50 | + (e@Some(Err(_)), _ ) => e, |
| 51 | + (Some(Ok(prev)), Ok(x)) if prev == x => Some(Ok(prev)), |
| 52 | + _ => Some(Err(::Error::Header)) |
| 53 | + } |
| 54 | + }) |
| 55 | + .unwrap_or(Err(::Error::Header)) |
| 56 | + .map(ContentLength) |
| 57 | + } |
| 58 | +} |
| 59 | + |
| 60 | +impl HeaderFormat for ContentLength { |
| 61 | + #[inline] |
| 62 | + fn fmt_header(&self, f: &mut fmt::Formatter) -> fmt::Result { |
| 63 | + fmt::Display::fmt(&self.0, f) |
34 | 64 | }
|
35 | 65 | }
|
36 | 66 |
|
| 67 | +impl fmt::Display for ContentLength { |
| 68 | + #[inline] |
| 69 | + fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result { |
| 70 | + fmt::Display::fmt(&self.0, f) |
| 71 | + } |
| 72 | +} |
| 73 | + |
| 74 | +__hyper__deref!(ContentLength => u64); |
| 75 | +__hyper_generate_header_serialization!(ContentLength); |
| 76 | + |
| 77 | +__hyper__tm!(ContentLength, tests { |
| 78 | + // Testcase from RFC |
| 79 | + test_header!(test1, vec![b"3495"], Some(HeaderField(3495))); |
| 80 | + |
| 81 | + test_header!(test_invalid, vec![b"34v95"], None); |
| 82 | + test_header!(test_duplicates, vec![b"5", b"5"], Some(HeaderField(5))); |
| 83 | + test_header!(test_duplicates_vary, vec![b"5", b"6", b"5"], None); |
| 84 | +}); |
| 85 | + |
37 | 86 | bench_header!(bench, ContentLength, { vec![b"42349984".to_vec()] });
|
0 commit comments