Skip to content

Commit 5916167

Browse files
authored
Merge pull request rust-lang#28 from xfix/serde
Implement optional Serde 1.0 support
2 parents 9951058 + 9fa5931 commit 5916167

File tree

4 files changed

+86
-0
lines changed

4 files changed

+86
-0
lines changed

Cargo.toml

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -11,13 +11,15 @@ description = "A hash table with consistent order and fast iteration."
1111
bench = false
1212

1313
[dependencies]
14+
serde = { version = "1.0", optional = true }
1415

1516
[dev-dependencies]
1617
itertools = "0.5.1"
1718
rand = "0.3"
1819
quickcheck = { version = "0.4", default-features = false }
1920
fnv = "1.0"
2021
lazy_static = "0.2"
22+
serde_test = "1.0.5"
2123

2224
[features]
2325
# for testing only, of course

src/lib.rs

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,8 @@
22
#![doc(html_root_url = "https://docs.rs/ordermap/0.2/")]
33

44
mod macros;
5+
#[cfg(feature = "serde")]
6+
mod serde;
57
mod util;
68

79
use std::hash::Hash;

src/serde.rs

Lines changed: 63 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,63 @@
1+
2+
extern crate serde;
3+
4+
use self::serde::ser::{Serialize, Serializer, SerializeMap};
5+
use self::serde::de::{Deserialize, Deserializer, MapAccess, Visitor};
6+
7+
use std::fmt::{self, Formatter};
8+
use std::hash::{BuildHasher, Hash};
9+
use std::marker::PhantomData;
10+
11+
use OrderMap;
12+
13+
impl<K, V, S> Serialize for OrderMap<K, V, S>
14+
where K: Serialize + Hash + Eq,
15+
V: Serialize,
16+
S: BuildHasher
17+
{
18+
fn serialize<T>(&self, serializer: T) -> Result<T::Ok, T::Error>
19+
where T: Serializer
20+
{
21+
let mut map_serializer = try!(serializer.serialize_map(Some(self.len())));
22+
for (key, value) in self {
23+
try!(map_serializer.serialize_entry(key, value));
24+
}
25+
map_serializer.end()
26+
}
27+
}
28+
29+
struct OrderMapVisitor<K, V>(PhantomData<(K, V)>);
30+
31+
impl<'de, K, V> Visitor<'de> for OrderMapVisitor<K, V>
32+
where K: Deserialize<'de> + Eq + Hash,
33+
V: Deserialize<'de>
34+
{
35+
type Value = OrderMap<K, V>;
36+
37+
fn expecting(&self, formatter: &mut Formatter) -> fmt::Result {
38+
write!(formatter, "a map")
39+
}
40+
41+
fn visit_map<A>(self, mut map: A) -> Result<Self::Value, A::Error>
42+
where A: MapAccess<'de>
43+
{
44+
let mut values = OrderMap::with_capacity(map.size_hint().unwrap_or(0));
45+
46+
while let Some((key, value)) = try!(map.next_entry()) {
47+
values.insert(key, value);
48+
}
49+
50+
Ok(values)
51+
}
52+
}
53+
54+
impl<'de, K, V> Deserialize<'de> for OrderMap<K, V>
55+
where K: Deserialize<'de> + Eq + Hash,
56+
V: Deserialize<'de>
57+
{
58+
fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
59+
where D: Deserializer<'de>
60+
{
61+
deserializer.deserialize_map(OrderMapVisitor(PhantomData))
62+
}
63+
}

tests/serde.rs

Lines changed: 19 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
#![cfg(feature = "serde")]
2+
3+
#[macro_use]
4+
extern crate ordermap;
5+
extern crate serde_test;
6+
7+
use serde_test::{Token, assert_tokens};
8+
9+
#[test]
10+
fn test_serde() {
11+
let map = ordermap! { 1 => 2, 3 => 4 };
12+
assert_tokens(&map,
13+
&[Token::Map { len: Some(2) },
14+
Token::I32(1),
15+
Token::I32(2),
16+
Token::I32(3),
17+
Token::I32(4),
18+
Token::MapEnd]);
19+
}

0 commit comments

Comments
 (0)