Skip to content

Commit 20a585e

Browse files
committed
feat(headers): add AcceptLanguage header
Adds header support for Accept-Language, using a struct for the primary and sub components of the language.
1 parent 361e451 commit 20a585e

File tree

2 files changed

+88
-0
lines changed

2 files changed

+88
-0
lines changed

src/header/common/accept_language.rs

+86
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,86 @@
1+
use header::{self, QualityItem};
2+
use std::str::FromStr;
3+
use std::fmt;
4+
5+
/// A language tag.
6+
/// See http://www.w3.org/Protocols/rfc2616/rfc2616-sec3.html#sec3.10
7+
#[derive(Clone, PartialEq, Debug)]
8+
pub struct Language{
9+
primary: String,
10+
sub: Option<String>
11+
}
12+
13+
impl FromStr for Language {
14+
type Err = ();
15+
fn from_str(s: &str) -> Result<Language, ()> {
16+
let mut i = s.split_str("-");
17+
let p = i.next();
18+
let s = i.next();
19+
match (p, s) {
20+
(Some(p),Some(s)) => Ok(Language{primary: p.to_string(),
21+
sub: Some(s.to_string())}),
22+
(Some(p),_) => Ok(Language{primary: p.to_string(), sub: None}),
23+
_ => Err(())
24+
}
25+
}
26+
}
27+
28+
impl fmt::Display for Language {
29+
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
30+
try!(write!(f, "{}", self.primary));
31+
match self.sub {
32+
Some(ref s) => write!(f, "-{}", s),
33+
None => Ok(())
34+
}
35+
}
36+
}
37+
38+
/// The `Accept-Language` header
39+
///
40+
/// The `Accept-Language` header can be used by clients to indicate what
41+
/// response languages they accept.
42+
#[derive(Clone, PartialEq, Debug)]
43+
pub struct AcceptLanguage(pub Vec<QualityItem<Language>>);
44+
45+
impl_list_header!(AcceptLanguage,
46+
"Accept-Language",
47+
Vec<QualityItem<Language>>);
48+
49+
bench_header!(bench, AcceptLanguage,
50+
{ vec![b"en-us;q=1.0, en;q=0.5, fr".to_vec()] });
51+
52+
#[test]
53+
fn test_parse_header() {
54+
let a: AcceptLanguage = header::Header::parse_header(
55+
[b"en-us;q=1.0, en;q=0.5, fr".to_vec()].as_slice()).unwrap();
56+
let b = AcceptLanguage(vec![
57+
QualityItem { item: Language{primary: "en".to_string(),
58+
sub: Some("us".to_string())},
59+
quality: 1f32 },
60+
QualityItem { item: Language{primary: "en".to_string(), sub: None},
61+
quality: 0.5f32 },
62+
QualityItem { item: Language{primary: "fr".to_string(), sub: None},
63+
quality: 1f32 },
64+
]);
65+
assert_eq!(format!("{}", a), format!("{}", b));
66+
assert_eq!(a, b);
67+
}
68+
69+
#[test]
70+
fn test_display() {
71+
assert_eq!("en".to_string(),
72+
format!("{}", Language{primary: "en".to_string(),
73+
sub: None}));
74+
assert_eq!("en-us".to_string(),
75+
format!("{}", Language{primary: "en".to_string(),
76+
sub: Some("us".to_string())}));
77+
}
78+
79+
#[test]
80+
fn test_from_str() {
81+
assert_eq!(Language { primary: "en".to_string(), sub: None },
82+
"en".parse().unwrap());
83+
assert_eq!(Language { primary: "en".to_string(),
84+
sub: Some("us".to_string()) },
85+
"en-us".parse().unwrap());
86+
}

src/header/common/mod.rs

+2
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99
pub use self::access_control::*;
1010
pub use self::accept::Accept;
1111
pub use self::accept_encoding::AcceptEncoding;
12+
pub use self::accept_language::AcceptLanguage;
1213
pub use self::allow::Allow;
1314
pub use self::authorization::{Authorization, Scheme, Basic};
1415
pub use self::cache_control::{CacheControl, CacheDirective};
@@ -147,6 +148,7 @@ macro_rules! impl_header(
147148
mod access_control;
148149
mod accept;
149150
mod accept_encoding;
151+
mod accept_language;
150152
mod allow;
151153
mod authorization;
152154
mod cache_control;

0 commit comments

Comments
 (0)