mirror of
https://github.com/pezkuwichain/serde.git
synced 2026-06-15 13:11:01 +00:00
Compare commits
1 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 37c0ad19bb |
+1
-2
@@ -1,3 +1,2 @@
|
|||||||
Cargo.lock
|
|
||||||
target
|
target
|
||||||
*.sw[po]
|
Cargo.lock
|
||||||
|
|||||||
+30
-31
@@ -1,35 +1,34 @@
|
|||||||
sudo: false
|
|
||||||
language: rust
|
language: rust
|
||||||
rust:
|
rust:
|
||||||
- stable
|
- stable
|
||||||
- nightly
|
- beta
|
||||||
- 1.8.0
|
- nightly
|
||||||
- 1.9.0
|
|
||||||
- beta
|
|
||||||
addons:
|
|
||||||
apt:
|
|
||||||
packages:
|
|
||||||
- libcurl4-openssl-dev
|
|
||||||
- libelf-dev
|
|
||||||
- libdw-dev
|
|
||||||
before_script:
|
|
||||||
- pip install 'travis-cargo<0.2' --user
|
|
||||||
- export PATH=$HOME/.local/bin:$PATH
|
|
||||||
script:
|
|
||||||
- (cd serde && travis-cargo build)
|
|
||||||
- (cd serde && travis-cargo --skip nightly test)
|
|
||||||
- (cd serde && travis-cargo --only nightly test -- --features unstable-testing)
|
|
||||||
- (cd serde && travis-cargo build -- --no-default-features)
|
|
||||||
- (cd serde && travis-cargo --only nightly build -- --no-default-features --features alloc)
|
|
||||||
- (cd serde && travis-cargo --only nightly build -- --no-default-features --features collections)
|
|
||||||
- (cd testing && travis-cargo --skip nightly test)
|
|
||||||
- (cd testing && travis-cargo --only nightly test -- --features unstable-testing)
|
|
||||||
- (cd serde_macros && travis-cargo --only nightly test -- --features unstable-testing)
|
|
||||||
- (cd examples/serde-syntex-example && travis-cargo --skip nightly run)
|
|
||||||
- (cd examples/serde-syntex-example && travis-cargo --only nightly run -- --no-default-features --features unstable)
|
|
||||||
- (cd serde && travis-cargo --only stable doc)
|
|
||||||
after_success:
|
|
||||||
- (cd testing && travis-cargo --only stable coveralls --no-sudo)
|
|
||||||
env:
|
env:
|
||||||
global:
|
global:
|
||||||
- TRAVIS_CARGO_NIGHTLY_FEATURE=""
|
secure: HO41LMpMXkF2In9+1sxWVu7fgolL+y9+4Q5PI6wZX2L5pDwpPJCjxaQarQXCEnoIxED1PlP03JuF7ULNz0zw1ylYhAOfOSdkxFZRnE2wMZqq6qvXBHwyMiDrAociIzoPKSGv7JVrKPsjsnd+96K6xxueIodQZrmAdyq7N/M82Mc=
|
||||||
|
matrix:
|
||||||
|
- CRATE=serde_tests TARGET=test
|
||||||
|
matrix:
|
||||||
|
include:
|
||||||
|
- rust: nightly
|
||||||
|
env: CRATE=serde_macros TARGET=test
|
||||||
|
- rust: nightly
|
||||||
|
env: CRATE=serde_macros TARGET=bench
|
||||||
|
- rust: nightly
|
||||||
|
env: CRATE=serde_tests TARGET=bench
|
||||||
|
script:
|
||||||
|
- (cd $CRATE && cargo $TARGET)
|
||||||
|
after_success: |
|
||||||
|
[ $TRAVIS_BRANCH = "master" ] &&
|
||||||
|
[ $TRAVIS_PULL_REQUEST = false ] &&
|
||||||
|
mkdir -p target/doc &&
|
||||||
|
(cd serde && cargo doc --no-deps) &&
|
||||||
|
(cd serde_codegen && cargo doc --no-deps) &&
|
||||||
|
(cd serde_macros && cargo doc --no-deps) &&
|
||||||
|
cp -r serde/target/doc target/doc/serde &&
|
||||||
|
cp -r serde_codegen/target/doc target/doc/serde_codegen &&
|
||||||
|
cp -r serde_macros/target/doc target/doc/serde_macros &&
|
||||||
|
echo "<meta http-equiv=refresh content=0;url=`echo $TRAVIS_REPO_SLUG | cut -d '/' -f 2`/index.html>" > target/doc/index.html &&
|
||||||
|
sudo pip install ghp-import &&
|
||||||
|
ghp-import -n target/doc &&
|
||||||
|
git push -fq https://${GH_TOKEN}@github.com/${TRAVIS_REPO_SLUG}.git gh-pages
|
||||||
|
|||||||
@@ -1,45 +0,0 @@
|
|||||||
# Contributing to Serde
|
|
||||||
|
|
||||||
Serde welcomes contribution from everyone. Here are the guidelines if you are
|
|
||||||
thinking of helping us:
|
|
||||||
|
|
||||||
## Contributions
|
|
||||||
|
|
||||||
Contributions to Serde or its dependencies should be made in the form of GitHub
|
|
||||||
pull requests. Each pull request will be reviewed by a core contributor
|
|
||||||
(someone with permission to land patches) and either landed in the main tree or
|
|
||||||
given feedback for changes that would be required. All contributions should
|
|
||||||
follow this format, even those from core contributors.
|
|
||||||
|
|
||||||
Should you wish to work on an issue, please claim it first by commenting on
|
|
||||||
the GitHub issue that you want to work on it. This is to prevent duplicated
|
|
||||||
efforts from contributors on the same issue.
|
|
||||||
|
|
||||||
## Pull Request Checklist
|
|
||||||
|
|
||||||
- Branch from the master branch and, if needed, rebase to the current master
|
|
||||||
branch before submitting your pull request. If it doesn't merge cleanly with
|
|
||||||
master you may be asked to rebase your changes.
|
|
||||||
|
|
||||||
- Commits should be as small as possible, while ensuring that each commit is
|
|
||||||
correct independently (i.e., each commit should compile and pass tests).
|
|
||||||
|
|
||||||
- If your patch is not getting reviewed or you need a specific person to review
|
|
||||||
it, you can @-reply a reviewer asking for a review in the pull request or a
|
|
||||||
comment, or you can ask for a review in `#serde` on `irc.mozilla.org`.
|
|
||||||
|
|
||||||
- Add tests relevant to the fixed bug or new feature.
|
|
||||||
|
|
||||||
## Conduct
|
|
||||||
|
|
||||||
In all Serde-related forums, we follow the [Rust Code of
|
|
||||||
Conduct](https://www.rust-lang.org/conduct.html). For escalation or moderation
|
|
||||||
issues, please contact Erick (erick.tryzelaar@gmail.com) instead of the Rust
|
|
||||||
moderation team.
|
|
||||||
|
|
||||||
## Communication
|
|
||||||
|
|
||||||
Beyond opening tickets on the
|
|
||||||
[serde-rs/serde](https://github.com/serde-rs/serde) project, Serde contributors
|
|
||||||
frequent the `#serde` channel on
|
|
||||||
[`irc.mozilla.org`](https://wiki.mozilla.org/IRC).
|
|
||||||
@@ -1,69 +1,424 @@
|
|||||||
# Serde   [](https://travis-ci.org/serde-rs/serde) [](https://crates.io/crates/serde)
|
Serde Rust Serialization Framework
|
||||||
|
==================================
|
||||||
|
|
||||||
**Serde is a framework for *ser*ializing and *de*serializing Rust data structures efficiently and generically.**
|
[](https://travis-ci.org/serde-rs/serde)
|
||||||
|
[](https://crates.io/crates/serde)
|
||||||
|
|
||||||
---
|
Serde is a powerful framework that enables serialization libraries to
|
||||||
|
generically serialize Rust data structures without the overhead of runtime type
|
||||||
|
information. In many situations, the handshake protocol between serializers and
|
||||||
|
serializees can be completely optimized away, leaving Serde to perform roughly
|
||||||
|
the same speed as a hand written serializer for a specific type.
|
||||||
|
|
||||||
You may be looking for:
|
Documentation is available at http://serde-rs.github.io/serde/serde
|
||||||
|
|
||||||
- [An overview of Serde](https://serde.rs/)
|
Making a Type Serializable
|
||||||
- [Data formats supported by Serde](https://serde.rs/#data-formats)
|
==========================
|
||||||
- [Setting up `#[derive(Serialize, Deserialize)]`](https://serde.rs/codegen.html)
|
|
||||||
- [Examples](https://serde.rs/examples.html)
|
|
||||||
- [API documentation](https://docs.serde.rs/serde/)
|
|
||||||
|
|
||||||
## Serde in action
|
The simplest way to make a type serializable is to use the `serde_macros`
|
||||||
|
syntax extension, which comes with a `#[derive(Serialize, Deserialize)]`
|
||||||
|
annotation, which automatically generates implementations of
|
||||||
|
[Serialize](http://serde-rs.github.io/serde/serde/ser/trait.Serialize.html)
|
||||||
|
and
|
||||||
|
[Deserialize](http://serde-rs.github.io/serde/serde/de/trait.Deserialize.html)
|
||||||
|
for the annotated type:
|
||||||
|
|
||||||
```rust
|
```rust
|
||||||
#![feature(plugin, custom_derive)]
|
#![feature(custom_derive, plugin)]
|
||||||
#![plugin(serde_macros)]
|
#![plugin(serde_macros)]
|
||||||
|
|
||||||
extern crate serde_json;
|
extern crate serde;
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, Debug)]
|
...
|
||||||
|
|
||||||
|
#[derive(Serialize, Deserialize)]
|
||||||
|
struct Point {
|
||||||
|
x: i32,
|
||||||
|
y: i32,
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Serde bundles a high performance JSON serializer and deserializer,
|
||||||
|
[serde::json](http://serde-rs.github.io/serde/serde/json/index.html),
|
||||||
|
which comes with the helper functions
|
||||||
|
[to_string](http://serde-rs.github.io/serde/serde/json/ser/fn.to_string.html)
|
||||||
|
and
|
||||||
|
[from_str](http://serde-rs.github.io/serde/serde/json/de/fn.from_str.html)
|
||||||
|
that make it easy to go to and from JSON:
|
||||||
|
|
||||||
|
```rust
|
||||||
|
use serde::json;
|
||||||
|
|
||||||
|
...
|
||||||
|
|
||||||
|
let point = Point { x: 1, y: 2 };
|
||||||
|
let serialized_point = json::to_string(&point).unwrap();
|
||||||
|
|
||||||
|
println!("{}", serialized_point); // prints: {"x":1,"y":2}
|
||||||
|
|
||||||
|
let deserialize_point: Point = json::from_str(&serialized_point).unwrap();
|
||||||
|
```
|
||||||
|
|
||||||
|
[serde::json](http://serde-rs.github.io/serde/serde/json/index.html) also
|
||||||
|
supports a generic
|
||||||
|
[Value](http://serde-rs.github.io/serde/serde/json/value/enum.Value.html)
|
||||||
|
type, which can represent any JSON value. Also, any
|
||||||
|
[Serialize](http://serde-rs.github.io/serde/serde/ser/trait.Serialize.html)
|
||||||
|
and
|
||||||
|
[Deserialize](http://serde-rs.github.io/serde/serde/de/trait.Deserialize.html)
|
||||||
|
can be converted into a
|
||||||
|
[Value](http://serde-rs.github.io/serde/serde/json/value/enum.Value.html)
|
||||||
|
with the methods
|
||||||
|
[to_value](http://serde-rs.github.io/serde/serde/json/value/fn.to_value.html)
|
||||||
|
and
|
||||||
|
[from_value](http://serde-rs.github.io/serde/serde/json/value/fn.from_value.html):
|
||||||
|
|
||||||
|
```rust
|
||||||
|
let point = Point { x: 1, y: 2 };
|
||||||
|
let point_value = json::to_value(&point).unwrap();
|
||||||
|
|
||||||
|
println!("{}", point_value.find("x")); // prints: Some(1)
|
||||||
|
|
||||||
|
let deserialize_point: Point = json::from_value(point_value).unwrap();
|
||||||
|
```
|
||||||
|
|
||||||
|
Serialization without Macros
|
||||||
|
============================
|
||||||
|
|
||||||
|
Under the covers, Serde extensively uses the Visitor pattern to thread state
|
||||||
|
between the
|
||||||
|
[Serializer](http://serde-rs.github.io/serde/serde/ser/trait.Serializer.html)
|
||||||
|
and
|
||||||
|
[Serialize](http://serde-rs.github.io/serde/serde/ser/trait.Serialize.html)
|
||||||
|
without the two having specific information about each other's concrete type.
|
||||||
|
This has many of the same benefits as frameworks that use runtime type
|
||||||
|
information without the overhead. In fact, when compiling with optimizations,
|
||||||
|
Rust is able to remove most or all the visitor state, and generate code that's
|
||||||
|
nearly as fast as a hand written serializer format for a specific type.
|
||||||
|
|
||||||
|
To see it in action, lets look at how a simple type like `i32` is serialized.
|
||||||
|
The
|
||||||
|
[Serializer](http://serde-rs.github.io/serde/serde/ser/trait.Serializer.html)
|
||||||
|
is threaded through the type:
|
||||||
|
|
||||||
|
```rust
|
||||||
|
impl serde::Serialize for i32 {
|
||||||
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
|
where S: serde::Serializer,
|
||||||
|
{
|
||||||
|
serializer.visit_i32(*self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
As you can see it's pretty simple. More complex types like `BTreeMap` need to
|
||||||
|
pass a
|
||||||
|
[MapVisitor](http://serde-rs.github.io/serde/serde/ser/trait.MapVisitor.html)
|
||||||
|
to the
|
||||||
|
[Serializer](http://serde-rs.github.io/serde/serde/ser/trait.Serializer.html)
|
||||||
|
in order to walk through the type:
|
||||||
|
|
||||||
|
```rust
|
||||||
|
impl<K, V> Serialize for BTreeMap<K, V>
|
||||||
|
where K: Serialize + Ord,
|
||||||
|
V: Serialize,
|
||||||
|
{
|
||||||
|
#[inline]
|
||||||
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
|
where S: Serializer,
|
||||||
|
{
|
||||||
|
serializer.visit_map(MapIteratorVisitor::new(self.iter(), Some(self.len())))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct MapIteratorVisitor<Iter> {
|
||||||
|
iter: Iter,
|
||||||
|
len: Option<usize>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<K, V, Iter> MapIteratorVisitor<Iter>
|
||||||
|
where Iter: Iterator<Item=(K, V)>
|
||||||
|
{
|
||||||
|
#[inline]
|
||||||
|
pub fn new(iter: Iter, len: Option<usize>) -> MapIteratorVisitor<Iter> {
|
||||||
|
MapIteratorVisitor {
|
||||||
|
iter: iter,
|
||||||
|
len: len,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<K, V, I> MapVisitor for MapIteratorVisitor<I>
|
||||||
|
where K: Serialize,
|
||||||
|
V: Serialize,
|
||||||
|
I: Iterator<Item=(K, V)>,
|
||||||
|
{
|
||||||
|
#[inline]
|
||||||
|
fn visit<S>(&mut self, serializer: &mut S) -> Result<Option<()>, S::Error>
|
||||||
|
where S: Serializer,
|
||||||
|
{
|
||||||
|
match self.iter.next() {
|
||||||
|
Some((key, value)) => {
|
||||||
|
let value = try!(serializer.visit_map_elt(key, value));
|
||||||
|
Ok(Some(value))
|
||||||
|
}
|
||||||
|
None => Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn len(&self) -> Option<usize> {
|
||||||
|
self.len
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
Serializing structs follow this same pattern. In fact, structs are represented
|
||||||
|
as a named map. It's visitor uses a simple state machine to iterate through all
|
||||||
|
the fields:
|
||||||
|
|
||||||
|
```rust
|
||||||
struct Point {
|
struct Point {
|
||||||
x: i32,
|
x: i32,
|
||||||
y: i32,
|
y: i32,
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
impl serde::Serialize for Point {
|
||||||
let point = Point { x: 1, y: 2 };
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
|
where S: serde::Serializer
|
||||||
|
{
|
||||||
|
serializer.visit_named_map("Point", PointMapVisitor {
|
||||||
|
value: self,
|
||||||
|
state: 0,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Convert the Point to a JSON string.
|
struct PointMapVisitor<'a> {
|
||||||
let serialized = serde_json::to_string(&point).unwrap();
|
value: &'a Point,
|
||||||
|
state: u8,
|
||||||
|
}
|
||||||
|
|
||||||
// Prints serialized = {"x":1,"y":2}
|
impl<'a> serde::ser::MapVisitor for PointMapVisitor<'a> {
|
||||||
println!("serialized = {}", serialized);
|
fn visit<S>(&mut self, serializer: &mut S) -> Result<Option<()>, S::Error>
|
||||||
|
where S: serde::Serializer
|
||||||
// Convert the JSON string back to a Point.
|
{
|
||||||
let deserialized: Point = serde_json::from_str(&serialized).unwrap();
|
match self.state {
|
||||||
|
0 => {
|
||||||
// Prints deserialized = Point { x: 1, y: 2 }
|
self.state += 1;
|
||||||
println!("deserialized = {:?}", deserialized);
|
Ok(Some(try!(serializer.visit_map_elt("x", &self.value.x))))
|
||||||
|
}
|
||||||
|
1 => {
|
||||||
|
self.state += 1;
|
||||||
|
Ok(Some(try!(serializer.visit_map_elt("y", &self.value.y))))
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
## Getting help
|
Deserialization without Macros
|
||||||
|
==============================
|
||||||
|
|
||||||
Serde developers live in the #serde channel on
|
Deserialization is a little more complicated since there's a bit more error
|
||||||
[`irc.mozilla.org`](https://wiki.mozilla.org/IRC). The #rust channel is also a
|
handling that needs to occur. Let's start with the simple `i32`
|
||||||
good resource with generally faster response time but less specific knowledge
|
[Deserialize](http://serde-rs.github.io/serde/serde/de/trait.Deserialize.html)
|
||||||
about Serde. If IRC is not your thing, we are happy to respond to [GitHub
|
implementation. It passes a
|
||||||
issues](https://github.com/serde-rs/serde/issues/new) as well.
|
[Visitor](http://serde-rs.github.io/serde/serde/de/trait.Visitor.html) to the
|
||||||
|
[Deserializer](http://serde-rs.github.io/serde/serde/de/trait.Deserializer.html).
|
||||||
|
The [Visitor](http://serde-rs.github.io/serde/serde/de/trait.Visitor.html)
|
||||||
|
can create the `i32` from a variety of different types:
|
||||||
|
|
||||||
## License
|
```rust
|
||||||
|
impl Deserialize for i32 {
|
||||||
|
fn deserialize<D>(deserializer: &mut D) -> Result<i32, D::Error>
|
||||||
|
where D: serde::Deserializer,
|
||||||
|
{
|
||||||
|
deserializer.visit(I32Visitor)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Serde is licensed under either of
|
struct I32Visitor;
|
||||||
|
|
||||||
* Apache License, Version 2.0, ([LICENSE-APACHE](LICENSE-APACHE) or
|
impl serde::de::Visitor for I32Visitor {
|
||||||
http://www.apache.org/licenses/LICENSE-2.0)
|
type Value = i32;
|
||||||
* MIT license ([LICENSE-MIT](LICENSE-MIT) or
|
|
||||||
http://opensource.org/licenses/MIT)
|
|
||||||
|
|
||||||
at your option.
|
fn visit_i16<E>(&mut self, value: i16) -> Result<i16, E>
|
||||||
|
where E: Error,
|
||||||
|
{
|
||||||
|
self.visit_i32(value as i32)
|
||||||
|
}
|
||||||
|
|
||||||
### Contribution
|
fn visit_i32<E>(&mut self, value: i32) -> Result<i32, E>
|
||||||
|
where E: Error,
|
||||||
|
{
|
||||||
|
Ok(value)
|
||||||
|
}
|
||||||
|
|
||||||
Unless you explicitly state otherwise, any contribution intentionally submitted
|
...
|
||||||
for inclusion in Serde by you, as defined in the Apache-2.0 license, shall be
|
|
||||||
dual licensed as above, without any additional terms or conditions.
|
```
|
||||||
|
|
||||||
|
Since it's possible for this type to get passed an unexpected type, we need a
|
||||||
|
way to error out. This is done by way of the
|
||||||
|
[Error](http://serde-rs.github.io/serde/serde/de/trait.Error.html) trait,
|
||||||
|
which allows a
|
||||||
|
[Deserialize](http://serde-rs.github.io/serde/serde/de/trait.Deserialize.html)
|
||||||
|
to generate an error for a few common error conditions. Here's how it could be used:
|
||||||
|
|
||||||
|
```rust
|
||||||
|
...
|
||||||
|
|
||||||
|
fn visit_string<E>(&mut self, _: String) -> Result<i32, E>
|
||||||
|
where E: Error,
|
||||||
|
{
|
||||||
|
Err(serde::de::Error::syntax_error())
|
||||||
|
}
|
||||||
|
|
||||||
|
...
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
Maps follow a similar pattern as before, and use a
|
||||||
|
[MapVisitor](http://serde-rs.github.io/serde/serde/de/trait.MapVisitor.html)
|
||||||
|
to walk through the values generated by the
|
||||||
|
[Deserializer](http://serde-rs.github.io/serde/serde/de/trait.Deserializer.html).
|
||||||
|
|
||||||
|
```rust
|
||||||
|
impl<K, V> serde::Deserialize for BTreeMap<K, V>
|
||||||
|
where K: serde::Deserialize + Eq + Ord,
|
||||||
|
V: serde::Deserialize,
|
||||||
|
{
|
||||||
|
fn deserialize<D>(deserializer: &mut D) -> Result<BTreeMap<K, V>, D::Error>
|
||||||
|
where D: serde::Deserializer,
|
||||||
|
{
|
||||||
|
deserializer.visit(BTreeMapVisitor::new())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct BTreeMapVisitor<K, V> {
|
||||||
|
marker: PhantomData<BTreeMap<K, V>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<K, V> BTreeMapVisitor<K, V> {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
BTreeMapVisitor {
|
||||||
|
marker: PhantomData,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<K, V> serde::de::Visitor for BTreeMapVisitor<K, V>
|
||||||
|
where K: serde::de::Deserialize + Ord,
|
||||||
|
V: serde::de::Deserialize
|
||||||
|
{
|
||||||
|
type Value = BTreeMap<K, V>;
|
||||||
|
|
||||||
|
fn visit_unit<E>(&mut self) -> Result<BTreeMap<K, V>, E>
|
||||||
|
where E: Error,
|
||||||
|
{
|
||||||
|
Ok(BTreeMap::new())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_map<V_>(&mut self, mut visitor: V_) -> Result<BTreeMap<K, V>, V_::Error>
|
||||||
|
where V_: MapVisitor,
|
||||||
|
{
|
||||||
|
let mut values = BTreeMap::new();
|
||||||
|
|
||||||
|
while let Some((key, value)) = try!(visitor.visit()) {
|
||||||
|
values.insert(key, value);
|
||||||
|
}
|
||||||
|
|
||||||
|
try!(visitor.end());
|
||||||
|
|
||||||
|
Ok(values)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
Deserializing structs goes a step further in order to support not allocating a
|
||||||
|
`String` to hold the field names. This is done by custom field enum that
|
||||||
|
deserializes an enum variant from a string. So for our `Point` example from
|
||||||
|
before, we need to generate:
|
||||||
|
|
||||||
|
```rust
|
||||||
|
enum PointField {
|
||||||
|
X,
|
||||||
|
Y,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl serde::Deserialize for PointField {
|
||||||
|
fn deserialize<D>(deserializer: &mut D) -> Result<PointField, D::Error>
|
||||||
|
where D: serde::de::Deserializer
|
||||||
|
{
|
||||||
|
struct FieldVisitor;
|
||||||
|
|
||||||
|
impl serde::de::Visitor for FieldVisitor {
|
||||||
|
type Value = Field;
|
||||||
|
|
||||||
|
fn visit_str<E>(&mut self, value: &str) -> Result<PointField, E>
|
||||||
|
where E: serde::de::Error
|
||||||
|
{
|
||||||
|
match value {
|
||||||
|
"x" => Ok(Field::X),
|
||||||
|
"y" => Ok(Field::Y),
|
||||||
|
_ => Err(serde::de::Error::syntax_error()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
deserializer.visit(FieldVisitor)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
This is then used in our actual deserializer:
|
||||||
|
|
||||||
|
```rust
|
||||||
|
impl serde::Deserialize for Point {
|
||||||
|
fn deserialize<D>(deserializer: &mut D) -> Result<Point, D::Error>
|
||||||
|
where D: serde::de::Deserializer
|
||||||
|
{
|
||||||
|
deserializer.visit_named_map("Point", PointVisitor)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct PointVisitor;
|
||||||
|
|
||||||
|
impl serde::de::Visitor for PointVisitor {
|
||||||
|
type Value = Point;
|
||||||
|
|
||||||
|
fn visit_map<V>(&mut self, mut visitor: V) -> Result<Point, V::Error>
|
||||||
|
where V: serde::de::MapVisitor
|
||||||
|
{
|
||||||
|
let mut x = None;
|
||||||
|
let mut y = None;
|
||||||
|
|
||||||
|
loop {
|
||||||
|
match try!(visitor.visit_key()) {
|
||||||
|
Some(Field::X) => { x = Some(try!(visitor.visit_value())); }
|
||||||
|
Some(Field::Y) => { y = Some(try!(visitor.visit_value())); }
|
||||||
|
None => { break; }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let x = match x {
|
||||||
|
Some(x) => x,
|
||||||
|
None => try!(visitor.missing_field("x")),
|
||||||
|
};
|
||||||
|
|
||||||
|
let y = match y {
|
||||||
|
Some(y) => y,
|
||||||
|
None => try!(visitor.missing_field("y")),
|
||||||
|
};
|
||||||
|
|
||||||
|
try!(visitor.end());
|
||||||
|
|
||||||
|
Ok(Point{ x: x, y: y })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|||||||
@@ -1,2 +0,0 @@
|
|||||||
target
|
|
||||||
Cargo.lock
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "serde-syntex-example"
|
|
||||||
version = "0.1.0"
|
|
||||||
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
|
||||||
build = "build.rs"
|
|
||||||
|
|
||||||
[features]
|
|
||||||
default = ["serde_codegen"]
|
|
||||||
unstable = ["serde_macros"]
|
|
||||||
|
|
||||||
[build-dependencies]
|
|
||||||
serde_codegen = { version = "^0.8", optional = true, path = "../../serde_codegen" }
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
serde = "^0.8"
|
|
||||||
serde_json = "^0.8"
|
|
||||||
serde_macros = { version = "^0.8", optional = true, path = "../../serde_macros" }
|
|
||||||
@@ -1,20 +0,0 @@
|
|||||||
This example demonstrates how to use Serde with Syntex. On stable or nightly
|
|
||||||
with Syntex, it can be built with:
|
|
||||||
|
|
||||||
```
|
|
||||||
% rustup run stable cargo run
|
|
||||||
Running `target/debug/serde-syntex-example`
|
|
||||||
{"x":1,"y":2}
|
|
||||||
Point { x: 1, y: 2 }
|
|
||||||
|
|
||||||
% rustup run nightly cargo run
|
|
||||||
Running `target/debug/serde-syntex-example`
|
|
||||||
{"x":1,"y":2}
|
|
||||||
Point { x: 1, y: 2 }
|
|
||||||
```
|
|
||||||
|
|
||||||
On nightly, it can use a plugin with:
|
|
||||||
|
|
||||||
```
|
|
||||||
% rustup run nightly cargo run --features unstable --no-default-features
|
|
||||||
```
|
|
||||||
@@ -1,25 +0,0 @@
|
|||||||
#[cfg(not(feature = "serde_macros"))]
|
|
||||||
mod inner {
|
|
||||||
extern crate serde_codegen;
|
|
||||||
|
|
||||||
use std::env;
|
|
||||||
use std::path::Path;
|
|
||||||
|
|
||||||
pub fn main() {
|
|
||||||
let out_dir = env::var_os("OUT_DIR").unwrap();
|
|
||||||
|
|
||||||
let src = Path::new("src/main.rs.in");
|
|
||||||
let dst = Path::new(&out_dir).join("main.rs");
|
|
||||||
|
|
||||||
serde_codegen::expand(&src, &dst).unwrap();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "serde_macros")]
|
|
||||||
mod inner {
|
|
||||||
pub fn main() {}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn main() {
|
|
||||||
inner::main();
|
|
||||||
}
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
#![cfg_attr(feature = "serde_macros", feature(custom_derive, plugin))]
|
|
||||||
#![cfg_attr(feature = "serde_macros", plugin(serde_macros))]
|
|
||||||
|
|
||||||
extern crate serde;
|
|
||||||
extern crate serde_json;
|
|
||||||
|
|
||||||
#[cfg(feature = "serde_macros")]
|
|
||||||
include!("main.rs.in");
|
|
||||||
|
|
||||||
#[cfg(not(feature = "serde_macros"))]
|
|
||||||
include!(concat!(env!("OUT_DIR"), "/main.rs"));
|
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
#[derive(Serialize, Deserialize, Debug)]
|
|
||||||
struct Point {
|
|
||||||
x: i32,
|
|
||||||
y: i32,
|
|
||||||
}
|
|
||||||
|
|
||||||
fn main() {
|
|
||||||
let point = Point { x: 1, y: 2 };
|
|
||||||
let serialized = serde_json::to_string(&point).unwrap();
|
|
||||||
|
|
||||||
println!("{}", serialized);
|
|
||||||
|
|
||||||
let deserialized: Point = serde_json::from_str(&serialized).unwrap();
|
|
||||||
|
|
||||||
println!("{:?}", deserialized);
|
|
||||||
}
|
|
||||||
+4
-15
@@ -1,24 +1,13 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "serde"
|
name = "serde"
|
||||||
version = "0.8.5"
|
version = "0.4.2"
|
||||||
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
||||||
license = "MIT/Apache-2.0"
|
license = "MIT/Apache-2.0"
|
||||||
description = "A generic serialization/deserialization framework"
|
description = "A generic serialization/deserialization framework"
|
||||||
homepage = "https://serde.rs"
|
|
||||||
repository = "https://github.com/serde-rs/serde"
|
repository = "https://github.com/serde-rs/serde"
|
||||||
documentation = "https://docs.serde.rs/serde/"
|
documentation = "http://serde-rs.github.io/serde/serde"
|
||||||
readme = "../README.md"
|
readme = "../README.md"
|
||||||
keywords = ["serde", "serialization"]
|
keywords = ["serialization"]
|
||||||
include = ["Cargo.toml", "src/**/*.rs"]
|
|
||||||
|
|
||||||
[features]
|
|
||||||
default = ["std"]
|
|
||||||
|
|
||||||
std = []
|
|
||||||
unstable = []
|
|
||||||
alloc = ["unstable"]
|
|
||||||
collections = ["alloc"]
|
|
||||||
unstable-testing = ["clippy", "unstable", "std"]
|
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
clippy = { version = "^0.*", optional = true }
|
num = "*"
|
||||||
|
|||||||
@@ -0,0 +1,94 @@
|
|||||||
|
use std::cmp;
|
||||||
|
use std::io;
|
||||||
|
use std::slice;
|
||||||
|
|
||||||
|
trait IntoBufRead {
|
||||||
|
type IntoBuf: io::BufRead + BufReadExt;
|
||||||
|
|
||||||
|
fn into_buf_read(self) -> Self::IntoBuf;
|
||||||
|
}
|
||||||
|
|
||||||
|
trait BufReadExt {
|
||||||
|
fn get_buf(&self) -> &[u8];
|
||||||
|
fn read_u8(&mut self) -> io::Result<Option<u8>>;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct SliceReader<'a> {
|
||||||
|
buf: &'a [u8],
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> io::Read for SliceReader<'a> {
|
||||||
|
#[inline]
|
||||||
|
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
|
||||||
|
let amt = cmp::min(buf.len(), self.buf.len());
|
||||||
|
let (a, b) = self.buf.split_at(amt);
|
||||||
|
slice::bytes::copy_memory(buf, a);
|
||||||
|
*self.buf = b;
|
||||||
|
Ok(amt)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> io::BufRead for SliceReader<'a> {
|
||||||
|
fn fill_buf(&mut self) -> io::Result<&[u8]> { Ok(*self) }
|
||||||
|
fn consume(&mut self, amt: usize) { *self.buf = &self.buf[amt..]; }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> BufReadExt for SliceReader<'a> {
|
||||||
|
fn get_buf(&self) -> &[u8] { self.buf }
|
||||||
|
fn read_u8(&mut self) -> io::Result<Option<u8>> {
|
||||||
|
let byte = self.buf.get(0);
|
||||||
|
*self.buf = &self.buf[1..];
|
||||||
|
byte
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct BufReader<R> {
|
||||||
|
inner: R,
|
||||||
|
buf: io::Cursor<Vec<u8>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<R> BufReader<R> where R: io::Read {
|
||||||
|
fn new(inner: R) -> Self {
|
||||||
|
BufferedReader::with_capacity(io::DEFAULT_BUF_SIZE, inner)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn new(cap: usize, inner: R) -> Self {
|
||||||
|
BufferedReader {
|
||||||
|
inner: inner,
|
||||||
|
buf: io::Cursor::new(Vec::with_capacity(cap)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn into_inner(self) -> R {
|
||||||
|
self.inner
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<R> Read for BufReader<R> where R: io::Read {
|
||||||
|
fn read(&mut self, buf: &mut [u8]) -> io::Result<usize> {
|
||||||
|
// If we don't have any buffered data and we're doing a massive read
|
||||||
|
// (larger than our internal buffer), bypass our internal buffer
|
||||||
|
// entirely.
|
||||||
|
if self.buf.get_ref().len() == self.buf.position() as usize &&
|
||||||
|
buf.len() >= self.buf.get_ref().capacity() {
|
||||||
|
return self.inner.read(buf);
|
||||||
|
}
|
||||||
|
try!(self.fill_buf());
|
||||||
|
self.buf.read(buf)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<R> BufReadExt for BufReader<R> {
|
||||||
|
fn get_buf(&self) -> &[u8] {
|
||||||
|
self.buf.get_ref()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn read_u8(&mut self) -> io::Result<Option<u8>> {
|
||||||
|
if self.buf.get_ref().len() == self.buf.position() as usize {
|
||||||
|
|
||||||
|
}
|
||||||
|
let byte = self.buf.get(0);
|
||||||
|
*self.buf = &self.buf[1..];
|
||||||
|
byte
|
||||||
|
}
|
||||||
|
}
|
||||||
+132
-159
@@ -1,15 +1,11 @@
|
|||||||
//! Helper module to enable serializing bytes more efficiently
|
//! Helper module to enable serializing bytes more efficiently
|
||||||
|
|
||||||
use core::{ops, fmt, char, iter, slice};
|
use std::ops;
|
||||||
use core::fmt::Write;
|
use std::fmt;
|
||||||
|
use std::ascii;
|
||||||
|
|
||||||
use ser;
|
use ser;
|
||||||
|
use de;
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
pub use self::bytebuf::{ByteBuf, ByteBufVisitor};
|
|
||||||
|
|
||||||
#[cfg(feature = "collections")]
|
|
||||||
use collections::Vec;
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
@@ -21,11 +17,7 @@ pub struct Bytes<'a> {
|
|||||||
|
|
||||||
impl<'a> fmt::Debug for Bytes<'a> {
|
impl<'a> fmt::Debug for Bytes<'a> {
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
try!(f.write_str("b\""));
|
write!(f, "b\"{}\"", escape_bytestring(self.bytes))
|
||||||
for c in escape_bytestring(self.bytes) {
|
|
||||||
try!(f.write_char(c));
|
|
||||||
}
|
|
||||||
f.write_char('"')
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -37,11 +29,10 @@ impl<'a> From<&'a [u8]> for Bytes<'a> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
impl<'a> From<&'a Vec<u8>> for Bytes<'a> {
|
impl<'a> From<&'a Vec<u8>> for Bytes<'a> {
|
||||||
fn from(bytes: &'a Vec<u8>) -> Self {
|
fn from(bytes: &'a Vec<u8>) -> Self {
|
||||||
Bytes {
|
Bytes {
|
||||||
bytes: bytes,
|
bytes: &bytes,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -63,178 +54,160 @@ impl<'a> ser::Serialize for Bytes<'a> {
|
|||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
where S: ser::Serializer
|
where S: ser::Serializer
|
||||||
{
|
{
|
||||||
serializer.serialize_bytes(self.bytes)
|
serializer.visit_bytes(self.bytes)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
/// `ByteBuf` wraps a `Vec<u8>` in order to hook into serialize and from deserialize a byte array.
|
||||||
mod bytebuf {
|
#[derive(Clone, Eq, Hash, PartialEq, PartialOrd, Ord)]
|
||||||
use core::ops;
|
pub struct ByteBuf {
|
||||||
use core::fmt;
|
bytes: Vec<u8>,
|
||||||
use core::fmt::Write;
|
}
|
||||||
|
|
||||||
use ser;
|
impl ByteBuf {
|
||||||
use de;
|
pub fn new() -> Self {
|
||||||
|
ByteBuf {
|
||||||
#[cfg(feature = "collections")]
|
bytes: Vec::new(),
|
||||||
use collections::Vec;
|
|
||||||
|
|
||||||
/// `ByteBuf` wraps a `Vec<u8>` and serializes as a byte array.
|
|
||||||
#[derive(Clone, Default, Eq, Hash, PartialEq, PartialOrd, Ord)]
|
|
||||||
pub struct ByteBuf {
|
|
||||||
bytes: Vec<u8>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ByteBuf {
|
|
||||||
/// Construct a new, empty `ByteBuf`.
|
|
||||||
pub fn new() -> Self {
|
|
||||||
ByteBuf {
|
|
||||||
bytes: Vec::new(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Construct a new, empty `ByteBuf` with the specified capacity.
|
|
||||||
pub fn with_capacity(cap: usize) -> Self {
|
|
||||||
ByteBuf {
|
|
||||||
bytes: Vec::with_capacity(cap)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl fmt::Debug for ByteBuf {
|
pub fn with_capacity(cap: usize) -> Self {
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
ByteBuf {
|
||||||
try!(f.write_str("b\""));
|
bytes: Vec::with_capacity(cap)
|
||||||
for c in super::escape_bytestring(self.bytes.as_ref()) {
|
|
||||||
try!(f.write_char(c));
|
|
||||||
}
|
|
||||||
f.write_char('"')
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Into<Vec<u8>> for ByteBuf {
|
impl fmt::Debug for ByteBuf {
|
||||||
fn into(self) -> Vec<u8> {
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
self.bytes
|
write!(f, "b\"{}\"", escape_bytestring(self.bytes.as_ref()))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Into<Vec<u8>> for ByteBuf {
|
||||||
|
fn into(self) -> Vec<u8> {
|
||||||
|
self.bytes
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<Vec<u8>> for ByteBuf {
|
||||||
|
fn from(bytes: Vec<u8>) -> Self {
|
||||||
|
ByteBuf {
|
||||||
|
bytes: bytes,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl From<Vec<u8>> for ByteBuf {
|
impl AsRef<Vec<u8>> for ByteBuf {
|
||||||
fn from(bytes: Vec<u8>) -> Self {
|
fn as_ref(&self) -> &Vec<u8> {
|
||||||
ByteBuf {
|
&self.bytes
|
||||||
bytes: bytes,
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
impl AsRef<[u8]> for ByteBuf {
|
||||||
|
fn as_ref(&self) -> &[u8] {
|
||||||
|
&self.bytes
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AsMut<Vec<u8>> for ByteBuf {
|
||||||
|
fn as_mut(&mut self) -> &mut Vec<u8> {
|
||||||
|
&mut self.bytes
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AsMut<[u8]> for ByteBuf {
|
||||||
|
fn as_mut(&mut self) -> &mut [u8] {
|
||||||
|
&mut self.bytes
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ops::Deref for ByteBuf {
|
||||||
|
type Target = [u8];
|
||||||
|
|
||||||
|
fn deref(&self) -> &[u8] { &self.bytes[..] }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ops::DerefMut for ByteBuf {
|
||||||
|
fn deref_mut(&mut self) -> &mut [u8] { &mut self.bytes[..] }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ser::Serialize for ByteBuf {
|
||||||
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
|
where S: ser::Serializer
|
||||||
|
{
|
||||||
|
serializer.visit_bytes(&self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ByteBufVisitor;
|
||||||
|
|
||||||
|
impl de::Visitor for ByteBufVisitor {
|
||||||
|
type Value = ByteBuf;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_unit<E>(&mut self) -> Result<ByteBuf, E>
|
||||||
|
where E: de::Error,
|
||||||
|
{
|
||||||
|
Ok(ByteBuf {
|
||||||
|
bytes: Vec::new(),
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AsRef<Vec<u8>> for ByteBuf {
|
#[inline]
|
||||||
fn as_ref(&self) -> &Vec<u8> {
|
fn visit_seq<V>(&mut self, mut visitor: V) -> Result<ByteBuf, V::Error>
|
||||||
&self.bytes
|
where V: de::SeqVisitor,
|
||||||
|
{
|
||||||
|
let (len, _) = visitor.size_hint();
|
||||||
|
let mut values = Vec::with_capacity(len);
|
||||||
|
|
||||||
|
while let Some(value) = try!(visitor.visit()) {
|
||||||
|
values.push(value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
try!(visitor.end());
|
||||||
|
|
||||||
|
Ok(ByteBuf {
|
||||||
|
bytes: values,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AsRef<[u8]> for ByteBuf {
|
#[inline]
|
||||||
fn as_ref(&self) -> &[u8] {
|
fn visit_bytes<E>(&mut self, v: &[u8]) -> Result<ByteBuf, E>
|
||||||
&self.bytes
|
where E: de::Error,
|
||||||
}
|
{
|
||||||
|
self.visit_byte_buf(v.to_vec())
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AsMut<Vec<u8>> for ByteBuf {
|
#[inline]
|
||||||
fn as_mut(&mut self) -> &mut Vec<u8> {
|
fn visit_byte_buf<E>(&mut self, v: Vec<u8>) -> Result<ByteBuf, E>
|
||||||
&mut self.bytes
|
where E: de::Error,
|
||||||
}
|
{
|
||||||
|
Ok(ByteBuf {
|
||||||
|
bytes: v,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl AsMut<[u8]> for ByteBuf {
|
impl de::Deserialize for ByteBuf {
|
||||||
fn as_mut(&mut self) -> &mut [u8] {
|
#[inline]
|
||||||
&mut self.bytes
|
fn deserialize<D>(deserializer: &mut D) -> Result<ByteBuf, D::Error>
|
||||||
}
|
where D: de::Deserializer
|
||||||
}
|
{
|
||||||
|
deserializer.visit_bytes(ByteBufVisitor)
|
||||||
impl ops::Deref for ByteBuf {
|
|
||||||
type Target = [u8];
|
|
||||||
|
|
||||||
fn deref(&self) -> &[u8] { &self.bytes[..] }
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ops::DerefMut for ByteBuf {
|
|
||||||
fn deref_mut(&mut self) -> &mut [u8] { &mut self.bytes[..] }
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ser::Serialize for ByteBuf {
|
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
|
||||||
where S: ser::Serializer
|
|
||||||
{
|
|
||||||
serializer.serialize_bytes(self)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// This type implements the `serde::de::Visitor` trait for a `ByteBuf`.
|
|
||||||
pub struct ByteBufVisitor;
|
|
||||||
|
|
||||||
impl de::Visitor for ByteBufVisitor {
|
|
||||||
type Value = ByteBuf;
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn visit_unit<E>(&mut self) -> Result<ByteBuf, E>
|
|
||||||
where E: de::Error,
|
|
||||||
{
|
|
||||||
Ok(ByteBuf {
|
|
||||||
bytes: Vec::new(),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn visit_seq<V>(&mut self, mut visitor: V) -> Result<ByteBuf, V::Error>
|
|
||||||
where V: de::SeqVisitor,
|
|
||||||
{
|
|
||||||
let (len, _) = visitor.size_hint();
|
|
||||||
let mut values = Vec::with_capacity(len);
|
|
||||||
|
|
||||||
while let Some(value) = try!(visitor.visit()) {
|
|
||||||
values.push(value);
|
|
||||||
}
|
|
||||||
|
|
||||||
try!(visitor.end());
|
|
||||||
|
|
||||||
Ok(ByteBuf {
|
|
||||||
bytes: values,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn visit_bytes<E>(&mut self, v: &[u8]) -> Result<ByteBuf, E>
|
|
||||||
where E: de::Error,
|
|
||||||
{
|
|
||||||
self.visit_byte_buf(v.to_vec())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn visit_byte_buf<E>(&mut self, v: Vec<u8>) -> Result<ByteBuf, E>
|
|
||||||
where E: de::Error,
|
|
||||||
{
|
|
||||||
Ok(ByteBuf {
|
|
||||||
bytes: v,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl de::Deserialize for ByteBuf {
|
|
||||||
#[inline]
|
|
||||||
fn deserialize<D>(deserializer: &mut D) -> Result<ByteBuf, D::Error>
|
|
||||||
where D: de::Deserializer
|
|
||||||
{
|
|
||||||
deserializer.deserialize_bytes(ByteBufVisitor)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
#[inline]
|
fn escape_bytestring(bytes: &[u8]) -> String {
|
||||||
fn escape_bytestring<'a>(bytes: &'a [u8]) -> iter::FlatMap<slice::Iter<'a, u8>, char::EscapeDefault, fn(&u8) -> char::EscapeDefault> {
|
let mut result = String::new();
|
||||||
fn f(b: &u8) -> char::EscapeDefault {
|
for &b in bytes {
|
||||||
char::from_u32(*b as u32).unwrap().escape_default()
|
for esc in ascii::escape_default(b) {
|
||||||
|
result.push(esc as char);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
bytes.iter().flat_map(f as fn(&u8) -> char::EscapeDefault)
|
result
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,409 +0,0 @@
|
|||||||
// Copyright 2013-2014 The Rust Project Developers. See the COPYRIGHT
|
|
||||||
// file at the top-level directory of this distribution and at
|
|
||||||
// http://rust-lang.org/COPYRIGHT.
|
|
||||||
//
|
|
||||||
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
||||||
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
||||||
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
||||||
// option. This file may not be copied, modified, or distributed
|
|
||||||
// except according to those terms.
|
|
||||||
|
|
||||||
// Extracted from https://github.com/rust-num/num.
|
|
||||||
|
|
||||||
// Rust 1.5 is unhappy that this private module is undocumented.
|
|
||||||
#![allow(missing_docs)]
|
|
||||||
|
|
||||||
use core::{usize, u8, u16, u32, u64};
|
|
||||||
use core::{isize, i8, i16, i32, i64};
|
|
||||||
use core::{f32, f64};
|
|
||||||
use core::mem::size_of;
|
|
||||||
|
|
||||||
/// Numbers which have upper and lower bounds
|
|
||||||
pub trait Bounded {
|
|
||||||
// FIXME (#5527): These should be associated constants
|
|
||||||
/// returns the smallest finite number this type can represent
|
|
||||||
fn min_value() -> Self;
|
|
||||||
/// returns the largest finite number this type can represent
|
|
||||||
fn max_value() -> Self;
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! bounded_impl {
|
|
||||||
($t:ty, $min:expr, $max:expr) => {
|
|
||||||
impl Bounded for $t {
|
|
||||||
#[inline]
|
|
||||||
fn min_value() -> $t { $min }
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn max_value() -> $t { $max }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bounded_impl!(usize, usize::MIN, usize::MAX);
|
|
||||||
bounded_impl!(u8, u8::MIN, u8::MAX);
|
|
||||||
bounded_impl!(u16, u16::MIN, u16::MAX);
|
|
||||||
bounded_impl!(u32, u32::MIN, u32::MAX);
|
|
||||||
bounded_impl!(u64, u64::MIN, u64::MAX);
|
|
||||||
|
|
||||||
bounded_impl!(isize, isize::MIN, isize::MAX);
|
|
||||||
bounded_impl!(i8, i8::MIN, i8::MAX);
|
|
||||||
bounded_impl!(i16, i16::MIN, i16::MAX);
|
|
||||||
bounded_impl!(i32, i32::MIN, i32::MAX);
|
|
||||||
bounded_impl!(i64, i64::MIN, i64::MAX);
|
|
||||||
|
|
||||||
bounded_impl!(f32, f32::MIN, f32::MAX);
|
|
||||||
bounded_impl!(f64, f64::MIN, f64::MAX);
|
|
||||||
|
|
||||||
/// A generic trait for converting a value to a number.
|
|
||||||
pub trait ToPrimitive {
|
|
||||||
/// Converts the value of `self` to an `isize`.
|
|
||||||
#[inline]
|
|
||||||
fn to_isize(&self) -> Option<isize> {
|
|
||||||
self.to_i64().and_then(|x| x.to_isize())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Converts the value of `self` to an `i8`.
|
|
||||||
#[inline]
|
|
||||||
fn to_i8(&self) -> Option<i8> {
|
|
||||||
self.to_i64().and_then(|x| x.to_i8())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Converts the value of `self` to an `i16`.
|
|
||||||
#[inline]
|
|
||||||
fn to_i16(&self) -> Option<i16> {
|
|
||||||
self.to_i64().and_then(|x| x.to_i16())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Converts the value of `self` to an `i32`.
|
|
||||||
#[inline]
|
|
||||||
fn to_i32(&self) -> Option<i32> {
|
|
||||||
self.to_i64().and_then(|x| x.to_i32())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Converts the value of `self` to an `i64`.
|
|
||||||
fn to_i64(&self) -> Option<i64>;
|
|
||||||
|
|
||||||
/// Converts the value of `self` to a `usize`.
|
|
||||||
#[inline]
|
|
||||||
fn to_usize(&self) -> Option<usize> {
|
|
||||||
self.to_u64().and_then(|x| x.to_usize())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Converts the value of `self` to an `u8`.
|
|
||||||
#[inline]
|
|
||||||
fn to_u8(&self) -> Option<u8> {
|
|
||||||
self.to_u64().and_then(|x| x.to_u8())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Converts the value of `self` to an `u16`.
|
|
||||||
#[inline]
|
|
||||||
fn to_u16(&self) -> Option<u16> {
|
|
||||||
self.to_u64().and_then(|x| x.to_u16())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Converts the value of `self` to an `u32`.
|
|
||||||
#[inline]
|
|
||||||
fn to_u32(&self) -> Option<u32> {
|
|
||||||
self.to_u64().and_then(|x| x.to_u32())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Converts the value of `self` to an `u64`.
|
|
||||||
#[inline]
|
|
||||||
fn to_u64(&self) -> Option<u64>;
|
|
||||||
|
|
||||||
/// Converts the value of `self` to an `f32`.
|
|
||||||
#[inline]
|
|
||||||
fn to_f32(&self) -> Option<f32> {
|
|
||||||
self.to_f64().and_then(|x| x.to_f32())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Converts the value of `self` to an `f64`.
|
|
||||||
#[inline]
|
|
||||||
fn to_f64(&self) -> Option<f64> {
|
|
||||||
self.to_i64().and_then(|x| x.to_f64())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! impl_to_primitive_int_to_int {
|
|
||||||
($SrcT:ty, $DstT:ty, $slf:expr) => (
|
|
||||||
{
|
|
||||||
if size_of::<$SrcT>() <= size_of::<$DstT>() {
|
|
||||||
Some($slf as $DstT)
|
|
||||||
} else {
|
|
||||||
let n = $slf as i64;
|
|
||||||
let min_value: $DstT = Bounded::min_value();
|
|
||||||
let max_value: $DstT = Bounded::max_value();
|
|
||||||
if min_value as i64 <= n && n <= max_value as i64 {
|
|
||||||
Some($slf as $DstT)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! impl_to_primitive_int_to_uint {
|
|
||||||
($SrcT:ty, $DstT:ty, $slf:expr) => (
|
|
||||||
{
|
|
||||||
let zero: $SrcT = 0;
|
|
||||||
let max_value: $DstT = Bounded::max_value();
|
|
||||||
if zero <= $slf && $slf as u64 <= max_value as u64 {
|
|
||||||
Some($slf as $DstT)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! impl_to_primitive_int {
|
|
||||||
($T:ty) => (
|
|
||||||
impl ToPrimitive for $T {
|
|
||||||
#[inline]
|
|
||||||
fn to_isize(&self) -> Option<isize> { impl_to_primitive_int_to_int!($T, isize, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_i8(&self) -> Option<i8> { impl_to_primitive_int_to_int!($T, i8, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_i16(&self) -> Option<i16> { impl_to_primitive_int_to_int!($T, i16, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_i32(&self) -> Option<i32> { impl_to_primitive_int_to_int!($T, i32, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_i64(&self) -> Option<i64> { impl_to_primitive_int_to_int!($T, i64, *self) }
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn to_usize(&self) -> Option<usize> { impl_to_primitive_int_to_uint!($T, usize, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_u8(&self) -> Option<u8> { impl_to_primitive_int_to_uint!($T, u8, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_u16(&self) -> Option<u16> { impl_to_primitive_int_to_uint!($T, u16, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_u32(&self) -> Option<u32> { impl_to_primitive_int_to_uint!($T, u32, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_u64(&self) -> Option<u64> { impl_to_primitive_int_to_uint!($T, u64, *self) }
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn to_f32(&self) -> Option<f32> { Some(*self as f32) }
|
|
||||||
#[inline]
|
|
||||||
fn to_f64(&self) -> Option<f64> { Some(*self as f64) }
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
impl_to_primitive_int! { isize }
|
|
||||||
impl_to_primitive_int! { i8 }
|
|
||||||
impl_to_primitive_int! { i16 }
|
|
||||||
impl_to_primitive_int! { i32 }
|
|
||||||
impl_to_primitive_int! { i64 }
|
|
||||||
|
|
||||||
macro_rules! impl_to_primitive_uint_to_int {
|
|
||||||
($DstT:ty, $slf:expr) => (
|
|
||||||
{
|
|
||||||
let max_value: $DstT = Bounded::max_value();
|
|
||||||
if $slf as u64 <= max_value as u64 {
|
|
||||||
Some($slf as $DstT)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! impl_to_primitive_uint_to_uint {
|
|
||||||
($SrcT:ty, $DstT:ty, $slf:expr) => (
|
|
||||||
{
|
|
||||||
if size_of::<$SrcT>() <= size_of::<$DstT>() {
|
|
||||||
Some($slf as $DstT)
|
|
||||||
} else {
|
|
||||||
let zero: $SrcT = 0;
|
|
||||||
let max_value: $DstT = Bounded::max_value();
|
|
||||||
if zero <= $slf && $slf as u64 <= max_value as u64 {
|
|
||||||
Some($slf as $DstT)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! impl_to_primitive_uint {
|
|
||||||
($T:ty) => (
|
|
||||||
impl ToPrimitive for $T {
|
|
||||||
#[inline]
|
|
||||||
fn to_isize(&self) -> Option<isize> { impl_to_primitive_uint_to_int!(isize, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_i8(&self) -> Option<i8> { impl_to_primitive_uint_to_int!(i8, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_i16(&self) -> Option<i16> { impl_to_primitive_uint_to_int!(i16, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_i32(&self) -> Option<i32> { impl_to_primitive_uint_to_int!(i32, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_i64(&self) -> Option<i64> { impl_to_primitive_uint_to_int!(i64, *self) }
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn to_usize(&self) -> Option<usize> {
|
|
||||||
impl_to_primitive_uint_to_uint!($T, usize, *self)
|
|
||||||
}
|
|
||||||
#[inline]
|
|
||||||
fn to_u8(&self) -> Option<u8> { impl_to_primitive_uint_to_uint!($T, u8, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_u16(&self) -> Option<u16> { impl_to_primitive_uint_to_uint!($T, u16, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_u32(&self) -> Option<u32> { impl_to_primitive_uint_to_uint!($T, u32, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_u64(&self) -> Option<u64> { impl_to_primitive_uint_to_uint!($T, u64, *self) }
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn to_f32(&self) -> Option<f32> { Some(*self as f32) }
|
|
||||||
#[inline]
|
|
||||||
fn to_f64(&self) -> Option<f64> { Some(*self as f64) }
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
impl_to_primitive_uint! { usize }
|
|
||||||
impl_to_primitive_uint! { u8 }
|
|
||||||
impl_to_primitive_uint! { u16 }
|
|
||||||
impl_to_primitive_uint! { u32 }
|
|
||||||
impl_to_primitive_uint! { u64 }
|
|
||||||
|
|
||||||
macro_rules! impl_to_primitive_float_to_float {
|
|
||||||
($SrcT:ident, $DstT:ident, $slf:expr) => (
|
|
||||||
if size_of::<$SrcT>() <= size_of::<$DstT>() {
|
|
||||||
Some($slf as $DstT)
|
|
||||||
} else {
|
|
||||||
let n = $slf as f64;
|
|
||||||
let max_value: $SrcT = ::core::$SrcT::MAX;
|
|
||||||
if -max_value as f64 <= n && n <= max_value as f64 {
|
|
||||||
Some($slf as $DstT)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! impl_to_primitive_float {
|
|
||||||
($T:ident) => (
|
|
||||||
impl ToPrimitive for $T {
|
|
||||||
#[inline]
|
|
||||||
fn to_isize(&self) -> Option<isize> { Some(*self as isize) }
|
|
||||||
#[inline]
|
|
||||||
fn to_i8(&self) -> Option<i8> { Some(*self as i8) }
|
|
||||||
#[inline]
|
|
||||||
fn to_i16(&self) -> Option<i16> { Some(*self as i16) }
|
|
||||||
#[inline]
|
|
||||||
fn to_i32(&self) -> Option<i32> { Some(*self as i32) }
|
|
||||||
#[inline]
|
|
||||||
fn to_i64(&self) -> Option<i64> { Some(*self as i64) }
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn to_usize(&self) -> Option<usize> { Some(*self as usize) }
|
|
||||||
#[inline]
|
|
||||||
fn to_u8(&self) -> Option<u8> { Some(*self as u8) }
|
|
||||||
#[inline]
|
|
||||||
fn to_u16(&self) -> Option<u16> { Some(*self as u16) }
|
|
||||||
#[inline]
|
|
||||||
fn to_u32(&self) -> Option<u32> { Some(*self as u32) }
|
|
||||||
#[inline]
|
|
||||||
fn to_u64(&self) -> Option<u64> { Some(*self as u64) }
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn to_f32(&self) -> Option<f32> { impl_to_primitive_float_to_float!($T, f32, *self) }
|
|
||||||
#[inline]
|
|
||||||
fn to_f64(&self) -> Option<f64> { impl_to_primitive_float_to_float!($T, f64, *self) }
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
impl_to_primitive_float! { f32 }
|
|
||||||
impl_to_primitive_float! { f64 }
|
|
||||||
|
|
||||||
pub trait FromPrimitive: Sized {
|
|
||||||
#[inline]
|
|
||||||
fn from_isize(n: isize) -> Option<Self> {
|
|
||||||
FromPrimitive::from_i64(n as i64)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn from_i8(n: i8) -> Option<Self> {
|
|
||||||
FromPrimitive::from_i64(n as i64)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn from_i16(n: i16) -> Option<Self> {
|
|
||||||
FromPrimitive::from_i64(n as i64)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn from_i32(n: i32) -> Option<Self> {
|
|
||||||
FromPrimitive::from_i64(n as i64)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn from_i64(n: i64) -> Option<Self>;
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn from_usize(n: usize) -> Option<Self> {
|
|
||||||
FromPrimitive::from_u64(n as u64)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn from_u8(n: u8) -> Option<Self> {
|
|
||||||
FromPrimitive::from_u64(n as u64)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn from_u16(n: u16) -> Option<Self> {
|
|
||||||
FromPrimitive::from_u64(n as u64)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn from_u32(n: u32) -> Option<Self> {
|
|
||||||
FromPrimitive::from_u64(n as u64)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn from_u64(n: u64) -> Option<Self>;
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn from_f32(n: f32) -> Option<Self> {
|
|
||||||
FromPrimitive::from_f64(n as f64)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
fn from_f64(n: f64) -> Option<Self> {
|
|
||||||
FromPrimitive::from_i64(n as i64)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! impl_from_primitive {
|
|
||||||
($T:ty, $to_ty:ident) => (
|
|
||||||
impl FromPrimitive for $T {
|
|
||||||
#[inline] fn from_i8(n: i8) -> Option<$T> { n.$to_ty() }
|
|
||||||
#[inline] fn from_i16(n: i16) -> Option<$T> { n.$to_ty() }
|
|
||||||
#[inline] fn from_i32(n: i32) -> Option<$T> { n.$to_ty() }
|
|
||||||
#[inline] fn from_i64(n: i64) -> Option<$T> { n.$to_ty() }
|
|
||||||
|
|
||||||
#[inline] fn from_u8(n: u8) -> Option<$T> { n.$to_ty() }
|
|
||||||
#[inline] fn from_u16(n: u16) -> Option<$T> { n.$to_ty() }
|
|
||||||
#[inline] fn from_u32(n: u32) -> Option<$T> { n.$to_ty() }
|
|
||||||
#[inline] fn from_u64(n: u64) -> Option<$T> { n.$to_ty() }
|
|
||||||
|
|
||||||
#[inline] fn from_f32(n: f32) -> Option<$T> { n.$to_ty() }
|
|
||||||
#[inline] fn from_f64(n: f64) -> Option<$T> { n.$to_ty() }
|
|
||||||
}
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
impl_from_primitive! { isize, to_isize }
|
|
||||||
impl_from_primitive! { i8, to_i8 }
|
|
||||||
impl_from_primitive! { i16, to_i16 }
|
|
||||||
impl_from_primitive! { i32, to_i32 }
|
|
||||||
impl_from_primitive! { i64, to_i64 }
|
|
||||||
impl_from_primitive! { usize, to_usize }
|
|
||||||
impl_from_primitive! { u8, to_u8 }
|
|
||||||
impl_from_primitive! { u16, to_u16 }
|
|
||||||
impl_from_primitive! { u32, to_u32 }
|
|
||||||
impl_from_primitive! { u64, to_u64 }
|
|
||||||
impl_from_primitive! { f32, to_f32 }
|
|
||||||
impl_from_primitive! { f64, to_f64 }
|
|
||||||
+339
-812
File diff suppressed because it is too large
Load Diff
+150
-567
@@ -1,287 +1,23 @@
|
|||||||
//! Generic deserialization framework.
|
//! Generic deserialization framework.
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
use std::error;
|
|
||||||
#[cfg(not(feature = "std"))]
|
|
||||||
use error;
|
|
||||||
|
|
||||||
#[cfg(all(not(feature = "std"), feature = "collections"))]
|
|
||||||
use collections::{String, Vec};
|
|
||||||
|
|
||||||
use core::fmt;
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
/// Macro helper to not have to re-implement all the defaulted methods.
|
|
||||||
/// Every given method ignores all arguments and forwards to `deserialize`.
|
|
||||||
/// Note that `deserialize_enum` simply returns an `Error::invalid_type`.
|
|
||||||
macro_rules! de_forward_to_deserialize {
|
|
||||||
($($func:ident),*) => {
|
|
||||||
$(de_forward_to_deserialize!{func: $func})*
|
|
||||||
};
|
|
||||||
(func: deserialize_unit_struct) => {
|
|
||||||
de_forward_to_deserialize!{named: deserialize_unit_struct}
|
|
||||||
};
|
|
||||||
(func: deserialize_newtype_struct) => {
|
|
||||||
de_forward_to_deserialize!{named: deserialize_newtype_struct}
|
|
||||||
};
|
|
||||||
(func: deserialize_tuple) => {
|
|
||||||
de_forward_to_deserialize!{tup_fn: deserialize_tuple}
|
|
||||||
};
|
|
||||||
(func: deserialize_seq_fixed_size) => {
|
|
||||||
de_forward_to_deserialize!{tup_fn: deserialize_seq_fixed_size}
|
|
||||||
};
|
|
||||||
(func: deserialize_tuple_struct) => {
|
|
||||||
#[inline]
|
|
||||||
fn deserialize_tuple_struct<__V>(&mut self, _: &str, _: usize, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: $crate::de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
};
|
|
||||||
(func: deserialize_struct) => {
|
|
||||||
#[inline]
|
|
||||||
fn deserialize_struct<__V>(&mut self, _: &str, _: &[&str], visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: $crate::de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
};
|
|
||||||
(func: deserialize_enum) => {
|
|
||||||
#[inline]
|
|
||||||
fn deserialize_enum<__V>(&mut self, _: &str, _: &[&str], _: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: $crate::de::EnumVisitor {
|
|
||||||
Err($crate::de::Error::invalid_type($crate::de::Type::Enum))
|
|
||||||
}
|
|
||||||
};
|
|
||||||
(named: $func:ident) => {
|
|
||||||
#[inline]
|
|
||||||
fn $func<__V>(&mut self, _: &str, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: $crate::de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
};
|
|
||||||
(tup_fn: $func: ident) => {
|
|
||||||
#[inline]
|
|
||||||
fn $func<__V>(&mut self, _: usize, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: $crate::de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
};
|
|
||||||
(func: $func:ident) => {
|
|
||||||
#[inline]
|
|
||||||
fn $func<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: $crate::de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
// modules come after macros
|
|
||||||
|
|
||||||
pub mod impls;
|
pub mod impls;
|
||||||
pub mod value;
|
pub mod value;
|
||||||
mod from_primitive;
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
/// `Error` is a trait that allows a `Deserialize` to generically create a
|
pub trait Error {
|
||||||
/// `Deserializer` error.
|
fn syntax_error() -> Self;
|
||||||
pub trait Error: Sized + error::Error {
|
|
||||||
/// Raised when there is general error when deserializing a type.
|
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
fn custom<T: Into<String>>(msg: T) -> Self;
|
|
||||||
|
|
||||||
/// Raised when there is general error when deserializing a type.
|
fn end_of_stream_error() -> Self;
|
||||||
#[cfg(all(not(feature = "std"), not(feature = "collections")))]
|
|
||||||
fn custom<T: Into<&'static str>>(msg: T) -> Self;
|
|
||||||
|
|
||||||
/// Raised when a `Deserialize` type unexpectedly hit the end of the stream.
|
fn unknown_field_error(field: &str) -> Self;
|
||||||
fn end_of_stream() -> Self;
|
|
||||||
|
|
||||||
/// Raised when a `Deserialize` was passed an incorrect type.
|
fn missing_field_error(field: &'static str) -> Self;
|
||||||
fn invalid_type(ty: Type) -> Self {
|
|
||||||
Error::custom(format!("Invalid type. Expected `{:?}`", ty))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Raised when a `Deserialize` was passed an incorrect value.
|
|
||||||
fn invalid_value(msg: &str) -> Self {
|
|
||||||
Error::custom(format!("Invalid value: {}", msg))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Raised when a fixed sized sequence or map was passed in the wrong amount of arguments.
|
|
||||||
///
|
|
||||||
/// The parameter `len` is the number of arguments found in the serialization. The sequence
|
|
||||||
/// may either expect more arguments or less arguments.
|
|
||||||
fn invalid_length(len: usize) -> Self {
|
|
||||||
Error::custom(format!("Invalid length: {}", len))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Raised when a `Deserialize` enum type received an unexpected variant.
|
|
||||||
fn unknown_variant(field: &str) -> Self {
|
|
||||||
Error::custom(format!("Unknown variant `{}`", field))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Raised when a `Deserialize` struct type received an unexpected struct field.
|
|
||||||
fn unknown_field(field: &str) -> Self {
|
|
||||||
Error::custom(format!("Unknown field `{}`", field))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// raised when a `deserialize` struct type did not receive a field.
|
|
||||||
fn missing_field(field: &'static str) -> Self {
|
|
||||||
Error::custom(format!("Missing field `{}`", field))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Raised when a `Deserialize` struct type received more than one of the
|
|
||||||
/// same struct field.
|
|
||||||
fn duplicate_field(field: &'static str) -> Self {
|
|
||||||
Error::custom(format!("Duplicate field `{}`", field))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// `Type` represents all the primitive types that can be deserialized. This is used by
|
|
||||||
/// `Error::invalid_type`.
|
|
||||||
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
|
|
||||||
pub enum Type {
|
|
||||||
/// Represents a `bool` type.
|
|
||||||
Bool,
|
|
||||||
|
|
||||||
/// Represents a `usize` type.
|
|
||||||
Usize,
|
|
||||||
|
|
||||||
/// Represents a `u8` type.
|
|
||||||
U8,
|
|
||||||
|
|
||||||
/// Represents a `u16` type.
|
|
||||||
U16,
|
|
||||||
|
|
||||||
/// Represents a `u32` type.
|
|
||||||
U32,
|
|
||||||
|
|
||||||
/// Represents a `u64` type.
|
|
||||||
U64,
|
|
||||||
|
|
||||||
/// Represents a `isize` type.
|
|
||||||
Isize,
|
|
||||||
|
|
||||||
/// Represents a `i8` type.
|
|
||||||
I8,
|
|
||||||
|
|
||||||
/// Represents a `i16` type.
|
|
||||||
I16,
|
|
||||||
|
|
||||||
/// Represents a `i32` type.
|
|
||||||
I32,
|
|
||||||
|
|
||||||
/// Represents a `i64` type.
|
|
||||||
I64,
|
|
||||||
|
|
||||||
/// Represents a `f32` type.
|
|
||||||
F32,
|
|
||||||
|
|
||||||
/// Represents a `f64` type.
|
|
||||||
F64,
|
|
||||||
|
|
||||||
/// Represents a `char` type.
|
|
||||||
Char,
|
|
||||||
|
|
||||||
/// Represents a `&str` type.
|
|
||||||
Str,
|
|
||||||
|
|
||||||
/// Represents a `String` type.
|
|
||||||
String,
|
|
||||||
|
|
||||||
/// Represents a `()` type.
|
|
||||||
Unit,
|
|
||||||
|
|
||||||
/// Represents an `Option<T>` type.
|
|
||||||
Option,
|
|
||||||
|
|
||||||
/// Represents a sequence type.
|
|
||||||
Seq,
|
|
||||||
|
|
||||||
/// Represents a map type.
|
|
||||||
Map,
|
|
||||||
|
|
||||||
/// Represents a unit struct type.
|
|
||||||
UnitStruct,
|
|
||||||
|
|
||||||
/// Represents a newtype type.
|
|
||||||
NewtypeStruct,
|
|
||||||
|
|
||||||
/// Represents a tuple struct type.
|
|
||||||
TupleStruct,
|
|
||||||
|
|
||||||
/// Represents a struct type.
|
|
||||||
Struct,
|
|
||||||
|
|
||||||
/// Represents a struct field name.
|
|
||||||
FieldName,
|
|
||||||
|
|
||||||
/// Represents a tuple type.
|
|
||||||
Tuple,
|
|
||||||
|
|
||||||
/// Represents an `enum` type.
|
|
||||||
Enum,
|
|
||||||
|
|
||||||
/// Represents an enum variant name.
|
|
||||||
VariantName,
|
|
||||||
|
|
||||||
/// Represents a struct variant.
|
|
||||||
StructVariant,
|
|
||||||
|
|
||||||
/// Represents a tuple variant.
|
|
||||||
TupleVariant,
|
|
||||||
|
|
||||||
/// Represents a unit variant.
|
|
||||||
UnitVariant,
|
|
||||||
|
|
||||||
/// Represents a `&[u8]` type.
|
|
||||||
Bytes,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl fmt::Display for Type {
|
|
||||||
fn fmt(&self, formatter: &mut fmt::Formatter) -> Result<(), fmt::Error> {
|
|
||||||
let display = match *self {
|
|
||||||
Type::Bool => "bool",
|
|
||||||
Type::Usize => "usize",
|
|
||||||
Type::U8 => "u8",
|
|
||||||
Type::U16 => "u16",
|
|
||||||
Type::U32 => "u32",
|
|
||||||
Type::U64 => "u64",
|
|
||||||
Type::Isize => "isize",
|
|
||||||
Type::I8 => "i8",
|
|
||||||
Type::I16 => "i16",
|
|
||||||
Type::I32 => "i32",
|
|
||||||
Type::I64 => "i64",
|
|
||||||
Type::F32 => "f32",
|
|
||||||
Type::F64 => "f64",
|
|
||||||
Type::Char => "char",
|
|
||||||
Type::Str => "str",
|
|
||||||
Type::String => "string",
|
|
||||||
Type::Unit => "unit",
|
|
||||||
Type::Option => "option",
|
|
||||||
Type::Seq => "seq",
|
|
||||||
Type::Map => "map",
|
|
||||||
Type::UnitStruct => "unit struct",
|
|
||||||
Type::NewtypeStruct => "newtype struct",
|
|
||||||
Type::TupleStruct => "tuple struct",
|
|
||||||
Type::Struct => "struct",
|
|
||||||
Type::FieldName => "field name",
|
|
||||||
Type::Tuple => "tuple",
|
|
||||||
Type::Enum => "enum",
|
|
||||||
Type::VariantName => "variant name",
|
|
||||||
Type::StructVariant => "struct variant",
|
|
||||||
Type::TupleVariant => "tuple variant",
|
|
||||||
Type::UnitVariant => "unit variant",
|
|
||||||
Type::Bytes => "bytes",
|
|
||||||
};
|
|
||||||
display.fmt(formatter)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
/// `Deserialize` represents a type that can be deserialized.
|
pub trait Deserialize {
|
||||||
pub trait Deserialize: Sized {
|
|
||||||
/// Deserialize this value given this `Deserializer`.
|
/// Deserialize this value given this `Deserializer`.
|
||||||
fn deserialize<D>(deserializer: &mut D) -> Result<Self, D::Error>
|
fn deserialize<D>(deserializer: &mut D) -> Result<Self, D::Error>
|
||||||
where D: Deserializer;
|
where D: Deserializer;
|
||||||
@@ -289,383 +25,253 @@ pub trait Deserialize: Sized {
|
|||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
/// `Deserializer` is a trait that can deserialize values by threading a `Visitor` trait through a
|
/// `Deserializer` is an abstract trait that can deserialize values into a `Visitor`.
|
||||||
/// value. It supports two entry point styles which enables different kinds of deserialization.
|
|
||||||
///
|
|
||||||
/// 1) The `deserialize` method. File formats like JSON embed the type of its construct in its file
|
|
||||||
/// format. This allows the `Deserializer` to deserialize into a generic type like
|
|
||||||
/// `json::Value`, which can represent all JSON types.
|
|
||||||
///
|
|
||||||
/// 2) The `deserialize_*` methods. File formats like bincode do not embed in its format how to
|
|
||||||
/// decode its values. It relies instead on the `Deserialize` type to hint to the `Deserializer`
|
|
||||||
/// with the `deserialize_*` methods how it should parse the next value. One downside though to
|
|
||||||
/// only supporting the `deserialize_*` types is that it does not allow for deserializing into a
|
|
||||||
/// generic `json::Value`-esque type.
|
|
||||||
pub trait Deserializer {
|
pub trait Deserializer {
|
||||||
/// The error type that can be returned if some error occurs during deserialization.
|
|
||||||
type Error: Error;
|
type Error: Error;
|
||||||
|
|
||||||
/// This method walks a visitor through a value as it is being deserialized.
|
/// The `visit` method walks a visitor through a value as it is being deserialized.
|
||||||
fn deserialize<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
fn visit<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
where V: Visitor;
|
where V: Visitor;
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a `bool` value.
|
/// The `visit_option` method allows a `Deserialize` type to inform the `Deserializer` that
|
||||||
fn deserialize_bool<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
/// it's expecting an optional value. This allows deserializers that encode an optional value
|
||||||
where V: Visitor;
|
/// as a nullable value to convert the null value into a `None`, and a regular value as
|
||||||
|
/// `Some(value)`.
|
||||||
|
#[inline]
|
||||||
|
fn visit_option<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where V: Visitor,
|
||||||
|
{
|
||||||
|
self.visit(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting an `usize` value.
|
/// The `visit_seq` method allows a `Deserialize` type to inform the `Deserializer` that it's
|
||||||
/// A reasonable default is to forward to `deserialize_u64`.
|
/// expecting a sequence of values. This allows deserializers to parse sequences that aren't
|
||||||
fn deserialize_usize<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
/// tagged as sequences.
|
||||||
where V: Visitor;
|
#[inline]
|
||||||
|
fn visit_seq<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where V: Visitor,
|
||||||
|
{
|
||||||
|
self.visit(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting an `u8` value.
|
/// The `visit_map` method allows a `Deserialize` type to inform the `Deserializer` that it's
|
||||||
/// A reasonable default is to forward to `deserialize_u64`.
|
/// expecting a map of values. This allows deserializers to parse sequences that aren't tagged
|
||||||
fn deserialize_u8<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
/// as maps.
|
||||||
where V: Visitor;
|
#[inline]
|
||||||
|
fn visit_map<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where V: Visitor,
|
||||||
|
{
|
||||||
|
self.visit(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting an `u16` value.
|
/// The `visit_named_unit` method allows a `Deserialize` type to inform the `Deserializer` that
|
||||||
/// A reasonable default is to forward to `deserialize_u64`.
|
/// it's expecting a named unit. This allows deserializers to a named unit that aren't tagged
|
||||||
fn deserialize_u16<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
/// as a named unit.
|
||||||
where V: Visitor;
|
#[inline]
|
||||||
|
fn visit_named_unit<V>(&mut self, _name: &str, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where V: Visitor,
|
||||||
|
{
|
||||||
|
self.visit(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting an `u32` value.
|
/// The `visit_named_seq` method allows a `Deserialize` type to inform the `Deserializer` that
|
||||||
/// A reasonable default is to forward to `deserialize_u64`.
|
/// it's expecting a named sequence of values. This allows deserializers to parse sequences
|
||||||
fn deserialize_u32<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
/// that aren't tagged as sequences.
|
||||||
where V: Visitor;
|
#[inline]
|
||||||
|
fn visit_named_seq<V>(&mut self, _name: &str, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where V: Visitor,
|
||||||
|
{
|
||||||
|
self.visit_seq(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting an `u64` value.
|
/// The `visit_named_map` method allows a `Deserialize` type to inform the `Deserializer` that
|
||||||
fn deserialize_u64<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
/// it's expecting a map of values. This allows deserializers to parse sequences that aren't
|
||||||
where V: Visitor;
|
/// tagged as maps.
|
||||||
|
#[inline]
|
||||||
|
fn visit_named_map<V>(&mut self, _name: &str, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where V: Visitor,
|
||||||
|
{
|
||||||
|
self.visit_map(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting an `isize` value.
|
/// The `visit_enum` method allows a `Deserialize` type to inform the `Deserializer` that it's
|
||||||
/// A reasonable default is to forward to `deserialize_i64`.
|
/// expecting an enum value. This allows deserializers that provide a custom enumeration
|
||||||
fn deserialize_isize<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
/// serialization to properly deserialize the type.
|
||||||
where V: Visitor;
|
#[inline]
|
||||||
|
fn visit_enum<V>(&mut self, _enum: &str, _visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where V: EnumVisitor,
|
||||||
|
{
|
||||||
|
Err(Error::syntax_error())
|
||||||
|
}
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting an `i8` value.
|
/// The `visit_bytes` method allows a `Deserialize` type to inform the `Deserializer` that it's
|
||||||
/// A reasonable default is to forward to `deserialize_i64`.
|
/// expecting a `Vec<u8>`. This allows deserializers that provide a custom byte vector
|
||||||
fn deserialize_i8<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
/// serialization to properly deserialize the type.
|
||||||
where V: Visitor;
|
#[inline]
|
||||||
|
fn visit_bytes<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
|
where V: Visitor,
|
||||||
|
{
|
||||||
|
self.visit(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting an `i16` value.
|
/// Specify a format string for the deserializer.
|
||||||
/// A reasonable default is to forward to `deserialize_i64`.
|
///
|
||||||
fn deserialize_i16<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
/// The deserializer format is used to determine which format
|
||||||
where V: Visitor;
|
/// specific field attributes should be used with the
|
||||||
|
/// deserializer.
|
||||||
/// This method hints that the `Deserialize` type is expecting an `i32` value.
|
fn format() -> &'static str {
|
||||||
/// A reasonable default is to forward to `deserialize_i64`.
|
""
|
||||||
fn deserialize_i32<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
}
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting an `i64` value.
|
|
||||||
fn deserialize_i64<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a `f32` value.
|
|
||||||
/// A reasonable default is to forward to `deserialize_f64`.
|
|
||||||
fn deserialize_f32<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a `f64` value.
|
|
||||||
fn deserialize_f64<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a `char` value.
|
|
||||||
fn deserialize_char<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a `&str` value.
|
|
||||||
fn deserialize_str<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a `String` value.
|
|
||||||
fn deserialize_string<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting an `unit` value.
|
|
||||||
fn deserialize_unit<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting an `Option` value. This allows
|
|
||||||
/// deserializers that encode an optional value as a nullable value to convert the null value
|
|
||||||
/// into a `None`, and a regular value as `Some(value)`.
|
|
||||||
fn deserialize_option<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a sequence value. This allows
|
|
||||||
/// deserializers to parse sequences that aren't tagged as sequences.
|
|
||||||
fn deserialize_seq<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a fixed size array. This allows
|
|
||||||
/// deserializers to parse arrays that aren't tagged as arrays.
|
|
||||||
fn deserialize_seq_fixed_size<V>(&mut self,
|
|
||||||
len: usize,
|
|
||||||
visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a `Vec<u8>`. This allows
|
|
||||||
/// deserializers that provide a custom byte vector serialization to properly deserialize the
|
|
||||||
/// type.
|
|
||||||
fn deserialize_bytes<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a map of values. This allows
|
|
||||||
/// deserializers to parse sequences that aren't tagged as maps.
|
|
||||||
fn deserialize_map<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a unit struct. This allows
|
|
||||||
/// deserializers to a unit struct that aren't tagged as a unit struct.
|
|
||||||
fn deserialize_unit_struct<V>(&mut self,
|
|
||||||
name: &'static str,
|
|
||||||
visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a newtype struct. This allows
|
|
||||||
/// deserializers to a newtype struct that aren't tagged as a newtype struct.
|
|
||||||
/// A reasonable default is to simply deserialize the expected value directly.
|
|
||||||
fn deserialize_newtype_struct<V>(&mut self,
|
|
||||||
name: &'static str,
|
|
||||||
visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a tuple struct. This allows
|
|
||||||
/// deserializers to parse sequences that aren't tagged as sequences.
|
|
||||||
fn deserialize_tuple_struct<V>(&mut self,
|
|
||||||
name: &'static str,
|
|
||||||
len: usize,
|
|
||||||
visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a struct. This allows
|
|
||||||
/// deserializers to parse sequences that aren't tagged as maps.
|
|
||||||
fn deserialize_struct<V>(&mut self,
|
|
||||||
name: &'static str,
|
|
||||||
fields: &'static [&'static str],
|
|
||||||
visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting some sort of struct field
|
|
||||||
/// name. This allows deserializers to choose between &str, usize, or &[u8] to properly
|
|
||||||
/// deserialize a struct field.
|
|
||||||
fn deserialize_struct_field<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting a tuple value. This allows
|
|
||||||
/// deserializers that provide a custom tuple serialization to properly deserialize the type.
|
|
||||||
fn deserialize_tuple<V>(&mut self, len: usize, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type is expecting an enum value. This allows
|
|
||||||
/// deserializers that provide a custom enumeration serialization to properly deserialize the
|
|
||||||
/// type.
|
|
||||||
fn deserialize_enum<V>(&mut self,
|
|
||||||
name: &'static str,
|
|
||||||
variants: &'static [&'static str],
|
|
||||||
visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: EnumVisitor;
|
|
||||||
|
|
||||||
/// This method hints that the `Deserialize` type needs to deserialize a value whose type
|
|
||||||
/// doesn't matter because it is ignored.
|
|
||||||
fn deserialize_ignored_any<V>(&mut self, visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
/// This trait represents a visitor that walks through a deserializer.
|
|
||||||
pub trait Visitor {
|
pub trait Visitor {
|
||||||
/// The value produced by this visitor.
|
|
||||||
type Value: Deserialize;
|
type Value: Deserialize;
|
||||||
|
|
||||||
/// `visit_bool` deserializes a `bool` into a `Value`.
|
fn visit_bool<E>(&mut self, _v: bool) -> Result<Self::Value, E>
|
||||||
fn visit_bool<E>(&mut self, v: bool) -> Result<Self::Value, E>
|
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
let _ = v;
|
Err(Error::syntax_error())
|
||||||
Err(Error::invalid_type(Type::Bool))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_isize` deserializes a `isize` into a `Value`.
|
|
||||||
fn visit_isize<E>(&mut self, v: isize) -> Result<Self::Value, E>
|
fn visit_isize<E>(&mut self, v: isize) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
self.visit_i64(v as i64)
|
self.visit_i64(v as i64)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_i8` deserializes a `i8` into a `Value`.
|
|
||||||
fn visit_i8<E>(&mut self, v: i8) -> Result<Self::Value, E>
|
fn visit_i8<E>(&mut self, v: i8) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
self.visit_i64(v as i64)
|
self.visit_i64(v as i64)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_i16` deserializes a `i16` into a `Value`.
|
|
||||||
fn visit_i16<E>(&mut self, v: i16) -> Result<Self::Value, E>
|
fn visit_i16<E>(&mut self, v: i16) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
self.visit_i64(v as i64)
|
self.visit_i64(v as i64)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_i32` deserializes a `i32` into a `Value`.
|
|
||||||
fn visit_i32<E>(&mut self, v: i32) -> Result<Self::Value, E>
|
fn visit_i32<E>(&mut self, v: i32) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
self.visit_i64(v as i64)
|
self.visit_i64(v as i64)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_i64` deserializes a `i64` into a `Value`.
|
fn visit_i64<E>(&mut self, _v: i64) -> Result<Self::Value, E>
|
||||||
fn visit_i64<E>(&mut self, v: i64) -> Result<Self::Value, E>
|
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
let _ = v;
|
Err(Error::syntax_error())
|
||||||
Err(Error::invalid_type(Type::I64))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_usize` deserializes a `usize` into a `Value`.
|
|
||||||
fn visit_usize<E>(&mut self, v: usize) -> Result<Self::Value, E>
|
fn visit_usize<E>(&mut self, v: usize) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
self.visit_u64(v as u64)
|
self.visit_u64(v as u64)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_u8` deserializes a `u8` into a `Value`.
|
|
||||||
fn visit_u8<E>(&mut self, v: u8) -> Result<Self::Value, E>
|
fn visit_u8<E>(&mut self, v: u8) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
self.visit_u64(v as u64)
|
self.visit_u64(v as u64)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_u16` deserializes a `u16` into a `Value`.
|
|
||||||
fn visit_u16<E>(&mut self, v: u16) -> Result<Self::Value, E>
|
fn visit_u16<E>(&mut self, v: u16) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
self.visit_u64(v as u64)
|
self.visit_u64(v as u64)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_u32` deserializes a `u32` into a `Value`.
|
|
||||||
fn visit_u32<E>(&mut self, v: u32) -> Result<Self::Value, E>
|
fn visit_u32<E>(&mut self, v: u32) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
self.visit_u64(v as u64)
|
self.visit_u64(v as u64)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_u64` deserializes a `u64` into a `Value`.
|
fn visit_u64<E>(&mut self, _v: u64) -> Result<Self::Value, E>
|
||||||
fn visit_u64<E>(&mut self, v: u64) -> Result<Self::Value, E>
|
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
let _ = v;
|
Err(Error::syntax_error())
|
||||||
Err(Error::invalid_type(Type::U64))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_f32` deserializes a `f32` into a `Value`.
|
|
||||||
fn visit_f32<E>(&mut self, v: f32) -> Result<Self::Value, E>
|
fn visit_f32<E>(&mut self, v: f32) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
self.visit_f64(v as f64)
|
self.visit_f64(v as f64)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_f64` deserializes a `f64` into a `Value`.
|
fn visit_f64<E>(&mut self, _v: f64) -> Result<Self::Value, E>
|
||||||
fn visit_f64<E>(&mut self, v: f64) -> Result<Self::Value, E>
|
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
let _ = v;
|
Err(Error::syntax_error())
|
||||||
Err(Error::invalid_type(Type::F64))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_char` deserializes a `char` into a `Value`.
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn visit_char<E>(&mut self, v: char) -> Result<Self::Value, E>
|
fn visit_char<E>(&mut self, v: char) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
self.visit_str(::utils::encode_utf8(v).as_str())
|
// FIXME: this allocation is required in order to be compatible with stable rust, which
|
||||||
|
// doesn't support encoding a `char` into a stack buffer.
|
||||||
|
self.visit_string(v.to_string())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_str` deserializes a `&str` into a `Value`.
|
fn visit_str<E>(&mut self, _v: &str) -> Result<Self::Value, E>
|
||||||
fn visit_str<E>(&mut self, v: &str) -> Result<Self::Value, E>
|
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
let _ = v;
|
Err(Error::syntax_error())
|
||||||
Err(Error::invalid_type(Type::Str))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_string` deserializes a `String` into a `Value`. This allows a deserializer to avoid
|
|
||||||
/// a copy if it is deserializing a string from a `String` type. By default it passes a `&str`
|
|
||||||
/// to the `visit_str` method.
|
|
||||||
#[inline]
|
#[inline]
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
fn visit_string<E>(&mut self, v: String) -> Result<Self::Value, E>
|
fn visit_string<E>(&mut self, v: String) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
self.visit_str(&v)
|
self.visit_str(&v)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_unit` deserializes a `()` into a `Value`.
|
|
||||||
fn visit_unit<E>(&mut self) -> Result<Self::Value, E>
|
fn visit_unit<E>(&mut self) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
Err(Error::invalid_type(Type::Unit))
|
Err(Error::syntax_error())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_unit_struct` deserializes a unit struct into a `Value`.
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn visit_unit_struct<E>(&mut self, name: &'static str) -> Result<Self::Value, E>
|
fn visit_named_unit<E>(&mut self, _name: &str) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
let _ = name;
|
|
||||||
self.visit_unit()
|
self.visit_unit()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_none` deserializes a none value into a `Value`.
|
|
||||||
fn visit_none<E>(&mut self) -> Result<Self::Value, E>
|
fn visit_none<E>(&mut self) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
Err(Error::invalid_type(Type::Option))
|
Err(Error::syntax_error())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_some` deserializes a value into a `Value`.
|
fn visit_some<D>(&mut self, _deserializer: &mut D) -> Result<Self::Value, D::Error>
|
||||||
fn visit_some<D>(&mut self, deserializer: &mut D) -> Result<Self::Value, D::Error>
|
|
||||||
where D: Deserializer,
|
where D: Deserializer,
|
||||||
{
|
{
|
||||||
let _ = deserializer;
|
Err(Error::syntax_error())
|
||||||
Err(Error::invalid_type(Type::Option))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_newtype_struct` deserializes a value into a `Value`.
|
fn visit_seq<V>(&mut self, _visitor: V) -> Result<Self::Value, V::Error>
|
||||||
fn visit_newtype_struct<D>(&mut self, deserializer: &mut D) -> Result<Self::Value, D::Error>
|
|
||||||
where D: Deserializer,
|
|
||||||
{
|
|
||||||
let _ = deserializer;
|
|
||||||
Err(Error::invalid_type(Type::NewtypeStruct))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// `visit_seq` deserializes a `SeqVisitor` into a `Value`.
|
|
||||||
fn visit_seq<V>(&mut self, visitor: V) -> Result<Self::Value, V::Error>
|
|
||||||
where V: SeqVisitor,
|
where V: SeqVisitor,
|
||||||
{
|
{
|
||||||
let _ = visitor;
|
Err(Error::syntax_error())
|
||||||
Err(Error::invalid_type(Type::Seq))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_map` deserializes a `MapVisitor` into a `Value`.
|
fn visit_map<V>(&mut self, _visitor: V) -> Result<Self::Value, V::Error>
|
||||||
fn visit_map<V>(&mut self, visitor: V) -> Result<Self::Value, V::Error>
|
|
||||||
where V: MapVisitor,
|
where V: MapVisitor,
|
||||||
{
|
{
|
||||||
let _ = visitor;
|
Err(Error::syntax_error())
|
||||||
Err(Error::invalid_type(Type::Map))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_bytes` deserializes a `&[u8]` into a `Value`.
|
fn visit_bytes<E>(&mut self, _v: &[u8]) -> Result<Self::Value, E>
|
||||||
fn visit_bytes<E>(&mut self, v: &[u8]) -> Result<Self::Value, E>
|
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
let _ = v;
|
Err(Error::syntax_error())
|
||||||
Err(Error::invalid_type(Type::Bytes))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_byte_buf` deserializes a `Vec<u8>` into a `Value`.
|
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
fn visit_byte_buf<E>(&mut self, v: Vec<u8>) -> Result<Self::Value, E>
|
fn visit_byte_buf<E>(&mut self, v: Vec<u8>) -> Result<Self::Value, E>
|
||||||
where E: Error,
|
where E: Error,
|
||||||
{
|
{
|
||||||
@@ -675,23 +281,14 @@ pub trait Visitor {
|
|||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
/// `SeqVisitor` visits each item in a sequence.
|
|
||||||
///
|
|
||||||
/// This is a trait that a `Deserializer` passes to a `Visitor` implementation, which deserializes
|
|
||||||
/// each item in a sequence.
|
|
||||||
pub trait SeqVisitor {
|
pub trait SeqVisitor {
|
||||||
/// The error type that can be returned if some error occurs during deserialization.
|
|
||||||
type Error: Error;
|
type Error: Error;
|
||||||
|
|
||||||
/// This returns a `Ok(Some(value))` for the next value in the sequence, or `Ok(None)` if there
|
|
||||||
/// are no more remaining items.
|
|
||||||
fn visit<T>(&mut self) -> Result<Option<T>, Self::Error>
|
fn visit<T>(&mut self) -> Result<Option<T>, Self::Error>
|
||||||
where T: Deserialize;
|
where T: Deserialize;
|
||||||
|
|
||||||
/// This signals to the `SeqVisitor` that the `Visitor` does not expect any more items.
|
|
||||||
fn end(&mut self) -> Result<(), Self::Error>;
|
fn end(&mut self) -> Result<(), Self::Error>;
|
||||||
|
|
||||||
/// Return the lower and upper bound of items remaining in the sequence.
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn size_hint(&self) -> (usize, Option<usize>) {
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
||||||
(0, None)
|
(0, None)
|
||||||
@@ -721,15 +318,9 @@ impl<'a, V> SeqVisitor for &'a mut V where V: SeqVisitor {
|
|||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
/// `MapVisitor` visits each item in a sequence.
|
|
||||||
///
|
|
||||||
/// This is a trait that a `Deserializer` passes to a `Visitor` implementation.
|
|
||||||
pub trait MapVisitor {
|
pub trait MapVisitor {
|
||||||
/// The error type that can be returned if some error occurs during deserialization.
|
|
||||||
type Error: Error;
|
type Error: Error;
|
||||||
|
|
||||||
/// This returns a `Ok(Some((key, value)))` for the next (key-value) pair in the map, or
|
|
||||||
/// `Ok(None)` if there are no more remaining items.
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn visit<K, V>(&mut self) -> Result<Option<(K, V)>, Self::Error>
|
fn visit<K, V>(&mut self) -> Result<Option<(K, V)>, Self::Error>
|
||||||
where K: Deserialize,
|
where K: Deserialize,
|
||||||
@@ -744,29 +335,23 @@ pub trait MapVisitor {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// This returns a `Ok(Some(key))` for the next key in the map, or `Ok(None)` if there are no
|
|
||||||
/// more remaining items.
|
|
||||||
fn visit_key<K>(&mut self) -> Result<Option<K>, Self::Error>
|
fn visit_key<K>(&mut self) -> Result<Option<K>, Self::Error>
|
||||||
where K: Deserialize;
|
where K: Deserialize;
|
||||||
|
|
||||||
/// This returns a `Ok(value)` for the next value in the map.
|
|
||||||
fn visit_value<V>(&mut self) -> Result<V, Self::Error>
|
fn visit_value<V>(&mut self) -> Result<V, Self::Error>
|
||||||
where V: Deserialize;
|
where V: Deserialize;
|
||||||
|
|
||||||
/// This signals to the `MapVisitor` that the `Visitor` does not expect any more items.
|
|
||||||
fn end(&mut self) -> Result<(), Self::Error>;
|
fn end(&mut self) -> Result<(), Self::Error>;
|
||||||
|
|
||||||
/// Return the lower and upper bound of items remaining in the sequence.
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn size_hint(&self) -> (usize, Option<usize>) {
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
||||||
(0, None)
|
(0, None)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Report that the struct has a field that wasn't deserialized
|
|
||||||
fn missing_field<V>(&mut self, field: &'static str) -> Result<V, Self::Error>
|
fn missing_field<V>(&mut self, field: &'static str) -> Result<V, Self::Error>
|
||||||
where V: Deserialize,
|
where V: Deserialize,
|
||||||
{
|
{
|
||||||
Err(Error::missing_field(field))
|
Err(Error::missing_field_error(field))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -811,10 +396,8 @@ impl<'a, V_> MapVisitor for &'a mut V_ where V_: MapVisitor {
|
|||||||
/// `EnumVisitor` is a visitor that is created by the `Deserialize` and passed to the
|
/// `EnumVisitor` is a visitor that is created by the `Deserialize` and passed to the
|
||||||
/// `Deserializer` in order to deserialize enums.
|
/// `Deserializer` in order to deserialize enums.
|
||||||
pub trait EnumVisitor {
|
pub trait EnumVisitor {
|
||||||
/// The value produced by this visitor.
|
|
||||||
type Value;
|
type Value;
|
||||||
|
|
||||||
/// Visit the specific variant with the `VariantVisitor`.
|
|
||||||
fn visit<V>(&mut self, visitor: V) -> Result<Self::Value, V::Error>
|
fn visit<V>(&mut self, visitor: V) -> Result<Self::Value, V::Error>
|
||||||
where V: VariantVisitor;
|
where V: VariantVisitor;
|
||||||
}
|
}
|
||||||
@@ -824,7 +407,6 @@ pub trait EnumVisitor {
|
|||||||
/// `VariantVisitor` is a visitor that is created by the `Deserializer` and passed to the
|
/// `VariantVisitor` is a visitor that is created by the `Deserializer` and passed to the
|
||||||
/// `Deserialize` in order to deserialize a specific enum variant.
|
/// `Deserialize` in order to deserialize a specific enum variant.
|
||||||
pub trait VariantVisitor {
|
pub trait VariantVisitor {
|
||||||
/// The error type that can be returned if some error occurs during deserialization.
|
|
||||||
type Error: Error;
|
type Error: Error;
|
||||||
|
|
||||||
/// `visit_variant` is called to identify which variant to deserialize.
|
/// `visit_variant` is called to identify which variant to deserialize.
|
||||||
@@ -833,29 +415,22 @@ pub trait VariantVisitor {
|
|||||||
|
|
||||||
/// `visit_unit` is called when deserializing a variant with no values.
|
/// `visit_unit` is called when deserializing a variant with no values.
|
||||||
fn visit_unit(&mut self) -> Result<(), Self::Error> {
|
fn visit_unit(&mut self) -> Result<(), Self::Error> {
|
||||||
Err(Error::invalid_type(Type::UnitVariant))
|
Err(Error::syntax_error())
|
||||||
}
|
}
|
||||||
|
|
||||||
/// `visit_newtype` is called when deserializing a variant with a single value.
|
/// `visit_seq` is called when deserializing a tuple-like variant.
|
||||||
/// A good default is often to use the `visit_tuple` method to deserialize a `(value,)`.
|
fn visit_seq<V>(&mut self, _visitor: V) -> Result<V::Value, Self::Error>
|
||||||
fn visit_newtype<T>(&mut self) -> Result<T, Self::Error>
|
where V: Visitor
|
||||||
where T: Deserialize;
|
{
|
||||||
|
Err(Error::syntax_error())
|
||||||
|
}
|
||||||
|
|
||||||
/// `visit_tuple` is called when deserializing a tuple-like variant.
|
/// `visit_map` is called when deserializing a struct-like variant.
|
||||||
/// If no tuple variants are expected, yield a
|
fn visit_map<V>(&mut self, _visitor: V) -> Result<V::Value, Self::Error>
|
||||||
/// `Err(serde::de::Error::invalid_type(serde::de::Type::TupleVariant))`
|
where V: Visitor
|
||||||
fn visit_tuple<V>(&mut self,
|
{
|
||||||
len: usize,
|
Err(Error::syntax_error())
|
||||||
visitor: V) -> Result<V::Value, Self::Error>
|
}
|
||||||
where V: Visitor;
|
|
||||||
|
|
||||||
/// `visit_struct` is called when deserializing a struct-like variant.
|
|
||||||
/// If no struct variants are expected, yield a
|
|
||||||
/// `Err(serde::de::Error::invalid_type(serde::de::Type::StructVariant))`
|
|
||||||
fn visit_struct<V>(&mut self,
|
|
||||||
fields: &'static [&'static str],
|
|
||||||
visitor: V) -> Result<V::Value, Self::Error>
|
|
||||||
where V: Visitor;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a, T> VariantVisitor for &'a mut T where T: VariantVisitor {
|
impl<'a, T> VariantVisitor for &'a mut T where T: VariantVisitor {
|
||||||
@@ -871,25 +446,33 @@ impl<'a, T> VariantVisitor for &'a mut T where T: VariantVisitor {
|
|||||||
(**self).visit_unit()
|
(**self).visit_unit()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn visit_newtype<D>(&mut self) -> Result<D, T::Error>
|
fn visit_seq<V>(&mut self, visitor: V) -> Result<V::Value, T::Error>
|
||||||
where D: Deserialize,
|
|
||||||
{
|
|
||||||
(**self).visit_newtype()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_tuple<V>(&mut self,
|
|
||||||
len: usize,
|
|
||||||
visitor: V) -> Result<V::Value, T::Error>
|
|
||||||
where V: Visitor,
|
where V: Visitor,
|
||||||
{
|
{
|
||||||
(**self).visit_tuple(len, visitor)
|
(**self).visit_seq(visitor)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn visit_struct<V>(&mut self,
|
fn visit_map<V>(&mut self, visitor: V) -> Result<V::Value, T::Error>
|
||||||
fields: &'static [&'static str],
|
|
||||||
visitor: V) -> Result<V::Value, T::Error>
|
|
||||||
where V: Visitor,
|
where V: Visitor,
|
||||||
{
|
{
|
||||||
(**self).visit_struct(fields, visitor)
|
(**self).visit_map(visitor)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
pub trait EnumSeqVisitor {
|
||||||
|
type Value;
|
||||||
|
|
||||||
|
fn visit<V>(&mut self, visitor: V) -> Result<Self::Value, V::Error>
|
||||||
|
where V: SeqVisitor;
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
pub trait EnumMapVisitor {
|
||||||
|
type Value;
|
||||||
|
|
||||||
|
fn visit<V>(&mut self, visitor: V) -> Result<Self::Value, V::Error>
|
||||||
|
where V: MapVisitor;
|
||||||
|
}
|
||||||
|
|||||||
+136
-716
File diff suppressed because it is too large
Load Diff
@@ -1,44 +0,0 @@
|
|||||||
//! A stand-in for `std::error`
|
|
||||||
use core::any::TypeId;
|
|
||||||
use core::fmt::{Debug, Display};
|
|
||||||
|
|
||||||
|
|
||||||
/// A stand-in for `std::error::Error`, which requires no allocation.
|
|
||||||
#[cfg(feature = "unstable")]
|
|
||||||
pub trait Error: Debug + Display + ::core::marker::Reflect {
|
|
||||||
/// A short description of the error.
|
|
||||||
///
|
|
||||||
/// The description should not contain newlines or sentence-ending
|
|
||||||
/// punctuation, to facilitate embedding in larger user-facing
|
|
||||||
/// strings.
|
|
||||||
fn description(&self) -> &str;
|
|
||||||
|
|
||||||
/// The lower-level cause of this error, if any.
|
|
||||||
fn cause(&self) -> Option<&Error> { None }
|
|
||||||
|
|
||||||
/// Get the `TypeId` of `self`
|
|
||||||
#[doc(hidden)]
|
|
||||||
fn type_id(&self) -> TypeId where Self: 'static {
|
|
||||||
TypeId::of::<Self>()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// A stand-in for `std::error::Error`, which requires no allocation.
|
|
||||||
#[cfg(not(feature = "unstable"))]
|
|
||||||
pub trait Error: Debug + Display {
|
|
||||||
/// A short description of the error.
|
|
||||||
///
|
|
||||||
/// The description should not contain newlines or sentence-ending
|
|
||||||
/// punctuation, to facilitate embedding in larger user-facing
|
|
||||||
/// strings.
|
|
||||||
fn description(&self) -> &str;
|
|
||||||
|
|
||||||
/// The lower-level cause of this error, if any.
|
|
||||||
fn cause(&self) -> Option<&Error> { None }
|
|
||||||
|
|
||||||
/// Stubbed! Returns type_id of `()`
|
|
||||||
#[doc(hidden)]
|
|
||||||
fn type_id(&self) -> TypeId where Self: 'static {
|
|
||||||
TypeId::of::<()>()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,9 +1,6 @@
|
|||||||
//! Module that contains helper iterators.
|
|
||||||
|
|
||||||
use std::io;
|
use std::io;
|
||||||
use std::iter::Peekable;
|
use std::iter::Peekable;
|
||||||
|
|
||||||
/// Iterator over a byte stream that tracks the current position's line and column.
|
|
||||||
pub struct LineColIterator<Iter: Iterator<Item=io::Result<u8>>> {
|
pub struct LineColIterator<Iter: Iterator<Item=io::Result<u8>>> {
|
||||||
iter: Iter,
|
iter: Iter,
|
||||||
line: usize,
|
line: usize,
|
||||||
@@ -11,7 +8,6 @@ pub struct LineColIterator<Iter: Iterator<Item=io::Result<u8>>> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl<Iter: Iterator<Item=io::Result<u8>>> LineColIterator<Iter> {
|
impl<Iter: Iterator<Item=io::Result<u8>>> LineColIterator<Iter> {
|
||||||
/// Construct a new `LineColIterator<Iter>`.
|
|
||||||
pub fn new(iter: Iter) -> LineColIterator<Iter> {
|
pub fn new(iter: Iter) -> LineColIterator<Iter> {
|
||||||
LineColIterator {
|
LineColIterator {
|
||||||
iter: iter,
|
iter: iter,
|
||||||
|
|||||||
@@ -0,0 +1,84 @@
|
|||||||
|
// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT
|
||||||
|
// file at the top-level directory of this distribution and at
|
||||||
|
// http://rust-lang.org/COPYRIGHT.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
||||||
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
||||||
|
// option. This file may not be copied, modified, or distributed
|
||||||
|
// except according to those terms.
|
||||||
|
|
||||||
|
use std::collections::BTreeMap;
|
||||||
|
|
||||||
|
use ser::{self, Serialize};
|
||||||
|
use json::value::{self, Value};
|
||||||
|
|
||||||
|
pub struct ArrayBuilder {
|
||||||
|
array: Vec<Value>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ArrayBuilder {
|
||||||
|
pub fn new() -> ArrayBuilder {
|
||||||
|
ArrayBuilder { array: Vec::new() }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn unwrap(self) -> Value {
|
||||||
|
Value::Array(self.array)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn push<T: ser::Serialize>(mut self, v: T) -> ArrayBuilder {
|
||||||
|
self.array.push(value::to_value(&v));
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn push_array<F>(mut self, f: F) -> ArrayBuilder where
|
||||||
|
F: FnOnce(ArrayBuilder) -> ArrayBuilder
|
||||||
|
{
|
||||||
|
let builder = ArrayBuilder::new();
|
||||||
|
self.array.push(f(builder).unwrap());
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn push_object<F>(mut self, f: F) -> ArrayBuilder where
|
||||||
|
F: FnOnce(ObjectBuilder) -> ObjectBuilder
|
||||||
|
{
|
||||||
|
let builder = ObjectBuilder::new();
|
||||||
|
self.array.push(f(builder).unwrap());
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct ObjectBuilder {
|
||||||
|
object: BTreeMap<String, Value>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ObjectBuilder {
|
||||||
|
pub fn new() -> ObjectBuilder {
|
||||||
|
ObjectBuilder { object: BTreeMap::new() }
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn unwrap(self) -> Value {
|
||||||
|
Value::Object(self.object)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn insert<V: ser::Serialize>(mut self, k: String, v: V) -> ObjectBuilder {
|
||||||
|
self.object.insert(k, value::to_value(&v));
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn insert_array<F>(mut self, key: String, f: F) -> ObjectBuilder where
|
||||||
|
F: FnOnce(ArrayBuilder) -> ArrayBuilder
|
||||||
|
{
|
||||||
|
let builder = ArrayBuilder::new();
|
||||||
|
self.object.insert(key, f(builder).unwrap());
|
||||||
|
self
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn insert_object<F>(mut self, key: String, f: F) -> ObjectBuilder where
|
||||||
|
F: FnOnce(ObjectBuilder) -> ObjectBuilder
|
||||||
|
{
|
||||||
|
let builder = ObjectBuilder::new();
|
||||||
|
self.object.insert(key, f(builder).unwrap());
|
||||||
|
self
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,694 @@
|
|||||||
|
use std::char;
|
||||||
|
use std::i32;
|
||||||
|
use std::io;
|
||||||
|
use std::str;
|
||||||
|
|
||||||
|
use de;
|
||||||
|
use iter::LineColIterator;
|
||||||
|
|
||||||
|
use super::error::{Error, ErrorCode};
|
||||||
|
|
||||||
|
pub struct Deserializer<Iter: Iterator<Item=io::Result<u8>>> {
|
||||||
|
rdr: LineColIterator<Iter>,
|
||||||
|
ch: Option<u8>,
|
||||||
|
str_buf: Vec<u8>,
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! try_or_invalid {
|
||||||
|
($self_:expr, $e:expr) => {
|
||||||
|
match $e {
|
||||||
|
Some(v) => v,
|
||||||
|
None => { return Err($self_.error(ErrorCode::InvalidNumber)); }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<Iter> Deserializer<Iter>
|
||||||
|
where Iter: Iterator<Item=io::Result<u8>>,
|
||||||
|
{
|
||||||
|
/// Creates the JSON parser from an `std::iter::Iterator`.
|
||||||
|
#[inline]
|
||||||
|
pub fn new(rdr: Iter) -> Result<Deserializer<Iter>, Error> {
|
||||||
|
let mut deserializer = Deserializer {
|
||||||
|
rdr: LineColIterator::new(rdr),
|
||||||
|
ch: None,
|
||||||
|
str_buf: Vec::with_capacity(128),
|
||||||
|
};
|
||||||
|
|
||||||
|
try!(deserializer.bump());
|
||||||
|
|
||||||
|
Ok(deserializer)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn end(&mut self) -> Result<(), Error> {
|
||||||
|
try!(self.parse_whitespace());
|
||||||
|
if self.eof() {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(self.error(ErrorCode::TrailingCharacters))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn eof(&self) -> bool { self.ch.is_none() }
|
||||||
|
|
||||||
|
fn ch_or_null(&self) -> u8 { self.ch.unwrap_or(b'\x00') }
|
||||||
|
|
||||||
|
fn bump(&mut self) -> Result<(), Error> {
|
||||||
|
self.ch = match self.rdr.next() {
|
||||||
|
Some(Err(err)) => { return Err(Error::IoError(err)); }
|
||||||
|
Some(Ok(ch)) => Some(ch),
|
||||||
|
None => None,
|
||||||
|
};
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn next_char(&mut self) -> Result<Option<u8>, Error> {
|
||||||
|
try!(self.bump());
|
||||||
|
Ok(self.ch)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn ch_is(&self, c: u8) -> bool {
|
||||||
|
self.ch == Some(c)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn error(&mut self, reason: ErrorCode) -> Error {
|
||||||
|
Error::SyntaxError(reason, self.rdr.line(), self.rdr.col())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_whitespace(&mut self) -> Result<(), Error> {
|
||||||
|
while self.ch_is(b' ') ||
|
||||||
|
self.ch_is(b'\n') ||
|
||||||
|
self.ch_is(b'\t') ||
|
||||||
|
self.ch_is(b'\r') { try!(self.bump()); }
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_value<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
try!(self.parse_whitespace());
|
||||||
|
|
||||||
|
if self.eof() {
|
||||||
|
return Err(self.error(ErrorCode::EOFWhileParsingValue));
|
||||||
|
}
|
||||||
|
|
||||||
|
let value = match self.ch_or_null() {
|
||||||
|
b'n' => {
|
||||||
|
try!(self.parse_ident(b"ull"));
|
||||||
|
visitor.visit_unit()
|
||||||
|
}
|
||||||
|
b't' => {
|
||||||
|
try!(self.parse_ident(b"rue"));
|
||||||
|
visitor.visit_bool(true)
|
||||||
|
}
|
||||||
|
b'f' => {
|
||||||
|
try!(self.parse_ident(b"alse"));
|
||||||
|
visitor.visit_bool(false)
|
||||||
|
}
|
||||||
|
b'0' ... b'9' | b'-' => self.parse_number(visitor),
|
||||||
|
b'"' => {
|
||||||
|
try!(self.parse_string());
|
||||||
|
let s = str::from_utf8(&self.str_buf).unwrap();
|
||||||
|
visitor.visit_str(s)
|
||||||
|
}
|
||||||
|
b'[' => {
|
||||||
|
try!(self.bump());
|
||||||
|
visitor.visit_seq(SeqVisitor::new(self))
|
||||||
|
}
|
||||||
|
b'{' => {
|
||||||
|
try!(self.bump());
|
||||||
|
visitor.visit_map(MapVisitor::new(self))
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
Err(self.error(ErrorCode::ExpectedSomeValue))
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
match value {
|
||||||
|
Ok(value) => Ok(value),
|
||||||
|
Err(Error::SyntaxError(code, _, _)) => Err(self.error(code)),
|
||||||
|
Err(err) => Err(err),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_ident(&mut self, ident: &[u8]) -> Result<(), Error> {
|
||||||
|
for c in ident {
|
||||||
|
if Some(*c) != try!(self.next_char()) {
|
||||||
|
return Err(self.error(ErrorCode::ExpectedSomeIdent));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try!(self.bump());
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_number<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
let mut neg = false;
|
||||||
|
|
||||||
|
if self.ch_is(b'-') {
|
||||||
|
try!(self.bump());
|
||||||
|
neg = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
let res = try!(self.parse_integer());
|
||||||
|
|
||||||
|
if self.ch_is(b'.') || self.ch_is(b'e') || self.ch_is(b'E') {
|
||||||
|
let mut res = res as f64;
|
||||||
|
|
||||||
|
if self.ch_is(b'.') {
|
||||||
|
res = try!(self.parse_decimal(res));
|
||||||
|
}
|
||||||
|
|
||||||
|
if self.ch_is(b'e') || self.ch_is(b'E') {
|
||||||
|
res = try!(self.parse_exponent(res));
|
||||||
|
}
|
||||||
|
|
||||||
|
if neg {
|
||||||
|
visitor.visit_f64(-res)
|
||||||
|
} else {
|
||||||
|
visitor.visit_f64(res)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if neg {
|
||||||
|
let res = -(res as i64);
|
||||||
|
|
||||||
|
// Make sure we didn't underflow.
|
||||||
|
if res > 0 {
|
||||||
|
Err(self.error(ErrorCode::InvalidNumber))
|
||||||
|
} else {
|
||||||
|
visitor.visit_i64(res)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
visitor.visit_u64(res)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_integer(&mut self) -> Result<u64, Error> {
|
||||||
|
let mut accum: u64 = 0;
|
||||||
|
|
||||||
|
match self.ch_or_null() {
|
||||||
|
b'0' => {
|
||||||
|
try!(self.bump());
|
||||||
|
|
||||||
|
// There can be only one leading '0'.
|
||||||
|
match self.ch_or_null() {
|
||||||
|
b'0' ... b'9' => {
|
||||||
|
return Err(self.error(ErrorCode::InvalidNumber));
|
||||||
|
}
|
||||||
|
_ => ()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
b'1' ... b'9' => {
|
||||||
|
while !self.eof() {
|
||||||
|
match self.ch_or_null() {
|
||||||
|
c @ b'0' ... b'9' => {
|
||||||
|
accum = try_or_invalid!(self, accum.checked_mul(10));
|
||||||
|
accum = try_or_invalid!(self, accum.checked_add((c as u64) - ('0' as u64)));
|
||||||
|
|
||||||
|
try!(self.bump());
|
||||||
|
}
|
||||||
|
_ => break,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => { return Err(self.error(ErrorCode::InvalidNumber)); }
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(accum)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_decimal(&mut self, res: f64) -> Result<f64, Error> {
|
||||||
|
try!(self.bump());
|
||||||
|
|
||||||
|
// Make sure a digit follows the decimal place.
|
||||||
|
match self.ch_or_null() {
|
||||||
|
b'0' ... b'9' => (),
|
||||||
|
_ => { return Err(self.error(ErrorCode::InvalidNumber)); }
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut res = res;
|
||||||
|
let mut dec = 1.0;
|
||||||
|
while !self.eof() {
|
||||||
|
match self.ch_or_null() {
|
||||||
|
c @ b'0' ... b'9' => {
|
||||||
|
dec /= 10.0;
|
||||||
|
res += (((c as u64) - (b'0' as u64)) as f64) * dec;
|
||||||
|
try!(self.bump());
|
||||||
|
}
|
||||||
|
_ => break,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(res)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_exponent(&mut self, mut res: f64) -> Result<f64, Error> {
|
||||||
|
try!(self.bump());
|
||||||
|
|
||||||
|
let mut exp: u64 = 0;
|
||||||
|
let mut neg_exp = false;
|
||||||
|
|
||||||
|
if self.ch_is(b'+') {
|
||||||
|
try!(self.bump());
|
||||||
|
} else if self.ch_is(b'-') {
|
||||||
|
try!(self.bump());
|
||||||
|
neg_exp = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Make sure a digit follows the exponent place.
|
||||||
|
match self.ch_or_null() {
|
||||||
|
b'0' ... b'9' => (),
|
||||||
|
_ => { return Err(self.error(ErrorCode::InvalidNumber)); }
|
||||||
|
}
|
||||||
|
while !self.eof() {
|
||||||
|
match self.ch_or_null() {
|
||||||
|
c @ b'0' ... b'9' => {
|
||||||
|
exp = try_or_invalid!(self, exp.checked_mul(10));
|
||||||
|
exp = try_or_invalid!(self, exp.checked_add((c as u64) - (b'0' as u64)));
|
||||||
|
|
||||||
|
try!(self.bump());
|
||||||
|
}
|
||||||
|
_ => break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let exp = if exp <= i32::MAX as u64 {
|
||||||
|
10_f64.powi(exp as i32)
|
||||||
|
} else {
|
||||||
|
return Err(self.error(ErrorCode::InvalidNumber));
|
||||||
|
};
|
||||||
|
|
||||||
|
if neg_exp {
|
||||||
|
res /= exp;
|
||||||
|
} else {
|
||||||
|
res *= exp;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(res)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn decode_hex_escape(&mut self) -> Result<u16, Error> {
|
||||||
|
let mut i = 0;
|
||||||
|
let mut n = 0u16;
|
||||||
|
while i < 4 && !self.eof() {
|
||||||
|
try!(self.bump());
|
||||||
|
n = match self.ch_or_null() {
|
||||||
|
c @ b'0' ... b'9' => n * 16_u16 + ((c as u16) - (b'0' as u16)),
|
||||||
|
b'a' | b'A' => n * 16_u16 + 10_u16,
|
||||||
|
b'b' | b'B' => n * 16_u16 + 11_u16,
|
||||||
|
b'c' | b'C' => n * 16_u16 + 12_u16,
|
||||||
|
b'd' | b'D' => n * 16_u16 + 13_u16,
|
||||||
|
b'e' | b'E' => n * 16_u16 + 14_u16,
|
||||||
|
b'f' | b'F' => n * 16_u16 + 15_u16,
|
||||||
|
_ => { return Err(self.error(ErrorCode::InvalidEscape)); }
|
||||||
|
};
|
||||||
|
|
||||||
|
i += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Error out if we didn't parse 4 digits.
|
||||||
|
if i != 4 {
|
||||||
|
return Err(self.error(ErrorCode::InvalidEscape));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(n)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_string(&mut self) -> Result<(), Error> {
|
||||||
|
self.str_buf.clear();
|
||||||
|
|
||||||
|
loop {
|
||||||
|
let ch = match try!(self.next_char()) {
|
||||||
|
Some(ch) => ch,
|
||||||
|
None => { return Err(self.error(ErrorCode::EOFWhileParsingString)); }
|
||||||
|
};
|
||||||
|
|
||||||
|
match ch {
|
||||||
|
b'"' => {
|
||||||
|
try!(self.bump());
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
b'\\' => {
|
||||||
|
let ch = match try!(self.next_char()) {
|
||||||
|
Some(ch) => ch,
|
||||||
|
None => { return Err(self.error(ErrorCode::EOFWhileParsingString)); }
|
||||||
|
};
|
||||||
|
|
||||||
|
match ch {
|
||||||
|
b'"' => self.str_buf.push(b'"'),
|
||||||
|
b'\\' => self.str_buf.push(b'\\'),
|
||||||
|
b'/' => self.str_buf.push(b'/'),
|
||||||
|
b'b' => self.str_buf.push(b'\x08'),
|
||||||
|
b'f' => self.str_buf.push(b'\x0c'),
|
||||||
|
b'n' => self.str_buf.push(b'\n'),
|
||||||
|
b'r' => self.str_buf.push(b'\r'),
|
||||||
|
b't' => self.str_buf.push(b'\t'),
|
||||||
|
b'u' => {
|
||||||
|
let c = match try!(self.decode_hex_escape()) {
|
||||||
|
0xDC00 ... 0xDFFF => {
|
||||||
|
return Err(self.error(ErrorCode::LoneLeadingSurrogateInHexEscape));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Non-BMP characters are encoded as a sequence of
|
||||||
|
// two hex escapes, representing UTF-16 surrogates.
|
||||||
|
n1 @ 0xD800 ... 0xDBFF => {
|
||||||
|
match (try!(self.next_char()), try!(self.next_char())) {
|
||||||
|
(Some(b'\\'), Some(b'u')) => (),
|
||||||
|
_ => {
|
||||||
|
return Err(self.error(ErrorCode::UnexpectedEndOfHexEscape));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let n2 = try!(self.decode_hex_escape());
|
||||||
|
|
||||||
|
if n2 < 0xDC00 || n2 > 0xDFFF {
|
||||||
|
return Err(self.error(ErrorCode::LoneLeadingSurrogateInHexEscape));
|
||||||
|
}
|
||||||
|
|
||||||
|
let n = (((n1 - 0xD800) as u32) << 10 |
|
||||||
|
(n2 - 0xDC00) as u32) + 0x1_0000;
|
||||||
|
|
||||||
|
match char::from_u32(n as u32) {
|
||||||
|
Some(c) => c,
|
||||||
|
None => {
|
||||||
|
return Err(self.error(ErrorCode::InvalidUnicodeCodePoint));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
n => {
|
||||||
|
match char::from_u32(n as u32) {
|
||||||
|
Some(c) => c,
|
||||||
|
None => {
|
||||||
|
return Err(self.error(ErrorCode::InvalidUnicodeCodePoint));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// FIXME: this allocation is required in order to be compatible with stable
|
||||||
|
// rust, which doesn't support encoding a `char` into a stack buffer.
|
||||||
|
let buf = c.to_string();
|
||||||
|
self.str_buf.extend(buf.bytes());
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
return Err(self.error(ErrorCode::InvalidEscape));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ch => {
|
||||||
|
self.str_buf.push(ch);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn parse_object_colon(&mut self) -> Result<(), Error> {
|
||||||
|
try!(self.parse_whitespace());
|
||||||
|
|
||||||
|
if self.ch_is(b':') {
|
||||||
|
try!(self.bump());
|
||||||
|
Ok(())
|
||||||
|
} else if self.eof() {
|
||||||
|
Err(self.error(ErrorCode::EOFWhileParsingObject))
|
||||||
|
} else {
|
||||||
|
Err(self.error(ErrorCode::ExpectedColon))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<Iter> de::Deserializer for Deserializer<Iter>
|
||||||
|
where Iter: Iterator<Item=io::Result<u8>>,
|
||||||
|
{
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit<V>(&mut self, visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
self.parse_value(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_option<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
try!(self.parse_whitespace());
|
||||||
|
|
||||||
|
if self.eof() {
|
||||||
|
return Err(self.error(ErrorCode::EOFWhileParsingValue));
|
||||||
|
}
|
||||||
|
|
||||||
|
if self.ch_is(b'n') {
|
||||||
|
try!(self.parse_ident(b"ull"));
|
||||||
|
visitor.visit_none()
|
||||||
|
} else {
|
||||||
|
visitor.visit_some(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_enum<V>(&mut self, _name: &str, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::EnumVisitor,
|
||||||
|
{
|
||||||
|
try!(self.parse_whitespace());
|
||||||
|
|
||||||
|
if self.ch_is(b'{') {
|
||||||
|
try!(self.bump());
|
||||||
|
try!(self.parse_whitespace());
|
||||||
|
|
||||||
|
let value = {
|
||||||
|
try!(visitor.visit(&mut *self))
|
||||||
|
};
|
||||||
|
|
||||||
|
try!(self.parse_whitespace());
|
||||||
|
|
||||||
|
if self.ch_is(b'}') {
|
||||||
|
try!(self.bump());
|
||||||
|
Ok(value)
|
||||||
|
} else {
|
||||||
|
Err(self.error(ErrorCode::ExpectedSomeValue))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Err(self.error(ErrorCode::ExpectedSomeValue))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn format() -> &'static str {
|
||||||
|
"json"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct SeqVisitor<'a, Iter: 'a + Iterator<Item=io::Result<u8>>> {
|
||||||
|
de: &'a mut Deserializer<Iter>,
|
||||||
|
first: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, Iter: Iterator<Item=io::Result<u8>>> SeqVisitor<'a, Iter> {
|
||||||
|
fn new(de: &'a mut Deserializer<Iter>) -> Self {
|
||||||
|
SeqVisitor {
|
||||||
|
de: de,
|
||||||
|
first: true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, Iter> de::SeqVisitor for SeqVisitor<'a, Iter>
|
||||||
|
where Iter: Iterator<Item=io::Result<u8>>,
|
||||||
|
{
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit<T>(&mut self) -> Result<Option<T>, Error>
|
||||||
|
where T: de::Deserialize,
|
||||||
|
{
|
||||||
|
try!(self.de.parse_whitespace());
|
||||||
|
|
||||||
|
if self.de.ch_is(b']') {
|
||||||
|
return Ok(None);
|
||||||
|
}
|
||||||
|
|
||||||
|
if self.first {
|
||||||
|
self.first = false;
|
||||||
|
} else {
|
||||||
|
if self.de.ch_is(b',') {
|
||||||
|
try!(self.de.bump());
|
||||||
|
} else if self.de.eof() {
|
||||||
|
return Err(self.de.error(ErrorCode::EOFWhileParsingList));
|
||||||
|
} else {
|
||||||
|
return Err(self.de.error(ErrorCode::ExpectedListCommaOrEnd));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let value = try!(de::Deserialize::deserialize(self.de));
|
||||||
|
Ok(Some(value))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
|
try!(self.de.parse_whitespace());
|
||||||
|
|
||||||
|
if self.de.ch_is(b']') {
|
||||||
|
self.de.bump()
|
||||||
|
} else if self.de.eof() {
|
||||||
|
Err(self.de.error(ErrorCode::EOFWhileParsingList))
|
||||||
|
} else {
|
||||||
|
Err(self.de.error(ErrorCode::TrailingCharacters))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct MapVisitor<'a, Iter: 'a + Iterator<Item=io::Result<u8>>> {
|
||||||
|
de: &'a mut Deserializer<Iter>,
|
||||||
|
first: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, Iter: Iterator<Item=io::Result<u8>>> MapVisitor<'a, Iter> {
|
||||||
|
fn new(de: &'a mut Deserializer<Iter>) -> Self {
|
||||||
|
MapVisitor {
|
||||||
|
de: de,
|
||||||
|
first: true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, Iter> de::MapVisitor for MapVisitor<'a, Iter>
|
||||||
|
where Iter: Iterator<Item=io::Result<u8>>
|
||||||
|
{
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit_key<K>(&mut self) -> Result<Option<K>, Error>
|
||||||
|
where K: de::Deserialize,
|
||||||
|
{
|
||||||
|
try!(self.de.parse_whitespace());
|
||||||
|
|
||||||
|
if self.de.ch_is(b'}') {
|
||||||
|
return Ok(None);
|
||||||
|
}
|
||||||
|
|
||||||
|
if self.first {
|
||||||
|
self.first = false;
|
||||||
|
} else {
|
||||||
|
if self.de.ch_is(b',') {
|
||||||
|
try!(self.de.bump());
|
||||||
|
try!(self.de.parse_whitespace());
|
||||||
|
} else if self.de.eof() {
|
||||||
|
return Err(self.de.error(ErrorCode::EOFWhileParsingObject));
|
||||||
|
} else {
|
||||||
|
return Err(self.de.error(ErrorCode::ExpectedObjectCommaOrEnd));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if self.de.eof() {
|
||||||
|
return Err(self.de.error(ErrorCode::EOFWhileParsingValue));
|
||||||
|
}
|
||||||
|
|
||||||
|
if !self.de.ch_is(b'"') {
|
||||||
|
return Err(self.de.error(ErrorCode::KeyMustBeAString));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(Some(try!(de::Deserialize::deserialize(self.de))))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_value<V>(&mut self) -> Result<V, Error>
|
||||||
|
where V: de::Deserialize,
|
||||||
|
{
|
||||||
|
try!(self.de.parse_object_colon());
|
||||||
|
|
||||||
|
Ok(try!(de::Deserialize::deserialize(self.de)))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
|
try!(self.de.parse_whitespace());
|
||||||
|
|
||||||
|
if self.de.ch_is(b'}') {
|
||||||
|
try!(self.de.bump());
|
||||||
|
Ok(())
|
||||||
|
} else if self.de.eof() {
|
||||||
|
Err(self.de.error(ErrorCode::EOFWhileParsingObject))
|
||||||
|
} else {
|
||||||
|
Err(self.de.error(ErrorCode::TrailingCharacters))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn missing_field<V>(&mut self, _field: &'static str) -> Result<V, Error>
|
||||||
|
where V: de::Deserialize,
|
||||||
|
{
|
||||||
|
let mut de = de::value::ValueDeserializer::into_deserializer(());
|
||||||
|
Ok(try!(de::Deserialize::deserialize(&mut de)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<Iter> de::VariantVisitor for Deserializer<Iter>
|
||||||
|
where Iter: Iterator<Item=io::Result<u8>>,
|
||||||
|
{
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit_variant<V>(&mut self) -> Result<V, Error>
|
||||||
|
where V: de::Deserialize
|
||||||
|
{
|
||||||
|
de::Deserialize::deserialize(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_unit(&mut self) -> Result<(), Error> {
|
||||||
|
try!(self.parse_object_colon());
|
||||||
|
|
||||||
|
de::Deserialize::deserialize(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_seq<V>(&mut self, visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
try!(self.parse_object_colon());
|
||||||
|
|
||||||
|
de::Deserializer::visit(self, visitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_map<V>(&mut self, visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
try!(self.parse_object_colon());
|
||||||
|
|
||||||
|
de::Deserializer::visit(self, visitor)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Decodes a json value from a `std::io::Read`.
|
||||||
|
pub fn from_iter<I, T>(iter: I) -> Result<T, Error>
|
||||||
|
where I: Iterator<Item=io::Result<u8>>,
|
||||||
|
T: de::Deserialize,
|
||||||
|
{
|
||||||
|
let mut de = try!(Deserializer::new(iter));
|
||||||
|
let value = try!(de::Deserialize::deserialize(&mut de));
|
||||||
|
|
||||||
|
// Make sure the whole stream has been consumed.
|
||||||
|
try!(de.end());
|
||||||
|
Ok(value)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Decodes a json value from a `std::io::Read`.
|
||||||
|
pub fn from_reader<R, T>(rdr: R) -> Result<T, Error>
|
||||||
|
where R: io::Read,
|
||||||
|
T: de::Deserialize,
|
||||||
|
{
|
||||||
|
from_iter(rdr.bytes())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Decodes a json value from a `&str`.
|
||||||
|
pub fn from_slice<T>(v: &[u8]) -> Result<T, Error>
|
||||||
|
where T: de::Deserialize
|
||||||
|
{
|
||||||
|
from_iter(v.iter().map(|byte| Ok(*byte)))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Decodes a json value from a `&str`.
|
||||||
|
pub fn from_str<T>(s: &str) -> Result<T, Error>
|
||||||
|
where T: de::Deserialize
|
||||||
|
{
|
||||||
|
from_slice(s.as_bytes())
|
||||||
|
}
|
||||||
@@ -0,0 +1,185 @@
|
|||||||
|
use std::error;
|
||||||
|
use std::fmt;
|
||||||
|
use std::io;
|
||||||
|
|
||||||
|
use de;
|
||||||
|
|
||||||
|
/// The errors that can arise while parsing a JSON stream.
|
||||||
|
#[derive(Clone, PartialEq)]
|
||||||
|
pub enum ErrorCode {
|
||||||
|
EOFWhileParsingList,
|
||||||
|
EOFWhileParsingObject,
|
||||||
|
EOFWhileParsingString,
|
||||||
|
EOFWhileParsingValue,
|
||||||
|
ExpectedColon,
|
||||||
|
ExpectedConversion,
|
||||||
|
ExpectedEnumEnd,
|
||||||
|
ExpectedEnumEndToken,
|
||||||
|
ExpectedEnumMapStart,
|
||||||
|
ExpectedEnumToken,
|
||||||
|
ExpectedEnumVariantString,
|
||||||
|
ExpectedListCommaOrEnd,
|
||||||
|
ExpectedName,
|
||||||
|
ExpectedObjectCommaOrEnd,
|
||||||
|
ExpectedSomeIdent,
|
||||||
|
ExpectedSomeValue,
|
||||||
|
InvalidEscape,
|
||||||
|
InvalidNumber,
|
||||||
|
InvalidUnicodeCodePoint,
|
||||||
|
KeyMustBeAString,
|
||||||
|
LoneLeadingSurrogateInHexEscape,
|
||||||
|
UnknownField(String),
|
||||||
|
MissingField(&'static str),
|
||||||
|
NotFourDigit,
|
||||||
|
NotUtf8,
|
||||||
|
TrailingCharacters,
|
||||||
|
UnexpectedEndOfHexEscape,
|
||||||
|
UnknownVariant,
|
||||||
|
UnrecognizedHex,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Debug for ErrorCode {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
use std::fmt::Debug;
|
||||||
|
|
||||||
|
match *self {
|
||||||
|
//ErrorCode::ConversionError(ref token) => write!(f, "failed to convert {}", token),
|
||||||
|
ErrorCode::EOFWhileParsingList => "EOF While parsing list".fmt(f),
|
||||||
|
ErrorCode::EOFWhileParsingObject => "EOF While parsing object".fmt(f),
|
||||||
|
ErrorCode::EOFWhileParsingString => "EOF While parsing string".fmt(f),
|
||||||
|
ErrorCode::EOFWhileParsingValue => "EOF While parsing value".fmt(f),
|
||||||
|
ErrorCode::ExpectedColon => "expected `:`".fmt(f),
|
||||||
|
ErrorCode::ExpectedConversion => "expected conversion".fmt(f),
|
||||||
|
ErrorCode::ExpectedEnumEnd => "expected enum end".fmt(f),
|
||||||
|
ErrorCode::ExpectedEnumEndToken => "expected enum map end".fmt(f),
|
||||||
|
ErrorCode::ExpectedEnumMapStart => "expected enum map start".fmt(f),
|
||||||
|
ErrorCode::ExpectedEnumToken => "expected enum token".fmt(f),
|
||||||
|
ErrorCode::ExpectedEnumVariantString => "expected variant".fmt(f),
|
||||||
|
ErrorCode::ExpectedListCommaOrEnd => "expected `,` or `]`".fmt(f),
|
||||||
|
ErrorCode::ExpectedName => "expected name".fmt(f),
|
||||||
|
ErrorCode::ExpectedObjectCommaOrEnd => "expected `,` or `}`".fmt(f),
|
||||||
|
ErrorCode::ExpectedSomeIdent => "expected ident".fmt(f),
|
||||||
|
ErrorCode::ExpectedSomeValue => "expected value".fmt(f),
|
||||||
|
//ErrorCode::ExpectedTokens(ref token, tokens) => write!(f, "expected {}, found {}", tokens, token),
|
||||||
|
ErrorCode::InvalidEscape => "invalid escape".fmt(f),
|
||||||
|
ErrorCode::InvalidNumber => "invalid number".fmt(f),
|
||||||
|
ErrorCode::InvalidUnicodeCodePoint => "invalid unicode code point".fmt(f),
|
||||||
|
ErrorCode::KeyMustBeAString => "key must be a string".fmt(f),
|
||||||
|
ErrorCode::LoneLeadingSurrogateInHexEscape => "lone leading surrogate in hex escape".fmt(f),
|
||||||
|
ErrorCode::UnknownField(ref field) => write!(f, "unknown field \"{}\"", field),
|
||||||
|
ErrorCode::MissingField(ref field) => write!(f, "missing field \"{}\"", field),
|
||||||
|
ErrorCode::NotFourDigit => "invalid \\u escape (not four digits)".fmt(f),
|
||||||
|
ErrorCode::NotUtf8 => "contents not utf-8".fmt(f),
|
||||||
|
ErrorCode::TrailingCharacters => "trailing characters".fmt(f),
|
||||||
|
ErrorCode::UnexpectedEndOfHexEscape => "unexpected end of hex escape".fmt(f),
|
||||||
|
//ErrorCode::UnexpectedName(ref name) => write!(f, "unexpected name {}", name),
|
||||||
|
ErrorCode::UnknownVariant => "unknown variant".fmt(f),
|
||||||
|
ErrorCode::UnrecognizedHex => "invalid \\u escape (unrecognized hex)".fmt(f),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub enum Error {
|
||||||
|
/// msg, line, col
|
||||||
|
SyntaxError(ErrorCode, usize, usize),
|
||||||
|
IoError(io::Error),
|
||||||
|
/*
|
||||||
|
ExpectedError(String, String),
|
||||||
|
*/
|
||||||
|
MissingFieldError(&'static str),
|
||||||
|
/*
|
||||||
|
UnknownVariantError(String),
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
|
||||||
|
impl error::Error for Error {
|
||||||
|
fn description(&self) -> &str {
|
||||||
|
match *self {
|
||||||
|
Error::SyntaxError(..) => "syntax error",
|
||||||
|
Error::IoError(ref error) => error::Error::description(error),
|
||||||
|
/*
|
||||||
|
Error::ExpectedError(ref expected, _) => &expected,
|
||||||
|
*/
|
||||||
|
Error::MissingFieldError(_) => "missing field",
|
||||||
|
/*
|
||||||
|
Error::UnknownVariantError(_) => "unknown variant",
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn cause(&self) -> Option<&error::Error> {
|
||||||
|
match *self {
|
||||||
|
Error::IoError(ref error) => Some(error),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Display for Error {
|
||||||
|
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
match *self {
|
||||||
|
Error::SyntaxError(ref code, line, col) => {
|
||||||
|
write!(fmt, "{:?} at line {} column {}", code, line, col)
|
||||||
|
}
|
||||||
|
Error::IoError(ref error) => fmt::Display::fmt(error, fmt),
|
||||||
|
/*
|
||||||
|
Error::ExpectedError(ref expected, ref found) => {
|
||||||
|
Some(format!("expected {}, found {}", expected, found))
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
Error::MissingFieldError(ref field) => {
|
||||||
|
write!(fmt, "missing field {}", field)
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
Error::UnknownVariantError(ref variant) => {
|
||||||
|
Some(format!("unknown variant {}", variant))
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<io::Error> for Error {
|
||||||
|
fn from(error: io::Error) -> Error {
|
||||||
|
Error::IoError(error)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl From<de::value::Error> for Error {
|
||||||
|
fn from(error: de::value::Error) -> Error {
|
||||||
|
match error {
|
||||||
|
de::value::Error::SyntaxError => {
|
||||||
|
de::Error::syntax_error()
|
||||||
|
}
|
||||||
|
de::value::Error::EndOfStreamError => {
|
||||||
|
de::Error::end_of_stream_error()
|
||||||
|
}
|
||||||
|
de::value::Error::UnknownFieldError(field) => {
|
||||||
|
Error::SyntaxError(ErrorCode::UnknownField(field), 0, 0)
|
||||||
|
}
|
||||||
|
de::value::Error::MissingFieldError(field) => {
|
||||||
|
de::Error::missing_field_error(field)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl de::Error for Error {
|
||||||
|
fn syntax_error() -> Error {
|
||||||
|
Error::SyntaxError(ErrorCode::ExpectedSomeValue, 0, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn end_of_stream_error() -> Error {
|
||||||
|
Error::SyntaxError(ErrorCode::EOFWhileParsingValue, 0, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn unknown_field_error(field: &str) -> Error {
|
||||||
|
Error::SyntaxError(ErrorCode::UnknownField(field.to_string()), 0, 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn missing_field_error(field: &'static str) -> Error {
|
||||||
|
Error::MissingFieldError(field)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,113 @@
|
|||||||
|
//! JSON and serialization
|
||||||
|
//!
|
||||||
|
//! # What is JSON?
|
||||||
|
//!
|
||||||
|
//! JSON (JavaScript Object Notation) is a way to write data in JavaScript. Like XML, it allows to
|
||||||
|
//! encode structured data in a text format that can be easily read by humans. Its simple syntax
|
||||||
|
//! and native compatibility with JavaScript have made it a widely used format.
|
||||||
|
//!
|
||||||
|
//! Data types that can be encoded are JavaScript types (see the `serde::json:Value` enum for more
|
||||||
|
//! details):
|
||||||
|
//!
|
||||||
|
//! * `Boolean`: equivalent to rust's `bool`
|
||||||
|
//! * `I64`: equivalent to rust's `i64`
|
||||||
|
//! * `U64`: equivalent to rust's `u64`
|
||||||
|
//! * `F64`: equivalent to rust's `i64`
|
||||||
|
//! * `String`: equivalent to rust's `String`
|
||||||
|
//! * `Array`: equivalent to rust's `Vec<T>`, but also allowing objects of different types in the
|
||||||
|
//! same array
|
||||||
|
//! * `Object`: equivalent to rust's `BTreeMap<String, serde::json::Value>`
|
||||||
|
//! * `Null`
|
||||||
|
//!
|
||||||
|
//! An object is a series of string keys mapping to values, in `"key": value` format. Arrays are
|
||||||
|
//! enclosed in square brackets ([ ... ]) and objects in curly brackets ({ ... }). A simple JSON
|
||||||
|
//! document encoding a person, his/her age, address and phone numbers could look like
|
||||||
|
//!
|
||||||
|
//! ```ignore
|
||||||
|
//! {
|
||||||
|
//! "FirstName": "John",
|
||||||
|
//! "LastName": "Doe",
|
||||||
|
//! "Age": 43,
|
||||||
|
//! "Address": {
|
||||||
|
//! "Street": "Downing Street 10",
|
||||||
|
//! "City": "London",
|
||||||
|
//! "Country": "Great Britain"
|
||||||
|
//! },
|
||||||
|
//! "PhoneNumbers": [
|
||||||
|
//! "+44 1234567",
|
||||||
|
//! "+44 2345678"
|
||||||
|
//! ]
|
||||||
|
//! }
|
||||||
|
//! ```
|
||||||
|
//!
|
||||||
|
//! # Type-based Serialization and Deserialization
|
||||||
|
//!
|
||||||
|
//! Serde provides a mechanism for low boilerplate serialization & deserialization of values to and
|
||||||
|
//! from JSON via the serialization API. To be able to serialize a piece of data, it must implement
|
||||||
|
//! the `serde::Serialize` trait. To be able to deserialize a piece of data, it must implement the
|
||||||
|
//! `serde::Deserialize` trait. Serde provides provides an annotation to automatically generate
|
||||||
|
//! the code for these traits: `#[derive(Serialize, Deserialize)]`.
|
||||||
|
//!
|
||||||
|
//! The JSON API also provides an enum `serde::json::Value` and a method `to_value` to serialize
|
||||||
|
//! objects. A `serde::json::Value` value can be serialized as a string or buffer using the
|
||||||
|
//! functions described above. You can also use the `json::Serializer` object, which implements the
|
||||||
|
//! `Serializer` trait.
|
||||||
|
//!
|
||||||
|
//! # Examples of use
|
||||||
|
//!
|
||||||
|
//! ## Parsing a `str` to `Value` and reading the result
|
||||||
|
//!
|
||||||
|
//! ```rust
|
||||||
|
//! //#![feature(custom_derive, plugin)]
|
||||||
|
//! //#![plugin(serde_macros)]
|
||||||
|
//!
|
||||||
|
//! extern crate serde;
|
||||||
|
//!
|
||||||
|
//! use serde::json::{self, Value};
|
||||||
|
//!
|
||||||
|
//! fn main() {
|
||||||
|
//! let data: Value = json::from_str("{\"foo\": 13, \"bar\": \"baz\"}").unwrap();
|
||||||
|
//! println!("data: {:?}", data);
|
||||||
|
//! // data: {"bar":"baz","foo":13}
|
||||||
|
//! println!("object? {}", data.is_object());
|
||||||
|
//! // object? true
|
||||||
|
//!
|
||||||
|
//! let obj = data.as_object().unwrap();
|
||||||
|
//! let foo = obj.get("foo").unwrap();
|
||||||
|
//!
|
||||||
|
//! println!("array? {:?}", foo.as_array());
|
||||||
|
//! // array? None
|
||||||
|
//! println!("u64? {:?}", foo.as_u64());
|
||||||
|
//! // u64? Some(13u64)
|
||||||
|
//!
|
||||||
|
//! for (key, value) in obj.iter() {
|
||||||
|
//! println!("{}: {}", key, match *value {
|
||||||
|
//! Value::U64(v) => format!("{} (u64)", v),
|
||||||
|
//! Value::String(ref v) => format!("{} (string)", v),
|
||||||
|
//! _ => format!("other")
|
||||||
|
//! });
|
||||||
|
//! }
|
||||||
|
//! // bar: baz (string)
|
||||||
|
//! // foo: 13 (u64)
|
||||||
|
//! }
|
||||||
|
//! ```
|
||||||
|
|
||||||
|
pub use self::de::{Deserializer, from_str};
|
||||||
|
pub use self::error::{Error, ErrorCode};
|
||||||
|
pub use self::ser::{
|
||||||
|
Serializer,
|
||||||
|
to_writer,
|
||||||
|
to_writer_pretty,
|
||||||
|
to_vec,
|
||||||
|
to_vec_pretty,
|
||||||
|
to_string,
|
||||||
|
to_string_pretty,
|
||||||
|
escape_str,
|
||||||
|
};
|
||||||
|
pub use self::value::{Value, to_value, from_value};
|
||||||
|
|
||||||
|
pub mod builder;
|
||||||
|
pub mod de;
|
||||||
|
pub mod error;
|
||||||
|
pub mod ser;
|
||||||
|
pub mod value;
|
||||||
@@ -0,0 +1,524 @@
|
|||||||
|
use std::io;
|
||||||
|
use std::num::FpCategory;
|
||||||
|
use std::string::FromUtf8Error;
|
||||||
|
|
||||||
|
use ser;
|
||||||
|
|
||||||
|
/// A structure for implementing serialization to JSON.
|
||||||
|
pub struct Serializer<W, F=CompactFormatter> {
|
||||||
|
writer: W,
|
||||||
|
formatter: F,
|
||||||
|
|
||||||
|
/// `first` is used to signify if we should print a comma when we are walking through a
|
||||||
|
/// sequence.
|
||||||
|
first: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<W> Serializer<W>
|
||||||
|
where W: io::Write,
|
||||||
|
{
|
||||||
|
/// Creates a new JSON serializer.
|
||||||
|
#[inline]
|
||||||
|
pub fn new(writer: W) -> Self {
|
||||||
|
Serializer::with_formatter(writer, CompactFormatter)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, W> Serializer<W, PrettyFormatter<'a>>
|
||||||
|
where W: io::Write,
|
||||||
|
{
|
||||||
|
/// Creates a new JSON pretty print serializer.
|
||||||
|
#[inline]
|
||||||
|
pub fn pretty(writer: W) -> Self {
|
||||||
|
Serializer::with_formatter(writer, PrettyFormatter::new())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<W, F> Serializer<W, F>
|
||||||
|
where W: io::Write,
|
||||||
|
F: Formatter,
|
||||||
|
{
|
||||||
|
/// Creates a new JSON visitor whose output will be written to the writer
|
||||||
|
/// specified.
|
||||||
|
#[inline]
|
||||||
|
pub fn with_formatter(writer: W, formatter: F) -> Self {
|
||||||
|
Serializer {
|
||||||
|
writer: writer,
|
||||||
|
formatter: formatter,
|
||||||
|
first: false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Unwrap the `Writer` from the `Serializer`.
|
||||||
|
#[inline]
|
||||||
|
pub fn into_inner(self) -> W {
|
||||||
|
self.writer
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<W, F> ser::Serializer for Serializer<W, F>
|
||||||
|
where W: io::Write,
|
||||||
|
F: Formatter,
|
||||||
|
{
|
||||||
|
type Error = io::Error;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_bool(&mut self, value: bool) -> io::Result<()> {
|
||||||
|
if value {
|
||||||
|
self.writer.write_all(b"true")
|
||||||
|
} else {
|
||||||
|
self.writer.write_all(b"false")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_isize(&mut self, value: isize) -> io::Result<()> {
|
||||||
|
write!(&mut self.writer, "{}", value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_i8(&mut self, value: i8) -> io::Result<()> {
|
||||||
|
write!(&mut self.writer, "{}", value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_i16(&mut self, value: i16) -> io::Result<()> {
|
||||||
|
write!(&mut self.writer, "{}", value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_i32(&mut self, value: i32) -> io::Result<()> {
|
||||||
|
write!(&mut self.writer, "{}", value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_i64(&mut self, value: i64) -> io::Result<()> {
|
||||||
|
write!(&mut self.writer, "{}", value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_usize(&mut self, value: usize) -> io::Result<()> {
|
||||||
|
write!(&mut self.writer, "{}", value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_u8(&mut self, value: u8) -> io::Result<()> {
|
||||||
|
write!(&mut self.writer, "{}", value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_u16(&mut self, value: u16) -> io::Result<()> {
|
||||||
|
write!(&mut self.writer, "{}", value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_u32(&mut self, value: u32) -> io::Result<()> {
|
||||||
|
write!(&mut self.writer, "{}", value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_u64(&mut self, value: u64) -> io::Result<()> {
|
||||||
|
write!(&mut self.writer, "{}", value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_f32(&mut self, value: f32) -> io::Result<()> {
|
||||||
|
fmt_f32_or_null(&mut self.writer, value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_f64(&mut self, value: f64) -> io::Result<()> {
|
||||||
|
fmt_f64_or_null(&mut self.writer, value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_char(&mut self, value: char) -> io::Result<()> {
|
||||||
|
escape_char(&mut self.writer, value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_str(&mut self, value: &str) -> io::Result<()> {
|
||||||
|
escape_str(&mut self.writer, value)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_none(&mut self) -> io::Result<()> {
|
||||||
|
self.visit_unit()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_some<V>(&mut self, value: V) -> io::Result<()>
|
||||||
|
where V: ser::Serialize
|
||||||
|
{
|
||||||
|
value.serialize(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_unit(&mut self) -> io::Result<()> {
|
||||||
|
self.writer.write_all(b"null")
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_enum_unit(&mut self, _name: &str, variant: &str) -> io::Result<()> {
|
||||||
|
try!(self.formatter.open(&mut self.writer, b'{'));
|
||||||
|
try!(self.formatter.comma(&mut self.writer, true));
|
||||||
|
try!(self.visit_str(variant));
|
||||||
|
try!(self.formatter.colon(&mut self.writer));
|
||||||
|
try!(self.writer.write_all(b"[]"));
|
||||||
|
self.formatter.close(&mut self.writer, b'}')
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_seq<V>(&mut self, mut visitor: V) -> io::Result<()>
|
||||||
|
where V: ser::SeqVisitor,
|
||||||
|
{
|
||||||
|
match visitor.len() {
|
||||||
|
Some(len) if len == 0 => {
|
||||||
|
self.writer.write_all(b"[]")
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
try!(self.formatter.open(&mut self.writer, b'['));
|
||||||
|
|
||||||
|
self.first = true;
|
||||||
|
|
||||||
|
while let Some(()) = try!(visitor.visit(self)) { }
|
||||||
|
|
||||||
|
self.formatter.close(&mut self.writer, b']')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_enum_seq<V>(&mut self, _name: &str, variant: &str, visitor: V) -> io::Result<()>
|
||||||
|
where V: ser::SeqVisitor,
|
||||||
|
{
|
||||||
|
try!(self.formatter.open(&mut self.writer, b'{'));
|
||||||
|
try!(self.formatter.comma(&mut self.writer, true));
|
||||||
|
try!(self.visit_str(variant));
|
||||||
|
try!(self.formatter.colon(&mut self.writer));
|
||||||
|
try!(self.visit_seq(visitor));
|
||||||
|
self.formatter.close(&mut self.writer, b'}')
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_seq_elt<T>(&mut self, value: T) -> io::Result<()>
|
||||||
|
where T: ser::Serialize,
|
||||||
|
{
|
||||||
|
try!(self.formatter.comma(&mut self.writer, self.first));
|
||||||
|
self.first = false;
|
||||||
|
|
||||||
|
value.serialize(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_map<V>(&mut self, mut visitor: V) -> io::Result<()>
|
||||||
|
where V: ser::MapVisitor,
|
||||||
|
{
|
||||||
|
match visitor.len() {
|
||||||
|
Some(len) if len == 0 => {
|
||||||
|
self.writer.write_all(b"{}")
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
try!(self.formatter.open(&mut self.writer, b'{'));
|
||||||
|
|
||||||
|
self.first = true;
|
||||||
|
|
||||||
|
while let Some(()) = try!(visitor.visit(self)) { }
|
||||||
|
|
||||||
|
self.formatter.close(&mut self.writer, b'}')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_enum_map<V>(&mut self, _name: &str, variant: &str, visitor: V) -> io::Result<()>
|
||||||
|
where V: ser::MapVisitor,
|
||||||
|
{
|
||||||
|
try!(self.formatter.open(&mut self.writer, b'{'));
|
||||||
|
try!(self.formatter.comma(&mut self.writer, true));
|
||||||
|
try!(self.visit_str(variant));
|
||||||
|
try!(self.formatter.colon(&mut self.writer));
|
||||||
|
try!(self.visit_map(visitor));
|
||||||
|
|
||||||
|
self.formatter.close(&mut self.writer, b'}')
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_map_elt<K, V>(&mut self, key: K, value: V) -> io::Result<()>
|
||||||
|
where K: ser::Serialize,
|
||||||
|
V: ser::Serialize,
|
||||||
|
{
|
||||||
|
try!(self.formatter.comma(&mut self.writer, self.first));
|
||||||
|
self.first = false;
|
||||||
|
|
||||||
|
try!(key.serialize(self));
|
||||||
|
try!(self.formatter.colon(&mut self.writer));
|
||||||
|
value.serialize(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn format() -> &'static str {
|
||||||
|
"json"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub trait Formatter {
|
||||||
|
fn open<W>(&mut self, writer: &mut W, ch: u8) -> io::Result<()>
|
||||||
|
where W: io::Write;
|
||||||
|
|
||||||
|
fn comma<W>(&mut self, writer: &mut W, first: bool) -> io::Result<()>
|
||||||
|
where W: io::Write;
|
||||||
|
|
||||||
|
fn colon<W>(&mut self, writer: &mut W) -> io::Result<()>
|
||||||
|
where W: io::Write;
|
||||||
|
|
||||||
|
fn close<W>(&mut self, writer: &mut W, ch: u8) -> io::Result<()>
|
||||||
|
where W: io::Write;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct CompactFormatter;
|
||||||
|
|
||||||
|
impl Formatter for CompactFormatter {
|
||||||
|
fn open<W>(&mut self, writer: &mut W, ch: u8) -> io::Result<()>
|
||||||
|
where W: io::Write,
|
||||||
|
{
|
||||||
|
writer.write_all(&[ch])
|
||||||
|
}
|
||||||
|
|
||||||
|
fn comma<W>(&mut self, writer: &mut W, first: bool) -> io::Result<()>
|
||||||
|
where W: io::Write,
|
||||||
|
{
|
||||||
|
if first {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
writer.write_all(b",")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn colon<W>(&mut self, writer: &mut W) -> io::Result<()>
|
||||||
|
where W: io::Write,
|
||||||
|
{
|
||||||
|
writer.write_all(b":")
|
||||||
|
}
|
||||||
|
|
||||||
|
fn close<W>(&mut self, writer: &mut W, ch: u8) -> io::Result<()>
|
||||||
|
where W: io::Write,
|
||||||
|
{
|
||||||
|
writer.write_all(&[ch])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct PrettyFormatter<'a> {
|
||||||
|
current_indent: usize,
|
||||||
|
indent: &'a [u8],
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> PrettyFormatter<'a> {
|
||||||
|
fn new() -> Self {
|
||||||
|
PrettyFormatter::with_indent(b" ")
|
||||||
|
}
|
||||||
|
|
||||||
|
fn with_indent(indent: &'a [u8]) -> Self {
|
||||||
|
PrettyFormatter {
|
||||||
|
current_indent: 0,
|
||||||
|
indent: indent,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> Formatter for PrettyFormatter<'a> {
|
||||||
|
fn open<W>(&mut self, writer: &mut W, ch: u8) -> io::Result<()>
|
||||||
|
where W: io::Write,
|
||||||
|
{
|
||||||
|
self.current_indent += 1;
|
||||||
|
writer.write_all(&[ch])
|
||||||
|
}
|
||||||
|
|
||||||
|
fn comma<W>(&mut self, writer: &mut W, first: bool) -> io::Result<()>
|
||||||
|
where W: io::Write,
|
||||||
|
{
|
||||||
|
if first {
|
||||||
|
try!(writer.write_all(b"\n"));
|
||||||
|
} else {
|
||||||
|
try!(writer.write_all(b",\n"));
|
||||||
|
}
|
||||||
|
|
||||||
|
indent(writer, self.current_indent, self.indent)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn colon<W>(&mut self, writer: &mut W) -> io::Result<()>
|
||||||
|
where W: io::Write,
|
||||||
|
{
|
||||||
|
writer.write_all(b": ")
|
||||||
|
}
|
||||||
|
|
||||||
|
fn close<W>(&mut self, writer: &mut W, ch: u8) -> io::Result<()>
|
||||||
|
where W: io::Write,
|
||||||
|
{
|
||||||
|
self.current_indent -= 1;
|
||||||
|
try!(writer.write(b"\n"));
|
||||||
|
try!(indent(writer, self.current_indent, self.indent));
|
||||||
|
|
||||||
|
writer.write_all(&[ch])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn escape_bytes<W>(wr: &mut W, bytes: &[u8]) -> io::Result<()>
|
||||||
|
where W: io::Write
|
||||||
|
{
|
||||||
|
try!(wr.write_all(b"\""));
|
||||||
|
|
||||||
|
let mut start = 0;
|
||||||
|
|
||||||
|
for (i, byte) in bytes.iter().enumerate() {
|
||||||
|
let escaped = match *byte {
|
||||||
|
b'"' => b"\\\"",
|
||||||
|
b'\\' => b"\\\\",
|
||||||
|
b'\x08' => b"\\b",
|
||||||
|
b'\x0c' => b"\\f",
|
||||||
|
b'\n' => b"\\n",
|
||||||
|
b'\r' => b"\\r",
|
||||||
|
b'\t' => b"\\t",
|
||||||
|
_ => { continue; }
|
||||||
|
};
|
||||||
|
|
||||||
|
if start < i {
|
||||||
|
try!(wr.write_all(&bytes[start..i]));
|
||||||
|
}
|
||||||
|
|
||||||
|
try!(wr.write_all(escaped));
|
||||||
|
|
||||||
|
start = i + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if start != bytes.len() {
|
||||||
|
try!(wr.write_all(&bytes[start..]));
|
||||||
|
}
|
||||||
|
|
||||||
|
try!(wr.write_all(b"\""));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn escape_str<W>(wr: &mut W, value: &str) -> io::Result<()>
|
||||||
|
where W: io::Write
|
||||||
|
{
|
||||||
|
escape_bytes(wr, value.as_bytes())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn escape_char<W>(wr: &mut W, value: char) -> io::Result<()>
|
||||||
|
where W: io::Write
|
||||||
|
{
|
||||||
|
// FIXME: this allocation is required in order to be compatible with stable
|
||||||
|
// rust, which doesn't support encoding a `char` into a stack buffer.
|
||||||
|
escape_bytes(wr, value.to_string().as_bytes())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn fmt_f32_or_null<W>(wr: &mut W, value: f32) -> io::Result<()>
|
||||||
|
where W: io::Write
|
||||||
|
{
|
||||||
|
match value.classify() {
|
||||||
|
FpCategory::Nan | FpCategory::Infinite => wr.write_all(b"null"),
|
||||||
|
_ => {
|
||||||
|
let s = format!("{:?}", value);
|
||||||
|
try!(wr.write_all(s.as_bytes()));
|
||||||
|
if !s.contains('.') {
|
||||||
|
try!(wr.write_all(b".0"))
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn fmt_f64_or_null<W>(wr: &mut W, value: f64) -> io::Result<()>
|
||||||
|
where W: io::Write
|
||||||
|
{
|
||||||
|
match value.classify() {
|
||||||
|
FpCategory::Nan | FpCategory::Infinite => wr.write_all(b"null"),
|
||||||
|
_ => {
|
||||||
|
let s = format!("{:?}", value);
|
||||||
|
try!(wr.write_all(s.as_bytes()));
|
||||||
|
if !s.contains('.') {
|
||||||
|
try!(wr.write_all(b".0"))
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Encode the specified struct into a json `[u8]` writer.
|
||||||
|
#[inline]
|
||||||
|
pub fn to_writer<W, T>(writer: &mut W, value: &T) -> io::Result<()>
|
||||||
|
where W: io::Write,
|
||||||
|
T: ser::Serialize,
|
||||||
|
{
|
||||||
|
let mut ser = Serializer::new(writer);
|
||||||
|
try!(value.serialize(&mut ser));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Encode the specified struct into a json `[u8]` writer.
|
||||||
|
#[inline]
|
||||||
|
pub fn to_writer_pretty<W, T>(writer: &mut W, value: &T) -> io::Result<()>
|
||||||
|
where W: io::Write,
|
||||||
|
T: ser::Serialize,
|
||||||
|
{
|
||||||
|
let mut ser = Serializer::pretty(writer);
|
||||||
|
try!(value.serialize(&mut ser));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Encode the specified struct into a json `[u8]` buffer.
|
||||||
|
#[inline]
|
||||||
|
pub fn to_vec<T>(value: &T) -> Vec<u8>
|
||||||
|
where T: ser::Serialize,
|
||||||
|
{
|
||||||
|
// We are writing to a Vec, which doesn't fail. So we can ignore
|
||||||
|
// the error.
|
||||||
|
let mut writer = Vec::with_capacity(128);
|
||||||
|
to_writer(&mut writer, value).unwrap();
|
||||||
|
writer
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Encode the specified struct into a json `[u8]` buffer.
|
||||||
|
#[inline]
|
||||||
|
pub fn to_vec_pretty<T>(value: &T) -> Vec<u8>
|
||||||
|
where T: ser::Serialize,
|
||||||
|
{
|
||||||
|
// We are writing to a Vec, which doesn't fail. So we can ignore
|
||||||
|
// the error.
|
||||||
|
let mut writer = Vec::with_capacity(128);
|
||||||
|
to_writer_pretty(&mut writer, value).unwrap();
|
||||||
|
writer
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Encode the specified struct into a json `String` buffer.
|
||||||
|
#[inline]
|
||||||
|
pub fn to_string<T>(value: &T) -> Result<String, FromUtf8Error>
|
||||||
|
where T: ser::Serialize
|
||||||
|
{
|
||||||
|
let vec = to_vec(value);
|
||||||
|
String::from_utf8(vec)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Encode the specified struct into a json `String` buffer.
|
||||||
|
#[inline]
|
||||||
|
pub fn to_string_pretty<T>(value: &T) -> Result<String, FromUtf8Error>
|
||||||
|
where T: ser::Serialize
|
||||||
|
{
|
||||||
|
let vec = to_vec_pretty(value);
|
||||||
|
String::from_utf8(vec)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn indent<W>(wr: &mut W, n: usize, s: &[u8]) -> io::Result<()>
|
||||||
|
where W: io::Write,
|
||||||
|
{
|
||||||
|
for _ in 0 .. n {
|
||||||
|
try!(wr.write_all(s));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
@@ -0,0 +1,915 @@
|
|||||||
|
use std::collections::{BTreeMap, btree_map};
|
||||||
|
use std::fmt;
|
||||||
|
use std::io;
|
||||||
|
use std::str;
|
||||||
|
use std::vec;
|
||||||
|
|
||||||
|
use num::NumCast;
|
||||||
|
|
||||||
|
use de;
|
||||||
|
use ser;
|
||||||
|
use super::error::Error;
|
||||||
|
|
||||||
|
#[derive(Clone, PartialEq)]
|
||||||
|
pub enum Value {
|
||||||
|
Null,
|
||||||
|
Bool(bool),
|
||||||
|
I64(i64),
|
||||||
|
U64(u64),
|
||||||
|
F64(f64),
|
||||||
|
String(String),
|
||||||
|
Array(Vec<Value>),
|
||||||
|
Object(BTreeMap<String, Value>),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Value {
|
||||||
|
/// If the `Value` is an Object, returns the value associated with the provided key.
|
||||||
|
/// Otherwise, returns None.
|
||||||
|
pub fn find<'a>(&'a self, key: &str) -> Option<&'a Value>{
|
||||||
|
match self {
|
||||||
|
&Value::Object(ref map) => map.get(key),
|
||||||
|
_ => None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Attempts to get a nested Value Object for each key in `keys`.
|
||||||
|
/// If any key is found not to exist, find_path will return None.
|
||||||
|
/// Otherwise, it will return the `Value` associated with the final key.
|
||||||
|
pub fn find_path<'a>(&'a self, keys: &[&str]) -> Option<&'a Value>{
|
||||||
|
let mut target = self;
|
||||||
|
for key in keys {
|
||||||
|
match target.find(key) {
|
||||||
|
Some(t) => { target = t; },
|
||||||
|
None => return None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some(target)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Looks up a value by path.
|
||||||
|
///
|
||||||
|
/// This is a convenience method that splits the path by `'.'`
|
||||||
|
/// and then feeds the sequence of keys into the `find_path`
|
||||||
|
/// method.
|
||||||
|
///
|
||||||
|
/// ``` ignore
|
||||||
|
/// let obj: Value = json::from_str(r#"{"x": {"a": 1}}"#).unwrap();
|
||||||
|
///
|
||||||
|
/// assert!(obj.lookup("x.a").unwrap() == &Value::U64(1));
|
||||||
|
/// ```
|
||||||
|
pub fn lookup<'a>(&'a self, path: &str) -> Option<&'a Value> {
|
||||||
|
let mut target = self;
|
||||||
|
for key in path.split('.') {
|
||||||
|
match target.find(key) {
|
||||||
|
Some(t) => { target = t; },
|
||||||
|
None => return None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some(target)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// If the `Value` is an Object, performs a depth-first search until
|
||||||
|
/// a value associated with the provided key is found. If no value is found
|
||||||
|
/// or the `Value` is not an Object, returns None.
|
||||||
|
pub fn search<'a>(&'a self, key: &str) -> Option<&'a Value> {
|
||||||
|
match self {
|
||||||
|
&Value::Object(ref map) => {
|
||||||
|
match map.get(key) {
|
||||||
|
Some(json_value) => Some(json_value),
|
||||||
|
None => {
|
||||||
|
for (_, v) in map.iter() {
|
||||||
|
match v.search(key) {
|
||||||
|
x if x.is_some() => return x,
|
||||||
|
_ => ()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
_ => None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if the `Value` is an Object. Returns false otherwise.
|
||||||
|
pub fn is_object<'a>(&'a self) -> bool {
|
||||||
|
self.as_object().is_some()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// If the `Value` is an Object, returns the associated BTreeMap.
|
||||||
|
/// Returns None otherwise.
|
||||||
|
pub fn as_object<'a>(&'a self) -> Option<&'a BTreeMap<String, Value>> {
|
||||||
|
match self {
|
||||||
|
&Value::Object(ref map) => Some(map),
|
||||||
|
_ => None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// If the `Value` is an Object, returns the associated mutable BTreeMap.
|
||||||
|
/// Returns None otherwise.
|
||||||
|
pub fn as_object_mut<'a>(&'a mut self) -> Option<&'a mut BTreeMap<String, Value>> {
|
||||||
|
match self {
|
||||||
|
&mut Value::Object(ref mut map) => Some(map),
|
||||||
|
_ => None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if the `Value` is an Array. Returns false otherwise.
|
||||||
|
pub fn is_array<'a>(&'a self) -> bool {
|
||||||
|
self.as_array().is_some()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// If the `Value` is an Array, returns the associated vector.
|
||||||
|
/// Returns None otherwise.
|
||||||
|
pub fn as_array<'a>(&'a self) -> Option<&'a Vec<Value>> {
|
||||||
|
match self {
|
||||||
|
&Value::Array(ref array) => Some(&*array),
|
||||||
|
_ => None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// If the `Value` is an Array, returns the associated mutable vector.
|
||||||
|
/// Returns None otherwise.
|
||||||
|
pub fn as_array_mut<'a>(&'a mut self) -> Option<&'a mut Vec<Value>> {
|
||||||
|
match self {
|
||||||
|
&mut Value::Array(ref mut list) => Some(list),
|
||||||
|
_ => None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if the `Value` is a String. Returns false otherwise.
|
||||||
|
pub fn is_string<'a>(&'a self) -> bool {
|
||||||
|
self.as_string().is_some()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// If the `Value` is a String, returns the associated str.
|
||||||
|
/// Returns None otherwise.
|
||||||
|
pub fn as_string<'a>(&'a self) -> Option<&'a str> {
|
||||||
|
match *self {
|
||||||
|
Value::String(ref s) => Some(&s),
|
||||||
|
_ => None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if the `Value` is a Number. Returns false otherwise.
|
||||||
|
pub fn is_number(&self) -> bool {
|
||||||
|
match *self {
|
||||||
|
Value::I64(_) | Value::U64(_) | Value::F64(_) => true,
|
||||||
|
_ => false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if the `Value` is a i64. Returns false otherwise.
|
||||||
|
pub fn is_i64(&self) -> bool {
|
||||||
|
match *self {
|
||||||
|
Value::I64(_) => true,
|
||||||
|
_ => false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if the `Value` is a u64. Returns false otherwise.
|
||||||
|
pub fn is_u64(&self) -> bool {
|
||||||
|
match *self {
|
||||||
|
Value::U64(_) => true,
|
||||||
|
_ => false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if the `Value` is a f64. Returns false otherwise.
|
||||||
|
pub fn is_f64(&self) -> bool {
|
||||||
|
match *self {
|
||||||
|
Value::F64(_) => true,
|
||||||
|
_ => false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// If the `Value` is a number, return or cast it to a i64.
|
||||||
|
/// Returns None otherwise.
|
||||||
|
pub fn as_i64(&self) -> Option<i64> {
|
||||||
|
match *self {
|
||||||
|
Value::I64(n) => Some(n),
|
||||||
|
Value::U64(n) => NumCast::from(n),
|
||||||
|
_ => None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// If the `Value` is a number, return or cast it to a u64.
|
||||||
|
/// Returns None otherwise.
|
||||||
|
pub fn as_u64(&self) -> Option<u64> {
|
||||||
|
match *self {
|
||||||
|
Value::I64(n) => NumCast::from(n),
|
||||||
|
Value::U64(n) => Some(n),
|
||||||
|
_ => None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// If the `Value` is a number, return or cast it to a f64.
|
||||||
|
/// Returns None otherwise.
|
||||||
|
pub fn as_f64(&self) -> Option<f64> {
|
||||||
|
match *self {
|
||||||
|
Value::I64(n) => NumCast::from(n),
|
||||||
|
Value::U64(n) => NumCast::from(n),
|
||||||
|
Value::F64(n) => Some(n),
|
||||||
|
_ => None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if the `Value` is a Boolean. Returns false otherwise.
|
||||||
|
pub fn is_boolean(&self) -> bool {
|
||||||
|
self.as_boolean().is_some()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// If the `Value` is a Boolean, returns the associated bool.
|
||||||
|
/// Returns None otherwise.
|
||||||
|
pub fn as_boolean(&self) -> Option<bool> {
|
||||||
|
match self {
|
||||||
|
&Value::Bool(b) => Some(b),
|
||||||
|
_ => None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Returns true if the `Value` is a Null. Returns false otherwise.
|
||||||
|
pub fn is_null(&self) -> bool {
|
||||||
|
self.as_null().is_some()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// If the `Value` is a Null, returns ().
|
||||||
|
/// Returns None otherwise.
|
||||||
|
pub fn as_null(&self) -> Option<()> {
|
||||||
|
match self {
|
||||||
|
&Value::Null => Some(()),
|
||||||
|
_ => None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ser::Serialize for Value {
|
||||||
|
#[inline]
|
||||||
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
|
where S: ser::Serializer,
|
||||||
|
{
|
||||||
|
match *self {
|
||||||
|
Value::Null => serializer.visit_unit(),
|
||||||
|
Value::Bool(v) => serializer.visit_bool(v),
|
||||||
|
Value::I64(v) => serializer.visit_i64(v),
|
||||||
|
Value::U64(v) => serializer.visit_u64(v),
|
||||||
|
Value::F64(v) => serializer.visit_f64(v),
|
||||||
|
Value::String(ref v) => serializer.visit_str(&v),
|
||||||
|
Value::Array(ref v) => v.serialize(serializer),
|
||||||
|
Value::Object(ref v) => v.serialize(serializer),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl de::Deserialize for Value {
|
||||||
|
#[inline]
|
||||||
|
fn deserialize<D>(deserializer: &mut D) -> Result<Value, D::Error>
|
||||||
|
where D: de::Deserializer,
|
||||||
|
{
|
||||||
|
struct ValueVisitor;
|
||||||
|
|
||||||
|
impl de::Visitor for ValueVisitor {
|
||||||
|
type Value = Value;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_bool<E>(&mut self, value: bool) -> Result<Value, E> {
|
||||||
|
Ok(Value::Bool(value))
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_i64<E>(&mut self, value: i64) -> Result<Value, E> {
|
||||||
|
if value < 0 {
|
||||||
|
Ok(Value::I64(value))
|
||||||
|
} else {
|
||||||
|
Ok(Value::U64(value as u64))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_u64<E>(&mut self, value: u64) -> Result<Value, E> {
|
||||||
|
Ok(Value::U64(value))
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_f64<E>(&mut self, value: f64) -> Result<Value, E> {
|
||||||
|
Ok(Value::F64(value))
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_str<E>(&mut self, value: &str) -> Result<Value, E>
|
||||||
|
where E: de::Error,
|
||||||
|
{
|
||||||
|
self.visit_string(value.to_string())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_string<E>(&mut self, value: String) -> Result<Value, E> {
|
||||||
|
Ok(Value::String(value))
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_none<E>(&mut self) -> Result<Value, E> {
|
||||||
|
Ok(Value::Null)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_some<D>(&mut self, deserializer: &mut D) -> Result<Value, D::Error>
|
||||||
|
where D: de::Deserializer,
|
||||||
|
{
|
||||||
|
de::Deserialize::deserialize(deserializer)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_unit<E>(&mut self) -> Result<Value, E> {
|
||||||
|
Ok(Value::Null)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_seq<V>(&mut self, visitor: V) -> Result<Value, V::Error>
|
||||||
|
where V: de::SeqVisitor,
|
||||||
|
{
|
||||||
|
let values = try!(de::impls::VecVisitor::new().visit_seq(visitor));
|
||||||
|
Ok(Value::Array(values))
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_map<V>(&mut self, visitor: V) -> Result<Value, V::Error>
|
||||||
|
where V: de::MapVisitor,
|
||||||
|
{
|
||||||
|
let values = try!(de::impls::BTreeMapVisitor::new().visit_map(visitor));
|
||||||
|
Ok(Value::Object(values))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
deserializer.visit(ValueVisitor)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct WriterFormatter<'a, 'b: 'a> {
|
||||||
|
inner: &'a mut fmt::Formatter<'b>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, 'b> io::Write for WriterFormatter<'a, 'b> {
|
||||||
|
fn write(&mut self, buf: &[u8]) -> io::Result<usize> {
|
||||||
|
match self.inner.write_str(str::from_utf8(buf).unwrap()) {
|
||||||
|
Ok(_) => Ok(buf.len()),
|
||||||
|
Err(_) => Err(io::Error::last_os_error()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn flush(&mut self) -> io::Result<()> {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl fmt::Debug for Value {
|
||||||
|
/// Serializes a json value into a string
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
||||||
|
let mut wr = WriterFormatter { inner: f };
|
||||||
|
super::ser::to_writer(&mut wr, self).map_err(|_| fmt::Error)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
enum State {
|
||||||
|
Value(Value),
|
||||||
|
Array(Vec<Value>),
|
||||||
|
Object(BTreeMap<String, Value>),
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Serializer {
|
||||||
|
state: Vec<State>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Serializer {
|
||||||
|
pub fn new() -> Serializer {
|
||||||
|
Serializer {
|
||||||
|
state: Vec::with_capacity(4),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn unwrap(mut self) -> Value {
|
||||||
|
match self.state.pop().unwrap() {
|
||||||
|
State::Value(value) => value,
|
||||||
|
state => panic!("expected value, found {:?}", state),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ser::Serializer for Serializer {
|
||||||
|
type Error = ();
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_bool(&mut self, value: bool) -> Result<(), ()> {
|
||||||
|
self.state.push(State::Value(Value::Bool(value)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_i64(&mut self, value: i64) -> Result<(), ()> {
|
||||||
|
if value < 0 {
|
||||||
|
self.state.push(State::Value(Value::I64(value)));
|
||||||
|
} else {
|
||||||
|
self.state.push(State::Value(Value::U64(value as u64)));
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_u64(&mut self, value: u64) -> Result<(), ()> {
|
||||||
|
self.state.push(State::Value(Value::U64(value)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_f64(&mut self, value: f64) -> Result<(), ()> {
|
||||||
|
self.state.push(State::Value(Value::F64(value as f64)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_char(&mut self, value: char) -> Result<(), ()> {
|
||||||
|
self.state.push(State::Value(Value::String(value.to_string())));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_str(&mut self, value: &str) -> Result<(), ()> {
|
||||||
|
self.state.push(State::Value(Value::String(value.to_string())));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_none(&mut self) -> Result<(), ()> {
|
||||||
|
self.visit_unit()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_some<V>(&mut self, value: V) -> Result<(), ()>
|
||||||
|
where V: ser::Serialize,
|
||||||
|
{
|
||||||
|
value.serialize(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_unit(&mut self) -> Result<(), ()> {
|
||||||
|
self.state.push(State::Value(Value::Null));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_enum_unit(&mut self, _name: &str, variant: &str) -> Result<(), ()> {
|
||||||
|
let mut values = BTreeMap::new();
|
||||||
|
values.insert(variant.to_string(), Value::Array(vec![]));
|
||||||
|
|
||||||
|
self.state.push(State::Value(Value::Object(values)));
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_seq<V>(&mut self, mut visitor: V) -> Result<(), ()>
|
||||||
|
where V: ser::SeqVisitor,
|
||||||
|
{
|
||||||
|
let len = visitor.len().unwrap_or(0);
|
||||||
|
let values = Vec::with_capacity(len);
|
||||||
|
|
||||||
|
self.state.push(State::Array(values));
|
||||||
|
|
||||||
|
while let Some(()) = try!(visitor.visit(self)) { }
|
||||||
|
|
||||||
|
let values = match self.state.pop().unwrap() {
|
||||||
|
State::Array(values) => values,
|
||||||
|
state => panic!("Expected array, found {:?}", state),
|
||||||
|
};
|
||||||
|
|
||||||
|
self.state.push(State::Value(Value::Array(values)));
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_enum_seq<V>(&mut self, _name: &str, variant: &str, visitor: V) -> Result<(), ()>
|
||||||
|
where V: ser::SeqVisitor,
|
||||||
|
{
|
||||||
|
try!(self.visit_seq(visitor));
|
||||||
|
|
||||||
|
let value = match self.state.pop().unwrap() {
|
||||||
|
State::Value(value) => value,
|
||||||
|
state => panic!("expected value, found {:?}", state),
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut object = BTreeMap::new();
|
||||||
|
|
||||||
|
object.insert(variant.to_string(), value);
|
||||||
|
|
||||||
|
self.state.push(State::Value(Value::Object(object)));
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_seq_elt<T>(&mut self, value: T) -> Result<(), ()>
|
||||||
|
where T: ser::Serialize,
|
||||||
|
{
|
||||||
|
try!(value.serialize(self));
|
||||||
|
|
||||||
|
let value = match self.state.pop().unwrap() {
|
||||||
|
State::Value(value) => value,
|
||||||
|
state => panic!("expected value, found {:?}", state),
|
||||||
|
};
|
||||||
|
|
||||||
|
match *self.state.last_mut().unwrap() {
|
||||||
|
State::Array(ref mut values) => { values.push(value); }
|
||||||
|
ref state => panic!("expected array, found {:?}", state),
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_map<V>(&mut self, mut visitor: V) -> Result<(), ()>
|
||||||
|
where V: ser::MapVisitor,
|
||||||
|
{
|
||||||
|
let values = BTreeMap::new();
|
||||||
|
|
||||||
|
self.state.push(State::Object(values));
|
||||||
|
|
||||||
|
while let Some(()) = try!(visitor.visit(self)) { }
|
||||||
|
|
||||||
|
let values = match self.state.pop().unwrap() {
|
||||||
|
State::Object(values) => values,
|
||||||
|
state => panic!("expected object, found {:?}", state),
|
||||||
|
};
|
||||||
|
|
||||||
|
self.state.push(State::Value(Value::Object(values)));
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_enum_map<V>(&mut self, _name: &str, variant: &str, visitor: V) -> Result<(), ()>
|
||||||
|
where V: ser::MapVisitor,
|
||||||
|
{
|
||||||
|
try!(self.visit_map(visitor));
|
||||||
|
|
||||||
|
let value = match self.state.pop().unwrap() {
|
||||||
|
State::Value(value) => value,
|
||||||
|
state => panic!("expected value, found {:?}", state),
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut object = BTreeMap::new();
|
||||||
|
|
||||||
|
object.insert(variant.to_string(), value);
|
||||||
|
|
||||||
|
self.state.push(State::Value(Value::Object(object)));
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_map_elt<K, V>(&mut self, key: K, value: V) -> Result<(), ()>
|
||||||
|
where K: ser::Serialize,
|
||||||
|
V: ser::Serialize,
|
||||||
|
{
|
||||||
|
try!(key.serialize(self));
|
||||||
|
|
||||||
|
let key = match self.state.pop().unwrap() {
|
||||||
|
State::Value(Value::String(value)) => value,
|
||||||
|
state => panic!("expected key, found {:?}", state),
|
||||||
|
};
|
||||||
|
|
||||||
|
try!(value.serialize(self));
|
||||||
|
|
||||||
|
let value = match self.state.pop().unwrap() {
|
||||||
|
State::Value(value) => value,
|
||||||
|
state => panic!("expected value, found {:?}", state),
|
||||||
|
};
|
||||||
|
|
||||||
|
match *self.state.last_mut().unwrap() {
|
||||||
|
State::Object(ref mut values) => { values.insert(key, value); }
|
||||||
|
ref state => panic!("expected object, found {:?}", state),
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn format() -> &'static str {
|
||||||
|
"json"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Deserializer {
|
||||||
|
value: Option<Value>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Deserializer {
|
||||||
|
/// Creates a new deserializer instance for deserializing the specified JSON value.
|
||||||
|
pub fn new(value: Value) -> Deserializer {
|
||||||
|
Deserializer {
|
||||||
|
value: Some(value),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl de::Deserializer for Deserializer {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
let value = match self.value.take() {
|
||||||
|
Some(value) => value,
|
||||||
|
None => { return Err(de::Error::end_of_stream_error()); }
|
||||||
|
};
|
||||||
|
|
||||||
|
match value {
|
||||||
|
Value::Null => visitor.visit_unit(),
|
||||||
|
Value::Bool(v) => visitor.visit_bool(v),
|
||||||
|
Value::I64(v) => visitor.visit_i64(v),
|
||||||
|
Value::U64(v) => visitor.visit_u64(v),
|
||||||
|
Value::F64(v) => visitor.visit_f64(v),
|
||||||
|
Value::String(v) => visitor.visit_string(v),
|
||||||
|
Value::Array(v) => {
|
||||||
|
let len = v.len();
|
||||||
|
visitor.visit_seq(SeqDeserializer {
|
||||||
|
de: self,
|
||||||
|
iter: v.into_iter(),
|
||||||
|
len: len,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
Value::Object(v) => {
|
||||||
|
let len = v.len();
|
||||||
|
visitor.visit_map(MapDeserializer {
|
||||||
|
de: self,
|
||||||
|
iter: v.into_iter(),
|
||||||
|
value: None,
|
||||||
|
len: len,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_option<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
match self.value {
|
||||||
|
Some(Value::Null) => visitor.visit_none(),
|
||||||
|
Some(_) => visitor.visit_some(self),
|
||||||
|
None => Err(de::Error::end_of_stream_error()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit_enum<V>(&mut self, _name: &str, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::EnumVisitor,
|
||||||
|
{
|
||||||
|
let value = match self.value.take() {
|
||||||
|
Some(Value::Object(value)) => value,
|
||||||
|
Some(_) => { return Err(de::Error::syntax_error()); }
|
||||||
|
None => { return Err(de::Error::end_of_stream_error()); }
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut iter = value.into_iter();
|
||||||
|
|
||||||
|
let value = match iter.next() {
|
||||||
|
Some((variant, Value::Array(fields))) => {
|
||||||
|
self.value = Some(Value::String(variant));
|
||||||
|
|
||||||
|
let len = fields.len();
|
||||||
|
try!(visitor.visit(SeqDeserializer {
|
||||||
|
de: self,
|
||||||
|
iter: fields.into_iter(),
|
||||||
|
len: len,
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
Some((variant, Value::Object(fields))) => {
|
||||||
|
let len = fields.len();
|
||||||
|
try!(visitor.visit(MapDeserializer {
|
||||||
|
de: self,
|
||||||
|
iter: fields.into_iter(),
|
||||||
|
value: Some(Value::String(variant)),
|
||||||
|
len: len,
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
Some(_) => { return Err(de::Error::syntax_error()); }
|
||||||
|
None => { return Err(de::Error::syntax_error()); }
|
||||||
|
};
|
||||||
|
|
||||||
|
match iter.next() {
|
||||||
|
Some(_) => Err(de::Error::syntax_error()),
|
||||||
|
None => Ok(value)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn format() -> &'static str {
|
||||||
|
"json"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct SeqDeserializer<'a> {
|
||||||
|
de: &'a mut Deserializer,
|
||||||
|
iter: vec::IntoIter<Value>,
|
||||||
|
len: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> de::Deserializer for SeqDeserializer<'a> {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
if self.len == 0 {
|
||||||
|
visitor.visit_unit()
|
||||||
|
} else {
|
||||||
|
visitor.visit_seq(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> de::SeqVisitor for SeqDeserializer<'a> {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit<T>(&mut self) -> Result<Option<T>, Error>
|
||||||
|
where T: de::Deserialize
|
||||||
|
{
|
||||||
|
match self.iter.next() {
|
||||||
|
Some(value) => {
|
||||||
|
self.len -= 1;
|
||||||
|
self.de.value = Some(value);
|
||||||
|
Ok(Some(try!(de::Deserialize::deserialize(self.de))))
|
||||||
|
}
|
||||||
|
None => Ok(None),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
|
if self.len == 0 {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(de::Error::end_of_stream_error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
||||||
|
(self.len, Some(self.len))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> de::VariantVisitor for SeqDeserializer<'a> {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit_variant<V>(&mut self) -> Result<V, Error>
|
||||||
|
where V: de::Deserialize,
|
||||||
|
{
|
||||||
|
de::Deserialize::deserialize(self.de)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_unit(&mut self) -> Result<(), Error>
|
||||||
|
{
|
||||||
|
de::Deserialize::deserialize(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_seq<V>(&mut self, visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
de::Deserializer::visit(self, visitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_map<V>(&mut self, visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
de::Deserializer::visit(self, visitor)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct MapDeserializer<'a> {
|
||||||
|
de: &'a mut Deserializer,
|
||||||
|
iter: btree_map::IntoIter<String, Value>,
|
||||||
|
value: Option<Value>,
|
||||||
|
len: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> de::MapVisitor for MapDeserializer<'a> {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit_key<T>(&mut self) -> Result<Option<T>, Error>
|
||||||
|
where T: de::Deserialize
|
||||||
|
{
|
||||||
|
match self.iter.next() {
|
||||||
|
Some((key, value)) => {
|
||||||
|
self.len -= 1;
|
||||||
|
self.value = Some(value);
|
||||||
|
self.de.value = Some(Value::String(key));
|
||||||
|
Ok(Some(try!(de::Deserialize::deserialize(self.de))))
|
||||||
|
}
|
||||||
|
None => Ok(None),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_value<T>(&mut self) -> Result<T, Error>
|
||||||
|
where T: de::Deserialize
|
||||||
|
{
|
||||||
|
let value = self.value.take().unwrap();
|
||||||
|
self.de.value = Some(value);
|
||||||
|
Ok(try!(de::Deserialize::deserialize(self.de)))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
|
if self.len == 0 {
|
||||||
|
Ok(())
|
||||||
|
} else {
|
||||||
|
Err(de::Error::end_of_stream_error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn missing_field<V>(&mut self, _field: &'static str) -> Result<V, Error>
|
||||||
|
where V: de::Deserialize,
|
||||||
|
{
|
||||||
|
// See if the type can deserialize from a unit.
|
||||||
|
struct UnitDeserializer;
|
||||||
|
|
||||||
|
impl de::Deserializer for UnitDeserializer {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
visitor.visit_unit()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_option<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
visitor.visit_none()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(try!(de::Deserialize::deserialize(&mut UnitDeserializer)))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
||||||
|
(self.len, Some(self.len))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> de::Deserializer for MapDeserializer<'a> {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn visit<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
println!("MapDeserializer!");
|
||||||
|
visitor.visit_map(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> de::VariantVisitor for MapDeserializer<'a> {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit_variant<V>(&mut self) -> Result<V, Error>
|
||||||
|
where V: de::Deserialize,
|
||||||
|
{
|
||||||
|
self.de.value = self.value.take();
|
||||||
|
de::Deserialize::deserialize(self.de)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_unit(&mut self) -> Result<(), Error> {
|
||||||
|
de::Deserialize::deserialize(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_seq<V>(&mut self, visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
de::Deserializer::visit(self, visitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_map<V>(&mut self, visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
de::Deserializer::visit(self, visitor)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Shortcut function to encode a `T` into a JSON `Value`
|
||||||
|
pub fn to_value<T>(value: &T) -> Value
|
||||||
|
where T: ser::Serialize
|
||||||
|
{
|
||||||
|
let mut ser = Serializer::new();
|
||||||
|
value.serialize(&mut ser).ok().unwrap();
|
||||||
|
ser.unwrap()
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Shortcut function to decode a JSON `Value` into a `T`
|
||||||
|
pub fn from_value<T>(value: Value) -> Result<T, Error>
|
||||||
|
where T: de::Deserialize
|
||||||
|
{
|
||||||
|
let mut de = Deserializer::new(value);
|
||||||
|
de::Deserialize::deserialize(&mut de)
|
||||||
|
}
|
||||||
+3
-39
@@ -5,51 +5,15 @@
|
|||||||
//! handshake protocol between serializers and serializees can be completely optimized away,
|
//! handshake protocol between serializers and serializees can be completely optimized away,
|
||||||
//! leaving serde to perform roughly the same speed as a hand written serializer for a specific
|
//! leaving serde to perform roughly the same speed as a hand written serializer for a specific
|
||||||
//! type.
|
//! type.
|
||||||
//!
|
#![doc(html_root_url="http://erickt.github.io/rust-serde")]
|
||||||
//! For a detailed tutorial on the different ways to use serde please check out the
|
|
||||||
//! [github repository](https://github.com/serde-rs/serde)
|
|
||||||
|
|
||||||
#![doc(html_root_url="https://docs.serde.rs")]
|
extern crate num;
|
||||||
#![cfg_attr(not(feature = "std"), no_std)]
|
|
||||||
#![cfg_attr(feature = "unstable", feature(reflect_marker, unicode, nonzero, plugin, step_trait, zero_one))]
|
|
||||||
#![cfg_attr(feature = "alloc", feature(alloc))]
|
|
||||||
#![cfg_attr(feature = "collections", feature(collections, enumset))]
|
|
||||||
#![cfg_attr(feature = "clippy", plugin(clippy))]
|
|
||||||
#![cfg_attr(feature = "clippy", allow(linkedlist))]
|
|
||||||
|
|
||||||
#![cfg_attr(any(not(feature = "std"), feature = "unstable"), allow(unused_variables, unused_imports, unused_features, dead_code))]
|
|
||||||
|
|
||||||
#![deny(missing_docs)]
|
|
||||||
|
|
||||||
#[cfg(all(feature = "unstable", feature = "collections"))]
|
|
||||||
extern crate collections;
|
|
||||||
|
|
||||||
#[cfg(all(feature = "unstable", feature = "alloc"))]
|
|
||||||
extern crate alloc;
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
mod core {
|
|
||||||
pub use std::{ops, hash, fmt, cmp, marker, mem, i8, i16, i32, i64, u8, u16, u32, u64, isize,
|
|
||||||
usize, f32, f64, char, str, num, slice, iter};
|
|
||||||
#[cfg(feature = "unstable")]
|
|
||||||
extern crate core;
|
|
||||||
#[cfg(feature = "unstable")]
|
|
||||||
pub use self::core::nonzero;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub use ser::{Serialize, Serializer};
|
pub use ser::{Serialize, Serializer};
|
||||||
pub use de::{Deserialize, Deserializer, Error};
|
pub use de::{Deserialize, Deserializer, Error};
|
||||||
|
|
||||||
#[cfg(not(feature = "std"))]
|
|
||||||
macro_rules! format {
|
|
||||||
($s:expr, $($rest:tt)*) => ($s)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub mod bytes;
|
pub mod bytes;
|
||||||
pub mod de;
|
pub mod de;
|
||||||
#[cfg(feature = "std")]
|
|
||||||
pub mod iter;
|
pub mod iter;
|
||||||
|
pub mod json;
|
||||||
pub mod ser;
|
pub mod ser;
|
||||||
#[cfg(not(feature = "std"))]
|
|
||||||
pub mod error;
|
|
||||||
mod utils;
|
|
||||||
|
|||||||
+187
-383
@@ -1,75 +1,14 @@
|
|||||||
//! Implementations for all of Rust's builtin types. Tuples implement the `Serialize` trait if they
|
use std::collections::{BTreeMap, BTreeSet, HashMap, HashSet};
|
||||||
//! have at most 16 fields. Arrays implement the `Serialize` trait if their length is 32 or less.
|
use std::hash::Hash;
|
||||||
//! You can always forward array serialization to slice serialization, which works for any length.
|
|
||||||
//! Long tuples are best replaced by tuple structs, for which you can use `derive(Serialize)`. In
|
|
||||||
//! that case the number of fields is irrelevant.
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
use std::borrow::Cow;
|
|
||||||
#[cfg(all(feature = "collections", not(feature = "std")))]
|
|
||||||
use collections::borrow::Cow;
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
use std::collections::{
|
|
||||||
BinaryHeap,
|
|
||||||
BTreeMap,
|
|
||||||
BTreeSet,
|
|
||||||
LinkedList,
|
|
||||||
HashMap,
|
|
||||||
HashSet,
|
|
||||||
VecDeque,
|
|
||||||
};
|
|
||||||
#[cfg(all(feature = "collections", not(feature = "std")))]
|
|
||||||
use collections::{
|
|
||||||
BinaryHeap,
|
|
||||||
BTreeMap,
|
|
||||||
BTreeSet,
|
|
||||||
LinkedList,
|
|
||||||
VecDeque,
|
|
||||||
String,
|
|
||||||
Vec,
|
|
||||||
};
|
|
||||||
|
|
||||||
#[cfg(all(feature = "unstable", feature = "collections"))]
|
|
||||||
use collections::enum_set::{CLike, EnumSet};
|
|
||||||
#[cfg(all(feature = "unstable", feature = "collections"))]
|
|
||||||
use collections::borrow::ToOwned;
|
|
||||||
|
|
||||||
use core::hash::{Hash, BuildHasher};
|
|
||||||
#[cfg(feature = "unstable")]
|
|
||||||
use core::iter;
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
use std::net;
|
|
||||||
#[cfg(feature = "unstable")]
|
|
||||||
use core::num;
|
|
||||||
#[cfg(feature = "unstable")]
|
|
||||||
use core::ops;
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
use std::path;
|
use std::path;
|
||||||
#[cfg(feature = "std")]
|
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
#[cfg(all(feature = "alloc", not(feature = "std")))]
|
|
||||||
use alloc::rc::Rc;
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
use std::time::Duration;
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
#[cfg(all(feature = "alloc", not(feature = "std")))]
|
|
||||||
use alloc::arc::Arc;
|
|
||||||
|
|
||||||
#[cfg(all(feature = "alloc", not(feature = "std")))]
|
|
||||||
use alloc::boxed::Box;
|
|
||||||
|
|
||||||
use core::marker::PhantomData;
|
|
||||||
|
|
||||||
#[cfg(feature = "unstable")]
|
|
||||||
use core::nonzero::{NonZero, Zeroable};
|
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
Error,
|
|
||||||
Serialize,
|
Serialize,
|
||||||
Serializer,
|
Serializer,
|
||||||
|
SeqVisitor,
|
||||||
|
MapVisitor,
|
||||||
};
|
};
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
@@ -87,20 +26,20 @@ macro_rules! impl_visit {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl_visit!(bool, serialize_bool);
|
impl_visit!(bool, visit_bool);
|
||||||
impl_visit!(isize, serialize_isize);
|
impl_visit!(isize, visit_isize);
|
||||||
impl_visit!(i8, serialize_i8);
|
impl_visit!(i8, visit_i8);
|
||||||
impl_visit!(i16, serialize_i16);
|
impl_visit!(i16, visit_i16);
|
||||||
impl_visit!(i32, serialize_i32);
|
impl_visit!(i32, visit_i32);
|
||||||
impl_visit!(i64, serialize_i64);
|
impl_visit!(i64, visit_i64);
|
||||||
impl_visit!(usize, serialize_usize);
|
impl_visit!(usize, visit_usize);
|
||||||
impl_visit!(u8, serialize_u8);
|
impl_visit!(u8, visit_u8);
|
||||||
impl_visit!(u16, serialize_u16);
|
impl_visit!(u16, visit_u16);
|
||||||
impl_visit!(u32, serialize_u32);
|
impl_visit!(u32, visit_u32);
|
||||||
impl_visit!(u64, serialize_u64);
|
impl_visit!(u64, visit_u64);
|
||||||
impl_visit!(f32, serialize_f32);
|
impl_visit!(f32, visit_f32);
|
||||||
impl_visit!(f64, serialize_f64);
|
impl_visit!(f64, visit_f64);
|
||||||
impl_visit!(char, serialize_char);
|
impl_visit!(char, visit_char);
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
@@ -109,11 +48,10 @@ impl Serialize for str {
|
|||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
where S: Serializer,
|
where S: Serializer,
|
||||||
{
|
{
|
||||||
serializer.serialize_str(self)
|
serializer.visit_str(self)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
impl Serialize for String {
|
impl Serialize for String {
|
||||||
#[inline]
|
#[inline]
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
@@ -131,43 +69,67 @@ impl<T> Serialize for Option<T> where T: Serialize {
|
|||||||
where S: Serializer,
|
where S: Serializer,
|
||||||
{
|
{
|
||||||
match *self {
|
match *self {
|
||||||
Some(ref value) => serializer.serialize_some(value),
|
Some(ref value) => serializer.visit_some(value),
|
||||||
None => serializer.serialize_none(),
|
None => serializer.visit_none(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
impl<T> Serialize for PhantomData<T> {
|
pub struct SeqIteratorVisitor<Iter> {
|
||||||
|
iter: Iter,
|
||||||
|
len: Option<usize>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T, Iter> SeqIteratorVisitor<Iter>
|
||||||
|
where Iter: Iterator<Item=T>
|
||||||
|
{
|
||||||
#[inline]
|
#[inline]
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
pub fn new(iter: Iter, len: Option<usize>) -> SeqIteratorVisitor<Iter> {
|
||||||
where S: Serializer,
|
SeqIteratorVisitor {
|
||||||
{
|
iter: iter,
|
||||||
serializer.serialize_unit_struct("PhantomData")
|
len: len,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<T, Iter> SeqVisitor for SeqIteratorVisitor<Iter>
|
||||||
|
where T: Serialize,
|
||||||
|
Iter: Iterator<Item=T>,
|
||||||
|
{
|
||||||
|
#[inline]
|
||||||
|
fn visit<S>(&mut self, serializer: &mut S) -> Result<Option<()>, S::Error>
|
||||||
|
where S: Serializer,
|
||||||
|
{
|
||||||
|
match self.iter.next() {
|
||||||
|
Some(value) => {
|
||||||
|
let value = try!(serializer.visit_seq_elt(value));
|
||||||
|
Ok(Some(value))
|
||||||
|
}
|
||||||
|
None => Ok(None),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
fn len(&self) -> Option<usize> {
|
||||||
|
self.len
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
impl<T> Serialize for [T]
|
impl<'a, T> Serialize for &'a [T]
|
||||||
where T: Serialize,
|
where T: Serialize,
|
||||||
{
|
{
|
||||||
#[inline]
|
#[inline]
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
where S: Serializer,
|
where S: Serializer,
|
||||||
{
|
{
|
||||||
let mut state = try!(serializer.serialize_seq(Some(self.len())));
|
serializer.visit_seq(SeqIteratorVisitor::new(self.iter(), Some(self.len())))
|
||||||
for e in self {
|
|
||||||
try!(serializer.serialize_seq_elt(&mut state, e));
|
|
||||||
}
|
|
||||||
serializer.serialize_seq_end(state)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
macro_rules! array_impls {
|
macro_rules! array_impls {
|
||||||
($len:expr) => {
|
($len:expr) => {
|
||||||
impl<T> Serialize for [T; $len] where T: Serialize {
|
impl<T> Serialize for [T; $len] where T: Serialize {
|
||||||
@@ -175,11 +137,7 @@ macro_rules! array_impls {
|
|||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
where S: Serializer,
|
where S: Serializer,
|
||||||
{
|
{
|
||||||
let mut state = try!(serializer.serialize_seq_fixed_size($len));
|
serializer.visit_seq(SeqIteratorVisitor::new(self.iter(), Some($len)))
|
||||||
for e in self {
|
|
||||||
try!(serializer.serialize_seq_elt(&mut state, e));
|
|
||||||
}
|
|
||||||
serializer.serialize_seq_end(state)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -219,84 +177,34 @@ array_impls!(30);
|
|||||||
array_impls!(31);
|
array_impls!(31);
|
||||||
array_impls!(32);
|
array_impls!(32);
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
impl<T> Serialize for Vec<T> where T: Serialize {
|
||||||
|
#[inline]
|
||||||
macro_rules! serialize_seq {
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
() => {
|
where S: Serializer,
|
||||||
#[inline]
|
{
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
(&self[..]).serialize(serializer)
|
||||||
where S: Serializer,
|
|
||||||
{
|
|
||||||
let mut state = try!(serializer.serialize_seq(Some(self.len())));
|
|
||||||
for e in self {
|
|
||||||
try!(serializer.serialize_seq_elt(&mut state, e));
|
|
||||||
}
|
|
||||||
serializer.serialize_seq_end(state)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
impl<T> Serialize for BinaryHeap<T>
|
|
||||||
where T: Serialize + Ord
|
|
||||||
{
|
|
||||||
serialize_seq!();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
impl<T> Serialize for BTreeSet<T>
|
impl<T> Serialize for BTreeSet<T>
|
||||||
where T: Serialize + Ord,
|
where T: Serialize + Ord,
|
||||||
{
|
|
||||||
serialize_seq!();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(all(feature = "unstable", feature = "collections"))]
|
|
||||||
impl<T> Serialize for EnumSet<T>
|
|
||||||
where T: Serialize + CLike
|
|
||||||
{
|
|
||||||
serialize_seq!();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
impl<T, H> Serialize for HashSet<T, H>
|
|
||||||
where T: Serialize + Eq + Hash,
|
|
||||||
H: BuildHasher,
|
|
||||||
{
|
|
||||||
serialize_seq!();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
impl<T> Serialize for LinkedList<T>
|
|
||||||
where T: Serialize,
|
|
||||||
{
|
|
||||||
serialize_seq!();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
impl<T> Serialize for Vec<T> where T: Serialize {
|
|
||||||
serialize_seq!();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
impl<T> Serialize for VecDeque<T> where T: Serialize {
|
|
||||||
serialize_seq!();
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "unstable")]
|
|
||||||
impl<A> Serialize for ops::Range<A>
|
|
||||||
where A: Serialize + Clone + iter::Step + num::One,
|
|
||||||
for<'a> &'a A: ops::Add<&'a A, Output = A>,
|
|
||||||
{
|
{
|
||||||
#[inline]
|
#[inline]
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
where S: Serializer,
|
where S: Serializer,
|
||||||
{
|
{
|
||||||
let len = iter::Step::steps_between(&self.start, &self.end, &A::one());
|
serializer.visit_seq(SeqIteratorVisitor::new(self.iter(), Some(self.len())))
|
||||||
let mut state = try!(serializer.serialize_seq(len));
|
}
|
||||||
for e in self.clone() {
|
}
|
||||||
try!(serializer.serialize_seq_elt(&mut state, e));
|
|
||||||
}
|
impl<T> Serialize for HashSet<T>
|
||||||
serializer.serialize_seq_end(state)
|
where T: Serialize + Eq + Hash,
|
||||||
|
{
|
||||||
|
#[inline]
|
||||||
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
|
where S: Serializer,
|
||||||
|
{
|
||||||
|
serializer.visit_seq(SeqIteratorVisitor::new(self.iter(), Some(self.len())))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -307,7 +215,7 @@ impl Serialize for () {
|
|||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
where S: Serializer,
|
where S: Serializer,
|
||||||
{
|
{
|
||||||
serializer.serialize_unit()
|
serializer.visit_unit()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -325,18 +233,50 @@ macro_rules! tuple_impls {
|
|||||||
}
|
}
|
||||||
)+) => {
|
)+) => {
|
||||||
$(
|
$(
|
||||||
|
pub struct $TupleVisitor<'a, $($T: 'a),+> {
|
||||||
|
tuple: &'a ($($T,)+),
|
||||||
|
state: u8,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, $($T: 'a),+> $TupleVisitor<'a, $($T),+> {
|
||||||
|
pub fn new(tuple: &'a ($($T,)+)) -> $TupleVisitor<'a, $($T),+> {
|
||||||
|
$TupleVisitor {
|
||||||
|
tuple: tuple,
|
||||||
|
state: 0,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a, $($T),+> SeqVisitor for $TupleVisitor<'a, $($T),+>
|
||||||
|
where $($T: Serialize),+
|
||||||
|
{
|
||||||
|
fn visit<S>(&mut self, serializer: &mut S) -> Result<Option<()>, S::Error>
|
||||||
|
where S: Serializer,
|
||||||
|
{
|
||||||
|
match self.state {
|
||||||
|
$(
|
||||||
|
$state => {
|
||||||
|
self.state += 1;
|
||||||
|
Ok(Some(try!(serializer.visit_seq_elt(&e!(self.tuple.$idx)))))
|
||||||
|
}
|
||||||
|
)+
|
||||||
|
_ => {
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn len(&self) -> Option<usize> {
|
||||||
|
Some($len)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<$($T),+> Serialize for ($($T,)+)
|
impl<$($T),+> Serialize for ($($T,)+)
|
||||||
where $($T: Serialize),+
|
where $($T: Serialize),+
|
||||||
{
|
{
|
||||||
#[inline]
|
#[inline]
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S: Serializer>(&self, serializer: &mut S) -> Result<(), S::Error> {
|
||||||
where S: Serializer,
|
serializer.visit_seq($TupleVisitor::new(self))
|
||||||
{
|
|
||||||
let mut state = try!(serializer.serialize_tuple($len));
|
|
||||||
$(
|
|
||||||
try!(serializer.serialize_tuple_elt(&mut state, &e!(self.$idx)));
|
|
||||||
)+
|
|
||||||
serializer.serialize_tuple_end(state)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
)+
|
)+
|
||||||
@@ -446,109 +386,91 @@ tuple_impls! {
|
|||||||
10 => 10,
|
10 => 10,
|
||||||
11 => 11,
|
11 => 11,
|
||||||
}
|
}
|
||||||
TupleVisitor13 (13, T0, T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12) {
|
}
|
||||||
0 => 0,
|
|
||||||
1 => 1,
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
2 => 2,
|
|
||||||
3 => 3,
|
pub struct MapIteratorVisitor<Iter> {
|
||||||
4 => 4,
|
iter: Iter,
|
||||||
5 => 5,
|
len: Option<usize>,
|
||||||
6 => 6,
|
}
|
||||||
7 => 7,
|
|
||||||
8 => 8,
|
impl<K, V, Iter> MapIteratorVisitor<Iter>
|
||||||
9 => 9,
|
where Iter: Iterator<Item=(K, V)>
|
||||||
10 => 10,
|
{
|
||||||
11 => 11,
|
#[inline]
|
||||||
12 => 12,
|
pub fn new(iter: Iter, len: Option<usize>) -> MapIteratorVisitor<Iter> {
|
||||||
|
MapIteratorVisitor {
|
||||||
|
iter: iter,
|
||||||
|
len: len,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
TupleVisitor14 (14, T0, T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13) {
|
}
|
||||||
0 => 0,
|
|
||||||
1 => 1,
|
impl<K, V, I> MapVisitor for MapIteratorVisitor<I>
|
||||||
2 => 2,
|
where K: Serialize,
|
||||||
3 => 3,
|
V: Serialize,
|
||||||
4 => 4,
|
I: Iterator<Item=(K, V)>,
|
||||||
5 => 5,
|
{
|
||||||
6 => 6,
|
#[inline]
|
||||||
7 => 7,
|
fn visit<S>(&mut self, serializer: &mut S) -> Result<Option<()>, S::Error>
|
||||||
8 => 8,
|
where S: Serializer,
|
||||||
9 => 9,
|
{
|
||||||
10 => 10,
|
match self.iter.next() {
|
||||||
11 => 11,
|
Some((key, value)) => {
|
||||||
12 => 12,
|
let value = try!(serializer.visit_map_elt(key, value));
|
||||||
13 => 13,
|
Ok(Some(value))
|
||||||
|
}
|
||||||
|
None => Ok(None)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
TupleVisitor15 (15, T0, T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13, T14) {
|
|
||||||
0 => 0,
|
#[inline]
|
||||||
1 => 1,
|
fn len(&self) -> Option<usize> {
|
||||||
2 => 2,
|
self.len
|
||||||
3 => 3,
|
|
||||||
4 => 4,
|
|
||||||
5 => 5,
|
|
||||||
6 => 6,
|
|
||||||
7 => 7,
|
|
||||||
8 => 8,
|
|
||||||
9 => 9,
|
|
||||||
10 => 10,
|
|
||||||
11 => 11,
|
|
||||||
12 => 12,
|
|
||||||
13 => 13,
|
|
||||||
14 => 14,
|
|
||||||
}
|
|
||||||
TupleVisitor16 (16, T0, T1, T2, T3, T4, T5, T6, T7, T8, T9, T10, T11, T12, T13, T14, T15) {
|
|
||||||
0 => 0,
|
|
||||||
1 => 1,
|
|
||||||
2 => 2,
|
|
||||||
3 => 3,
|
|
||||||
4 => 4,
|
|
||||||
5 => 5,
|
|
||||||
6 => 6,
|
|
||||||
7 => 7,
|
|
||||||
8 => 8,
|
|
||||||
9 => 9,
|
|
||||||
10 => 10,
|
|
||||||
11 => 11,
|
|
||||||
12 => 12,
|
|
||||||
13 => 13,
|
|
||||||
14 => 14,
|
|
||||||
15 => 15,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
macro_rules! serialize_map {
|
|
||||||
() => {
|
|
||||||
#[inline]
|
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
|
||||||
where S: Serializer,
|
|
||||||
{
|
|
||||||
let mut state = try!(serializer.serialize_map(Some(self.len())));
|
|
||||||
for (k, v) in self {
|
|
||||||
try!(serializer.serialize_map_key(&mut state, k));
|
|
||||||
try!(serializer.serialize_map_value(&mut state, v));
|
|
||||||
}
|
|
||||||
serializer.serialize_map_end(state)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
impl<K, V> Serialize for BTreeMap<K, V>
|
impl<K, V> Serialize for BTreeMap<K, V>
|
||||||
where K: Serialize + Ord,
|
where K: Serialize + Ord,
|
||||||
V: Serialize,
|
V: Serialize,
|
||||||
{
|
{
|
||||||
serialize_map!();
|
#[inline]
|
||||||
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
|
where S: Serializer,
|
||||||
|
{
|
||||||
|
serializer.visit_map(MapIteratorVisitor::new(self.iter(), Some(self.len())))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
impl<K, V> Serialize for HashMap<K, V>
|
||||||
impl<K, V, H> Serialize for HashMap<K, V, H>
|
|
||||||
where K: Serialize + Eq + Hash,
|
where K: Serialize + Eq + Hash,
|
||||||
V: Serialize,
|
V: Serialize,
|
||||||
H: BuildHasher,
|
|
||||||
{
|
{
|
||||||
serialize_map!();
|
#[inline]
|
||||||
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
|
where S: Serializer,
|
||||||
|
{
|
||||||
|
serializer.visit_map(MapIteratorVisitor::new(self.iter(), Some(self.len())))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// FIXME: `VecMap` is unstable.
|
||||||
|
/*
|
||||||
|
impl<V> Serialize for VecMap<V>
|
||||||
|
where V: Serialize,
|
||||||
|
{
|
||||||
|
#[inline]
|
||||||
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
|
where S: Serializer,
|
||||||
|
{
|
||||||
|
serializer.visit_map(MapIteratorVisitor::new(self.iter(), Some(self.len())))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
impl<'a, T: ?Sized> Serialize for &'a T where T: Serialize {
|
impl<'a, T: ?Sized> Serialize for &'a T where T: Serialize {
|
||||||
@@ -569,7 +491,6 @@ impl<'a, T: ?Sized> Serialize for &'a mut T where T: Serialize {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
|
||||||
impl<T: ?Sized> Serialize for Box<T> where T: Serialize {
|
impl<T: ?Sized> Serialize for Box<T> where T: Serialize {
|
||||||
#[inline]
|
#[inline]
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
@@ -579,7 +500,6 @@ impl<T: ?Sized> Serialize for Box<T> where T: Serialize {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
|
||||||
impl<T> Serialize for Rc<T> where T: Serialize, {
|
impl<T> Serialize for Rc<T> where T: Serialize, {
|
||||||
#[inline]
|
#[inline]
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
@@ -589,7 +509,6 @@ impl<T> Serialize for Rc<T> where T: Serialize, {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
|
||||||
impl<T> Serialize for Arc<T> where T: Serialize, {
|
impl<T> Serialize for Arc<T> where T: Serialize, {
|
||||||
#[inline]
|
#[inline]
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
@@ -599,135 +518,20 @@ impl<T> Serialize for Arc<T> where T: Serialize, {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
impl<'a, T: ?Sized> Serialize for Cow<'a, T> where T: Serialize + ToOwned, {
|
|
||||||
#[inline]
|
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
|
||||||
where S: Serializer,
|
|
||||||
{
|
|
||||||
(**self).serialize(serializer)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
impl<T, E> Serialize for Result<T, E> where T: Serialize, E: Serialize {
|
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error> where S: Serializer {
|
|
||||||
match *self {
|
|
||||||
Result::Ok(ref value) => {
|
|
||||||
serializer.serialize_newtype_variant("Result", 0, "Ok", value)
|
|
||||||
}
|
|
||||||
Result::Err(ref value) => {
|
|
||||||
serializer.serialize_newtype_variant("Result", 1, "Err", value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
impl Serialize for Duration {
|
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
|
||||||
where S: Serializer,
|
|
||||||
{
|
|
||||||
let mut state = try!(serializer.serialize_struct("Duration", 2));
|
|
||||||
try!(serializer.serialize_struct_elt(&mut state, "secs", self.as_secs()));
|
|
||||||
try!(serializer.serialize_struct_elt(&mut state, "nanos", self.subsec_nanos()));
|
|
||||||
serializer.serialize_struct_end(state)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#[cfg(all(feature = "std", feature = "unstable"))]
|
|
||||||
impl Serialize for net::IpAddr {
|
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
|
||||||
where S: Serializer,
|
|
||||||
{
|
|
||||||
match *self {
|
|
||||||
net::IpAddr::V4(ref addr) => addr.serialize(serializer),
|
|
||||||
net::IpAddr::V6(ref addr) => addr.serialize(serializer),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
impl Serialize for net::Ipv4Addr {
|
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
|
||||||
where S: Serializer,
|
|
||||||
{
|
|
||||||
self.to_string().serialize(serializer)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
impl Serialize for net::Ipv6Addr {
|
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
|
||||||
where S: Serializer,
|
|
||||||
{
|
|
||||||
self.to_string().serialize(serializer)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
impl Serialize for net::SocketAddr {
|
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
|
||||||
where S: Serializer,
|
|
||||||
{
|
|
||||||
match *self {
|
|
||||||
net::SocketAddr::V4(ref addr) => addr.serialize(serializer),
|
|
||||||
net::SocketAddr::V6(ref addr) => addr.serialize(serializer),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
impl Serialize for net::SocketAddrV4 {
|
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
|
||||||
where S: Serializer,
|
|
||||||
{
|
|
||||||
self.to_string().serialize(serializer)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
impl Serialize for net::SocketAddrV6 {
|
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
|
||||||
where S: Serializer,
|
|
||||||
{
|
|
||||||
self.to_string().serialize(serializer)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
impl Serialize for path::Path {
|
impl Serialize for path::Path {
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
where S: Serializer,
|
where S: Serializer,
|
||||||
{
|
{
|
||||||
match self.to_str() {
|
self.to_str().unwrap().serialize(serializer)
|
||||||
Some(s) => s.serialize(serializer),
|
|
||||||
None => Err(Error::invalid_value("Path contains invalid UTF-8 characters")),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
impl Serialize for path::PathBuf {
|
impl Serialize for path::PathBuf {
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
where S: Serializer,
|
where S: Serializer,
|
||||||
{
|
{
|
||||||
self.as_path().serialize(serializer)
|
self.to_str().unwrap().serialize(serializer)
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "unstable")]
|
|
||||||
impl<T> Serialize for NonZero<T> where T: Serialize + Zeroable {
|
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error> where S: Serializer {
|
|
||||||
(**self).serialize(serializer)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
+167
-367
@@ -1,412 +1,212 @@
|
|||||||
//! Generic serialization framework.
|
//! Generic serialization framework.
|
||||||
//! # For Developers who want to serialize objects
|
|
||||||
//! Implement the `Serialize` trait for the type of objects you want to serialize. Call methods of
|
|
||||||
//! the `serializer` object. For which methods to call and how to do so, look at the documentation
|
|
||||||
//! of the `Serializer` trait.
|
|
||||||
//!
|
|
||||||
//! # For Serialization Format Developers
|
|
||||||
//! Implement the `Serializer` trait for a structure that contains fields that enable it to write
|
|
||||||
//! the serialization result to your target. When a method's argument is an object of type
|
|
||||||
//! `Serialize`, you can either forward the serializer object (`self`) or create a new one,
|
|
||||||
//! depending on the quirks of your format.
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
|
||||||
use std::error;
|
|
||||||
#[cfg(not(feature = "std"))]
|
|
||||||
use error;
|
|
||||||
|
|
||||||
#[cfg(all(feature = "collections", not(feature = "std")))]
|
|
||||||
use collections::String;
|
|
||||||
|
|
||||||
pub mod impls;
|
pub mod impls;
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
/// `Error` is a trait that allows a `Serialize` to generically create a
|
|
||||||
/// `Serializer` error.
|
|
||||||
pub trait Error: Sized + error::Error {
|
|
||||||
/// Raised when there is a general error when serializing a type.
|
|
||||||
#[cfg(any(feature = "std", feature = "collections"))]
|
|
||||||
fn custom<T: Into<String>>(msg: T) -> Self;
|
|
||||||
|
|
||||||
/// Raised when there is a general error when serializing a type.
|
|
||||||
#[cfg(all(not(feature = "std"), not(feature = "collections")))]
|
|
||||||
fn custom<T: Into<&'static str>>(msg: T) -> Self;
|
|
||||||
|
|
||||||
/// Raised when a `Serialize` was passed an incorrect value.
|
|
||||||
fn invalid_value(msg: &str) -> Self {
|
|
||||||
Error::custom(format!("invalid value: {}", msg))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
/// A trait that describes a type that can be serialized by a `Serializer`.
|
|
||||||
pub trait Serialize {
|
pub trait Serialize {
|
||||||
/// Serializes this value into this serializer.
|
|
||||||
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
fn serialize<S>(&self, serializer: &mut S) -> Result<(), S::Error>
|
||||||
where S: Serializer;
|
where S: Serializer;
|
||||||
}
|
}
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
/// A trait that describes a type that can serialize a stream of values into the underlying format.
|
|
||||||
///
|
|
||||||
/// # For `Serialize` Developers
|
|
||||||
/// Non-aggrergate types like integers and strings can be serialized directly by calling the
|
|
||||||
/// appropriate function. For Aggregate types there's an initial `serialize_T` method that yields
|
|
||||||
/// a State object that you should not interact with. For each part of the aggregate there's a
|
|
||||||
/// `serialize_T_elt` method that allows you to pass values or key/value pairs. The types of the
|
|
||||||
/// values or the keys may change between calls, but the serialization format may not necessarily
|
|
||||||
/// accept it. The `serialize_T_elt` method also takes a mutable reference to the state object.
|
|
||||||
/// Make sure that you always use the same state object and only the state object that was returned
|
|
||||||
/// by the `serialize_T` method. Finally, when your object is done, call the `serialize_T_end`
|
|
||||||
/// method and pass the state object by value
|
|
||||||
///
|
|
||||||
/// # For Serialization Format Developers
|
|
||||||
/// If your format has different situations where it accepts different types, create a
|
|
||||||
/// `Serializer` for each situation. You can create the sub-`Serializer` in one of the aggregate
|
|
||||||
/// `serialize_T` methods and return it as a state object. Remember to also set the corresponding
|
|
||||||
/// associated type `TState`. In the `serialize_T_elt` methods you will be given a mutable
|
|
||||||
/// reference to that state. You do not need to do any additional checks for the correctness of the
|
|
||||||
/// state object, as it is expected that the user will not modify it. Due to the generic nature
|
|
||||||
/// of the `Serialize` impls, modifying the object is impossible on stable Rust.
|
|
||||||
pub trait Serializer {
|
pub trait Serializer {
|
||||||
/// The error type that can be returned if some error occurs during serialization.
|
type Error;
|
||||||
type Error: Error;
|
|
||||||
|
|
||||||
/// A state object that is initialized by `serialize_seq`, passed to
|
/// `visit_bool` serializes a `bool` value.
|
||||||
/// `serialize_seq_elt`, and consumed by `serialize_seq_end`. Use `()` if no
|
fn visit_bool(&mut self, v: bool) -> Result<(), Self::Error>;
|
||||||
/// state is required.
|
|
||||||
type SeqState;
|
|
||||||
/// A state object that is initialized by `serialize_tuple`, passed to
|
|
||||||
/// `serialize_tuple_elt`, and consumed by `serialize_tuple_end`. Use `()`
|
|
||||||
/// if no state is required.
|
|
||||||
type TupleState;
|
|
||||||
/// A state object that is initialized by `serialize_tuple_struct`, passed
|
|
||||||
/// to `serialize_tuple_struct_elt`, and consumed by
|
|
||||||
/// `serialize_tuple_struct_end`. Use `()` if no state is required.
|
|
||||||
type TupleStructState;
|
|
||||||
/// A state object that is initialized by `serialize_tuple_variant`, passed
|
|
||||||
/// to `serialize_tuple_variant_elt`, and consumed by
|
|
||||||
/// `serialize_tuple_variant_end`. Use `()` if no state is required.
|
|
||||||
type TupleVariantState;
|
|
||||||
/// A state object that is initialized by `serialize_map`, passed to
|
|
||||||
/// `serialize_map_elt`, and consumed by `serialize_map_end`. Use `()` if no
|
|
||||||
/// state is required.
|
|
||||||
type MapState;
|
|
||||||
/// A state object that is initialized by `serialize_struct`, passed to
|
|
||||||
/// `serialize_struct_elt`, and consumed by `serialize_struct_end`. Use `()`
|
|
||||||
/// if no state is required.
|
|
||||||
type StructState;
|
|
||||||
/// A state object that is initialized by `serialize_struct_variant`, passed
|
|
||||||
/// to `serialize_struct_variant_elt`, and consumed by
|
|
||||||
/// `serialize_struct_variant_end`. Use `()` if no state is required.
|
|
||||||
type StructVariantState;
|
|
||||||
|
|
||||||
/// Serializes a `bool` value.
|
/// `visit_isize` serializes a `isize` value. By default it casts the value to a `i64` and
|
||||||
fn serialize_bool(&mut self, v: bool) -> Result<(), Self::Error>;
|
/// passes it to the `visit_i64` method.
|
||||||
|
#[inline]
|
||||||
|
fn visit_isize(&mut self, v: isize) -> Result<(), Self::Error> {
|
||||||
|
self.visit_i64(v as i64)
|
||||||
|
}
|
||||||
|
|
||||||
/// Serializes an `isize` value. If the format does not differentiate
|
/// `visit_i8` serializes a `i8` value. By default it casts the value to a `i64` and
|
||||||
/// between `isize` and `i64`, a reasonable implementation would be to cast
|
/// passes it to the `visit_i64` method.
|
||||||
/// the value to `i64` and forward to `serialize_i64`.
|
#[inline]
|
||||||
fn serialize_isize(&mut self, v: isize) -> Result<(), Self::Error>;
|
fn visit_i8(&mut self, v: i8) -> Result<(), Self::Error> {
|
||||||
|
self.visit_i64(v as i64)
|
||||||
|
}
|
||||||
|
|
||||||
/// Serializes an `i8` value. If the format does not differentiate between
|
/// `visit_i16` serializes a `i16` value. By default it casts the value to a `i64` and
|
||||||
/// `i8` and `i64`, a reasonable implementation would be to cast the value
|
/// passes it to the `visit_i64` method.
|
||||||
/// to `i64` and forward to `serialize_i64`.
|
#[inline]
|
||||||
fn serialize_i8(&mut self, v: i8) -> Result<(), Self::Error>;
|
fn visit_i16(&mut self, v: i16) -> Result<(), Self::Error> {
|
||||||
|
self.visit_i64(v as i64)
|
||||||
|
}
|
||||||
|
|
||||||
/// Serializes an `i16` value. If the format does not differentiate between
|
/// `visit_i32` serializes a `i32` value. By default it casts the value to a `i64` and
|
||||||
/// `i16` and `i64`, a reasonable implementation would be to cast the value
|
/// passes it to the `visit_i64` method.
|
||||||
/// to `i64` and forward to `serialize_i64`.
|
#[inline]
|
||||||
fn serialize_i16(&mut self, v: i16) -> Result<(), Self::Error>;
|
fn visit_i32(&mut self, v: i32) -> Result<(), Self::Error> {
|
||||||
|
self.visit_i64(v as i64)
|
||||||
|
}
|
||||||
|
|
||||||
/// Serializes an `i32` value. If the format does not differentiate between
|
/// `visit_i64` serializes a `i64` value.
|
||||||
/// `i32` and `i64`, a reasonable implementation would be to cast the value
|
#[inline]
|
||||||
/// to `i64` and forward to `serialize_i64`.
|
fn visit_i64(&mut self, v: i64) -> Result<(), Self::Error>;
|
||||||
fn serialize_i32(&mut self, v: i32) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Serializes an `i64` value.
|
/// `visit_usize` serializes a `usize` value. By default it casts the value to a `u64` and
|
||||||
fn serialize_i64(&mut self, v: i64) -> Result<(), Self::Error>;
|
/// passes it to the `visit_u64` method.
|
||||||
|
#[inline]
|
||||||
|
fn visit_usize(&mut self, v: usize) -> Result<(), Self::Error> {
|
||||||
|
self.visit_u64(v as u64)
|
||||||
|
}
|
||||||
|
|
||||||
/// Serializes a `usize` value. If the format does not differentiate between
|
/// `visit_u8` serializes a `u8` value. By default it casts the value to a `u64` and passes
|
||||||
/// `usize` and `u64`, a reasonable implementation would be to cast the
|
/// it to the `visit_u64` method.
|
||||||
/// value to `u64` and forward to `serialize_u64`.
|
#[inline]
|
||||||
fn serialize_usize(&mut self, v: usize) -> Result<(), Self::Error>;
|
fn visit_u8(&mut self, v: u8) -> Result<(), Self::Error> {
|
||||||
|
self.visit_u64(v as u64)
|
||||||
|
}
|
||||||
|
|
||||||
/// Serializes a `u8` value. If the format does not differentiate between
|
/// `visit_u32` serializes a `u32` value. By default it casts the value to a `u64` and passes
|
||||||
/// `u8` and `u64`, a reasonable implementation would be to cast the value
|
/// it to the `visit_u64` method.
|
||||||
/// to `u64` and forward to `serialize_u64`.
|
#[inline]
|
||||||
fn serialize_u8(&mut self, v: u8) -> Result<(), Self::Error>;
|
fn visit_u16(&mut self, v: u16) -> Result<(), Self::Error> {
|
||||||
|
self.visit_u64(v as u64)
|
||||||
|
}
|
||||||
|
|
||||||
/// Serializes a `u16` value. If the format does not differentiate between
|
/// `visit_u32` serializes a `u32` value. By default it casts the value to a `u64` and passes
|
||||||
/// `u16` and `u64`, a reasonable implementation would be to cast the value
|
/// it to the `visit_u64` method.
|
||||||
/// to `u64` and forward to `serialize_u64`.
|
#[inline]
|
||||||
fn serialize_u16(&mut self, v: u16) -> Result<(), Self::Error>;
|
fn visit_u32(&mut self, v: u32) -> Result<(), Self::Error> {
|
||||||
|
self.visit_u64(v as u64)
|
||||||
|
}
|
||||||
|
|
||||||
/// Serializes a `u32` value. If the format does not differentiate between
|
/// `visit_u64` serializes a `u64` value.
|
||||||
/// `u32` and `u64`, a reasonable implementation would be to cast the value
|
#[inline]
|
||||||
/// to `u64` and forward to `serialize_u64`.
|
fn visit_u64(&mut self, v: u64) -> Result<(), Self::Error>;
|
||||||
fn serialize_u32(&mut self, v: u32) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// `Serializes a `u64` value.
|
/// `visit_f32` serializes a `f32` value. By default it casts the value to a `f64` and passes
|
||||||
fn serialize_u64(&mut self, v: u64) -> Result<(), Self::Error>;
|
/// it to the `visit_f64` method.
|
||||||
|
#[inline]
|
||||||
|
fn visit_f32(&mut self, v: f32) -> Result<(), Self::Error> {
|
||||||
|
self.visit_f64(v as f64)
|
||||||
|
}
|
||||||
|
|
||||||
/// Serializes an `f32` value. If the format does not differentiate between
|
/// `visit_f64` serializes a `f64` value.
|
||||||
/// `f32` and `f64`, a reasonable implementation would be to cast the value
|
fn visit_f64(&mut self, v: f64) -> Result<(), Self::Error>;
|
||||||
/// to `f64` and forward to `serialize_f64`.
|
|
||||||
fn serialize_f32(&mut self, v: f32) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Serializes an `f64` value.
|
/// `visit_char` serializes a character. By default it serializes it as a `&str` containing a
|
||||||
fn serialize_f64(&mut self, v: f64) -> Result<(), Self::Error>;
|
/// single character.
|
||||||
|
#[inline]
|
||||||
|
fn visit_char(&mut self, v: char) -> Result<(), Self::Error> {
|
||||||
|
// FIXME: this allocation is required in order to be compatible with stable rust, which
|
||||||
|
// doesn't support encoding a `char` into a stack buffer.
|
||||||
|
self.visit_str(&v.to_string())
|
||||||
|
}
|
||||||
|
|
||||||
/// Serializes a character. If the format does not support characters,
|
/// `visit_str` serializes a `&str`.
|
||||||
/// it is reasonable to serialize it as a single element `str` or a `u32`.
|
fn visit_str(&mut self, value: &str) -> Result<(), Self::Error>;
|
||||||
fn serialize_char(&mut self, v: char) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Serializes a `&str`.
|
/// `visit_bytes` is a hook that enables those serialization formats that support serializing
|
||||||
fn serialize_str(&mut self, value: &str) -> Result<(), Self::Error>;
|
/// byte slices separately from generic arrays. By default it serializes as a regular array.
|
||||||
|
#[inline]
|
||||||
|
fn visit_bytes(&mut self, value: &[u8]) -> Result<(), Self::Error> {
|
||||||
|
self.visit_seq(impls::SeqIteratorVisitor::new(value.iter(), Some(value.len())))
|
||||||
|
}
|
||||||
|
|
||||||
/// Enables serializers to serialize byte slices more compactly or more
|
fn visit_unit(&mut self) -> Result<(), Self::Error>;
|
||||||
/// efficiently than other types of slices. If no efficient implementation
|
|
||||||
/// is available, a reasonable implementation would be to forward to
|
|
||||||
/// `serialize_seq`. If forwarded, the implementation looks usually just like this:
|
|
||||||
/// ```rust
|
|
||||||
/// let mut state = try!(self.serialize_seq(value));
|
|
||||||
/// for b in value {
|
|
||||||
/// try!(self.serialize_seq_elt(&mut state, b));
|
|
||||||
/// }
|
|
||||||
/// self.serialize_seq_end(state)
|
|
||||||
/// ```
|
|
||||||
fn serialize_bytes(&mut self, value: &[u8]) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Serializes a `()` value. It's reasonable to just not serialize anything.
|
#[inline]
|
||||||
fn serialize_unit(&mut self) -> Result<(), Self::Error>;
|
fn visit_named_unit(&mut self, _name: &str) -> Result<(), Self::Error> {
|
||||||
|
self.visit_unit()
|
||||||
|
}
|
||||||
|
|
||||||
/// Serializes a unit struct value. A reasonable implementation would be to
|
#[inline]
|
||||||
/// forward to `serialize_unit`.
|
fn visit_enum_unit(&mut self,
|
||||||
fn serialize_unit_struct(
|
_name: &str,
|
||||||
&mut self,
|
_variant: &str) -> Result<(), Self::Error> {
|
||||||
name: &'static str,
|
self.visit_unit()
|
||||||
) -> Result<(), Self::Error>;
|
}
|
||||||
|
|
||||||
/// Serializes a unit variant, otherwise known as a variant with no
|
fn visit_none(&mut self) -> Result<(), Self::Error>;
|
||||||
/// arguments. A reasonable implementation would be to forward to
|
|
||||||
/// `serialize_unit`.
|
|
||||||
fn serialize_unit_variant(
|
|
||||||
&mut self,
|
|
||||||
name: &'static str,
|
|
||||||
variant_index: usize,
|
|
||||||
variant: &'static str,
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Allows a tuple struct with a single element, also known as a newtype
|
fn visit_some<V>(&mut self, value: V) -> Result<(), Self::Error>
|
||||||
/// struct, to be more efficiently serialized than a tuple struct with
|
where V: Serialize;
|
||||||
/// multiple items. A reasonable implementation would be to forward to
|
|
||||||
/// `serialize_tuple_struct` or to just serialize the inner value without wrapping.
|
|
||||||
fn serialize_newtype_struct<T: Serialize>(
|
|
||||||
&mut self,
|
|
||||||
name: &'static str,
|
|
||||||
value: T,
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Allows a variant with a single item to be more efficiently serialized
|
fn visit_seq<V>(&mut self, visitor: V) -> Result<(), Self::Error>
|
||||||
/// than a variant with multiple items. A reasonable implementation would be
|
where V: SeqVisitor;
|
||||||
/// to forward to `serialize_tuple_variant`.
|
|
||||||
fn serialize_newtype_variant<T: Serialize>(
|
|
||||||
&mut self,
|
|
||||||
name: &'static str,
|
|
||||||
variant_index: usize,
|
|
||||||
variant: &'static str,
|
|
||||||
value: T,
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Serializes a `None` value.
|
#[inline]
|
||||||
fn serialize_none(&mut self) -> Result<(), Self::Error>;
|
fn visit_named_seq<V>(&mut self,
|
||||||
|
_name: &'static str,
|
||||||
|
visitor: V) -> Result<(), Self::Error>
|
||||||
|
where V: SeqVisitor,
|
||||||
|
{
|
||||||
|
self.visit_seq(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
/// Serializes a `Some(...)` value.
|
#[inline]
|
||||||
fn serialize_some<T: Serialize>(
|
fn visit_enum_seq<V>(&mut self,
|
||||||
&mut self,
|
_name: &'static str,
|
||||||
value: T,
|
_variant: &'static str,
|
||||||
) -> Result<(), Self::Error>;
|
visitor: V) -> Result<(), Self::Error>
|
||||||
|
where V: SeqVisitor,
|
||||||
|
{
|
||||||
|
self.visit_seq(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
/// Begins to serialize a sequence. This call must be followed by zero or
|
fn visit_seq_elt<T>(&mut self, value: T) -> Result<(), Self::Error>
|
||||||
/// more calls to `serialize_seq_elt`, then a call to `serialize_seq_end`.
|
where T: Serialize;
|
||||||
fn serialize_seq(
|
|
||||||
&mut self,
|
|
||||||
len: Option<usize>,
|
|
||||||
) -> Result<Self::SeqState, Self::Error>;
|
|
||||||
|
|
||||||
/// Serializes a sequence element. Must have previously called
|
fn visit_map<V>(&mut self, visitor: V) -> Result<(), Self::Error>
|
||||||
/// `serialize_seq`.
|
where V: MapVisitor;
|
||||||
fn serialize_seq_elt<T: Serialize>(
|
|
||||||
&mut self,
|
|
||||||
state: &mut Self::SeqState,
|
|
||||||
value: T,
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Finishes serializing a sequence.
|
#[inline]
|
||||||
fn serialize_seq_end(
|
fn visit_named_map<V>(&mut self,
|
||||||
&mut self,
|
_name: &'static str,
|
||||||
state: Self::SeqState,
|
visitor: V) -> Result<(), Self::Error>
|
||||||
) -> Result<(), Self::Error>;
|
where V: MapVisitor,
|
||||||
|
{
|
||||||
|
self.visit_map(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
/// Begins to serialize a sequence whose length will be known at
|
#[inline]
|
||||||
/// deserialization time. This call must be followed by zero or more calls
|
fn visit_enum_map<V>(&mut self,
|
||||||
/// to `serialize_seq_elt`, then a call to `serialize_seq_end`. A reasonable
|
_name: &'static str,
|
||||||
/// implementation would be to forward to `serialize_seq`.
|
_variant: &'static str,
|
||||||
fn serialize_seq_fixed_size(
|
visitor: V) -> Result<(), Self::Error>
|
||||||
&mut self,
|
where V: MapVisitor,
|
||||||
size: usize,
|
{
|
||||||
) -> Result<Self::SeqState, Self::Error>;
|
self.visit_map(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
/// Begins to serialize a tuple. This call must be followed by zero or more
|
fn visit_map_elt<K, V>(&mut self, key: K, value: V) -> Result<(), Self::Error>
|
||||||
/// calls to `serialize_tuple_elt`, then a call to `serialize_tuple_end`. A
|
where K: Serialize,
|
||||||
/// reasonable implementation would be to forward to `serialize_seq`.
|
V: Serialize;
|
||||||
fn serialize_tuple(
|
|
||||||
&mut self,
|
|
||||||
len: usize,
|
|
||||||
) -> Result<Self::TupleState, Self::Error>;
|
|
||||||
|
|
||||||
/// Serializes a tuple element. Must have previously called
|
/// Specify a format string for the serializer.
|
||||||
/// `serialize_tuple`.
|
///
|
||||||
fn serialize_tuple_elt<T: Serialize>(
|
/// The serializer format is used to determine which format
|
||||||
&mut self,
|
/// specific field attributes should be used with the serializer.
|
||||||
state: &mut Self::TupleState,
|
fn format() -> &'static str {
|
||||||
value: T,
|
""
|
||||||
) -> Result<(), Self::Error>;
|
}
|
||||||
|
}
|
||||||
/// Finishes serializing a tuple.
|
|
||||||
fn serialize_tuple_end(
|
pub trait SeqVisitor {
|
||||||
&mut self,
|
fn visit<S>(&mut self, serializer: &mut S) -> Result<Option<()>, S::Error>
|
||||||
state: Self::TupleState,
|
where S: Serializer;
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
/// Return the length of the sequence if known.
|
||||||
/// Begins to serialize a tuple struct. This call must be followed by zero
|
#[inline]
|
||||||
/// or more calls to `serialize_tuple_struct_elt`, then a call to
|
fn len(&self) -> Option<usize> {
|
||||||
/// `serialize_tuple_struct_end`. A reasonable implementation would be to
|
None
|
||||||
/// forward to `serialize_tuple`.
|
}
|
||||||
fn serialize_tuple_struct(
|
}
|
||||||
&mut self,
|
|
||||||
name: &'static str,
|
pub trait MapVisitor {
|
||||||
len: usize,
|
fn visit<S>(&mut self, serializer: &mut S) -> Result<Option<()>, S::Error>
|
||||||
) -> Result<Self::TupleStructState, Self::Error>;
|
where S: Serializer;
|
||||||
|
|
||||||
/// Serializes a tuple struct element. Must have previously called
|
/// Return the length of the map if known.
|
||||||
/// `serialize_tuple_struct`.
|
#[inline]
|
||||||
fn serialize_tuple_struct_elt<T: Serialize>(
|
fn len(&self) -> Option<usize> {
|
||||||
&mut self,
|
None
|
||||||
state: &mut Self::TupleStructState,
|
}
|
||||||
value: T,
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Finishes serializing a tuple struct.
|
|
||||||
fn serialize_tuple_struct_end(
|
|
||||||
&mut self,
|
|
||||||
state: Self::TupleStructState,
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Begins to serialize a tuple variant. This call must be followed by zero
|
|
||||||
/// or more calls to `serialize_tuple_variant_elt`, then a call to
|
|
||||||
/// `serialize_tuple_variant_end`. A reasonable implementation would be to
|
|
||||||
/// forward to `serialize_tuple_struct`.
|
|
||||||
fn serialize_tuple_variant(
|
|
||||||
&mut self,
|
|
||||||
name: &'static str,
|
|
||||||
variant_index: usize,
|
|
||||||
variant: &'static str,
|
|
||||||
len: usize,
|
|
||||||
) -> Result<Self::TupleVariantState, Self::Error>;
|
|
||||||
|
|
||||||
/// Serializes a tuple variant element. Must have previously called
|
|
||||||
/// `serialize_tuple_variant`.
|
|
||||||
fn serialize_tuple_variant_elt<T: Serialize>(
|
|
||||||
&mut self,
|
|
||||||
state: &mut Self::TupleVariantState,
|
|
||||||
value: T,
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Finishes serializing a tuple variant.
|
|
||||||
fn serialize_tuple_variant_end(
|
|
||||||
&mut self,
|
|
||||||
state: Self::TupleVariantState,
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Begins to serialize a map. This call must be followed by zero or more
|
|
||||||
/// calls to `serialize_map_key` and `serialize_map_value`, then a call to
|
|
||||||
/// `serialize_map_end`.
|
|
||||||
fn serialize_map(
|
|
||||||
&mut self,
|
|
||||||
len: Option<usize>,
|
|
||||||
) -> Result<Self::MapState, Self::Error>;
|
|
||||||
|
|
||||||
/// Serialize a map key. Must have previously called `serialize_map`.
|
|
||||||
fn serialize_map_key<T: Serialize>(
|
|
||||||
&mut self,
|
|
||||||
state: &mut Self::MapState,
|
|
||||||
key: T
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Serialize a map value. Must have previously called `serialize_map`.
|
|
||||||
fn serialize_map_value<T: Serialize>(
|
|
||||||
&mut self,
|
|
||||||
state: &mut Self::MapState,
|
|
||||||
value: T
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Finishes serializing a map.
|
|
||||||
fn serialize_map_end(
|
|
||||||
&mut self,
|
|
||||||
state: Self::MapState,
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Begins to serialize a struct. This call must be followed by zero or more
|
|
||||||
/// calls to `serialize_struct_elt`, then a call to `serialize_struct_end`.
|
|
||||||
fn serialize_struct(
|
|
||||||
&mut self,
|
|
||||||
name: &'static str,
|
|
||||||
len: usize,
|
|
||||||
) -> Result<Self::StructState, Self::Error>;
|
|
||||||
|
|
||||||
/// Serializes a struct field. Must have previously called
|
|
||||||
/// `serialize_struct`.
|
|
||||||
fn serialize_struct_elt<V: Serialize>(
|
|
||||||
&mut self,
|
|
||||||
state: &mut Self::StructState,
|
|
||||||
key: &'static str,
|
|
||||||
value: V,
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Finishes serializing a struct.
|
|
||||||
fn serialize_struct_end(
|
|
||||||
&mut self,
|
|
||||||
state: Self::StructState,
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Begins to serialize a struct variant. This call must be followed by zero
|
|
||||||
/// or more calls to `serialize_struct_variant_elt`, then a call to
|
|
||||||
/// `serialize_struct_variant_end`.
|
|
||||||
fn serialize_struct_variant(
|
|
||||||
&mut self,
|
|
||||||
name: &'static str,
|
|
||||||
variant_index: usize,
|
|
||||||
variant: &'static str,
|
|
||||||
len: usize,
|
|
||||||
) -> Result<Self::StructVariantState, Self::Error>;
|
|
||||||
|
|
||||||
/// Serialize a struct variant element. Must have previously called
|
|
||||||
/// `serialize_struct_variant`.
|
|
||||||
fn serialize_struct_variant_elt<V: Serialize>(
|
|
||||||
&mut self,
|
|
||||||
state: &mut Self::StructVariantState,
|
|
||||||
key: &'static str,
|
|
||||||
value: V,
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
|
|
||||||
/// Finishes serializing a struct variant.
|
|
||||||
fn serialize_struct_variant_end(
|
|
||||||
&mut self,
|
|
||||||
state: Self::StructVariantState,
|
|
||||||
) -> Result<(), Self::Error>;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,72 +0,0 @@
|
|||||||
//! Private utility functions
|
|
||||||
|
|
||||||
const TAG_CONT: u8 = 0b1000_0000;
|
|
||||||
const TAG_TWO_B: u8 = 0b1100_0000;
|
|
||||||
const TAG_THREE_B: u8 = 0b1110_0000;
|
|
||||||
const TAG_FOUR_B: u8 = 0b1111_0000;
|
|
||||||
const MAX_ONE_B: u32 = 0x80;
|
|
||||||
const MAX_TWO_B: u32 = 0x800;
|
|
||||||
const MAX_THREE_B: u32 = 0x10000;
|
|
||||||
|
|
||||||
#[inline]
|
|
||||||
pub fn encode_utf8(c: char) -> EncodeUtf8 {
|
|
||||||
let code = c as u32;
|
|
||||||
let mut buf = [0; 4];
|
|
||||||
let pos = if code < MAX_ONE_B {
|
|
||||||
buf[3] = code as u8;
|
|
||||||
3
|
|
||||||
} else if code < MAX_TWO_B {
|
|
||||||
buf[2] = (code >> 6 & 0x1F) as u8 | TAG_TWO_B;
|
|
||||||
buf[3] = (code & 0x3F) as u8 | TAG_CONT;
|
|
||||||
2
|
|
||||||
} else if code < MAX_THREE_B {
|
|
||||||
buf[1] = (code >> 12 & 0x0F) as u8 | TAG_THREE_B;
|
|
||||||
buf[2] = (code >> 6 & 0x3F) as u8 | TAG_CONT;
|
|
||||||
buf[3] = (code & 0x3F) as u8 | TAG_CONT;
|
|
||||||
1
|
|
||||||
} else {
|
|
||||||
buf[0] = (code >> 18 & 0x07) as u8 | TAG_FOUR_B;
|
|
||||||
buf[1] = (code >> 12 & 0x3F) as u8 | TAG_CONT;
|
|
||||||
buf[2] = (code >> 6 & 0x3F) as u8 | TAG_CONT;
|
|
||||||
buf[3] = (code & 0x3F) as u8 | TAG_CONT;
|
|
||||||
0
|
|
||||||
};
|
|
||||||
EncodeUtf8 { buf: buf, pos: pos }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct EncodeUtf8 {
|
|
||||||
buf: [u8; 4],
|
|
||||||
pos: usize,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl EncodeUtf8 {
|
|
||||||
// FIXME: use this from_utf8_unchecked, since we know it can never fail
|
|
||||||
pub fn as_str(&self) -> &str {
|
|
||||||
::core::str::from_utf8(&self.buf[self.pos..]).unwrap()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[allow(non_upper_case_globals)]
|
|
||||||
const Pattern_White_Space_table: &'static [(char, char)] = &[
|
|
||||||
('\u{9}', '\u{d}'), ('\u{20}', '\u{20}'), ('\u{85}', '\u{85}'), ('\u{200e}', '\u{200f}'),
|
|
||||||
('\u{2028}', '\u{2029}')
|
|
||||||
];
|
|
||||||
|
|
||||||
fn bsearch_range_table(c: char, r: &'static [(char, char)]) -> bool {
|
|
||||||
use core::cmp::Ordering::{Equal, Less, Greater};
|
|
||||||
r.binary_search_by(|&(lo, hi)| {
|
|
||||||
if c < lo {
|
|
||||||
Greater
|
|
||||||
} else if hi < c {
|
|
||||||
Less
|
|
||||||
} else {
|
|
||||||
Equal
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.is_ok()
|
|
||||||
}
|
|
||||||
|
|
||||||
#[allow(non_snake_case)]
|
|
||||||
pub fn Pattern_White_Space(c: char) -> bool {
|
|
||||||
bsearch_range_table(c, Pattern_White_Space_table)
|
|
||||||
}
|
|
||||||
+11
-25
@@ -1,38 +1,24 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "serde_codegen"
|
name = "serde_codegen"
|
||||||
version = "0.8.5"
|
version = "0.4.2"
|
||||||
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
||||||
license = "MIT/Apache-2.0"
|
license = "MIT/Apache-2.0"
|
||||||
description = "Macros to auto-generate implementations for the serde framework"
|
description = "Macros to auto-generate implementations for the serde framework"
|
||||||
homepage = "https://serde.rs"
|
repository = "https://github.com/erickt/rust-serde"
|
||||||
repository = "https://github.com/serde-rs/serde"
|
|
||||||
documentation = "https://serde.rs/codegen.html"
|
|
||||||
keywords = ["serde", "serialization"]
|
|
||||||
build = "build.rs"
|
build = "build.rs"
|
||||||
include = ["Cargo.toml", "build.rs", "src/**/*.rs", "src/lib.rs.in"]
|
|
||||||
|
|
||||||
[features]
|
[features]
|
||||||
default = ["with-syntex"]
|
default = ["with-syntex"]
|
||||||
unstable = ["quasi_macros"]
|
nightly = ["quasi_macros"]
|
||||||
unstable-testing = ["clippy"]
|
with-syntex = ["quasi/with-syntex", "quasi_codegen/with-syntex", "syntex", "syntex_syntax"]
|
||||||
with-syntex = [
|
|
||||||
"quasi/with-syntex",
|
|
||||||
"quasi_codegen",
|
|
||||||
"quasi_codegen/with-syntex",
|
|
||||||
"serde_codegen_internals/with-syntex",
|
|
||||||
"syntex",
|
|
||||||
"syntex_syntax",
|
|
||||||
]
|
|
||||||
|
|
||||||
[build-dependencies]
|
[build-dependencies]
|
||||||
quasi_codegen = { version = "^0.19.0", optional = true }
|
quasi_codegen = { verision = "*", optional = true }
|
||||||
syntex = { version = "^0.43.0", optional = true }
|
syntex = { version = "*", optional = true }
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
aster = { version = "^0.26.0", default-features = false }
|
aster = { version = "*", default-features = false }
|
||||||
clippy = { version = "^0.*", optional = true }
|
quasi = { verision = "*", default-features = false }
|
||||||
quasi = { version = "^0.19.0", default-features = false }
|
quasi_macros = { version = "*", optional = true }
|
||||||
quasi_macros = { version = "^0.19.0", optional = true }
|
syntex = { version = "*", optional = true }
|
||||||
serde_codegen_internals = { version = "=0.8.0", default-features = false, path = "../serde_codegen_internals" }
|
syntex_syntax = { version = "*", optional = true }
|
||||||
syntex = { version = "^0.43.0", optional = true }
|
|
||||||
syntex_syntax = { version = "^0.43.0", optional = true }
|
|
||||||
|
|||||||
@@ -1,20 +1,20 @@
|
|||||||
#[cfg(feature = "with-syntex")]
|
#[cfg(feature = "with-syntex")]
|
||||||
mod inner {
|
mod inner {
|
||||||
|
extern crate syntex;
|
||||||
extern crate quasi_codegen;
|
extern crate quasi_codegen;
|
||||||
|
|
||||||
use std::env;
|
use std::env;
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
use std::thread::spawn;
|
|
||||||
|
|
||||||
pub fn main() {
|
pub fn main() {
|
||||||
// put everything into a thread, so users can use `RUST_MIN_STACK` to increase the amount of stack
|
let out_dir = env::var_os("OUT_DIR").unwrap();
|
||||||
spawn(|| {
|
let mut registry = syntex::Registry::new();
|
||||||
let out_dir = env::var_os("OUT_DIR").unwrap();
|
quasi_codegen::register(&mut registry);
|
||||||
|
|
||||||
let src = Path::new("src/lib.rs.in");
|
let src = Path::new("src/lib.rs.in");
|
||||||
let dst = Path::new(&out_dir).join("lib.rs");
|
let dst = Path::new(&out_dir).join("lib.rs");
|
||||||
quasi_codegen::expand(&src, &dst).unwrap();
|
|
||||||
}).join().unwrap()
|
registry.expand("", &src, &dst).unwrap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,108 @@
|
|||||||
|
use std::collections::HashMap;
|
||||||
|
use std::collections::HashSet;
|
||||||
|
|
||||||
|
use syntax::ast;
|
||||||
|
use syntax::ext::base::ExtCtxt;
|
||||||
|
use syntax::ptr::P;
|
||||||
|
|
||||||
|
/// Represents field name information
|
||||||
|
pub enum FieldNames {
|
||||||
|
Global(P<ast::Expr>),
|
||||||
|
Format{
|
||||||
|
formats: HashMap<P<ast::Expr>, P<ast::Expr>>,
|
||||||
|
default: P<ast::Expr>,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Represents field attribute information
|
||||||
|
pub struct FieldAttrs {
|
||||||
|
names: FieldNames,
|
||||||
|
use_default: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl FieldAttrs {
|
||||||
|
|
||||||
|
/// Create a FieldAttr with a single default field name
|
||||||
|
pub fn new(default_value: bool, name: P<ast::Expr>) -> FieldAttrs {
|
||||||
|
FieldAttrs {
|
||||||
|
names: FieldNames::Global(name),
|
||||||
|
use_default: default_value,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create a FieldAttr with format specific field names
|
||||||
|
pub fn new_with_formats(
|
||||||
|
default_value: bool,
|
||||||
|
default_name: P<ast::Expr>,
|
||||||
|
formats: HashMap<P<ast::Expr>, P<ast::Expr>>,
|
||||||
|
) -> FieldAttrs {
|
||||||
|
FieldAttrs {
|
||||||
|
names: FieldNames::Format {
|
||||||
|
formats: formats,
|
||||||
|
default: default_name,
|
||||||
|
},
|
||||||
|
use_default: default_value,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Return a set of formats that the field has attributes for.
|
||||||
|
pub fn formats(&self) -> HashSet<P<ast::Expr>> {
|
||||||
|
match self.names {
|
||||||
|
FieldNames::Format{ref formats, default: _} => {
|
||||||
|
let mut set = HashSet::new();
|
||||||
|
for (fmt, _) in formats.iter() {
|
||||||
|
set.insert(fmt.clone());
|
||||||
|
};
|
||||||
|
set
|
||||||
|
},
|
||||||
|
_ => HashSet::new()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Return an expression for the field key name for serialisation.
|
||||||
|
///
|
||||||
|
/// The resulting expression assumes that `S` refers to a type
|
||||||
|
/// that implements `Serializer`.
|
||||||
|
pub fn serializer_key_expr(self, cx: &ExtCtxt) -> P<ast::Expr> {
|
||||||
|
match self.names {
|
||||||
|
FieldNames::Global(x) => x,
|
||||||
|
FieldNames::Format{formats, default} => {
|
||||||
|
let arms = formats.iter()
|
||||||
|
.map(|(fmt, lit)| {
|
||||||
|
quote_arm!(cx, $fmt => { $lit })
|
||||||
|
})
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
quote_expr!(cx,
|
||||||
|
{
|
||||||
|
match S::format() {
|
||||||
|
$arms
|
||||||
|
_ => { $default }
|
||||||
|
}
|
||||||
|
})
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Return the default field name for the field.
|
||||||
|
pub fn default_key_expr(&self) -> &P<ast::Expr> {
|
||||||
|
match self.names {
|
||||||
|
FieldNames::Global(ref expr) => expr,
|
||||||
|
FieldNames::Format{formats: _, ref default} => default
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Return the field name for the field in the specified format.
|
||||||
|
pub fn key_expr(&self, format: &P<ast::Expr>) -> &P<ast::Expr> {
|
||||||
|
match self.names {
|
||||||
|
FieldNames::Global(ref expr) =>
|
||||||
|
expr,
|
||||||
|
FieldNames::Format{ref formats, ref default} =>
|
||||||
|
formats.get(format).unwrap_or(default)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Predicate for using a field's default value
|
||||||
|
pub fn use_default(&self) -> bool {
|
||||||
|
self.use_default
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,127 +0,0 @@
|
|||||||
use std::collections::HashSet;
|
|
||||||
|
|
||||||
use aster::AstBuilder;
|
|
||||||
|
|
||||||
use syntax::ast;
|
|
||||||
use syntax::visit;
|
|
||||||
|
|
||||||
use internals::ast::Item;
|
|
||||||
use internals::attr;
|
|
||||||
|
|
||||||
// Remove the default from every type parameter because in the generated impls
|
|
||||||
// they look like associated types: "error: associated type bindings are not
|
|
||||||
// allowed here".
|
|
||||||
pub fn without_defaults(generics: &ast::Generics) -> ast::Generics {
|
|
||||||
ast::Generics {
|
|
||||||
ty_params: generics.ty_params.iter().map(|ty_param| {
|
|
||||||
ast::TyParam {
|
|
||||||
default: None,
|
|
||||||
.. ty_param.clone()
|
|
||||||
}}).collect(),
|
|
||||||
.. generics.clone()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn with_where_predicates(
|
|
||||||
builder: &AstBuilder,
|
|
||||||
generics: &ast::Generics,
|
|
||||||
predicates: &[ast::WherePredicate],
|
|
||||||
) -> ast::Generics {
|
|
||||||
builder.from_generics(generics.clone())
|
|
||||||
.with_predicates(predicates.to_vec())
|
|
||||||
.build()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn with_where_predicates_from_fields<F>(
|
|
||||||
builder: &AstBuilder,
|
|
||||||
item: &Item,
|
|
||||||
generics: &ast::Generics,
|
|
||||||
from_field: F,
|
|
||||||
) -> ast::Generics
|
|
||||||
where F: Fn(&attr::Field) -> Option<&[ast::WherePredicate]>,
|
|
||||||
{
|
|
||||||
builder.from_generics(generics.clone())
|
|
||||||
.with_predicates(
|
|
||||||
item.body.all_fields()
|
|
||||||
.flat_map(|field| from_field(&field.attrs))
|
|
||||||
.flat_map(|predicates| predicates.to_vec()))
|
|
||||||
.build()
|
|
||||||
}
|
|
||||||
|
|
||||||
// Puts the given bound on any generic type parameters that are used in fields
|
|
||||||
// for which filter returns true.
|
|
||||||
//
|
|
||||||
// For example, the following struct needs the bound `A: Serialize, B: Serialize`.
|
|
||||||
//
|
|
||||||
// struct S<'b, A, B: 'b, C> {
|
|
||||||
// a: A,
|
|
||||||
// b: Option<&'b B>
|
|
||||||
// #[serde(skip_serializing)]
|
|
||||||
// c: C,
|
|
||||||
// }
|
|
||||||
pub fn with_bound<F>(
|
|
||||||
builder: &AstBuilder,
|
|
||||||
item: &Item,
|
|
||||||
generics: &ast::Generics,
|
|
||||||
filter: F,
|
|
||||||
bound: &ast::Path,
|
|
||||||
) -> ast::Generics
|
|
||||||
where F: Fn(&attr::Field) -> bool,
|
|
||||||
{
|
|
||||||
struct FindTyParams {
|
|
||||||
// Set of all generic type parameters on the current struct (A, B, C in
|
|
||||||
// the example). Initialized up front.
|
|
||||||
all_ty_params: HashSet<ast::Name>,
|
|
||||||
// Set of generic type parameters used in fields for which filter
|
|
||||||
// returns true (A and B in the example). Filled in as the visitor sees
|
|
||||||
// them.
|
|
||||||
relevant_ty_params: HashSet<ast::Name>,
|
|
||||||
}
|
|
||||||
impl visit::Visitor for FindTyParams {
|
|
||||||
fn visit_path(&mut self, path: &ast::Path, _id: ast::NodeId) {
|
|
||||||
if let Some(seg) = path.segments.last() {
|
|
||||||
if seg.identifier.name.as_str() == "PhantomData" {
|
|
||||||
// Hardcoded exception, because PhantomData<T> implements
|
|
||||||
// Serialize and Deserialize whether or not T implements it.
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !path.global && path.segments.len() == 1 {
|
|
||||||
let id = path.segments[0].identifier.name;
|
|
||||||
if self.all_ty_params.contains(&id) {
|
|
||||||
self.relevant_ty_params.insert(id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
visit::walk_path(self, path);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let all_ty_params: HashSet<_> = generics.ty_params.iter()
|
|
||||||
.map(|ty_param| ty_param.ident.name)
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let relevant_tys = item.body.all_fields()
|
|
||||||
.filter(|&field| filter(&field.attrs))
|
|
||||||
.map(|field| &field.ty);
|
|
||||||
|
|
||||||
let mut visitor = FindTyParams {
|
|
||||||
all_ty_params: all_ty_params,
|
|
||||||
relevant_ty_params: HashSet::new(),
|
|
||||||
};
|
|
||||||
for ty in relevant_tys {
|
|
||||||
visit::walk_ty(&mut visitor, ty);
|
|
||||||
}
|
|
||||||
|
|
||||||
builder.from_generics(generics.clone())
|
|
||||||
.with_predicates(
|
|
||||||
generics.ty_params.iter()
|
|
||||||
.map(|ty_param| ty_param.ident.name)
|
|
||||||
.filter(|id| visitor.relevant_ty_params.contains(id))
|
|
||||||
.map(|id| builder.where_predicate()
|
|
||||||
// the type parameter that is being bounded e.g. T
|
|
||||||
.bound().build(builder.ty().id(id))
|
|
||||||
// the bound e.g. Serialize
|
|
||||||
.bound().trait_(bound.clone()).build()
|
|
||||||
.build()))
|
|
||||||
.build()
|
|
||||||
}
|
|
||||||
+492
-828
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,133 @@
|
|||||||
|
use std::collections::HashMap;
|
||||||
|
|
||||||
|
use aster;
|
||||||
|
|
||||||
|
use syntax::ast;
|
||||||
|
use syntax::attr;
|
||||||
|
use syntax::ext::base::ExtCtxt;
|
||||||
|
use syntax::ptr::P;
|
||||||
|
|
||||||
|
use attr::FieldAttrs;
|
||||||
|
|
||||||
|
enum Rename<'a> {
|
||||||
|
None,
|
||||||
|
Global(&'a ast::Lit),
|
||||||
|
Format(HashMap<P<ast::Expr>, &'a ast::Lit>)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn rename<'a>(
|
||||||
|
builder: &aster::AstBuilder,
|
||||||
|
mi: &'a ast::MetaItem,
|
||||||
|
) -> Option<Rename<'a>>
|
||||||
|
{
|
||||||
|
match mi.node {
|
||||||
|
ast::MetaNameValue(ref n, ref lit) => {
|
||||||
|
if n == &"rename" {
|
||||||
|
Some(Rename::Global(lit))
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
},
|
||||||
|
ast::MetaList(ref n, ref items) => {
|
||||||
|
if n == &"rename" {
|
||||||
|
let mut m = HashMap::new();
|
||||||
|
m.extend(
|
||||||
|
items.iter()
|
||||||
|
.filter_map(
|
||||||
|
|item|
|
||||||
|
match item.node {
|
||||||
|
ast::MetaNameValue(ref n, ref lit) =>
|
||||||
|
Some((builder.expr().str(n),
|
||||||
|
lit)),
|
||||||
|
_ => None
|
||||||
|
}));
|
||||||
|
Some(Rename::Format(m))
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
},
|
||||||
|
_ => None
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn default_value(mi: &ast::MetaItem) -> bool {
|
||||||
|
if let ast::MetaItem_::MetaWord(ref n) = mi.node {
|
||||||
|
n == &"default"
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn field_attrs<'a>(
|
||||||
|
builder: &aster::AstBuilder,
|
||||||
|
field: &'a ast::StructField,
|
||||||
|
) -> (Rename<'a>, bool) {
|
||||||
|
field.node.attrs.iter()
|
||||||
|
.find(|sa| {
|
||||||
|
if let ast::MetaList(ref n, _) = sa.node.value.node {
|
||||||
|
n == &"serde"
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.and_then(|sa| {
|
||||||
|
if let ast::MetaList(_, ref vals) = sa.node.value.node {
|
||||||
|
attr::mark_used(&sa);
|
||||||
|
Some((vals.iter()
|
||||||
|
.fold(None, |v, mi| v.or(rename(builder, mi)))
|
||||||
|
.unwrap_or(Rename::None),
|
||||||
|
vals.iter().any(|mi| default_value(mi))))
|
||||||
|
} else {
|
||||||
|
Some((Rename::None, false))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.unwrap_or((Rename::None, false))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn struct_field_attrs(
|
||||||
|
cx: &ExtCtxt,
|
||||||
|
builder: &aster::AstBuilder,
|
||||||
|
struct_def: &ast::StructDef,
|
||||||
|
) -> Vec<FieldAttrs> {
|
||||||
|
struct_def.fields.iter()
|
||||||
|
.map(|field| {
|
||||||
|
match field_attrs(builder, field) {
|
||||||
|
(Rename::Global(rename), default_value) =>
|
||||||
|
FieldAttrs::new(
|
||||||
|
default_value,
|
||||||
|
builder.expr().build_lit(P(rename.clone()))),
|
||||||
|
(Rename::Format(renames), default_value) => {
|
||||||
|
let mut res = HashMap::new();
|
||||||
|
res.extend(
|
||||||
|
renames.into_iter()
|
||||||
|
.map(|(k,v)|
|
||||||
|
(k, builder.expr().build_lit(P(v.clone())))));
|
||||||
|
FieldAttrs::new_with_formats(
|
||||||
|
default_value,
|
||||||
|
default_field_name(cx, builder, field.node.kind),
|
||||||
|
res)
|
||||||
|
},
|
||||||
|
(Rename::None, default_value) => {
|
||||||
|
FieldAttrs::new(
|
||||||
|
default_value,
|
||||||
|
default_field_name(cx, builder, field.node.kind))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn default_field_name(
|
||||||
|
cx: &ExtCtxt,
|
||||||
|
builder: &aster::AstBuilder,
|
||||||
|
kind: ast::StructFieldKind,
|
||||||
|
) -> P<ast::Expr> {
|
||||||
|
match kind {
|
||||||
|
ast::NamedField(name, _) => {
|
||||||
|
builder.expr().str(name)
|
||||||
|
}
|
||||||
|
ast::UnnamedField(_) => {
|
||||||
|
cx.bug("struct has named and unnamed fields")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
+26
-55
@@ -1,33 +1,20 @@
|
|||||||
#![cfg_attr(feature = "clippy", plugin(clippy))]
|
|
||||||
#![cfg_attr(feature = "clippy", feature(plugin))]
|
|
||||||
#![cfg_attr(feature = "clippy", allow(too_many_arguments))]
|
|
||||||
#![cfg_attr(feature = "clippy", allow(used_underscore_binding))]
|
|
||||||
#![cfg_attr(not(feature = "with-syntex"), feature(rustc_private, plugin))]
|
#![cfg_attr(not(feature = "with-syntex"), feature(rustc_private, plugin))]
|
||||||
#![cfg_attr(not(feature = "with-syntex"), plugin(quasi_macros))]
|
#![cfg_attr(not(feature = "with-syntex"), plugin(quasi_macros))]
|
||||||
|
|
||||||
extern crate aster;
|
extern crate aster;
|
||||||
extern crate quasi;
|
extern crate quasi;
|
||||||
extern crate serde_codegen_internals as internals;
|
|
||||||
|
|
||||||
#[cfg(feature = "with-syntex")]
|
#[cfg(feature = "with-syntex")]
|
||||||
extern crate syntex;
|
extern crate syntex;
|
||||||
|
|
||||||
#[cfg(feature = "with-syntex")]
|
#[cfg(feature = "with-syntex")]
|
||||||
#[macro_use]
|
|
||||||
extern crate syntex_syntax as syntax;
|
extern crate syntex_syntax as syntax;
|
||||||
|
|
||||||
#[cfg(not(feature = "with-syntex"))]
|
#[cfg(not(feature = "with-syntex"))]
|
||||||
#[macro_use]
|
|
||||||
extern crate syntax;
|
extern crate syntax;
|
||||||
|
|
||||||
#[cfg(not(feature = "with-syntex"))]
|
#[cfg(not(feature = "with-syntex"))]
|
||||||
extern crate rustc_plugin;
|
extern crate rustc;
|
||||||
|
|
||||||
#[cfg(feature = "with-syntex")]
|
|
||||||
use std::path::Path;
|
|
||||||
|
|
||||||
#[cfg(not(feature = "with-syntex"))]
|
|
||||||
use syntax::feature_gate::AttributeType;
|
|
||||||
|
|
||||||
#[cfg(feature = "with-syntex")]
|
#[cfg(feature = "with-syntex")]
|
||||||
include!(concat!(env!("OUT_DIR"), "/lib.rs"));
|
include!(concat!(env!("OUT_DIR"), "/lib.rs"));
|
||||||
@@ -36,58 +23,44 @@ include!(concat!(env!("OUT_DIR"), "/lib.rs"));
|
|||||||
include!("lib.rs.in");
|
include!("lib.rs.in");
|
||||||
|
|
||||||
#[cfg(feature = "with-syntex")]
|
#[cfg(feature = "with-syntex")]
|
||||||
pub fn expand<S, D>(src: S, dst: D) -> Result<(), syntex::Error>
|
pub fn register(reg: &mut syntex::Registry) {
|
||||||
where S: AsRef<Path>,
|
use syntax::{ast, fold};
|
||||||
D: AsRef<Path>,
|
|
||||||
{
|
|
||||||
let src = src.as_ref().to_owned();
|
|
||||||
let dst = dst.as_ref().to_owned();
|
|
||||||
|
|
||||||
let expand_thread = move || {
|
reg.add_attr("feature(custom_derive)");
|
||||||
use syntax::{ast, fold};
|
reg.add_attr("feature(custom_attribute)");
|
||||||
|
|
||||||
/// Strip the serde attributes from the crate.
|
reg.add_decorator("derive_Serialize", ser::expand_derive_serialize);
|
||||||
#[cfg(feature = "with-syntex")]
|
reg.add_decorator("derive_Deserialize", de::expand_derive_deserialize);
|
||||||
fn strip_attributes(krate: ast::Crate) -> ast::Crate {
|
|
||||||
/// Helper folder that strips the serde attributes after the extensions have been expanded.
|
|
||||||
struct StripAttributeFolder;
|
|
||||||
|
|
||||||
impl fold::Folder for StripAttributeFolder {
|
reg.add_post_expansion_pass(strip_attributes);
|
||||||
fn fold_attribute(&mut self, attr: ast::Attribute) -> Option<ast::Attribute> {
|
|
||||||
match attr.node.value.node {
|
|
||||||
ast::MetaItemKind::List(ref n, _) if n == &"serde" => { return None; }
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
|
|
||||||
Some(attr)
|
/// Strip the serde attributes from the crate.
|
||||||
|
#[cfg(feature = "with-syntex")]
|
||||||
|
fn strip_attributes(krate: ast::Crate) -> ast::Crate {
|
||||||
|
/// Helper folder that strips the serde attributes after the extensions have been expanded.
|
||||||
|
struct StripAttributeFolder;
|
||||||
|
|
||||||
|
impl fold::Folder for StripAttributeFolder {
|
||||||
|
fn fold_attribute(&mut self, attr: ast::Attribute) -> Option<ast::Attribute> {
|
||||||
|
match attr.node.value.node {
|
||||||
|
ast::MetaList(ref n, _) if n == &"serde" => { return None; }
|
||||||
|
_ => {}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn fold_mac(&mut self, mac: ast::Mac) -> ast::Mac {
|
Some(attr)
|
||||||
fold::noop_fold_mac(mac, self)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fold::Folder::fold_crate(&mut StripAttributeFolder, krate)
|
fn fold_mac(&mut self, mac: ast::Mac) -> ast::Mac {
|
||||||
|
fold::noop_fold_mac(mac, self)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut reg = syntex::Registry::new();
|
fold::Folder::fold_crate(&mut StripAttributeFolder, krate)
|
||||||
|
}
|
||||||
reg.add_attr("feature(custom_derive)");
|
|
||||||
reg.add_attr("feature(custom_attribute)");
|
|
||||||
|
|
||||||
reg.add_decorator("derive_Serialize", ser::expand_derive_serialize);
|
|
||||||
reg.add_decorator("derive_Deserialize", de::expand_derive_deserialize);
|
|
||||||
|
|
||||||
reg.add_post_expansion_pass(strip_attributes);
|
|
||||||
|
|
||||||
reg.expand("", src, dst)
|
|
||||||
};
|
|
||||||
|
|
||||||
syntex::with_extra_stack(expand_thread)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(not(feature = "with-syntex"))]
|
#[cfg(not(feature = "with-syntex"))]
|
||||||
pub fn register(reg: &mut rustc_plugin::Registry) {
|
pub fn register(reg: &mut rustc::plugin::Registry) {
|
||||||
reg.register_syntax_extension(
|
reg.register_syntax_extension(
|
||||||
syntax::parse::token::intern("derive_Serialize"),
|
syntax::parse::token::intern("derive_Serialize"),
|
||||||
syntax::ext::base::MultiDecorator(
|
syntax::ext::base::MultiDecorator(
|
||||||
@@ -97,6 +70,4 @@ pub fn register(reg: &mut rustc_plugin::Registry) {
|
|||||||
syntax::parse::token::intern("derive_Deserialize"),
|
syntax::parse::token::intern("derive_Deserialize"),
|
||||||
syntax::ext::base::MultiDecorator(
|
syntax::ext::base::MultiDecorator(
|
||||||
Box::new(de::expand_derive_deserialize)));
|
Box::new(de::expand_derive_deserialize)));
|
||||||
|
|
||||||
reg.register_attribute("serde".to_owned(), AttributeType::Normal);
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
mod bound;
|
mod attr;
|
||||||
mod de;
|
mod de;
|
||||||
|
mod field;
|
||||||
mod ser;
|
mod ser;
|
||||||
mod span;
|
|
||||||
|
|||||||
+424
-440
File diff suppressed because it is too large
Load Diff
@@ -1,43 +0,0 @@
|
|||||||
use syntax::ast;
|
|
||||||
use syntax::codemap::{self, ExpnId, Span};
|
|
||||||
use syntax::ext::base::{Annotatable, ExtCtxt};
|
|
||||||
use syntax::fold::{self, Folder};
|
|
||||||
use syntax::parse::token::intern;
|
|
||||||
use syntax::ptr::P;
|
|
||||||
|
|
||||||
pub fn record_expansion(
|
|
||||||
cx: &ExtCtxt,
|
|
||||||
item: P<ast::Item>,
|
|
||||||
derive: &str,
|
|
||||||
) -> Annotatable {
|
|
||||||
let info = codemap::ExpnInfo {
|
|
||||||
call_site: codemap::DUMMY_SP,
|
|
||||||
callee: codemap::NameAndSpan {
|
|
||||||
format: codemap::MacroAttribute(intern(&format!("derive({})", derive))),
|
|
||||||
span: None,
|
|
||||||
allow_internal_unstable: false,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
let expn_id = cx.codemap().record_expansion(info);
|
|
||||||
|
|
||||||
let mut respanner = Respanner { expn_id: expn_id };
|
|
||||||
let item = item.map(|item| respanner.fold_item_simple(item));
|
|
||||||
Annotatable::Item(item)
|
|
||||||
}
|
|
||||||
|
|
||||||
struct Respanner {
|
|
||||||
expn_id: ExpnId,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Folder for Respanner {
|
|
||||||
fn new_span(&mut self, span: Span) -> Span {
|
|
||||||
Span {
|
|
||||||
expn_id: self.expn_id,
|
|
||||||
.. span
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn fold_mac(&mut self, mac: ast::Mac) -> ast::Mac {
|
|
||||||
fold::noop_fold_mac(mac, self)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "serde_codegen_internals"
|
|
||||||
version = "0.8.0"
|
|
||||||
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
|
||||||
license = "MIT/Apache-2.0"
|
|
||||||
description = "AST representation used by Serde codegen. Unstable."
|
|
||||||
homepage = "https://serde.rs"
|
|
||||||
repository = "https://github.com/serde-rs/serde"
|
|
||||||
documentation = "https://docs.serde.rs/serde_codegen_internals/"
|
|
||||||
keywords = ["serde", "serialization"]
|
|
||||||
include = ["Cargo.toml", "src/**/*.rs"]
|
|
||||||
|
|
||||||
[features]
|
|
||||||
default = ["with-syntex"]
|
|
||||||
unstable-testing = ["clippy"]
|
|
||||||
with-syntex = ["syntex_syntax", "syntex_errors"]
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
clippy = { version = "^0.*", optional = true }
|
|
||||||
syntex_syntax = { version = "^0.43.0", optional = true }
|
|
||||||
syntex_errors = { version = "^0.43.0", optional = true }
|
|
||||||
@@ -1,142 +0,0 @@
|
|||||||
use syntax::ast;
|
|
||||||
use syntax::codemap;
|
|
||||||
use syntax::ext::base::ExtCtxt;
|
|
||||||
use syntax::ptr::P;
|
|
||||||
|
|
||||||
use attr;
|
|
||||||
use Error;
|
|
||||||
|
|
||||||
pub struct Item<'a> {
|
|
||||||
pub ident: ast::Ident,
|
|
||||||
pub span: codemap::Span,
|
|
||||||
pub attrs: attr::Item,
|
|
||||||
pub body: Body<'a>,
|
|
||||||
pub generics: &'a ast::Generics,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub enum Body<'a> {
|
|
||||||
Enum(Vec<Variant<'a>>),
|
|
||||||
Struct(Style, Vec<Field<'a>>),
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct Variant<'a> {
|
|
||||||
pub ident: ast::Ident,
|
|
||||||
pub span: codemap::Span,
|
|
||||||
pub attrs: attr::Variant,
|
|
||||||
pub style: Style,
|
|
||||||
pub fields: Vec<Field<'a>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct Field<'a> {
|
|
||||||
pub ident: Option<ast::Ident>,
|
|
||||||
pub span: codemap::Span,
|
|
||||||
pub attrs: attr::Field,
|
|
||||||
pub ty: &'a P<ast::Ty>,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub enum Style {
|
|
||||||
Struct,
|
|
||||||
Tuple,
|
|
||||||
Newtype,
|
|
||||||
Unit,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a> Item<'a> {
|
|
||||||
pub fn from_ast(
|
|
||||||
cx: &ExtCtxt,
|
|
||||||
item: &'a ast::Item,
|
|
||||||
) -> Result<Item<'a>, Error> {
|
|
||||||
let attrs = attr::Item::from_ast(cx, item);
|
|
||||||
|
|
||||||
let (body, generics) = match item.node {
|
|
||||||
ast::ItemKind::Enum(ref enum_def, ref generics) => {
|
|
||||||
let variants = enum_from_ast(cx, enum_def);
|
|
||||||
(Body::Enum(variants), generics)
|
|
||||||
}
|
|
||||||
ast::ItemKind::Struct(ref variant_data, ref generics) => {
|
|
||||||
let (style, fields) = struct_from_ast(cx, variant_data);
|
|
||||||
(Body::Struct(style, fields), generics)
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
return Err(Error::UnexpectedItemKind);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
Ok(Item {
|
|
||||||
ident: item.ident,
|
|
||||||
span: item.span,
|
|
||||||
attrs: attrs,
|
|
||||||
body: body,
|
|
||||||
generics: generics,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a> Body<'a> {
|
|
||||||
pub fn all_fields(&'a self) -> Box<Iterator<Item=&'a Field<'a>> + 'a> {
|
|
||||||
match *self {
|
|
||||||
Body::Enum(ref variants) => {
|
|
||||||
Box::new(variants.iter()
|
|
||||||
.flat_map(|variant| variant.fields.iter()))
|
|
||||||
}
|
|
||||||
Body::Struct(_, ref fields) => {
|
|
||||||
Box::new(fields.iter())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn enum_from_ast<'a>(
|
|
||||||
cx: &ExtCtxt,
|
|
||||||
enum_def: &'a ast::EnumDef,
|
|
||||||
) -> Vec<Variant<'a>> {
|
|
||||||
enum_def.variants.iter()
|
|
||||||
.map(|variant| {
|
|
||||||
let (style, fields) = struct_from_ast(cx, &variant.node.data);
|
|
||||||
Variant {
|
|
||||||
ident: variant.node.name,
|
|
||||||
span: variant.span,
|
|
||||||
attrs: attr::Variant::from_ast(cx, variant),
|
|
||||||
style: style,
|
|
||||||
fields: fields,
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.collect()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn struct_from_ast<'a>(
|
|
||||||
cx: &ExtCtxt,
|
|
||||||
variant_data: &'a ast::VariantData,
|
|
||||||
) -> (Style, Vec<Field<'a>>) {
|
|
||||||
match *variant_data {
|
|
||||||
ast::VariantData::Struct(ref fields, _) => {
|
|
||||||
(Style::Struct, fields_from_ast(cx, fields))
|
|
||||||
}
|
|
||||||
ast::VariantData::Tuple(ref fields, _) if fields.len() == 1 => {
|
|
||||||
(Style::Newtype, fields_from_ast(cx, fields))
|
|
||||||
}
|
|
||||||
ast::VariantData::Tuple(ref fields, _) => {
|
|
||||||
(Style::Tuple, fields_from_ast(cx, fields))
|
|
||||||
}
|
|
||||||
ast::VariantData::Unit(_) => {
|
|
||||||
(Style::Unit, Vec::new())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn fields_from_ast<'a>(
|
|
||||||
cx: &ExtCtxt,
|
|
||||||
fields: &'a [ast::StructField],
|
|
||||||
) -> Vec<Field<'a>> {
|
|
||||||
fields.iter()
|
|
||||||
.enumerate()
|
|
||||||
.map(|(i, field)| {
|
|
||||||
Field {
|
|
||||||
ident: field.ident,
|
|
||||||
span: field.span,
|
|
||||||
attrs: attr::Field::from_ast(cx, i, field),
|
|
||||||
ty: &field.ty,
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.collect()
|
|
||||||
}
|
|
||||||
@@ -1,660 +0,0 @@
|
|||||||
use std::rc::Rc;
|
|
||||||
|
|
||||||
use syntax::ast;
|
|
||||||
use syntax::attr::{self, HasAttrs};
|
|
||||||
use syntax::codemap::{Span, Spanned, respan};
|
|
||||||
use syntax::ext::base::ExtCtxt;
|
|
||||||
use syntax::fold::Folder;
|
|
||||||
use syntax::parse::parser::{Parser, PathStyle};
|
|
||||||
use syntax::parse::token::{self, InternedString};
|
|
||||||
use syntax::parse;
|
|
||||||
use syntax::print::pprust::{lit_to_string, meta_item_to_string, meta_list_item_to_string};
|
|
||||||
use syntax::tokenstream::{self, TokenTree};
|
|
||||||
|
|
||||||
// This module handles parsing of `#[serde(...)]` attributes. The entrypoints
|
|
||||||
// are `attr::Item::from_ast`, `attr::Variant::from_ast`, and
|
|
||||||
// `attr::Field::from_ast`. Each returns an instance of the corresponding
|
|
||||||
// struct. Note that none of them return a Result. Unrecognized, malformed, or
|
|
||||||
// duplicated attributes result in a span_err but otherwise are ignored. The
|
|
||||||
// user will see errors simultaneously for all bad attributes in the crate
|
|
||||||
// rather than just the first.
|
|
||||||
|
|
||||||
struct Attr<'a, 'b: 'a, T> {
|
|
||||||
cx: &'a ExtCtxt<'b>,
|
|
||||||
name: &'static str,
|
|
||||||
value: Option<Spanned<T>>,
|
|
||||||
}
|
|
||||||
impl<'a, 'b, T> Attr<'a, 'b, T> {
|
|
||||||
fn none(cx: &'a ExtCtxt<'b>, name: &'static str) -> Self {
|
|
||||||
Attr {
|
|
||||||
cx: cx,
|
|
||||||
name: name,
|
|
||||||
value: None,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn set(&mut self, span: Span, t: T) {
|
|
||||||
if let Some(Spanned { span: prev_span, .. }) = self.value {
|
|
||||||
let mut err = self.cx.struct_span_err(
|
|
||||||
span,
|
|
||||||
&format!("duplicate serde attribute `{}`", self.name));
|
|
||||||
err.span_help(prev_span, "previously set here");
|
|
||||||
err.emit();
|
|
||||||
} else {
|
|
||||||
self.value = Some(respan(span, t));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn set_opt(&mut self, v: Option<Spanned<T>>) {
|
|
||||||
if let Some(v) = v {
|
|
||||||
self.set(v.span, v.node);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn set_if_none(&mut self, span: Span, t: T) {
|
|
||||||
if self.value.is_none() {
|
|
||||||
self.value = Some(respan(span, t));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get(self) -> Option<T> {
|
|
||||||
self.value.map(|spanned| spanned.node)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get_spanned(self) -> Option<Spanned<T>> {
|
|
||||||
self.value
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct BoolAttr<'a, 'b: 'a>(Attr<'a, 'b, ()>);
|
|
||||||
impl<'a, 'b> BoolAttr<'a, 'b> {
|
|
||||||
fn none(cx: &'a ExtCtxt<'b>, name: &'static str) -> Self {
|
|
||||||
BoolAttr(Attr::none(cx, name))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn set_true(&mut self, span: Span) {
|
|
||||||
self.0.set(span, ());
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get(&self) -> bool {
|
|
||||||
self.0.value.is_some()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct Name {
|
|
||||||
serialize: InternedString,
|
|
||||||
deserialize: InternedString,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Name {
|
|
||||||
/// Return the container name for the container when serializing.
|
|
||||||
pub fn serialize_name(&self) -> InternedString {
|
|
||||||
self.serialize.clone()
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Return the container name for the container when deserializing.
|
|
||||||
pub fn deserialize_name(&self) -> InternedString {
|
|
||||||
self.deserialize.clone()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Represents container (e.g. struct) attribute information
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct Item {
|
|
||||||
name: Name,
|
|
||||||
deny_unknown_fields: bool,
|
|
||||||
ser_bound: Option<Vec<ast::WherePredicate>>,
|
|
||||||
de_bound: Option<Vec<ast::WherePredicate>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Item {
|
|
||||||
/// Extract out the `#[serde(...)]` attributes from an item.
|
|
||||||
pub fn from_ast(cx: &ExtCtxt, item: &ast::Item) -> Self {
|
|
||||||
let mut ser_name = Attr::none(cx, "rename");
|
|
||||||
let mut de_name = Attr::none(cx, "rename");
|
|
||||||
let mut deny_unknown_fields = BoolAttr::none(cx, "deny_unknown_fields");
|
|
||||||
let mut ser_bound = Attr::none(cx, "bound");
|
|
||||||
let mut de_bound = Attr::none(cx, "bound");
|
|
||||||
|
|
||||||
let ident = item.ident.name.as_str();
|
|
||||||
|
|
||||||
for meta_items in item.attrs().iter().filter_map(get_serde_meta_items) {
|
|
||||||
for meta_item in meta_items {
|
|
||||||
let span = meta_item.span;
|
|
||||||
match meta_item.node {
|
|
||||||
// Parse `#[serde(rename="foo")]`
|
|
||||||
ast::MetaItemKind::NameValue(ref name, ref lit) if name == &"rename" => {
|
|
||||||
if let Ok(s) = get_str_from_lit(cx, name, lit) {
|
|
||||||
ser_name.set(span, s.clone());
|
|
||||||
de_name.set(span, s);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(rename(serialize="foo", deserialize="bar"))]`
|
|
||||||
ast::MetaItemKind::List(ref name, ref meta_items) if name == &"rename" => {
|
|
||||||
if let Ok((ser, de)) = get_renames(cx, meta_items) {
|
|
||||||
ser_name.set_opt(ser);
|
|
||||||
de_name.set_opt(de);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(deny_unknown_fields)]`
|
|
||||||
ast::MetaItemKind::Word(ref name) if name == &"deny_unknown_fields" => {
|
|
||||||
deny_unknown_fields.set_true(span);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(bound="D: Serialize")]`
|
|
||||||
ast::MetaItemKind::NameValue(ref name, ref lit) if name == &"bound" => {
|
|
||||||
if let Ok(where_predicates) = parse_lit_into_where(cx, name, lit) {
|
|
||||||
ser_bound.set(span, where_predicates.clone());
|
|
||||||
de_bound.set(span, where_predicates);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(bound(serialize="D: Serialize", deserialize="D: Deserialize"))]`
|
|
||||||
ast::MetaItemKind::List(ref name, ref meta_items) if name == &"bound" => {
|
|
||||||
if let Ok((ser, de)) = get_where_predicates(cx, meta_items) {
|
|
||||||
ser_bound.set_opt(ser);
|
|
||||||
de_bound.set_opt(de);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_ => {
|
|
||||||
cx.span_err(
|
|
||||||
meta_item.span,
|
|
||||||
&format!("unknown serde container attribute `{}`",
|
|
||||||
meta_item_to_string(&meta_item)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Item {
|
|
||||||
name: Name {
|
|
||||||
serialize: ser_name.get().unwrap_or(ident.clone()),
|
|
||||||
deserialize: de_name.get().unwrap_or(ident),
|
|
||||||
},
|
|
||||||
deny_unknown_fields: deny_unknown_fields.get(),
|
|
||||||
ser_bound: ser_bound.get(),
|
|
||||||
de_bound: de_bound.get(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn name(&self) -> &Name {
|
|
||||||
&self.name
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn deny_unknown_fields(&self) -> bool {
|
|
||||||
self.deny_unknown_fields
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn ser_bound(&self) -> Option<&[ast::WherePredicate]> {
|
|
||||||
self.ser_bound.as_ref().map(|vec| &vec[..])
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn de_bound(&self) -> Option<&[ast::WherePredicate]> {
|
|
||||||
self.de_bound.as_ref().map(|vec| &vec[..])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Represents variant attribute information
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct Variant {
|
|
||||||
name: Name,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Variant {
|
|
||||||
pub fn from_ast(cx: &ExtCtxt, variant: &ast::Variant) -> Self {
|
|
||||||
let mut ser_name = Attr::none(cx, "rename");
|
|
||||||
let mut de_name = Attr::none(cx, "rename");
|
|
||||||
|
|
||||||
let ident = variant.node.name.name.as_str();
|
|
||||||
|
|
||||||
for meta_items in variant.node.attrs.iter().filter_map(get_serde_meta_items) {
|
|
||||||
for meta_item in meta_items {
|
|
||||||
let span = meta_item.span;
|
|
||||||
match meta_item.node {
|
|
||||||
// Parse `#[serde(rename="foo")]`
|
|
||||||
ast::MetaItemKind::NameValue(ref name, ref lit) if name == &"rename" => {
|
|
||||||
if let Ok(s) = get_str_from_lit(cx, name, lit) {
|
|
||||||
ser_name.set(span, s.clone());
|
|
||||||
de_name.set(span, s);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(rename(serialize="foo", deserialize="bar"))]`
|
|
||||||
ast::MetaItemKind::List(ref name, ref meta_items) if name == &"rename" => {
|
|
||||||
if let Ok((ser, de)) = get_renames(cx, meta_items) {
|
|
||||||
ser_name.set_opt(ser);
|
|
||||||
de_name.set_opt(de);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_ => {
|
|
||||||
cx.span_err(
|
|
||||||
meta_item.span,
|
|
||||||
&format!("unknown serde variant attribute `{}`",
|
|
||||||
meta_item_to_string(&meta_item)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Variant {
|
|
||||||
name: Name {
|
|
||||||
serialize: ser_name.get().unwrap_or(ident.clone()),
|
|
||||||
deserialize: de_name.get().unwrap_or(ident),
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn name(&self) -> &Name {
|
|
||||||
&self.name
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Represents field attribute information
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub struct Field {
|
|
||||||
name: Name,
|
|
||||||
skip_serializing: bool,
|
|
||||||
skip_deserializing: bool,
|
|
||||||
skip_serializing_if: Option<ast::Path>,
|
|
||||||
default: FieldDefault,
|
|
||||||
serialize_with: Option<ast::Path>,
|
|
||||||
deserialize_with: Option<ast::Path>,
|
|
||||||
ser_bound: Option<Vec<ast::WherePredicate>>,
|
|
||||||
de_bound: Option<Vec<ast::WherePredicate>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Represents the default to use for a field when deserializing.
|
|
||||||
#[derive(Debug, PartialEq)]
|
|
||||||
pub enum FieldDefault {
|
|
||||||
/// Field must always be specified because it does not have a default.
|
|
||||||
None,
|
|
||||||
/// The default is given by `std::default::Default::default()`.
|
|
||||||
Default,
|
|
||||||
/// The default is given by this function.
|
|
||||||
Path(ast::Path),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Field {
|
|
||||||
/// Extract out the `#[serde(...)]` attributes from a struct field.
|
|
||||||
pub fn from_ast(cx: &ExtCtxt,
|
|
||||||
index: usize,
|
|
||||||
field: &ast::StructField) -> Self {
|
|
||||||
let mut ser_name = Attr::none(cx, "rename");
|
|
||||||
let mut de_name = Attr::none(cx, "rename");
|
|
||||||
let mut skip_serializing = BoolAttr::none(cx, "skip_serializing");
|
|
||||||
let mut skip_deserializing = BoolAttr::none(cx, "skip_deserializing");
|
|
||||||
let mut skip_serializing_if = Attr::none(cx, "skip_serializing_if");
|
|
||||||
let mut default = Attr::none(cx, "default");
|
|
||||||
let mut serialize_with = Attr::none(cx, "serialize_with");
|
|
||||||
let mut deserialize_with = Attr::none(cx, "deserialize_with");
|
|
||||||
let mut ser_bound = Attr::none(cx, "bound");
|
|
||||||
let mut de_bound = Attr::none(cx, "bound");
|
|
||||||
|
|
||||||
let ident = match field.ident {
|
|
||||||
Some(ident) => ident.name.as_str(),
|
|
||||||
None => token::intern_and_get_ident(&index.to_string()),
|
|
||||||
};
|
|
||||||
|
|
||||||
for meta_items in field.attrs.iter().filter_map(get_serde_meta_items) {
|
|
||||||
for meta_item in meta_items {
|
|
||||||
let span = meta_item.span;
|
|
||||||
match meta_item.node {
|
|
||||||
// Parse `#[serde(rename="foo")]`
|
|
||||||
ast::MetaItemKind::NameValue(ref name, ref lit) if name == &"rename" => {
|
|
||||||
if let Ok(s) = get_str_from_lit(cx, name, lit) {
|
|
||||||
ser_name.set(span, s.clone());
|
|
||||||
de_name.set(span, s);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(rename(serialize="foo", deserialize="bar"))]`
|
|
||||||
ast::MetaItemKind::List(ref name, ref meta_items) if name == &"rename" => {
|
|
||||||
if let Ok((ser, de)) = get_renames(cx, meta_items) {
|
|
||||||
ser_name.set_opt(ser);
|
|
||||||
de_name.set_opt(de);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(default)]`
|
|
||||||
ast::MetaItemKind::Word(ref name) if name == &"default" => {
|
|
||||||
default.set(span, FieldDefault::Default);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(default="...")]`
|
|
||||||
ast::MetaItemKind::NameValue(ref name, ref lit) if name == &"default" => {
|
|
||||||
if let Ok(path) = parse_lit_into_path(cx, name, lit) {
|
|
||||||
default.set(span, FieldDefault::Path(path));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(skip_serializing)]`
|
|
||||||
ast::MetaItemKind::Word(ref name) if name == &"skip_serializing" => {
|
|
||||||
skip_serializing.set_true(span);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(skip_deserializing)]`
|
|
||||||
ast::MetaItemKind::Word(ref name) if name == &"skip_deserializing" => {
|
|
||||||
skip_deserializing.set_true(span);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(skip_serializing_if="...")]`
|
|
||||||
ast::MetaItemKind::NameValue(ref name, ref lit) if name == &"skip_serializing_if" => {
|
|
||||||
if let Ok(path) = parse_lit_into_path(cx, name, lit) {
|
|
||||||
skip_serializing_if.set(span, path);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(serialize_with="...")]`
|
|
||||||
ast::MetaItemKind::NameValue(ref name, ref lit) if name == &"serialize_with" => {
|
|
||||||
if let Ok(path) = parse_lit_into_path(cx, name, lit) {
|
|
||||||
serialize_with.set(span, path);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(deserialize_with="...")]`
|
|
||||||
ast::MetaItemKind::NameValue(ref name, ref lit) if name == &"deserialize_with" => {
|
|
||||||
if let Ok(path) = parse_lit_into_path(cx, name, lit) {
|
|
||||||
deserialize_with.set(span, path);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(bound="D: Serialize")]`
|
|
||||||
ast::MetaItemKind::NameValue(ref name, ref lit) if name == &"bound" => {
|
|
||||||
if let Ok(where_predicates) = parse_lit_into_where(cx, name, lit) {
|
|
||||||
ser_bound.set(span, where_predicates.clone());
|
|
||||||
de_bound.set(span, where_predicates);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse `#[serde(bound(serialize="D: Serialize", deserialize="D: Deserialize"))]`
|
|
||||||
ast::MetaItemKind::List(ref name, ref meta_items) if name == &"bound" => {
|
|
||||||
if let Ok((ser, de)) = get_where_predicates(cx, meta_items) {
|
|
||||||
ser_bound.set_opt(ser);
|
|
||||||
de_bound.set_opt(de);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_ => {
|
|
||||||
cx.span_err(
|
|
||||||
meta_item.span,
|
|
||||||
&format!("unknown serde field attribute `{}`",
|
|
||||||
meta_item_to_string(&meta_item)));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Is skip_deserializing, initialize the field to Default::default()
|
|
||||||
// unless a different default is specified by `#[serde(default="...")]`
|
|
||||||
if let Some(Spanned { span, .. }) = skip_deserializing.0.value {
|
|
||||||
default.set_if_none(span, FieldDefault::Default);
|
|
||||||
}
|
|
||||||
|
|
||||||
Field {
|
|
||||||
name: Name {
|
|
||||||
serialize: ser_name.get().unwrap_or(ident.clone()),
|
|
||||||
deserialize: de_name.get().unwrap_or(ident),
|
|
||||||
},
|
|
||||||
skip_serializing: skip_serializing.get(),
|
|
||||||
skip_deserializing: skip_deserializing.get(),
|
|
||||||
skip_serializing_if: skip_serializing_if.get(),
|
|
||||||
default: default.get().unwrap_or(FieldDefault::None),
|
|
||||||
serialize_with: serialize_with.get(),
|
|
||||||
deserialize_with: deserialize_with.get(),
|
|
||||||
ser_bound: ser_bound.get(),
|
|
||||||
de_bound: de_bound.get(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn name(&self) -> &Name {
|
|
||||||
&self.name
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn skip_serializing(&self) -> bool {
|
|
||||||
self.skip_serializing
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn skip_deserializing(&self) -> bool {
|
|
||||||
self.skip_deserializing
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn skip_serializing_if(&self) -> Option<&ast::Path> {
|
|
||||||
self.skip_serializing_if.as_ref()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn default(&self) -> &FieldDefault {
|
|
||||||
&self.default
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn serialize_with(&self) -> Option<&ast::Path> {
|
|
||||||
self.serialize_with.as_ref()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn deserialize_with(&self) -> Option<&ast::Path> {
|
|
||||||
self.deserialize_with.as_ref()
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn ser_bound(&self) -> Option<&[ast::WherePredicate]> {
|
|
||||||
self.ser_bound.as_ref().map(|vec| &vec[..])
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn de_bound(&self) -> Option<&[ast::WherePredicate]> {
|
|
||||||
self.de_bound.as_ref().map(|vec| &vec[..])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type SerAndDe<T> = (Option<Spanned<T>>, Option<Spanned<T>>);
|
|
||||||
|
|
||||||
fn get_ser_and_de<T, F>(
|
|
||||||
cx: &ExtCtxt,
|
|
||||||
attribute: &'static str,
|
|
||||||
items: &[ast::NestedMetaItem],
|
|
||||||
f: F
|
|
||||||
) -> Result<SerAndDe<T>, ()>
|
|
||||||
where F: Fn(&ExtCtxt, &str, &ast::Lit) -> Result<T, ()>,
|
|
||||||
{
|
|
||||||
let mut ser_item = Attr::none(cx, attribute);
|
|
||||||
let mut de_item = Attr::none(cx, attribute);
|
|
||||||
|
|
||||||
for item in items {
|
|
||||||
match item.node {
|
|
||||||
ast::NestedMetaItemKind::MetaItem(ref meta_item) => {
|
|
||||||
match meta_item.node {
|
|
||||||
ast::MetaItemKind::NameValue(ref name, ref lit) if name == &"serialize" => {
|
|
||||||
if let Ok(v) = f(cx, name, lit) {
|
|
||||||
ser_item.set(item.span, v);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ast::MetaItemKind::NameValue(ref name, ref lit) if name == &"deserialize" => {
|
|
||||||
if let Ok(v) = f(cx, name, lit) {
|
|
||||||
de_item.set(item.span, v);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_ => {
|
|
||||||
cx.span_err(
|
|
||||||
item.span,
|
|
||||||
&format!("unknown {} attribute `{}`",
|
|
||||||
attribute,
|
|
||||||
meta_item_to_string(meta_item)));
|
|
||||||
|
|
||||||
return Err(());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_ => {
|
|
||||||
cx.span_err(
|
|
||||||
item.span,
|
|
||||||
&format!("unknown {} attribute `{}`",
|
|
||||||
attribute,
|
|
||||||
meta_list_item_to_string(item)));
|
|
||||||
|
|
||||||
return Err(());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok((ser_item.get_spanned(), de_item.get_spanned()))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get_renames(
|
|
||||||
cx: &ExtCtxt,
|
|
||||||
items: &[ast::NestedMetaItem],
|
|
||||||
) -> Result<SerAndDe<InternedString>, ()> {
|
|
||||||
get_ser_and_de(cx, "rename", items, get_str_from_lit)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get_where_predicates(
|
|
||||||
cx: &ExtCtxt,
|
|
||||||
items: &[ast::NestedMetaItem],
|
|
||||||
) -> Result<SerAndDe<Vec<ast::WherePredicate>>, ()> {
|
|
||||||
get_ser_and_de(cx, "bound", items, parse_lit_into_where)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn get_serde_meta_items(attr: &ast::Attribute) -> Option<Vec<ast::MetaItem>> {
|
|
||||||
match attr.node.value.node {
|
|
||||||
ast::MetaItemKind::List(ref name, ref items) if name == &"serde" => {
|
|
||||||
attr::mark_used(attr);
|
|
||||||
Some(items.iter().filter_map(|item| {
|
|
||||||
match item.node {
|
|
||||||
ast::NestedMetaItemKind::MetaItem(ref meta_item) => {
|
|
||||||
Some((*meta_item.clone()).clone())
|
|
||||||
}
|
|
||||||
_ => None,
|
|
||||||
}
|
|
||||||
}).collect())
|
|
||||||
}
|
|
||||||
_ => None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// This syntax folder rewrites tokens to say their spans are coming from a macro context.
|
|
||||||
struct Respanner<'a, 'b: 'a> {
|
|
||||||
cx: &'a ExtCtxt<'b>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, 'b> Folder for Respanner<'a, 'b> {
|
|
||||||
fn fold_tt(&mut self, tt: &TokenTree) -> TokenTree {
|
|
||||||
match *tt {
|
|
||||||
TokenTree::Token(span, ref tok) => {
|
|
||||||
TokenTree::Token(
|
|
||||||
self.new_span(span),
|
|
||||||
self.fold_token(tok.clone())
|
|
||||||
)
|
|
||||||
}
|
|
||||||
TokenTree::Delimited(span, ref delimed) => {
|
|
||||||
TokenTree::Delimited(
|
|
||||||
self.new_span(span),
|
|
||||||
Rc::new(tokenstream::Delimited {
|
|
||||||
delim: delimed.delim,
|
|
||||||
open_span: delimed.open_span,
|
|
||||||
tts: self.fold_tts(&delimed.tts),
|
|
||||||
close_span: delimed.close_span,
|
|
||||||
})
|
|
||||||
)
|
|
||||||
}
|
|
||||||
TokenTree::Sequence(span, ref seq) => {
|
|
||||||
TokenTree::Sequence(
|
|
||||||
self.new_span(span),
|
|
||||||
Rc::new(tokenstream::SequenceRepetition {
|
|
||||||
tts: self.fold_tts(&seq.tts),
|
|
||||||
separator: seq.separator.clone().map(|tok| self.fold_token(tok)),
|
|
||||||
..**seq
|
|
||||||
})
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn new_span(&mut self, span: Span) -> Span {
|
|
||||||
Span {
|
|
||||||
lo: span.lo,
|
|
||||||
hi: span.hi,
|
|
||||||
expn_id: self.cx.backtrace(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn get_str_from_lit(cx: &ExtCtxt, name: &str, lit: &ast::Lit) -> Result<InternedString, ()> {
|
|
||||||
match lit.node {
|
|
||||||
ast::LitKind::Str(ref s, _) => Ok(s.clone()),
|
|
||||||
_ => {
|
|
||||||
cx.span_err(
|
|
||||||
lit.span,
|
|
||||||
&format!("serde annotation `{}` must be a string, not `{}`",
|
|
||||||
name,
|
|
||||||
lit_to_string(lit)));
|
|
||||||
|
|
||||||
Err(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// If we just parse a string literal from an attibute, any syntax errors in the
|
|
||||||
// source will only have spans that point inside the string and not back to the
|
|
||||||
// attribute. So to have better error reporting, we'll first parse the string
|
|
||||||
// into a token tree. Then we'll update those spans to say they're coming from a
|
|
||||||
// macro context that originally came from the attribnute, and then finally
|
|
||||||
// parse them into an expression or where-clause.
|
|
||||||
fn parse_string_via_tts<T, F>(cx: &ExtCtxt, name: &str, string: String, action: F) -> Result<T, ()>
|
|
||||||
where F: for<'a> Fn(&'a mut Parser) -> parse::PResult<'a, T>,
|
|
||||||
{
|
|
||||||
let tts = panictry!(parse::parse_tts_from_source_str(
|
|
||||||
format!("<serde {} expansion>", name),
|
|
||||||
string,
|
|
||||||
cx.cfg(),
|
|
||||||
cx.parse_sess()));
|
|
||||||
|
|
||||||
// Respan the spans to say they are all coming from this macro.
|
|
||||||
let tts = Respanner { cx: cx }.fold_tts(&tts);
|
|
||||||
|
|
||||||
let mut parser = parse::new_parser_from_tts(cx.parse_sess(), cx.cfg(), tts);
|
|
||||||
|
|
||||||
let path = match action(&mut parser) {
|
|
||||||
Ok(path) => path,
|
|
||||||
Err(mut e) => {
|
|
||||||
e.emit();
|
|
||||||
return Err(());
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Make sure to error out if there are trailing characters in the stream.
|
|
||||||
match parser.expect(&token::Eof) {
|
|
||||||
Ok(()) => { }
|
|
||||||
Err(mut e) => {
|
|
||||||
e.emit();
|
|
||||||
return Err(());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(path)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn parse_lit_into_path(cx: &ExtCtxt, name: &str, lit: &ast::Lit) -> Result<ast::Path, ()> {
|
|
||||||
let string = try!(get_str_from_lit(cx, name, lit)).to_string();
|
|
||||||
|
|
||||||
parse_string_via_tts(cx, name, string, |parser| {
|
|
||||||
parser.parse_path(PathStyle::Type)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn parse_lit_into_where(cx: &ExtCtxt, name: &str, lit: &ast::Lit) -> Result<Vec<ast::WherePredicate>, ()> {
|
|
||||||
let string = try!(get_str_from_lit(cx, name, lit));
|
|
||||||
if string.is_empty() {
|
|
||||||
return Ok(Vec::new());
|
|
||||||
}
|
|
||||||
|
|
||||||
let where_string = format!("where {}", string);
|
|
||||||
|
|
||||||
parse_string_via_tts(cx, name, where_string, |parser| {
|
|
||||||
let where_clause = try!(parser.parse_where_clause());
|
|
||||||
Ok(where_clause.predicates)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
@@ -1,19 +0,0 @@
|
|||||||
use std::error;
|
|
||||||
use std::fmt;
|
|
||||||
|
|
||||||
#[derive(Debug)]
|
|
||||||
pub enum Error {
|
|
||||||
UnexpectedItemKind,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl fmt::Display for Error {
|
|
||||||
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
||||||
write!(f, "expected a struct or enum")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl error::Error for Error {
|
|
||||||
fn description(&self) -> &str {
|
|
||||||
"expected a struct or enum"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
#![cfg_attr(feature = "clippy", plugin(clippy))]
|
|
||||||
#![cfg_attr(feature = "clippy", feature(plugin))]
|
|
||||||
#![cfg_attr(not(feature = "with-syntex"), feature(rustc_private, plugin))]
|
|
||||||
|
|
||||||
#[cfg(feature = "with-syntex")]
|
|
||||||
#[macro_use]
|
|
||||||
extern crate syntex_syntax as syntax;
|
|
||||||
#[cfg(feature = "with-syntex")]
|
|
||||||
extern crate syntex_errors as errors;
|
|
||||||
|
|
||||||
#[cfg(not(feature = "with-syntex"))]
|
|
||||||
#[macro_use]
|
|
||||||
extern crate syntax;
|
|
||||||
#[cfg(not(feature = "with-syntex"))]
|
|
||||||
extern crate rustc_errors as errors;
|
|
||||||
|
|
||||||
pub mod ast;
|
|
||||||
pub mod attr;
|
|
||||||
|
|
||||||
mod error;
|
|
||||||
pub use error::Error;
|
|
||||||
@@ -1,2 +0,0 @@
|
|||||||
# To prevent compiletest from seeing two versions of serde
|
|
||||||
paths = ["../serde"]
|
|
||||||
+6
-28
@@ -1,41 +1,19 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "serde_macros"
|
name = "serde_macros"
|
||||||
version = "0.8.5"
|
version = "0.4.2"
|
||||||
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
||||||
license = "MIT/Apache-2.0"
|
license = "MIT/Apache-2.0"
|
||||||
description = "Macros to auto-generate implementations for the serde framework"
|
description = "Macros to auto-generate implementations for the serde framework"
|
||||||
homepage = "https://serde.rs"
|
repository = "https://github.com/erickt/rust-serde"
|
||||||
repository = "https://github.com/serde-rs/serde"
|
|
||||||
documentation = "https://serde.rs/codegen.html"
|
|
||||||
keywords = ["serde", "serialization"]
|
|
||||||
include = ["Cargo.toml", "src/**/*.rs"]
|
|
||||||
|
|
||||||
[lib]
|
[lib]
|
||||||
name = "serde_macros"
|
name = "serde_macros"
|
||||||
plugin = true
|
plugin = true
|
||||||
|
|
||||||
[features]
|
|
||||||
unstable-testing = [
|
|
||||||
"clippy",
|
|
||||||
"serde/unstable-testing",
|
|
||||||
"serde_codegen/unstable-testing"
|
|
||||||
]
|
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
clippy = { version = "^0.*", optional = true }
|
serde_codegen = { version = "*", path = "../serde_codegen", default-features = false, features = ["nightly"] }
|
||||||
serde_codegen = { version = "=0.8.5", default-features = false, features = ["unstable"], path = "../serde_codegen" }
|
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
compiletest_rs = "^0.2.0"
|
num = "*"
|
||||||
fnv = "1.0"
|
rustc-serialize = "*"
|
||||||
rustc-serialize = "^0.3.16"
|
serde = { version = "*", path = "../serde" }
|
||||||
serde = { version = "0.8.5", path = "../serde" }
|
|
||||||
serde_test = { version = "0.8.5", path = "../serde_test" }
|
|
||||||
|
|
||||||
[[test]]
|
|
||||||
name = "test"
|
|
||||||
path = "tests/test.rs"
|
|
||||||
|
|
||||||
[[bench]]
|
|
||||||
name = "bench"
|
|
||||||
path = "benches/bench.rs"
|
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
#![feature(custom_attribute, custom_derive, plugin, test)]
|
#![feature(custom_attribute, custom_derive, plugin, test)]
|
||||||
#![cfg_attr(feature = "clippy", plugin(clippy))]
|
|
||||||
#![plugin(serde_macros)]
|
#![plugin(serde_macros)]
|
||||||
|
|
||||||
|
extern crate num;
|
||||||
extern crate rustc_serialize;
|
extern crate rustc_serialize;
|
||||||
extern crate serde;
|
extern crate serde;
|
||||||
extern crate test;
|
extern crate test;
|
||||||
|
|
||||||
include!("../../testing/benches/bench.rs.in");
|
include!("../../serde_tests/benches/bench.rs.in");
|
||||||
|
|||||||
@@ -0,0 +1,64 @@
|
|||||||
|
#![feature(custom_derive, plugin)]
|
||||||
|
#![plugin(serde_macros)]
|
||||||
|
|
||||||
|
extern crate serde;
|
||||||
|
|
||||||
|
use std::collections::BTreeMap;
|
||||||
|
use serde::json;
|
||||||
|
|
||||||
|
// Creating serializable types with serde is quite simple with `serde_macros`. It implements a
|
||||||
|
// syntax extension that automatically generates the necessary serde trait implementations.
|
||||||
|
#[derive(Debug, Serialize, Deserialize)]
|
||||||
|
struct Point {
|
||||||
|
x: i32,
|
||||||
|
y: i32,
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
let point = Point { x: 5, y: 6 };
|
||||||
|
|
||||||
|
// Serializing to JSON is pretty simple by using the `to_string` method:
|
||||||
|
let serialized_point = json::to_string(&point).unwrap();
|
||||||
|
|
||||||
|
println!("{}", serialized_point);
|
||||||
|
// prints:
|
||||||
|
//
|
||||||
|
// {"x":5,"y":6}
|
||||||
|
|
||||||
|
// There is also support for pretty printing using `to_string_pretty`:
|
||||||
|
let serialized_point = json::to_string_pretty(&point).unwrap();
|
||||||
|
|
||||||
|
println!("{}", serialized_point);
|
||||||
|
// prints:
|
||||||
|
//
|
||||||
|
// {
|
||||||
|
// "x":5,
|
||||||
|
// "y":6
|
||||||
|
// }
|
||||||
|
|
||||||
|
// Values can also be deserialized with the same style using `from_str`:
|
||||||
|
let deserialized_point: Point = json::from_str(&serialized_point).unwrap();
|
||||||
|
|
||||||
|
println!("{:?}", deserialized_point);
|
||||||
|
// prints:
|
||||||
|
//
|
||||||
|
// Point { x: 5, y: 6 }
|
||||||
|
|
||||||
|
// `Point`s aren't the only type that can be serialized to. Because `Point` members have the
|
||||||
|
// same type, they can be also serialized into a map. Also,
|
||||||
|
let deserialized_map: BTreeMap<String, i64> = json::from_str(&serialized_point).unwrap();
|
||||||
|
|
||||||
|
println!("{:?}", deserialized_map);
|
||||||
|
// prints:
|
||||||
|
//
|
||||||
|
// {"x": 5, "y": 6}
|
||||||
|
|
||||||
|
// If you need to accept arbitrary data, you can also deserialize into `json::Value`, which
|
||||||
|
// can represent all JSON values.
|
||||||
|
let deserialized_value: json::Value = json::from_str(&serialized_point).unwrap();
|
||||||
|
|
||||||
|
println!("{:?}", deserialized_value);
|
||||||
|
// prints:
|
||||||
|
//
|
||||||
|
// {"x":5,"y":6}
|
||||||
|
}
|
||||||
@@ -1,12 +1,10 @@
|
|||||||
#![feature(plugin_registrar, rustc_private)]
|
#![feature(plugin_registrar, rustc_private)]
|
||||||
#![cfg_attr(feature = "clippy", feature(plugin))]
|
|
||||||
#![cfg_attr(feature = "clippy", plugin(clippy))]
|
|
||||||
|
|
||||||
extern crate serde_codegen;
|
extern crate serde_codegen;
|
||||||
extern crate rustc_plugin;
|
extern crate rustc;
|
||||||
|
|
||||||
#[plugin_registrar]
|
#[plugin_registrar]
|
||||||
#[doc(hidden)]
|
#[doc(hidden)]
|
||||||
pub fn plugin_registrar(reg: &mut rustc_plugin::Registry) {
|
pub fn plugin_registrar(reg: &mut rustc::plugin::Registry) {
|
||||||
serde_codegen::register(reg);
|
serde_codegen::register(reg);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,32 +0,0 @@
|
|||||||
#![feature(custom_attribute, custom_derive, plugin)]
|
|
||||||
#![plugin(serde_macros)]
|
|
||||||
|
|
||||||
#[derive(Serialize)]
|
|
||||||
struct S {
|
|
||||||
#[serde(rename(serialize="x"))]
|
|
||||||
#[serde(rename(serialize="y"))] //~ ERROR: duplicate serde attribute `rename`
|
|
||||||
a: (),
|
|
||||||
|
|
||||||
#[serde(rename(serialize="x"))]
|
|
||||||
#[serde(rename="y")] //~ ERROR: duplicate serde attribute `rename`
|
|
||||||
b: (),
|
|
||||||
|
|
||||||
#[serde(rename(serialize="x"))]
|
|
||||||
#[serde(rename(deserialize="y"))] // ok
|
|
||||||
c: (),
|
|
||||||
|
|
||||||
#[serde(rename="x")]
|
|
||||||
#[serde(rename(deserialize="y"))] //~ ERROR: duplicate serde attribute `rename`
|
|
||||||
d: (),
|
|
||||||
|
|
||||||
#[serde(rename(serialize="x", serialize="y"))] //~ ERROR: duplicate serde attribute `rename`
|
|
||||||
e: (),
|
|
||||||
|
|
||||||
#[serde(rename="x", serialize="y")] //~ ERROR: unknown serde field attribute `serialize = "y"`
|
|
||||||
f: (),
|
|
||||||
|
|
||||||
#[serde(rename(serialize="x"), rename(serialize="y"))] //~ ERROR: duplicate serde attribute `rename`
|
|
||||||
g: (),
|
|
||||||
}
|
|
||||||
|
|
||||||
fn main() {}
|
|
||||||
@@ -1,30 +0,0 @@
|
|||||||
#![feature(custom_attribute, custom_derive, plugin)]
|
|
||||||
#![plugin(serde_macros)]
|
|
||||||
|
|
||||||
extern crate serde;
|
|
||||||
|
|
||||||
#[derive(Serialize)]
|
|
||||||
#[serde(abc="xyz")] //~ unknown serde container attribute `abc = "xyz"`
|
|
||||||
struct Foo {
|
|
||||||
x: u32,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize)]
|
|
||||||
#[serde(abc="xyz")] //~ unknown serde container attribute `abc = "xyz"`
|
|
||||||
struct Foo {
|
|
||||||
x: u32,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Serialize)]
|
|
||||||
struct Foo {
|
|
||||||
#[serde(abc="xyz")] //~ unknown serde field attribute `abc = "xyz"`
|
|
||||||
x: u32,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Deserialize)]
|
|
||||||
struct Foo {
|
|
||||||
#[serde(abc="xyz")] //~ unknown serde field attribute `abc = "xyz"`
|
|
||||||
x: u32,
|
|
||||||
}
|
|
||||||
|
|
||||||
fn main() { }
|
|
||||||
@@ -1,9 +0,0 @@
|
|||||||
#![feature(custom_attribute, custom_derive, plugin)]
|
|
||||||
#![plugin(serde_macros)]
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
struct Test<'a> {
|
|
||||||
s: &'a str, //~ ERROR: Serde does not support deserializing fields of type &str
|
|
||||||
}
|
|
||||||
|
|
||||||
fn main() {}
|
|
||||||
@@ -1,30 +0,0 @@
|
|||||||
extern crate compiletest_rs as compiletest;
|
|
||||||
|
|
||||||
use std::path::PathBuf;
|
|
||||||
use std::env::var;
|
|
||||||
|
|
||||||
fn run_mode(mode: &'static str) {
|
|
||||||
let mut config = compiletest::default_config();
|
|
||||||
|
|
||||||
let cfg_mode = mode.parse().ok().expect("Invalid mode");
|
|
||||||
|
|
||||||
config.target_rustcflags = Some("-L target/debug/ -L target/debug/deps/".to_owned());
|
|
||||||
if let Ok(name) = var::<&str>("TESTNAME") {
|
|
||||||
let s : String = name.to_owned();
|
|
||||||
config.filter = Some(s)
|
|
||||||
}
|
|
||||||
config.mode = cfg_mode;
|
|
||||||
config.src_base = PathBuf::from(format!("tests/{}", mode));
|
|
||||||
|
|
||||||
compiletest::run_tests(&config);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn compile_fail() {
|
|
||||||
run_mode("compile-fail");
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn run_pass() {
|
|
||||||
run_mode("run-pass");
|
|
||||||
}
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
#![feature(custom_derive, plugin)]
|
|
||||||
#![plugin(serde_macros, clippy)]
|
|
||||||
|
|
||||||
#![deny(identity_op)]
|
|
||||||
|
|
||||||
// The derived implementation uses 0+1 to add up the number of fields
|
|
||||||
// serialized, which Clippy warns about. If the expansion info is registered
|
|
||||||
// correctly, the Clippy lint is not triggered.
|
|
||||||
#[derive(Serialize)]
|
|
||||||
struct A { b: u8 }
|
|
||||||
|
|
||||||
fn main() {}
|
|
||||||
@@ -1,8 +1,7 @@
|
|||||||
#![feature(test, custom_attribute, custom_derive, plugin)]
|
#![feature(test, custom_attribute, custom_derive, plugin)]
|
||||||
#![plugin(serde_macros)]
|
#![plugin(serde_macros)]
|
||||||
|
|
||||||
|
extern crate serde;
|
||||||
extern crate test;
|
extern crate test;
|
||||||
|
|
||||||
include!("../../testing/tests/test.rs.in");
|
include!("../../serde_tests/tests/test.rs.in");
|
||||||
|
|
||||||
mod compile_tests;
|
|
||||||
|
|||||||
@@ -1,15 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "serde_test"
|
|
||||||
version = "0.8.5"
|
|
||||||
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
|
||||||
license = "MIT/Apache-2.0"
|
|
||||||
description = "Token De/Serializer for testing De/Serialize implementations"
|
|
||||||
homepage = "https://serde.rs"
|
|
||||||
repository = "https://github.com/serde-rs/serde"
|
|
||||||
documentation = "https://docs.serde.rs/serde_test/"
|
|
||||||
readme = "../README.md"
|
|
||||||
keywords = ["serde", "serialization"]
|
|
||||||
include = ["Cargo.toml", "src/**/*.rs"]
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
serde = { version = "0.8.5", path = "../serde" }
|
|
||||||
@@ -1,54 +0,0 @@
|
|||||||
use serde::{Serialize, Deserialize};
|
|
||||||
|
|
||||||
use de::Deserializer;
|
|
||||||
use error::Error;
|
|
||||||
use ser::Serializer;
|
|
||||||
use token::Token;
|
|
||||||
|
|
||||||
use std::fmt::Debug;
|
|
||||||
|
|
||||||
pub fn assert_tokens<T>(value: &T, tokens: &[Token<'static>])
|
|
||||||
where T: Serialize + Deserialize + PartialEq + Debug,
|
|
||||||
{
|
|
||||||
assert_ser_tokens(value, tokens);
|
|
||||||
assert_de_tokens(value, tokens);
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn assert_ser_tokens<T>(value: &T, tokens: &[Token])
|
|
||||||
where T: Serialize,
|
|
||||||
{
|
|
||||||
let mut ser = Serializer::new(tokens.iter());
|
|
||||||
assert_eq!(Serialize::serialize(value, &mut ser), Ok(()));
|
|
||||||
assert_eq!(ser.next_token(), None);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Expect an error serializing `T`.
|
|
||||||
pub fn assert_ser_tokens_error<T>(value: &T, tokens: &[Token], error: Error)
|
|
||||||
where T: Serialize + PartialEq + Debug,
|
|
||||||
{
|
|
||||||
let mut ser = Serializer::new(tokens.iter());
|
|
||||||
let v: Result<(), Error> = Serialize::serialize(value, &mut ser);
|
|
||||||
assert_eq!(v.as_ref(), Err(&error));
|
|
||||||
assert_eq!(ser.next_token(), None);
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn assert_de_tokens<T>(value: &T, tokens: &[Token<'static>])
|
|
||||||
where T: Deserialize + PartialEq + Debug,
|
|
||||||
{
|
|
||||||
let mut de = Deserializer::new(tokens.to_vec().into_iter());
|
|
||||||
let v: Result<T, Error> = Deserialize::deserialize(&mut de);
|
|
||||||
assert_eq!(v.as_ref(), Ok(value));
|
|
||||||
assert_eq!(de.next_token(), None);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Expect an error deserializing tokens into a `T`.
|
|
||||||
pub fn assert_de_tokens_error<T>(tokens: &[Token<'static>], error: Error)
|
|
||||||
where T: Deserialize + PartialEq + Debug,
|
|
||||||
{
|
|
||||||
let mut de = Deserializer::new(tokens.to_vec().into_iter());
|
|
||||||
let v: Result<T, Error> = Deserialize::deserialize(&mut de);
|
|
||||||
assert_eq!(v, Err(error));
|
|
||||||
// There may be one token left if a peek caused the error
|
|
||||||
de.next_token();
|
|
||||||
assert_eq!(de.next_token(), None);
|
|
||||||
}
|
|
||||||
@@ -1,946 +0,0 @@
|
|||||||
use std::iter;
|
|
||||||
|
|
||||||
use serde::de::{
|
|
||||||
self,
|
|
||||||
Deserialize,
|
|
||||||
EnumVisitor,
|
|
||||||
MapVisitor,
|
|
||||||
SeqVisitor,
|
|
||||||
VariantVisitor,
|
|
||||||
Visitor,
|
|
||||||
};
|
|
||||||
|
|
||||||
use error::Error;
|
|
||||||
use token::Token;
|
|
||||||
|
|
||||||
pub struct Deserializer<I>
|
|
||||||
where I: Iterator<Item=Token<'static>>,
|
|
||||||
{
|
|
||||||
tokens: iter::Peekable<I>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<I> Deserializer<I>
|
|
||||||
where I: Iterator<Item=Token<'static>>,
|
|
||||||
{
|
|
||||||
pub fn new(tokens: I) -> Deserializer<I> {
|
|
||||||
Deserializer {
|
|
||||||
tokens: tokens.peekable(),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn next_token(&mut self) -> Option<Token<'static>> {
|
|
||||||
self.tokens.next()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_seq<V>(&mut self, len: Option<usize>, mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
visitor.visit_seq(DeserializerSeqVisitor {
|
|
||||||
de: self,
|
|
||||||
len: len,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_array<V>(&mut self, len: usize, mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
visitor.visit_seq(DeserializerArrayVisitor {
|
|
||||||
de: self,
|
|
||||||
len: len,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_tuple<V>(&mut self, len: usize, mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
visitor.visit_seq(DeserializerTupleVisitor {
|
|
||||||
de: self,
|
|
||||||
len: len,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_tuple_struct<V>(&mut self, len: usize, mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
visitor.visit_seq(DeserializerTupleStructVisitor {
|
|
||||||
de: self,
|
|
||||||
len: len,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_variant_seq<V>(&mut self, len: Option<usize>, mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
visitor.visit_seq(DeserializerVariantSeqVisitor {
|
|
||||||
de: self,
|
|
||||||
len: len,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_map<V>(&mut self, len: Option<usize>, mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
visitor.visit_map(DeserializerMapVisitor {
|
|
||||||
de: self,
|
|
||||||
len: len,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_struct<V>(&mut self, len: Option<usize>, mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
visitor.visit_map(DeserializerStructVisitor {
|
|
||||||
de: self,
|
|
||||||
len: len,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_variant_map<V>(&mut self, len: Option<usize>, mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
visitor.visit_map(DeserializerVariantMapVisitor {
|
|
||||||
de: self,
|
|
||||||
len: len,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<I> de::Deserializer for Deserializer<I>
|
|
||||||
where I: Iterator<Item=Token<'static>>,
|
|
||||||
{
|
|
||||||
type Error = Error;
|
|
||||||
|
|
||||||
fn deserialize_seq<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_struct_field<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_map<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_unit<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_bytes<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_ignored_any<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_string<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_str<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_char<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_i64<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_i32<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_i16<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_i8<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_u64<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_u32<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_u16<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_u8<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_f32<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_f64<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_bool<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_usize<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_isize<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn deserialize<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
match self.tokens.next() {
|
|
||||||
Some(Token::Bool(v)) => visitor.visit_bool(v),
|
|
||||||
Some(Token::Isize(v)) => visitor.visit_isize(v),
|
|
||||||
Some(Token::I8(v)) => visitor.visit_i8(v),
|
|
||||||
Some(Token::I16(v)) => visitor.visit_i16(v),
|
|
||||||
Some(Token::I32(v)) => visitor.visit_i32(v),
|
|
||||||
Some(Token::I64(v)) => visitor.visit_i64(v),
|
|
||||||
Some(Token::Usize(v)) => visitor.visit_usize(v),
|
|
||||||
Some(Token::U8(v)) => visitor.visit_u8(v),
|
|
||||||
Some(Token::U16(v)) => visitor.visit_u16(v),
|
|
||||||
Some(Token::U32(v)) => visitor.visit_u32(v),
|
|
||||||
Some(Token::U64(v)) => visitor.visit_u64(v),
|
|
||||||
Some(Token::F32(v)) => visitor.visit_f32(v),
|
|
||||||
Some(Token::F64(v)) => visitor.visit_f64(v),
|
|
||||||
Some(Token::Char(v)) => visitor.visit_char(v),
|
|
||||||
Some(Token::Str(v)) => visitor.visit_str(v),
|
|
||||||
Some(Token::String(v)) => visitor.visit_string(v),
|
|
||||||
Some(Token::Bytes(v)) => visitor.visit_bytes(v),
|
|
||||||
Some(Token::Option(false)) => visitor.visit_none(),
|
|
||||||
Some(Token::Option(true)) => visitor.visit_some(self),
|
|
||||||
Some(Token::Unit) => visitor.visit_unit(),
|
|
||||||
Some(Token::UnitStruct(name)) => visitor.visit_unit_struct(name),
|
|
||||||
Some(Token::SeqStart(len)) => {
|
|
||||||
self.visit_seq(len, visitor)
|
|
||||||
}
|
|
||||||
Some(Token::SeqArrayStart(len))| Some(Token::TupleStructStart(_, len)) => {
|
|
||||||
self.visit_seq(Some(len), visitor)
|
|
||||||
}
|
|
||||||
Some(Token::MapStart(len)) => {
|
|
||||||
self.visit_map(len, visitor)
|
|
||||||
}
|
|
||||||
Some(Token::StructStart(_, len)) => {
|
|
||||||
self.visit_map(Some(len), visitor)
|
|
||||||
}
|
|
||||||
Some(token) => Err(Error::UnexpectedToken(token)),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Hook into `Option` deserializing so we can treat `Unit` as a
|
|
||||||
/// `None`, or a regular value as `Some(value)`.
|
|
||||||
fn deserialize_option<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
match self.tokens.peek() {
|
|
||||||
Some(&Token::Option(false)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
visitor.visit_none()
|
|
||||||
}
|
|
||||||
Some(&Token::Option(true)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
visitor.visit_some(self)
|
|
||||||
}
|
|
||||||
Some(&Token::Unit) => {
|
|
||||||
self.tokens.next();
|
|
||||||
visitor.visit_none()
|
|
||||||
}
|
|
||||||
Some(_) => visitor.visit_some(self),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn deserialize_enum<V>(&mut self,
|
|
||||||
name: &str,
|
|
||||||
_variants: &'static [&'static str],
|
|
||||||
mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: EnumVisitor,
|
|
||||||
{
|
|
||||||
match self.tokens.peek() {
|
|
||||||
Some(&Token::EnumStart(n)) if name == n => {
|
|
||||||
self.tokens.next();
|
|
||||||
|
|
||||||
visitor.visit(DeserializerVariantVisitor {
|
|
||||||
de: self,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
Some(&Token::EnumUnit(n, _))
|
|
||||||
| Some(&Token::EnumNewType(n, _))
|
|
||||||
| Some(&Token::EnumSeqStart(n, _, _))
|
|
||||||
| Some(&Token::EnumMapStart(n, _, _)) if name == n => {
|
|
||||||
visitor.visit(DeserializerVariantVisitor {
|
|
||||||
de: self,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
Some(_) => {
|
|
||||||
let token = self.tokens.next().unwrap();
|
|
||||||
Err(Error::UnexpectedToken(token))
|
|
||||||
}
|
|
||||||
None => { return Err(Error::EndOfStream); }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn deserialize_unit_struct<V>(&mut self, name: &str, mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
match self.tokens.peek() {
|
|
||||||
Some(&Token::UnitStruct(n)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
if name == n {
|
|
||||||
visitor.visit_unit()
|
|
||||||
} else {
|
|
||||||
Err(Error::InvalidName(n))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some(_) => self.deserialize(visitor),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn deserialize_newtype_struct<V>(&mut self,
|
|
||||||
name: &str,
|
|
||||||
mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
match self.tokens.peek() {
|
|
||||||
Some(&Token::StructNewType(n)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
if name == n {
|
|
||||||
visitor.visit_newtype_struct(self)
|
|
||||||
} else {
|
|
||||||
Err(Error::InvalidName(n))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some(_) => self.deserialize(visitor),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn deserialize_seq_fixed_size<V>(&mut self,
|
|
||||||
len: usize,
|
|
||||||
visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
match self.tokens.peek() {
|
|
||||||
Some(&Token::SeqArrayStart(_)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
self.visit_array(len, visitor)
|
|
||||||
}
|
|
||||||
Some(_) => self.deserialize(visitor),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn deserialize_tuple<V>(&mut self,
|
|
||||||
len: usize,
|
|
||||||
mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
match self.tokens.peek() {
|
|
||||||
Some(&Token::Unit) => {
|
|
||||||
self.tokens.next();
|
|
||||||
visitor.visit_unit()
|
|
||||||
}
|
|
||||||
Some(&Token::UnitStruct(_)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
visitor.visit_unit()
|
|
||||||
}
|
|
||||||
Some(&Token::SeqStart(_)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
self.visit_seq(Some(len), visitor)
|
|
||||||
}
|
|
||||||
Some(&Token::SeqArrayStart(_)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
self.visit_array(len, visitor)
|
|
||||||
}
|
|
||||||
Some(&Token::TupleStart(_)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
self.visit_tuple(len, visitor)
|
|
||||||
}
|
|
||||||
Some(&Token::TupleStructStart(_, _)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
self.visit_tuple_struct(len, visitor)
|
|
||||||
}
|
|
||||||
Some(_) => self.deserialize(visitor),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn deserialize_tuple_struct<V>(&mut self,
|
|
||||||
name: &str,
|
|
||||||
len: usize,
|
|
||||||
mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
match self.tokens.peek() {
|
|
||||||
Some(&Token::Unit) => {
|
|
||||||
self.tokens.next();
|
|
||||||
visitor.visit_unit()
|
|
||||||
}
|
|
||||||
Some(&Token::UnitStruct(n)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
if name == n {
|
|
||||||
visitor.visit_unit()
|
|
||||||
} else {
|
|
||||||
Err(Error::InvalidName(n))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some(&Token::SeqStart(_)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
self.visit_seq(Some(len), visitor)
|
|
||||||
}
|
|
||||||
Some(&Token::SeqArrayStart(_)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
self.visit_array(len, visitor)
|
|
||||||
}
|
|
||||||
Some(&Token::TupleStart(_)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
self.visit_tuple(len, visitor)
|
|
||||||
}
|
|
||||||
Some(&Token::TupleStructStart(n, _)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
if name == n {
|
|
||||||
self.visit_tuple_struct(len, visitor)
|
|
||||||
} else {
|
|
||||||
Err(Error::InvalidName(n))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some(_) => self.deserialize(visitor),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn deserialize_struct<V>(&mut self,
|
|
||||||
name: &str,
|
|
||||||
fields: &'static [&'static str],
|
|
||||||
visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
match self.tokens.peek() {
|
|
||||||
Some(&Token::StructStart(n, _)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
if name == n {
|
|
||||||
self.visit_struct(Some(fields.len()), visitor)
|
|
||||||
} else {
|
|
||||||
Err(Error::InvalidName(n))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some(&Token::MapStart(_)) => {
|
|
||||||
self.tokens.next();
|
|
||||||
self.visit_map(Some(fields.len()), visitor)
|
|
||||||
}
|
|
||||||
Some(_) => self.deserialize(visitor),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
struct DeserializerSeqVisitor<'a, I: 'a> where I: Iterator<Item=Token<'static>> {
|
|
||||||
de: &'a mut Deserializer<I>,
|
|
||||||
len: Option<usize>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, I> SeqVisitor for DeserializerSeqVisitor<'a, I>
|
|
||||||
where I: Iterator<Item=Token<'static>>,
|
|
||||||
{
|
|
||||||
type Error = Error;
|
|
||||||
|
|
||||||
fn visit<T>(&mut self) -> Result<Option<T>, Error>
|
|
||||||
where T: Deserialize,
|
|
||||||
{
|
|
||||||
match self.de.tokens.peek() {
|
|
||||||
Some(&Token::SeqSep) => {
|
|
||||||
self.de.tokens.next();
|
|
||||||
self.len = self.len.map(|len| len - 1);
|
|
||||||
Ok(Some(try!(Deserialize::deserialize(self.de))))
|
|
||||||
}
|
|
||||||
Some(&Token::SeqEnd) => Ok(None),
|
|
||||||
Some(_) => {
|
|
||||||
let token = self.de.tokens.next().unwrap();
|
|
||||||
Err(Error::UnexpectedToken(token))
|
|
||||||
}
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
|
||||||
//assert_eq!(self.len.unwrap_or(0), 0);
|
|
||||||
match self.de.tokens.next() {
|
|
||||||
Some(Token::SeqEnd) => Ok(()),
|
|
||||||
Some(token) => Err(Error::UnexpectedToken(token)),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn size_hint(&self) -> (usize, Option<usize>) {
|
|
||||||
let len = self.len.unwrap_or(0);
|
|
||||||
(len, self.len)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
struct DeserializerArrayVisitor<'a, I: 'a> where I: Iterator<Item=Token<'static>> {
|
|
||||||
de: &'a mut Deserializer<I>,
|
|
||||||
len: usize,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, I> SeqVisitor for DeserializerArrayVisitor<'a, I>
|
|
||||||
where I: Iterator<Item=Token<'static>>,
|
|
||||||
{
|
|
||||||
type Error = Error;
|
|
||||||
|
|
||||||
fn visit<T>(&mut self) -> Result<Option<T>, Error>
|
|
||||||
where T: Deserialize,
|
|
||||||
{
|
|
||||||
match self.de.tokens.peek() {
|
|
||||||
Some(&Token::SeqSep) => {
|
|
||||||
self.de.tokens.next();
|
|
||||||
self.len -= 1;
|
|
||||||
Ok(Some(try!(Deserialize::deserialize(self.de))))
|
|
||||||
}
|
|
||||||
Some(&Token::SeqEnd) => Ok(None),
|
|
||||||
Some(_) => {
|
|
||||||
let token = self.de.tokens.next().unwrap();
|
|
||||||
Err(Error::UnexpectedToken(token))
|
|
||||||
}
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.len, 0);
|
|
||||||
match self.de.tokens.next() {
|
|
||||||
Some(Token::SeqEnd) => Ok(()),
|
|
||||||
Some(token) => Err(Error::UnexpectedToken(token)),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn size_hint(&self) -> (usize, Option<usize>) {
|
|
||||||
(self.len, Some(self.len))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
struct DeserializerTupleVisitor<'a, I: 'a> where I: Iterator<Item=Token<'static>> {
|
|
||||||
de: &'a mut Deserializer<I>,
|
|
||||||
len: usize,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, I> SeqVisitor for DeserializerTupleVisitor<'a, I>
|
|
||||||
where I: Iterator<Item=Token<'static>>,
|
|
||||||
{
|
|
||||||
type Error = Error;
|
|
||||||
|
|
||||||
fn visit<T>(&mut self) -> Result<Option<T>, Error>
|
|
||||||
where T: Deserialize,
|
|
||||||
{
|
|
||||||
match self.de.tokens.peek() {
|
|
||||||
Some(&Token::TupleSep) => {
|
|
||||||
self.de.tokens.next();
|
|
||||||
self.len -= 1;
|
|
||||||
Ok(Some(try!(Deserialize::deserialize(self.de))))
|
|
||||||
}
|
|
||||||
Some(&Token::TupleEnd) => Ok(None),
|
|
||||||
Some(_) => {
|
|
||||||
let token = self.de.tokens.next().unwrap();
|
|
||||||
Err(Error::UnexpectedToken(token))
|
|
||||||
}
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.len, 0);
|
|
||||||
match self.de.tokens.next() {
|
|
||||||
Some(Token::TupleEnd) => Ok(()),
|
|
||||||
Some(token) => Err(Error::UnexpectedToken(token)),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn size_hint(&self) -> (usize, Option<usize>) {
|
|
||||||
(self.len, Some(self.len))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
struct DeserializerTupleStructVisitor<'a, I: 'a> where I: Iterator<Item=Token<'static>> {
|
|
||||||
de: &'a mut Deserializer<I>,
|
|
||||||
len: usize,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, I> SeqVisitor for DeserializerTupleStructVisitor<'a, I>
|
|
||||||
where I: Iterator<Item=Token<'static>>,
|
|
||||||
{
|
|
||||||
type Error = Error;
|
|
||||||
|
|
||||||
fn visit<T>(&mut self) -> Result<Option<T>, Error>
|
|
||||||
where T: Deserialize,
|
|
||||||
{
|
|
||||||
match self.de.tokens.peek() {
|
|
||||||
Some(&Token::TupleStructSep) => {
|
|
||||||
self.de.tokens.next();
|
|
||||||
self.len -= 1;
|
|
||||||
Ok(Some(try!(Deserialize::deserialize(self.de))))
|
|
||||||
}
|
|
||||||
Some(&Token::TupleStructEnd) => Ok(None),
|
|
||||||
Some(_) => {
|
|
||||||
let token = self.de.tokens.next().unwrap();
|
|
||||||
Err(Error::UnexpectedToken(token))
|
|
||||||
}
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.len, 0);
|
|
||||||
match self.de.tokens.next() {
|
|
||||||
Some(Token::TupleStructEnd) => Ok(()),
|
|
||||||
Some(token) => Err(Error::UnexpectedToken(token)),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn size_hint(&self) -> (usize, Option<usize>) {
|
|
||||||
(self.len, Some(self.len))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
struct DeserializerVariantSeqVisitor<'a, I: 'a> where I: Iterator<Item=Token<'static>> {
|
|
||||||
de: &'a mut Deserializer<I>,
|
|
||||||
len: Option<usize>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, I> SeqVisitor for DeserializerVariantSeqVisitor<'a, I>
|
|
||||||
where I: Iterator<Item=Token<'static>>,
|
|
||||||
{
|
|
||||||
type Error = Error;
|
|
||||||
|
|
||||||
fn visit<T>(&mut self) -> Result<Option<T>, Error>
|
|
||||||
where T: Deserialize,
|
|
||||||
{
|
|
||||||
match self.de.tokens.peek() {
|
|
||||||
Some(&Token::EnumSeqSep) => {
|
|
||||||
self.de.tokens.next();
|
|
||||||
self.len = self.len.map(|len| len - 1);
|
|
||||||
Ok(Some(try!(Deserialize::deserialize(self.de))))
|
|
||||||
}
|
|
||||||
Some(&Token::EnumSeqEnd) => Ok(None),
|
|
||||||
Some(_) => {
|
|
||||||
let token = self.de.tokens.next().unwrap();
|
|
||||||
Err(Error::UnexpectedToken(token))
|
|
||||||
}
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
|
||||||
//assert_eq!(self.len.unwrap_or(0), 0);
|
|
||||||
match self.de.tokens.next() {
|
|
||||||
Some(Token::EnumSeqEnd) => Ok(()),
|
|
||||||
Some(token) => Err(Error::UnexpectedToken(token)),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn size_hint(&self) -> (usize, Option<usize>) {
|
|
||||||
let len = self.len.unwrap_or(0);
|
|
||||||
(len, self.len)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
struct DeserializerMapVisitor<'a, I: 'a> where I: Iterator<Item=Token<'static>> {
|
|
||||||
de: &'a mut Deserializer<I>,
|
|
||||||
len: Option<usize>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, I> MapVisitor for DeserializerMapVisitor<'a, I>
|
|
||||||
where I: Iterator<Item=Token<'static>>,
|
|
||||||
{
|
|
||||||
type Error = Error;
|
|
||||||
|
|
||||||
fn visit_key<K>(&mut self) -> Result<Option<K>, Error>
|
|
||||||
where K: Deserialize,
|
|
||||||
{
|
|
||||||
match self.de.tokens.peek() {
|
|
||||||
Some(&Token::MapSep) => {
|
|
||||||
self.de.tokens.next();
|
|
||||||
self.len = self.len.map(|len| if len > 0 { len - 1} else { 0 });
|
|
||||||
Ok(Some(try!(Deserialize::deserialize(self.de))))
|
|
||||||
}
|
|
||||||
Some(&Token::MapEnd) => Ok(None),
|
|
||||||
Some(_) => {
|
|
||||||
let token = self.de.tokens.next().unwrap();
|
|
||||||
Err(Error::UnexpectedToken(token))
|
|
||||||
}
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_value<V>(&mut self) -> Result<V, Error>
|
|
||||||
where V: Deserialize,
|
|
||||||
{
|
|
||||||
Ok(try!(Deserialize::deserialize(self.de)))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
|
||||||
//assert_eq!(self.len.unwrap_or(0), 0);
|
|
||||||
match self.de.tokens.next() {
|
|
||||||
Some(Token::MapEnd) => Ok(()),
|
|
||||||
Some(token) => Err(Error::UnexpectedToken(token)),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn size_hint(&self) -> (usize, Option<usize>) {
|
|
||||||
let len = self.len.unwrap_or(0);
|
|
||||||
(len, self.len)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
struct DeserializerStructVisitor<'a, I: 'a> where I: Iterator<Item=Token<'static>> {
|
|
||||||
de: &'a mut Deserializer<I>,
|
|
||||||
len: Option<usize>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, I> MapVisitor for DeserializerStructVisitor<'a, I>
|
|
||||||
where I: Iterator<Item=Token<'static>>,
|
|
||||||
{
|
|
||||||
type Error = Error;
|
|
||||||
|
|
||||||
fn visit_key<K>(&mut self) -> Result<Option<K>, Error>
|
|
||||||
where K: Deserialize,
|
|
||||||
{
|
|
||||||
match self.de.tokens.peek() {
|
|
||||||
Some(&Token::StructSep) => {
|
|
||||||
self.de.tokens.next();
|
|
||||||
self.len = self.len.map(|len| if len > 0 { len - 1} else { 0 });
|
|
||||||
Ok(Some(try!(Deserialize::deserialize(self.de))))
|
|
||||||
}
|
|
||||||
Some(&Token::StructEnd) => Ok(None),
|
|
||||||
Some(_) => {
|
|
||||||
let token = self.de.tokens.next().unwrap();
|
|
||||||
Err(Error::UnexpectedToken(token))
|
|
||||||
}
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_value<V>(&mut self) -> Result<V, Error>
|
|
||||||
where V: Deserialize,
|
|
||||||
{
|
|
||||||
Ok(try!(Deserialize::deserialize(self.de)))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
|
||||||
//assert_eq!(self.len.unwrap_or(0), 0);
|
|
||||||
match self.de.tokens.next() {
|
|
||||||
Some(Token::StructEnd) => Ok(()),
|
|
||||||
Some(token) => Err(Error::UnexpectedToken(token)),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn size_hint(&self) -> (usize, Option<usize>) {
|
|
||||||
let len = self.len.unwrap_or(0);
|
|
||||||
(len, self.len)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
struct DeserializerVariantVisitor<'a, I: 'a> where I: Iterator<Item=Token<'static>> {
|
|
||||||
de: &'a mut Deserializer<I>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, I> VariantVisitor for DeserializerVariantVisitor<'a, I>
|
|
||||||
where I: Iterator<Item=Token<'static>>,
|
|
||||||
{
|
|
||||||
type Error = Error;
|
|
||||||
|
|
||||||
fn visit_variant<V>(&mut self) -> Result<V, Error>
|
|
||||||
where V: Deserialize,
|
|
||||||
{
|
|
||||||
match self.de.tokens.peek() {
|
|
||||||
Some(&Token::EnumUnit(_, v))
|
|
||||||
| Some(&Token::EnumNewType(_, v))
|
|
||||||
| Some(&Token::EnumSeqStart(_, v, _))
|
|
||||||
| Some(&Token::EnumMapStart(_, v, _)) => {
|
|
||||||
let mut de = de::value::ValueDeserializer::<Error>::into_deserializer(v);
|
|
||||||
let value = try!(Deserialize::deserialize(&mut de));
|
|
||||||
Ok(value)
|
|
||||||
}
|
|
||||||
Some(_) => {
|
|
||||||
Deserialize::deserialize(self.de)
|
|
||||||
}
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_unit(&mut self) -> Result<(), Error> {
|
|
||||||
match self.de.tokens.peek() {
|
|
||||||
Some(&Token::EnumUnit(_, _)) => {
|
|
||||||
self.de.tokens.next();
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
Some(_) => {
|
|
||||||
Deserialize::deserialize(self.de)
|
|
||||||
}
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_newtype<T>(&mut self) -> Result<T, Self::Error>
|
|
||||||
where T: Deserialize,
|
|
||||||
{
|
|
||||||
match self.de.tokens.peek() {
|
|
||||||
Some(&Token::EnumNewType(_, _)) => {
|
|
||||||
self.de.tokens.next();
|
|
||||||
Deserialize::deserialize(self.de)
|
|
||||||
}
|
|
||||||
Some(_) => {
|
|
||||||
Deserialize::deserialize(self.de)
|
|
||||||
}
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_tuple<V>(&mut self,
|
|
||||||
len: usize,
|
|
||||||
visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
match self.de.tokens.peek() {
|
|
||||||
Some(&Token::EnumSeqStart(_, _, enum_len)) => {
|
|
||||||
let token = self.de.tokens.next().unwrap();
|
|
||||||
|
|
||||||
if len == enum_len {
|
|
||||||
self.de.visit_variant_seq(Some(len), visitor)
|
|
||||||
} else {
|
|
||||||
Err(Error::UnexpectedToken(token))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some(&Token::SeqStart(Some(enum_len))) => {
|
|
||||||
let token = self.de.tokens.next().unwrap();
|
|
||||||
|
|
||||||
if len == enum_len {
|
|
||||||
self.de.visit_seq(Some(len), visitor)
|
|
||||||
} else {
|
|
||||||
Err(Error::UnexpectedToken(token))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some(_) => {
|
|
||||||
Deserialize::deserialize(self.de)
|
|
||||||
}
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_struct<V>(&mut self,
|
|
||||||
fields: &'static [&'static str],
|
|
||||||
visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: Visitor,
|
|
||||||
{
|
|
||||||
match self.de.tokens.peek() {
|
|
||||||
Some(&Token::EnumMapStart(_, _, enum_len)) => {
|
|
||||||
let token = self.de.tokens.next().unwrap();
|
|
||||||
|
|
||||||
if fields.len() == enum_len {
|
|
||||||
self.de.visit_variant_map(Some(fields.len()), visitor)
|
|
||||||
} else {
|
|
||||||
Err(Error::UnexpectedToken(token))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some(&Token::MapStart(Some(enum_len))) => {
|
|
||||||
let token = self.de.tokens.next().unwrap();
|
|
||||||
|
|
||||||
if fields.len() == enum_len {
|
|
||||||
self.de.visit_map(Some(fields.len()), visitor)
|
|
||||||
} else {
|
|
||||||
Err(Error::UnexpectedToken(token))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some(_) => {
|
|
||||||
Deserialize::deserialize(self.de)
|
|
||||||
}
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
struct DeserializerVariantMapVisitor<'a, I: 'a> where I: Iterator<Item=Token<'static>> {
|
|
||||||
de: &'a mut Deserializer<I>,
|
|
||||||
len: Option<usize>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, I> MapVisitor for DeserializerVariantMapVisitor<'a, I>
|
|
||||||
where I: Iterator<Item=Token<'static>>,
|
|
||||||
{
|
|
||||||
type Error = Error;
|
|
||||||
|
|
||||||
fn visit_key<K>(&mut self) -> Result<Option<K>, Error>
|
|
||||||
where K: Deserialize,
|
|
||||||
{
|
|
||||||
match self.de.tokens.peek() {
|
|
||||||
Some(&Token::EnumMapSep) => {
|
|
||||||
self.de.tokens.next();
|
|
||||||
self.len = self.len.map(|len| if len > 0 { len - 1} else { 0 });
|
|
||||||
Ok(Some(try!(Deserialize::deserialize(self.de))))
|
|
||||||
}
|
|
||||||
Some(&Token::EnumMapEnd) => Ok(None),
|
|
||||||
Some(_) => {
|
|
||||||
let token = self.de.tokens.next().unwrap();
|
|
||||||
Err(Error::UnexpectedToken(token))
|
|
||||||
}
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_value<V>(&mut self) -> Result<V, Error>
|
|
||||||
where V: Deserialize,
|
|
||||||
{
|
|
||||||
Ok(try!(Deserialize::deserialize(self.de)))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
|
||||||
//assert_eq!(self.len.unwrap_or(0), 0);
|
|
||||||
match self.de.tokens.next() {
|
|
||||||
Some(Token::EnumMapEnd) => Ok(()),
|
|
||||||
Some(token) => Err(Error::UnexpectedToken(token)),
|
|
||||||
None => Err(Error::EndOfStream),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn size_hint(&self) -> (usize, Option<usize>) {
|
|
||||||
let len = self.len.unwrap_or(0);
|
|
||||||
(len, self.len)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,87 +0,0 @@
|
|||||||
use std::{error, fmt};
|
|
||||||
|
|
||||||
use serde::{ser, de};
|
|
||||||
|
|
||||||
use token::Token;
|
|
||||||
|
|
||||||
#[derive(Clone, PartialEq, Debug)]
|
|
||||||
pub enum Error {
|
|
||||||
// Shared
|
|
||||||
Custom(String),
|
|
||||||
InvalidValue(String),
|
|
||||||
|
|
||||||
// De
|
|
||||||
EndOfStream,
|
|
||||||
InvalidType(de::Type),
|
|
||||||
InvalidLength(usize),
|
|
||||||
UnknownVariant(String),
|
|
||||||
UnknownField(String),
|
|
||||||
MissingField(&'static str),
|
|
||||||
DuplicateField(&'static str),
|
|
||||||
InvalidName(&'static str),
|
|
||||||
UnexpectedToken(Token<'static>),
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ser::Error for Error {
|
|
||||||
fn custom<T: Into<String>>(msg: T) -> Error {
|
|
||||||
Error::Custom(msg.into())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn invalid_value(msg: &str) -> Error {
|
|
||||||
Error::InvalidValue(msg.to_owned())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl de::Error for Error {
|
|
||||||
fn custom<T: Into<String>>(msg: T) -> Error {
|
|
||||||
Error::Custom(msg.into())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn end_of_stream() -> Error {
|
|
||||||
Error::EndOfStream
|
|
||||||
}
|
|
||||||
|
|
||||||
fn invalid_type(ty: de::Type) -> Error {
|
|
||||||
Error::InvalidType(ty)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn invalid_value(msg: &str) -> Error {
|
|
||||||
Error::InvalidValue(msg.to_owned())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn invalid_length(len: usize) -> Error {
|
|
||||||
Error::InvalidLength(len)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn unknown_variant(variant: &str) -> Error {
|
|
||||||
Error::UnknownVariant(variant.to_owned())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn unknown_field(field: &str) -> Error {
|
|
||||||
Error::UnknownField(field.to_owned())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn missing_field(field: &'static str) -> Error {
|
|
||||||
Error::MissingField(field)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn duplicate_field(field: &'static str) -> Error {
|
|
||||||
Error::DuplicateField(field)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl fmt::Display for Error {
|
|
||||||
fn fmt(&self, formatter: &mut fmt::Formatter) -> Result<(), fmt::Error> {
|
|
||||||
formatter.write_str(format!("{:?}", self).as_ref())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl error::Error for Error {
|
|
||||||
fn description(&self) -> &str {
|
|
||||||
"Serde Error"
|
|
||||||
}
|
|
||||||
|
|
||||||
fn cause(&self) -> Option<&error::Error> {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
extern crate serde;
|
|
||||||
|
|
||||||
mod assert;
|
|
||||||
pub use assert::{
|
|
||||||
assert_tokens,
|
|
||||||
assert_ser_tokens,
|
|
||||||
assert_ser_tokens_error,
|
|
||||||
assert_de_tokens,
|
|
||||||
assert_de_tokens_error,
|
|
||||||
};
|
|
||||||
|
|
||||||
mod ser;
|
|
||||||
pub use ser::Serializer;
|
|
||||||
|
|
||||||
mod de;
|
|
||||||
pub use de::Deserializer;
|
|
||||||
|
|
||||||
mod token;
|
|
||||||
pub use token::Token;
|
|
||||||
|
|
||||||
mod error;
|
|
||||||
pub use error::Error;
|
|
||||||
@@ -1,323 +0,0 @@
|
|||||||
use std::marker::PhantomData;
|
|
||||||
|
|
||||||
use serde::ser::{
|
|
||||||
self,
|
|
||||||
Serialize,
|
|
||||||
};
|
|
||||||
|
|
||||||
use error::Error;
|
|
||||||
use token::Token;
|
|
||||||
|
|
||||||
pub struct Serializer<'a, I>
|
|
||||||
where I: Iterator<Item=&'a Token<'a>>,
|
|
||||||
{
|
|
||||||
tokens: I,
|
|
||||||
phantom: PhantomData<&'a Token<'a>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, I> Serializer<'a, I>
|
|
||||||
where I: Iterator<Item=&'a Token<'a>>,
|
|
||||||
{
|
|
||||||
pub fn new(tokens: I) -> Serializer<'a, I> {
|
|
||||||
Serializer {
|
|
||||||
tokens: tokens,
|
|
||||||
phantom: PhantomData,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn next_token(&mut self) -> Option<&'a Token<'a>> {
|
|
||||||
self.tokens.next()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl<'a, I> ser::Serializer for Serializer<'a, I>
|
|
||||||
where I: Iterator<Item=&'a Token<'a>>,
|
|
||||||
{
|
|
||||||
type Error = Error;
|
|
||||||
type MapState = ();
|
|
||||||
type SeqState = ();
|
|
||||||
type TupleState = ();
|
|
||||||
type TupleStructState = ();
|
|
||||||
type TupleVariantState = ();
|
|
||||||
type StructState = ();
|
|
||||||
type StructVariantState = ();
|
|
||||||
|
|
||||||
fn serialize_unit(&mut self) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::Unit));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_newtype_variant<T>(&mut self,
|
|
||||||
name: &str,
|
|
||||||
_variant_index: usize,
|
|
||||||
variant: &str,
|
|
||||||
value: T) -> Result<(), Error>
|
|
||||||
where T: Serialize,
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::EnumNewType(name, variant)));
|
|
||||||
value.serialize(self)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_unit_struct(&mut self, name: &str) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::UnitStruct(name)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_unit_variant(&mut self,
|
|
||||||
name: &str,
|
|
||||||
_variant_index: usize,
|
|
||||||
variant: &str) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::EnumUnit(name, variant)));
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_bool(&mut self, v: bool) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::Bool(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_isize(&mut self, v: isize) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::Isize(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_i8(&mut self, v: i8) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::I8(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_i16(&mut self, v: i16) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::I16(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_i32(&mut self, v: i32) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::I32(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_i64(&mut self, v: i64) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::I64(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_usize(&mut self, v: usize) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::Usize(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_u8(&mut self, v: u8) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::U8(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_u16(&mut self, v: u16) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::U16(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_u32(&mut self, v: u32) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::U32(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_u64(&mut self, v: u64) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::U64(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_f32(&mut self, v: f32) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::F32(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_f64(&mut self, v: f64) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::F64(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_char(&mut self, v: char) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::Char(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_str(&mut self, v: &str) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::Str(v)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_none(&mut self) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::Option(false)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_some<V>(&mut self, value: V) -> Result<(), Error>
|
|
||||||
where V: Serialize,
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::Option(true)));
|
|
||||||
value.serialize(self)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_seq<'b>(&'b mut self, len: Option<usize>) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::SeqStart(len)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_seq_elt<T>(&mut self, _: &mut (), value: T) -> Result<(), Error>
|
|
||||||
where T: Serialize
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::SeqSep));
|
|
||||||
value.serialize(self)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_seq_end(&mut self, _: ()) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::SeqEnd));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_seq_fixed_size(&mut self, len: usize) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::SeqArrayStart(len)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple(&mut self, len: usize) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::TupleStart(len)));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_elt<T>(&mut self, _: &mut (), value: T) -> Result<(), Error>
|
|
||||||
where T: Serialize
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::TupleSep));
|
|
||||||
value.serialize(self)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_end(&mut self, _: ()) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::TupleEnd));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_newtype_struct<T>(&mut self,
|
|
||||||
name: &'static str,
|
|
||||||
value: T) -> Result<(), Error>
|
|
||||||
where T: Serialize,
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::StructNewType(name)));
|
|
||||||
value.serialize(self)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_struct(&mut self, name: &'static str, len: usize) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::TupleStructStart(name, len)));
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_struct_elt<T>(&mut self, _: &mut (), value: T) -> Result<(), Error>
|
|
||||||
where T: Serialize
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::TupleStructSep));
|
|
||||||
value.serialize(self)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_struct_end(&mut self, _: ()) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::TupleStructEnd));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_variant(&mut self,
|
|
||||||
name: &str,
|
|
||||||
_variant_index: usize,
|
|
||||||
variant: &str,
|
|
||||||
len: usize) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::EnumSeqStart(name, variant, len)));
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_variant_elt<T>(&mut self, _: &mut (), value: T) -> Result<(), Error>
|
|
||||||
where T: Serialize
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::EnumSeqSep));
|
|
||||||
value.serialize(self)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_variant_end(&mut self, _: ()) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::EnumSeqEnd));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_map(&mut self, len: Option<usize>) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::MapStart(len)));
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_map_key<T>(&mut self, _: &mut (), key: T) -> Result<(), Self::Error> where T: Serialize {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::MapSep));
|
|
||||||
key.serialize(self)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_map_value<T>(&mut self, _: &mut (), value: T) -> Result<(), Self::Error> where T: Serialize {
|
|
||||||
value.serialize(self)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_map_end(&mut self, _: ()) -> Result<(), Self::Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::MapEnd));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_struct(&mut self, name: &str, len: usize) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::StructStart(name, len)));
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_struct_elt<V>(&mut self, _: &mut (), key: &'static str, value: V) -> Result<(), Self::Error> where V: Serialize {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::StructSep));
|
|
||||||
try!(key.serialize(self));
|
|
||||||
value.serialize(self)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_struct_end(&mut self, _: ()) -> Result<(), Self::Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::StructEnd));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_struct_variant(&mut self,
|
|
||||||
name: &str,
|
|
||||||
_variant_index: usize,
|
|
||||||
variant: &str,
|
|
||||||
len: usize) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::EnumMapStart(name, variant, len)));
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_struct_variant_elt<V>(&mut self, _: &mut (), key: &'static str, value: V) -> Result<(), Self::Error> where V: Serialize {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::EnumMapSep));
|
|
||||||
try!(key.serialize(self));
|
|
||||||
value.serialize(self)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_struct_variant_end(&mut self, _: ()) -> Result<(), Self::Error> {
|
|
||||||
assert_eq!(self.tokens.next(), Some(&Token::EnumMapEnd));
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_bytes(&mut self, value: &[u8]) -> Result<(), Self::Error> {
|
|
||||||
let mut state = try!(self.serialize_seq(Some(value.len())));
|
|
||||||
for c in value {
|
|
||||||
try!(self.serialize_seq_elt(&mut state, c));
|
|
||||||
}
|
|
||||||
self.serialize_seq_end(state)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,60 +0,0 @@
|
|||||||
#[derive(Clone, PartialEq, Debug)]
|
|
||||||
pub enum Token<'a> {
|
|
||||||
Bool(bool),
|
|
||||||
Isize(isize),
|
|
||||||
I8(i8),
|
|
||||||
I16(i16),
|
|
||||||
I32(i32),
|
|
||||||
I64(i64),
|
|
||||||
Usize(usize),
|
|
||||||
U8(u8),
|
|
||||||
U16(u16),
|
|
||||||
U32(u32),
|
|
||||||
U64(u64),
|
|
||||||
F32(f32),
|
|
||||||
F64(f64),
|
|
||||||
Char(char),
|
|
||||||
Str(&'a str),
|
|
||||||
String(String),
|
|
||||||
Bytes(&'a [u8]),
|
|
||||||
|
|
||||||
Option(bool),
|
|
||||||
|
|
||||||
Unit,
|
|
||||||
UnitStruct(&'a str),
|
|
||||||
|
|
||||||
StructNewType(&'a str),
|
|
||||||
|
|
||||||
EnumStart(&'a str),
|
|
||||||
EnumUnit(&'a str, &'a str),
|
|
||||||
EnumNewType(&'a str, &'a str),
|
|
||||||
|
|
||||||
SeqStart(Option<usize>),
|
|
||||||
SeqArrayStart(usize),
|
|
||||||
SeqSep,
|
|
||||||
SeqEnd,
|
|
||||||
|
|
||||||
TupleStart(usize),
|
|
||||||
TupleSep,
|
|
||||||
TupleEnd,
|
|
||||||
|
|
||||||
TupleStructStart(&'a str, usize),
|
|
||||||
TupleStructSep,
|
|
||||||
TupleStructEnd,
|
|
||||||
|
|
||||||
MapStart(Option<usize>),
|
|
||||||
MapSep,
|
|
||||||
MapEnd,
|
|
||||||
|
|
||||||
StructStart(&'a str, usize),
|
|
||||||
StructSep,
|
|
||||||
StructEnd,
|
|
||||||
|
|
||||||
EnumSeqStart(&'a str, &'a str, usize),
|
|
||||||
EnumSeqSep,
|
|
||||||
EnumSeqEnd,
|
|
||||||
|
|
||||||
EnumMapStart(&'a str, &'a str, usize),
|
|
||||||
EnumMapSep,
|
|
||||||
EnumMapEnd,
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,30 @@
|
|||||||
|
[package]
|
||||||
|
name = "serde_tests"
|
||||||
|
version = "0.4.1"
|
||||||
|
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
||||||
|
license = "MIT/Apache-2.0"
|
||||||
|
description = "A generic serialization/deserialization framework"
|
||||||
|
repository = "https://github.com/serde-rs/serde"
|
||||||
|
documentation = "http://serde-rs.github.io/serde/serde"
|
||||||
|
readme = "README.md"
|
||||||
|
keywords = ["serialization"]
|
||||||
|
build = "build.rs"
|
||||||
|
|
||||||
|
[build-dependencies]
|
||||||
|
syntex = { version = "*", optional = true }
|
||||||
|
syntex_syntax = { version = "*" }
|
||||||
|
serde_codegen = { version = "*", path = "../serde_codegen", features = ["with-syntex"] }
|
||||||
|
|
||||||
|
[dev-dependencies]
|
||||||
|
num = "*"
|
||||||
|
rustc-serialize = "*"
|
||||||
|
serde = { version = "*", path = "../serde" }
|
||||||
|
syntex = "*"
|
||||||
|
|
||||||
|
[[test]]
|
||||||
|
name = "test"
|
||||||
|
path = "tests/test.rs"
|
||||||
|
|
||||||
|
[[bench]]
|
||||||
|
name = "bench"
|
||||||
|
path = "benches/bench.rs"
|
||||||
@@ -1,7 +1,6 @@
|
|||||||
#![feature(test)]
|
#![feature(test)]
|
||||||
#![cfg_attr(feature = "clippy", feature(plugin))]
|
|
||||||
#![cfg_attr(feature = "clippy", plugin(clippy))]
|
|
||||||
|
|
||||||
|
extern crate num;
|
||||||
extern crate rustc_serialize;
|
extern crate rustc_serialize;
|
||||||
extern crate serde;
|
extern crate serde;
|
||||||
extern crate test;
|
extern crate test;
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
mod bench_enum;
|
mod bench_enum;
|
||||||
|
mod bench_log;
|
||||||
mod bench_map;
|
mod bench_map;
|
||||||
mod bench_struct;
|
mod bench_struct;
|
||||||
mod bench_vec;
|
mod bench_vec;
|
||||||
@@ -1,9 +1,7 @@
|
|||||||
use test::Bencher;
|
use test::Bencher;
|
||||||
use std::error;
|
use rustc_serialize::{Decoder, Decodable};
|
||||||
use std::fmt;
|
|
||||||
use rustc_serialize::Decodable;
|
|
||||||
use serde;
|
use serde;
|
||||||
use serde::de::Deserialize;
|
use serde::de::{Deserializer, Deserialize};
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
@@ -17,34 +15,18 @@ pub enum Animal {
|
|||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum Error {
|
pub enum Error {
|
||||||
EndOfStream,
|
EndOfStreamError,
|
||||||
Syntax,
|
SyntaxError,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl serde::de::Error for Error {
|
impl serde::de::Error for Error {
|
||||||
fn custom<T: Into<String>>(_: T) -> Error { Error::Syntax }
|
fn syntax_error() -> Error { Error::SyntaxError }
|
||||||
|
|
||||||
fn end_of_stream() -> Error { Error::EndOfStream }
|
fn end_of_stream_error() -> Error { Error::EndOfStreamError }
|
||||||
|
|
||||||
fn unknown_field(_: &str) -> Error { Error::Syntax }
|
fn unknown_field_error(_: &str) -> Error { Error::SyntaxError }
|
||||||
|
|
||||||
fn missing_field(_: &'static str) -> Error { Error::Syntax }
|
fn missing_field_error(_: &'static str) -> Error { Error::SyntaxError }
|
||||||
}
|
|
||||||
|
|
||||||
impl fmt::Display for Error {
|
|
||||||
fn fmt(&self, formatter: &mut fmt::Formatter) -> Result<(), fmt::Error> {
|
|
||||||
formatter.write_str(format!("{:?}", self).as_ref())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl error::Error for Error {
|
|
||||||
fn description(&self) -> &str {
|
|
||||||
"Serde Deserialization Error"
|
|
||||||
}
|
|
||||||
|
|
||||||
fn cause(&self) -> Option<&error::Error> {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
@@ -54,11 +36,12 @@ mod decoder {
|
|||||||
|
|
||||||
use super::{Animal, Error};
|
use super::{Animal, Error};
|
||||||
use super::Animal::{Dog, Frog};
|
use super::Animal::{Dog, Frog};
|
||||||
|
use self::State::{AnimalState, IsizeState, StringState};
|
||||||
|
|
||||||
enum State {
|
enum State {
|
||||||
Animal(Animal),
|
AnimalState(Animal),
|
||||||
Isize(isize),
|
IsizeState(isize),
|
||||||
String(String),
|
StringState(String),
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct AnimalDecoder {
|
pub struct AnimalDecoder {
|
||||||
@@ -70,7 +53,7 @@ mod decoder {
|
|||||||
#[inline]
|
#[inline]
|
||||||
pub fn new(animal: Animal) -> AnimalDecoder {
|
pub fn new(animal: Animal) -> AnimalDecoder {
|
||||||
AnimalDecoder {
|
AnimalDecoder {
|
||||||
stack: vec!(State::Animal(animal)),
|
stack: vec!(AnimalState(animal)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -78,35 +61,35 @@ mod decoder {
|
|||||||
impl Decoder for AnimalDecoder {
|
impl Decoder for AnimalDecoder {
|
||||||
type Error = Error;
|
type Error = Error;
|
||||||
|
|
||||||
fn error(&mut self, _: &str) -> Error { Error::Syntax }
|
fn error(&mut self, _: &str) -> Error { Error::SyntaxError }
|
||||||
|
|
||||||
// Primitive types:
|
// Primitive types:
|
||||||
fn read_nil(&mut self) -> Result<(), Error> { Err(Error::Syntax) }
|
fn read_nil(&mut self) -> Result<(), Error> { Err(Error::SyntaxError) }
|
||||||
fn read_usize(&mut self) -> Result<usize, Error> { Err(Error::Syntax) }
|
fn read_usize(&mut self) -> Result<usize, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u64(&mut self) -> Result<u64, Error> { Err(Error::Syntax) }
|
fn read_u64(&mut self) -> Result<u64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u32(&mut self) -> Result<u32, Error> { Err(Error::Syntax) }
|
fn read_u32(&mut self) -> Result<u32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u16(&mut self) -> Result<u16, Error> { Err(Error::Syntax) }
|
fn read_u16(&mut self) -> Result<u16, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u8(&mut self) -> Result<u8, Error> { Err(Error::Syntax) }
|
fn read_u8(&mut self) -> Result<u8, Error> { Err(Error::SyntaxError) }
|
||||||
#[inline]
|
#[inline]
|
||||||
fn read_isize(&mut self) -> Result<isize, Error> {
|
fn read_isize(&mut self) -> Result<isize, Error> {
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Isize(x)) => Ok(x),
|
Some(IsizeState(x)) => Ok(x),
|
||||||
_ => Err(Error::Syntax),
|
_ => Err(Error::SyntaxError),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fn read_i64(&mut self) -> Result<i64, Error> { Err(Error::Syntax) }
|
fn read_i64(&mut self) -> Result<i64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i32(&mut self) -> Result<i32, Error> { Err(Error::Syntax) }
|
fn read_i32(&mut self) -> Result<i32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i16(&mut self) -> Result<i16, Error> { Err(Error::Syntax) }
|
fn read_i16(&mut self) -> Result<i16, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i8(&mut self) -> Result<i8, Error> { Err(Error::Syntax) }
|
fn read_i8(&mut self) -> Result<i8, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_bool(&mut self) -> Result<bool, Error> { Err(Error::Syntax) }
|
fn read_bool(&mut self) -> Result<bool, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_f64(&mut self) -> Result<f64, Error> { Err(Error::Syntax) }
|
fn read_f64(&mut self) -> Result<f64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_f32(&mut self) -> Result<f32, Error> { Err(Error::Syntax) }
|
fn read_f32(&mut self) -> Result<f32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_char(&mut self) -> Result<char, Error> { Err(Error::Syntax) }
|
fn read_char(&mut self) -> Result<char, Error> { Err(Error::SyntaxError) }
|
||||||
#[inline]
|
#[inline]
|
||||||
fn read_str(&mut self) -> Result<String, Error> {
|
fn read_str(&mut self) -> Result<String, Error> {
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::String(x)) => Ok(x),
|
Some(StringState(x)) => Ok(x),
|
||||||
_ => Err(Error::Syntax),
|
_ => Err(Error::SyntaxError),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -116,15 +99,15 @@ mod decoder {
|
|||||||
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Animal(animal)) => {
|
Some(AnimalState(animal)) => {
|
||||||
self.stack.push(State::Animal(animal));
|
self.stack.push(AnimalState(animal));
|
||||||
if name == "Animal" {
|
if name == "Animal" {
|
||||||
f(self)
|
f(self)
|
||||||
} else {
|
} else {
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_ => Err(Error::Syntax)
|
_ => Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -133,18 +116,18 @@ mod decoder {
|
|||||||
F: FnOnce(&mut AnimalDecoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
let name = match self.stack.pop() {
|
let name = match self.stack.pop() {
|
||||||
Some(State::Animal(Dog)) => "Dog",
|
Some(AnimalState(Dog)) => "Dog",
|
||||||
Some(State::Animal(Frog(x0, x1))) => {
|
Some(AnimalState(Frog(x0, x1))) => {
|
||||||
self.stack.push(State::Isize(x1));
|
self.stack.push(IsizeState(x1));
|
||||||
self.stack.push(State::String(x0));
|
self.stack.push(StringState(x0));
|
||||||
"Frog"
|
"Frog"
|
||||||
}
|
}
|
||||||
_ => { return Err(Error::Syntax); }
|
_ => { return Err(Error::SyntaxError); }
|
||||||
};
|
};
|
||||||
|
|
||||||
let idx = match names.iter().position(|n| *n == name) {
|
let idx = match names.iter().position(|n| *n == name) {
|
||||||
Some(idx) => idx,
|
Some(idx) => idx,
|
||||||
None => { return Err(Error::Syntax); }
|
None => { return Err(Error::SyntaxError); }
|
||||||
};
|
};
|
||||||
|
|
||||||
f(self, idx)
|
f(self, idx)
|
||||||
@@ -160,56 +143,56 @@ mod decoder {
|
|||||||
fn read_enum_struct_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
fn read_enum_struct_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut AnimalDecoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_struct_variant_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_enum_struct_variant_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_struct_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_struct_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple<T, F>(&mut self, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple<T, F>(&mut self, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_struct_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_struct_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Specialized types:
|
// Specialized types:
|
||||||
fn read_option<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
fn read_option<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut AnimalDecoder, bool) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder, bool) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
@@ -229,19 +212,19 @@ mod decoder {
|
|||||||
fn read_map<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
fn read_map<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut AnimalDecoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_map_elt_key<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
fn read_map_elt_key<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_map_elt_val<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
fn read_map_elt_val<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut AnimalDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -255,10 +238,10 @@ mod deserializer {
|
|||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
enum State {
|
enum State {
|
||||||
Animal(Animal),
|
AnimalState(Animal),
|
||||||
Isize(isize),
|
IsizeState(isize),
|
||||||
Str(&'static str),
|
StrState(&'static str),
|
||||||
String(String),
|
StringState(String),
|
||||||
UnitState,
|
UnitState,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -270,7 +253,7 @@ mod deserializer {
|
|||||||
#[inline]
|
#[inline]
|
||||||
pub fn new(animal: Animal) -> AnimalDeserializer {
|
pub fn new(animal: Animal) -> AnimalDeserializer {
|
||||||
AnimalDeserializer {
|
AnimalDeserializer {
|
||||||
stack: vec!(State::Animal(animal)),
|
stack: vec!(State::AnimalState(animal)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -279,60 +262,57 @@ mod deserializer {
|
|||||||
type Error = Error;
|
type Error = Error;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn deserialize<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
fn visit<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
where V: de::Visitor,
|
where V: de::Visitor,
|
||||||
{
|
{
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Isize(value)) => {
|
Some(State::IsizeState(value)) => {
|
||||||
visitor.visit_isize(value)
|
visitor.visit_isize(value)
|
||||||
}
|
}
|
||||||
Some(State::String(value)) => {
|
Some(State::StringState(value)) => {
|
||||||
visitor.visit_string(value)
|
visitor.visit_string(value)
|
||||||
}
|
}
|
||||||
Some(State::Str(value)) => {
|
Some(State::StrState(value)) => {
|
||||||
visitor.visit_str(value)
|
visitor.visit_str(value)
|
||||||
}
|
}
|
||||||
Some(State::UnitState) => {
|
Some(State::UnitState) => {
|
||||||
visitor.visit_unit()
|
visitor.visit_unit()
|
||||||
}
|
}
|
||||||
Some(_) => {
|
Some(_) => {
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
Err(Error::EndOfStream)
|
Err(Error::EndOfStreamError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn deserialize_enum<V>(&mut self,
|
fn visit_enum<V>(&mut self, _name: &str, mut visitor: V) -> Result<V::Value, Error>
|
||||||
_name: &str,
|
|
||||||
_variants: &[&str],
|
|
||||||
mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: de::EnumVisitor,
|
where V: de::EnumVisitor,
|
||||||
{
|
{
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Animal(Animal::Dog)) => {
|
Some(State::AnimalState(Animal::Dog)) => {
|
||||||
self.stack.push(State::UnitState);
|
self.stack.push(State::UnitState);
|
||||||
self.stack.push(State::Str("Dog"));
|
self.stack.push(State::StrState("Dog"));
|
||||||
visitor.visit(DogVisitor {
|
visitor.visit(DogVisitor {
|
||||||
de: self,
|
de: self,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
Some(State::Animal(Animal::Frog(x0, x1))) => {
|
Some(State::AnimalState(Animal::Frog(x0, x1))) => {
|
||||||
self.stack.push(State::Isize(x1));
|
self.stack.push(State::IsizeState(x1));
|
||||||
self.stack.push(State::String(x0));
|
self.stack.push(State::StringState(x0));
|
||||||
self.stack.push(State::Str("Frog"));
|
self.stack.push(State::StrState("Frog"));
|
||||||
visitor.visit(FrogVisitor {
|
visitor.visit(FrogVisitor {
|
||||||
de: self,
|
de: self,
|
||||||
state: 0,
|
state: 0,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
Some(_) => {
|
Some(_) => {
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
Err(Error::EndOfStream)
|
Err(Error::EndOfStreamError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -370,9 +350,7 @@ mod deserializer {
|
|||||||
de::Deserialize::deserialize(self.de)
|
de::Deserialize::deserialize(self.de)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn visit_tuple<V>(&mut self,
|
fn visit_seq<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
_len: usize,
|
|
||||||
mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: de::Visitor,
|
where V: de::Visitor,
|
||||||
{
|
{
|
||||||
visitor.visit_seq(self)
|
visitor.visit_seq(self)
|
||||||
@@ -404,7 +382,7 @@ mod deserializer {
|
|||||||
if self.state == 2 {
|
if self.state == 2 {
|
||||||
Ok(())
|
Ok(())
|
||||||
} else {
|
} else {
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -432,7 +410,7 @@ fn bench_decoder_dog(b: &mut Bencher) {
|
|||||||
#[bench]
|
#[bench]
|
||||||
fn bench_decoder_frog(b: &mut Bencher) {
|
fn bench_decoder_frog(b: &mut Bencher) {
|
||||||
b.iter(|| {
|
b.iter(|| {
|
||||||
let animal = Animal::Frog("Henry".to_owned(), 349);
|
let animal = Animal::Frog("Henry".to_string(), 349);
|
||||||
|
|
||||||
let mut d = decoder::AnimalDecoder::new(animal.clone());
|
let mut d = decoder::AnimalDecoder::new(animal.clone());
|
||||||
let value: Animal = Decodable::decode(&mut d).unwrap();
|
let value: Animal = Decodable::decode(&mut d).unwrap();
|
||||||
@@ -456,7 +434,7 @@ fn bench_deserializer_dog(b: &mut Bencher) {
|
|||||||
#[bench]
|
#[bench]
|
||||||
fn bench_deserializer_frog(b: &mut Bencher) {
|
fn bench_deserializer_frog(b: &mut Bencher) {
|
||||||
b.iter(|| {
|
b.iter(|| {
|
||||||
let animal = Animal::Frog("Henry".to_owned(), 349);
|
let animal = Animal::Frog("Henry".to_string(), 349);
|
||||||
|
|
||||||
let mut d = deserializer::AnimalDeserializer::new(animal.clone());
|
let mut d = deserializer::AnimalDeserializer::new(animal.clone());
|
||||||
let value: Animal = Deserialize::deserialize(&mut d).unwrap();
|
let value: Animal = Deserialize::deserialize(&mut d).unwrap();
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -1,6 +1,4 @@
|
|||||||
use std::fmt::Debug;
|
use std::fmt::Debug;
|
||||||
use std::fmt;
|
|
||||||
use std::error;
|
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use test::Bencher;
|
use test::Bencher;
|
||||||
|
|
||||||
@@ -14,37 +12,22 @@ use serde::de::{Deserializer, Deserialize};
|
|||||||
#[derive(PartialEq, Debug)]
|
#[derive(PartialEq, Debug)]
|
||||||
pub enum Error {
|
pub enum Error {
|
||||||
EndOfStream,
|
EndOfStream,
|
||||||
Syntax,
|
SyntaxError,
|
||||||
MissingField,
|
MissingField,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl serde::de::Error for Error {
|
impl serde::de::Error for Error {
|
||||||
fn custom<T: Into<String>>(_: T) -> Error { Error::Syntax }
|
fn syntax_error() -> Error { Error::SyntaxError }
|
||||||
|
|
||||||
fn end_of_stream() -> Error { Error::EndOfStream }
|
fn end_of_stream_error() -> Error { Error::EndOfStream }
|
||||||
|
|
||||||
fn unknown_field(_: &str) -> Error { Error::Syntax }
|
fn unknown_field_error(_: &str) -> Error { Error::SyntaxError }
|
||||||
|
|
||||||
fn missing_field(_: &'static str) -> Error {
|
fn missing_field_error(_: &'static str) -> Error {
|
||||||
Error::MissingField
|
Error::MissingField
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl fmt::Display for Error {
|
|
||||||
fn fmt(&self, formatter: &mut fmt::Formatter) -> Result<(), fmt::Error> {
|
|
||||||
formatter.write_str(format!("{:?}", self).as_ref())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl error::Error for Error {
|
|
||||||
fn description(&self) -> &str {
|
|
||||||
"Serde Deserialization Error"
|
|
||||||
}
|
|
||||||
|
|
||||||
fn cause(&self) -> Option<&error::Error> {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
//////////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
mod decoder {
|
mod decoder {
|
||||||
@@ -53,10 +36,11 @@ mod decoder {
|
|||||||
use rustc_serialize;
|
use rustc_serialize;
|
||||||
|
|
||||||
use super::Error;
|
use super::Error;
|
||||||
|
use self::Value::{StringValue, IsizeValue};
|
||||||
|
|
||||||
enum Value {
|
enum Value {
|
||||||
String(String),
|
StringValue(String),
|
||||||
Isize(isize),
|
IsizeValue(isize),
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct IsizeDecoder {
|
pub struct IsizeDecoder {
|
||||||
@@ -80,37 +64,37 @@ mod decoder {
|
|||||||
type Error = Error;
|
type Error = Error;
|
||||||
|
|
||||||
fn error(&mut self, _msg: &str) -> Error {
|
fn error(&mut self, _msg: &str) -> Error {
|
||||||
Error::Syntax
|
Error::SyntaxError
|
||||||
}
|
}
|
||||||
|
|
||||||
// Primitive types:
|
// Primitive types:
|
||||||
fn read_nil(&mut self) -> Result<(), Error> { Err(Error::Syntax) }
|
fn read_nil(&mut self) -> Result<(), Error> { Err(Error::SyntaxError) }
|
||||||
fn read_usize(&mut self) -> Result<usize, Error> { Err(Error::Syntax) }
|
fn read_usize(&mut self) -> Result<usize, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u64(&mut self) -> Result<u64, Error> { Err(Error::Syntax) }
|
fn read_u64(&mut self) -> Result<u64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u32(&mut self) -> Result<u32, Error> { Err(Error::Syntax) }
|
fn read_u32(&mut self) -> Result<u32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u16(&mut self) -> Result<u16, Error> { Err(Error::Syntax) }
|
fn read_u16(&mut self) -> Result<u16, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u8(&mut self) -> Result<u8, Error> { Err(Error::Syntax) }
|
fn read_u8(&mut self) -> Result<u8, Error> { Err(Error::SyntaxError) }
|
||||||
#[inline]
|
#[inline]
|
||||||
fn read_isize(&mut self) -> Result<isize, Error> {
|
fn read_isize(&mut self) -> Result<isize, Error> {
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(Value::Isize(x)) => Ok(x),
|
Some(IsizeValue(x)) => Ok(x),
|
||||||
Some(_) => Err(Error::Syntax),
|
Some(_) => Err(Error::SyntaxError),
|
||||||
None => Err(Error::EndOfStream),
|
None => Err(Error::EndOfStream),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fn read_i64(&mut self) -> Result<i64, Error> { Err(Error::Syntax) }
|
fn read_i64(&mut self) -> Result<i64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i32(&mut self) -> Result<i32, Error> { Err(Error::Syntax) }
|
fn read_i32(&mut self) -> Result<i32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i16(&mut self) -> Result<i16, Error> { Err(Error::Syntax) }
|
fn read_i16(&mut self) -> Result<i16, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i8(&mut self) -> Result<i8, Error> { Err(Error::Syntax) }
|
fn read_i8(&mut self) -> Result<i8, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_bool(&mut self) -> Result<bool, Error> { Err(Error::Syntax) }
|
fn read_bool(&mut self) -> Result<bool, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_f64(&mut self) -> Result<f64, Error> { Err(Error::Syntax) }
|
fn read_f64(&mut self) -> Result<f64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_f32(&mut self) -> Result<f32, Error> { Err(Error::Syntax) }
|
fn read_f32(&mut self) -> Result<f32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_char(&mut self) -> Result<char, Error> { Err(Error::Syntax) }
|
fn read_char(&mut self) -> Result<char, Error> { Err(Error::SyntaxError) }
|
||||||
#[inline]
|
#[inline]
|
||||||
fn read_str(&mut self) -> Result<String, Error> {
|
fn read_str(&mut self) -> Result<String, Error> {
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(Value::String(x)) => Ok(x),
|
Some(StringValue(x)) => Ok(x),
|
||||||
Some(_) => Err(Error::Syntax),
|
Some(_) => Err(Error::SyntaxError),
|
||||||
None => Err(Error::EndOfStream),
|
None => Err(Error::EndOfStream),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -119,86 +103,86 @@ mod decoder {
|
|||||||
fn read_enum<T, F>(&mut self, _name: &str, _f: F) -> Result<T, Error> where
|
fn read_enum<T, F>(&mut self, _name: &str, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
fn read_enum_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_variant_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_enum_variant_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_struct_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
fn read_enum_struct_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_struct_variant_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_enum_struct_variant_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_struct_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_struct_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple<T, F>(&mut self, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple<T, F>(&mut self, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_struct_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_struct_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Specialized types:
|
// Specialized types:
|
||||||
fn read_option<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
fn read_option<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder, bool) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder, bool) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_seq<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
fn read_seq<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_seq_elt<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
fn read_seq_elt<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut IsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
@@ -214,12 +198,12 @@ mod decoder {
|
|||||||
{
|
{
|
||||||
match self.iter.next() {
|
match self.iter.next() {
|
||||||
Some((key, value)) => {
|
Some((key, value)) => {
|
||||||
self.stack.push(Value::Isize(value));
|
self.stack.push(IsizeValue(value));
|
||||||
self.stack.push(Value::String(key));
|
self.stack.push(StringValue(key));
|
||||||
f(self)
|
f(self)
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -246,8 +230,8 @@ mod deserializer {
|
|||||||
#[derive(PartialEq, Debug)]
|
#[derive(PartialEq, Debug)]
|
||||||
enum State {
|
enum State {
|
||||||
StartState,
|
StartState,
|
||||||
Key(String),
|
KeyState(String),
|
||||||
Value(isize),
|
ValueState(isize),
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct IsizeDeserializer {
|
pub struct IsizeDeserializer {
|
||||||
@@ -268,17 +252,17 @@ mod deserializer {
|
|||||||
impl de::Deserializer for IsizeDeserializer {
|
impl de::Deserializer for IsizeDeserializer {
|
||||||
type Error = Error;
|
type Error = Error;
|
||||||
|
|
||||||
fn deserialize<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
fn visit<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
where V: de::Visitor,
|
where V: de::Visitor,
|
||||||
{
|
{
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::StartState) => {
|
Some(State::StartState) => {
|
||||||
visitor.visit_map(self)
|
visitor.visit_map(self)
|
||||||
}
|
}
|
||||||
Some(State::Key(key)) => {
|
Some(State::KeyState(key)) => {
|
||||||
visitor.visit_string(key)
|
visitor.visit_string(key)
|
||||||
}
|
}
|
||||||
Some(State::Value(value)) => {
|
Some(State::ValueState(value)) => {
|
||||||
visitor.visit_isize(value)
|
visitor.visit_isize(value)
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
@@ -296,8 +280,8 @@ mod deserializer {
|
|||||||
{
|
{
|
||||||
match self.iter.next() {
|
match self.iter.next() {
|
||||||
Some((key, value)) => {
|
Some((key, value)) => {
|
||||||
self.stack.push(State::Value(value));
|
self.stack.push(State::ValueState(value));
|
||||||
self.stack.push(State::Key(key));
|
self.stack.push(State::KeyState(key));
|
||||||
Ok(Some(try!(de::Deserialize::deserialize(self))))
|
Ok(Some(try!(de::Deserialize::deserialize(self))))
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
@@ -314,7 +298,7 @@ mod deserializer {
|
|||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
match self.iter.next() {
|
match self.iter.next() {
|
||||||
Some(_) => Err(Error::Syntax),
|
Some(_) => Err(Error::SyntaxError),
|
||||||
None => Ok(()),
|
None => Ok(()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -331,14 +315,14 @@ mod deserializer {
|
|||||||
#[inline]
|
#[inline]
|
||||||
fn next(&mut self) -> Option<Result<de::Token, Error>> {
|
fn next(&mut self) -> Option<Result<de::Token, Error>> {
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::StartState) => {
|
Some(StartState) => {
|
||||||
self.stack.push(KeyOrEndState);
|
self.stack.push(KeyOrEndState);
|
||||||
Some(Ok(de::Token::MapStart(self.len)))
|
Some(Ok(de::Token::MapStart(self.len)))
|
||||||
}
|
}
|
||||||
Some(State::KeyOrEndState) => {
|
Some(KeyOrEndState) => {
|
||||||
match self.iter.next() {
|
match self.iter.next() {
|
||||||
Some((key, value)) => {
|
Some((key, value)) => {
|
||||||
self.stack.push(Value(value));
|
self.stack.push(ValueState(value));
|
||||||
Some(Ok(de::Token::String(key)))
|
Some(Ok(de::Token::String(key)))
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
@@ -347,7 +331,7 @@ mod deserializer {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Some(State::Value(x)) => {
|
Some(ValueState(x)) => {
|
||||||
self.stack.push(KeyOrEndState);
|
self.stack.push(KeyOrEndState);
|
||||||
Some(Ok(de::Token::Isize(x)))
|
Some(Ok(de::Token::Isize(x)))
|
||||||
}
|
}
|
||||||
@@ -363,30 +347,30 @@ mod deserializer {
|
|||||||
|
|
||||||
impl de::Deserializer<Error> for IsizeDeserializer {
|
impl de::Deserializer<Error> for IsizeDeserializer {
|
||||||
#[inline]
|
#[inline]
|
||||||
fn end_of_stream(&mut self) -> Error {
|
fn end_of_stream_error(&mut self) -> Error {
|
||||||
EndOfStream
|
EndOfStream
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn syntax(&mut self, _token: de::Token, _expected: &[de::TokenKind]) -> Error {
|
fn syntax_error(&mut self, _token: de::Token, _expected: &[de::TokenKind]) -> Error {
|
||||||
Syntax
|
SyntaxError
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn unexpected_name(&mut self, _token: de::Token) -> Error {
|
fn unexpected_name_error(&mut self, _token: de::Token) -> Error {
|
||||||
Syntax
|
SyntaxError
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn conversion_error(&mut self, _token: de::Token) -> Error {
|
fn conversion_error(&mut self, _token: de::Token) -> Error {
|
||||||
Syntax
|
SyntaxError
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn missing_field<
|
fn missing_field<
|
||||||
T: de::Deserialize<IsizeDeserializer, Error>
|
T: de::Deserialize<IsizeDeserializer, Error>
|
||||||
>(&mut self, _field: &'static str) -> Result<T, Error> {
|
>(&mut self, _field: &'static str) -> Result<T, Error> {
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
@@ -415,7 +399,7 @@ fn bench_decoder_000(b: &mut Bencher) {
|
|||||||
fn bench_decoder_003(b: &mut Bencher) {
|
fn bench_decoder_003(b: &mut Bencher) {
|
||||||
b.iter(|| {
|
b.iter(|| {
|
||||||
let mut m: HashMap<String, isize> = HashMap::new();
|
let mut m: HashMap<String, isize> = HashMap::new();
|
||||||
for i in 0 .. 3 {
|
for i in (0 .. 3) {
|
||||||
m.insert(i.to_string(), i);
|
m.insert(i.to_string(), i);
|
||||||
}
|
}
|
||||||
run_decoder(decoder::IsizeDecoder::new(m.clone()), m)
|
run_decoder(decoder::IsizeDecoder::new(m.clone()), m)
|
||||||
@@ -426,21 +410,21 @@ fn bench_decoder_003(b: &mut Bencher) {
|
|||||||
fn bench_decoder_100(b: &mut Bencher) {
|
fn bench_decoder_100(b: &mut Bencher) {
|
||||||
b.iter(|| {
|
b.iter(|| {
|
||||||
let mut m: HashMap<String, isize> = HashMap::new();
|
let mut m: HashMap<String, isize> = HashMap::new();
|
||||||
for i in 0 .. 100 {
|
for i in (0 .. 100) {
|
||||||
m.insert(i.to_string(), i);
|
m.insert(i.to_string(), i);
|
||||||
}
|
}
|
||||||
run_decoder(decoder::IsizeDecoder::new(m.clone()), m)
|
run_decoder(decoder::IsizeDecoder::new(m.clone()), m)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run_deserializer<D, T>(mut d: D, value: T)
|
fn run_deserializer<
|
||||||
where D: Deserializer,
|
D: Deserializer<Error=E>,
|
||||||
D::Error: Debug + PartialEq,
|
E: Debug,
|
||||||
T: Clone + PartialEq + Debug + Deserialize
|
T: Clone + PartialEq + Debug + Deserialize
|
||||||
{
|
>(mut d: D, value: T) {
|
||||||
let v = T::deserialize(&mut d);
|
let v: T = Deserialize::deserialize(&mut d).unwrap();
|
||||||
|
|
||||||
assert_eq!(Ok(value), v);
|
assert_eq!(value, v);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[bench]
|
#[bench]
|
||||||
@@ -455,7 +439,7 @@ fn bench_deserializer_000(b: &mut Bencher) {
|
|||||||
fn bench_deserializer_003(b: &mut Bencher) {
|
fn bench_deserializer_003(b: &mut Bencher) {
|
||||||
b.iter(|| {
|
b.iter(|| {
|
||||||
let mut m: HashMap<String, isize> = HashMap::new();
|
let mut m: HashMap<String, isize> = HashMap::new();
|
||||||
for i in 0 .. 3 {
|
for i in (0 .. 3) {
|
||||||
m.insert(i.to_string(), i);
|
m.insert(i.to_string(), i);
|
||||||
}
|
}
|
||||||
run_deserializer(deserializer::IsizeDeserializer::new(m.clone()), m)
|
run_deserializer(deserializer::IsizeDeserializer::new(m.clone()), m)
|
||||||
@@ -466,7 +450,7 @@ fn bench_deserializer_003(b: &mut Bencher) {
|
|||||||
fn bench_deserializer_100(b: &mut Bencher) {
|
fn bench_deserializer_100(b: &mut Bencher) {
|
||||||
b.iter(|| {
|
b.iter(|| {
|
||||||
let mut m: HashMap<String, isize> = HashMap::new();
|
let mut m: HashMap<String, isize> = HashMap::new();
|
||||||
for i in 0 .. 100 {
|
for i in (0 .. 100) {
|
||||||
m.insert(i.to_string(), i);
|
m.insert(i.to_string(), i);
|
||||||
}
|
}
|
||||||
run_deserializer(deserializer::IsizeDeserializer::new(m.clone()), m)
|
run_deserializer(deserializer::IsizeDeserializer::new(m.clone()), m)
|
||||||
@@ -1,12 +1,10 @@
|
|||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use test::Bencher;
|
use test::Bencher;
|
||||||
use std::fmt;
|
|
||||||
use std::error;
|
|
||||||
|
|
||||||
use rustc_serialize::Decodable;
|
use rustc_serialize::{Decoder, Decodable};
|
||||||
|
|
||||||
use serde;
|
use serde;
|
||||||
use serde::de::Deserialize;
|
use serde::de::{Deserializer, Deserialize};
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
@@ -29,57 +27,54 @@ pub struct Outer {
|
|||||||
#[derive(Debug, PartialEq)]
|
#[derive(Debug, PartialEq)]
|
||||||
pub enum Error {
|
pub enum Error {
|
||||||
EndOfStream,
|
EndOfStream,
|
||||||
Syntax,
|
SyntaxError,
|
||||||
MissingField,
|
MissingField,
|
||||||
OtherError,
|
OtherError,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl serde::de::Error for Error {
|
impl serde::de::Error for Error {
|
||||||
fn custom<T: Into<String>>(_: T) -> Error { Error::Syntax }
|
fn syntax_error() -> Error { Error::SyntaxError }
|
||||||
|
|
||||||
fn end_of_stream() -> Error { Error::EndOfStream }
|
fn end_of_stream_error() -> Error { Error::EndOfStream }
|
||||||
|
|
||||||
fn unknown_field(_: &str) -> Error { Error::Syntax }
|
fn unknown_field_error(_: &str) -> Error { Error::SyntaxError }
|
||||||
|
|
||||||
fn missing_field(_: &'static str) -> Error {
|
fn missing_field_error(_: &'static str) -> Error {
|
||||||
Error::MissingField
|
Error::MissingField
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl fmt::Display for Error {
|
|
||||||
fn fmt(&self, formatter: &mut fmt::Formatter) -> Result<(), fmt::Error> {
|
|
||||||
formatter.write_str(format!("{:?}", self).as_ref())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl error::Error for Error {
|
|
||||||
fn description(&self) -> &str {
|
|
||||||
"Serde Deserialization Error"
|
|
||||||
}
|
|
||||||
|
|
||||||
fn cause(&self) -> Option<&error::Error> {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
mod decoder {
|
mod decoder {
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use rustc_serialize::Decoder;
|
use rustc_serialize::Decoder;
|
||||||
|
|
||||||
use super::{Outer, Inner, Error};
|
use super::{Outer, Inner, Error};
|
||||||
|
|
||||||
|
use self::State::{
|
||||||
|
OuterState,
|
||||||
|
InnerState,
|
||||||
|
NullState,
|
||||||
|
UsizeState,
|
||||||
|
CharState,
|
||||||
|
StringState,
|
||||||
|
FieldState,
|
||||||
|
VecState,
|
||||||
|
MapState,
|
||||||
|
OptionState,
|
||||||
|
};
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
enum State {
|
enum State {
|
||||||
Outer(Outer),
|
OuterState(Outer),
|
||||||
Inner(Inner),
|
InnerState(Inner),
|
||||||
Null,
|
NullState,
|
||||||
Usize(usize),
|
UsizeState(usize),
|
||||||
Char(char),
|
CharState(char),
|
||||||
String(String),
|
StringState(String),
|
||||||
Field(&'static str),
|
FieldState(&'static str),
|
||||||
Vec(Vec<Inner>),
|
VecState(Vec<Inner>),
|
||||||
Map(HashMap<String, Option<char>>),
|
MapState(HashMap<String, Option<char>>),
|
||||||
Option(bool),
|
OptionState(bool),
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct OuterDecoder {
|
pub struct OuterDecoder {
|
||||||
@@ -91,7 +86,7 @@ mod decoder {
|
|||||||
#[inline]
|
#[inline]
|
||||||
pub fn new(animal: Outer) -> OuterDecoder {
|
pub fn new(animal: Outer) -> OuterDecoder {
|
||||||
OuterDecoder {
|
OuterDecoder {
|
||||||
stack: vec!(State::Outer(animal)),
|
stack: vec!(OuterState(animal)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -107,41 +102,41 @@ mod decoder {
|
|||||||
#[inline]
|
#[inline]
|
||||||
fn read_nil(&mut self) -> Result<(), Error> {
|
fn read_nil(&mut self) -> Result<(), Error> {
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Null) => Ok(()),
|
Some(NullState) => Ok(()),
|
||||||
_ => Err(Error::Syntax),
|
_ => Err(Error::SyntaxError),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#[inline]
|
#[inline]
|
||||||
fn read_usize(&mut self) -> Result<usize, Error> {
|
fn read_usize(&mut self) -> Result<usize, Error> {
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Usize(value)) => Ok(value),
|
Some(UsizeState(value)) => Ok(value),
|
||||||
_ => Err(Error::Syntax),
|
_ => Err(Error::SyntaxError),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fn read_u64(&mut self) -> Result<u64, Error> { Err(Error::Syntax) }
|
fn read_u64(&mut self) -> Result<u64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u32(&mut self) -> Result<u32, Error> { Err(Error::Syntax) }
|
fn read_u32(&mut self) -> Result<u32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u16(&mut self) -> Result<u16, Error> { Err(Error::Syntax) }
|
fn read_u16(&mut self) -> Result<u16, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u8(&mut self) -> Result<u8, Error> { Err(Error::Syntax) }
|
fn read_u8(&mut self) -> Result<u8, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_isize(&mut self) -> Result<isize, Error> { Err(Error::Syntax) }
|
fn read_isize(&mut self) -> Result<isize, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i64(&mut self) -> Result<i64, Error> { Err(Error::Syntax) }
|
fn read_i64(&mut self) -> Result<i64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i32(&mut self) -> Result<i32, Error> { Err(Error::Syntax) }
|
fn read_i32(&mut self) -> Result<i32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i16(&mut self) -> Result<i16, Error> { Err(Error::Syntax) }
|
fn read_i16(&mut self) -> Result<i16, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i8(&mut self) -> Result<i8, Error> { Err(Error::Syntax) }
|
fn read_i8(&mut self) -> Result<i8, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_bool(&mut self) -> Result<bool, Error> { Err(Error::Syntax) }
|
fn read_bool(&mut self) -> Result<bool, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_f64(&mut self) -> Result<f64, Error> { Err(Error::Syntax) }
|
fn read_f64(&mut self) -> Result<f64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_f32(&mut self) -> Result<f32, Error> { Err(Error::Syntax) }
|
fn read_f32(&mut self) -> Result<f32, Error> { Err(Error::SyntaxError) }
|
||||||
#[inline]
|
#[inline]
|
||||||
fn read_char(&mut self) -> Result<char, Error> {
|
fn read_char(&mut self) -> Result<char, Error> {
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Char(c)) => Ok(c),
|
Some(CharState(c)) => Ok(c),
|
||||||
_ => Err(Error::Syntax),
|
_ => Err(Error::SyntaxError),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#[inline]
|
#[inline]
|
||||||
fn read_str(&mut self) -> Result<String, Error> {
|
fn read_str(&mut self) -> Result<String, Error> {
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::String(value)) => Ok(value),
|
Some(StringState(value)) => Ok(value),
|
||||||
_ => Err(Error::Syntax),
|
_ => Err(Error::SyntaxError),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -149,31 +144,31 @@ mod decoder {
|
|||||||
fn read_enum<T, F>(&mut self, _name: &str, _f: F) -> Result<T, Error> where
|
fn read_enum<T, F>(&mut self, _name: &str, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
fn read_enum_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut OuterDecoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_variant_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_enum_variant_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_struct_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
fn read_enum_struct_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut OuterDecoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_struct_variant_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_enum_struct_variant_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
@@ -181,31 +176,31 @@ mod decoder {
|
|||||||
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Outer(Outer { inner })) => {
|
Some(OuterState(Outer { inner })) => {
|
||||||
if s_name == "Outer" {
|
if s_name == "Outer" {
|
||||||
self.stack.push(State::Vec(inner));
|
self.stack.push(VecState(inner));
|
||||||
self.stack.push(State::Field("inner"));
|
self.stack.push(FieldState("inner"));
|
||||||
f(self)
|
f(self)
|
||||||
} else {
|
} else {
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Some(State::Inner(Inner { a: (), b, c })) => {
|
Some(InnerState(Inner { a: (), b, c })) => {
|
||||||
if s_name == "Inner" {
|
if s_name == "Inner" {
|
||||||
self.stack.push(State::Map(c));
|
self.stack.push(MapState(c));
|
||||||
self.stack.push(State::Field("c"));
|
self.stack.push(FieldState("c"));
|
||||||
|
|
||||||
self.stack.push(State::Usize(b));
|
self.stack.push(UsizeState(b));
|
||||||
self.stack.push(State::Field("b"));
|
self.stack.push(FieldState("b"));
|
||||||
|
|
||||||
self.stack.push(State::Null);
|
self.stack.push(NullState);
|
||||||
self.stack.push(State::Field("a"));
|
self.stack.push(FieldState("a"));
|
||||||
f(self)
|
f(self)
|
||||||
} else {
|
} else {
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_ => Err(Error::Syntax),
|
_ => Err(Error::SyntaxError),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#[inline]
|
#[inline]
|
||||||
@@ -213,39 +208,39 @@ mod decoder {
|
|||||||
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Field(name)) => {
|
Some(FieldState(name)) => {
|
||||||
if f_name == name {
|
if f_name == name {
|
||||||
f(self)
|
f(self)
|
||||||
} else {
|
} else {
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_ => Err(Error::Syntax)
|
_ => Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple<T, F>(&mut self, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple<T, F>(&mut self, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_struct_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_struct_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Specialized types:
|
// Specialized types:
|
||||||
@@ -254,8 +249,8 @@ mod decoder {
|
|||||||
F: FnOnce(&mut OuterDecoder, bool) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder, bool) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Option(b)) => f(self, b),
|
Some(OptionState(b)) => f(self, b),
|
||||||
_ => Err(Error::Syntax),
|
_ => Err(Error::SyntaxError),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -264,14 +259,14 @@ mod decoder {
|
|||||||
F: FnOnce(&mut OuterDecoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Vec(value)) => {
|
Some(VecState(value)) => {
|
||||||
let len = value.len();
|
let len = value.len();
|
||||||
for inner in value.into_iter().rev() {
|
for inner in value.into_iter().rev() {
|
||||||
self.stack.push(State::Inner(inner));
|
self.stack.push(InnerState(inner));
|
||||||
}
|
}
|
||||||
f(self, len)
|
f(self, len)
|
||||||
}
|
}
|
||||||
_ => Err(Error::Syntax)
|
_ => Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#[inline]
|
#[inline]
|
||||||
@@ -286,23 +281,23 @@ mod decoder {
|
|||||||
F: FnOnce(&mut OuterDecoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut OuterDecoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Map(map)) => {
|
Some(MapState(map)) => {
|
||||||
let len = map.len();
|
let len = map.len();
|
||||||
for (key, value) in map {
|
for (key, value) in map {
|
||||||
match value {
|
match value {
|
||||||
Some(c) => {
|
Some(c) => {
|
||||||
self.stack.push(State::Char(c));
|
self.stack.push(CharState(c));
|
||||||
self.stack.push(State::Option(true));
|
self.stack.push(OptionState(true));
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
self.stack.push(State::Option(false));
|
self.stack.push(OptionState(false));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
self.stack.push(State::String(key));
|
self.stack.push(StringState(key));
|
||||||
}
|
}
|
||||||
f(self, len)
|
f(self, len)
|
||||||
}
|
}
|
||||||
_ => Err(Error::Syntax),
|
_ => Err(Error::SyntaxError),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#[inline]
|
#[inline]
|
||||||
@@ -333,16 +328,16 @@ mod deserializer {
|
|||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
enum State {
|
enum State {
|
||||||
Outer(Outer),
|
OuterState(Outer),
|
||||||
Inner(Inner),
|
InnerState(Inner),
|
||||||
Str(&'static str),
|
StrState(&'static str),
|
||||||
Null,
|
NullState,
|
||||||
Usize(usize),
|
UsizeState(usize),
|
||||||
Char(char),
|
CharState(char),
|
||||||
String(String),
|
StringState(String),
|
||||||
Option(bool),
|
OptionState(bool),
|
||||||
Vec(Vec<Inner>),
|
VecState(Vec<Inner>),
|
||||||
Map(HashMap<String, Option<char>>),
|
MapState(HashMap<String, Option<char>>),
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct OuterDeserializer {
|
pub struct OuterDeserializer {
|
||||||
@@ -353,7 +348,7 @@ mod deserializer {
|
|||||||
#[inline]
|
#[inline]
|
||||||
pub fn new(outer: Outer) -> OuterDeserializer {
|
pub fn new(outer: Outer) -> OuterDeserializer {
|
||||||
OuterDeserializer {
|
OuterDeserializer {
|
||||||
stack: vec!(State::Outer(outer)),
|
stack: vec!(State::OuterState(outer)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -361,81 +356,78 @@ mod deserializer {
|
|||||||
impl de::Deserializer for OuterDeserializer {
|
impl de::Deserializer for OuterDeserializer {
|
||||||
type Error = Error;
|
type Error = Error;
|
||||||
|
|
||||||
fn deserialize<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
fn visit<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
where V: de::Visitor,
|
where V: de::Visitor,
|
||||||
{
|
{
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Vec(value)) => {
|
Some(State::VecState(value)) => {
|
||||||
visitor.visit_seq(OuterSeqVisitor {
|
visitor.visit_seq(OuterSeqVisitor {
|
||||||
de: self,
|
de: self,
|
||||||
iter: value.into_iter(),
|
iter: value.into_iter(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
Some(State::Map(value)) => {
|
Some(State::MapState(value)) => {
|
||||||
visitor.visit_map(MapVisitor {
|
visitor.visit_map(MapVisitor {
|
||||||
de: self,
|
de: self,
|
||||||
iter: value.into_iter(),
|
iter: value.into_iter(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
Some(State::Null) => {
|
Some(State::NullState) => {
|
||||||
visitor.visit_unit()
|
visitor.visit_unit()
|
||||||
}
|
}
|
||||||
Some(State::Usize(x)) => {
|
Some(State::UsizeState(x)) => {
|
||||||
visitor.visit_usize(x)
|
visitor.visit_usize(x)
|
||||||
}
|
}
|
||||||
Some(State::Char(x)) => {
|
Some(State::CharState(x)) => {
|
||||||
visitor.visit_char(x)
|
visitor.visit_char(x)
|
||||||
}
|
}
|
||||||
Some(State::Str(x)) => {
|
Some(State::StrState(x)) => {
|
||||||
visitor.visit_str(x)
|
visitor.visit_str(x)
|
||||||
}
|
}
|
||||||
Some(State::String(x)) => {
|
Some(State::StringState(x)) => {
|
||||||
visitor.visit_string(x)
|
visitor.visit_string(x)
|
||||||
}
|
}
|
||||||
Some(State::Option(false)) => {
|
Some(State::OptionState(false)) => {
|
||||||
visitor.visit_none()
|
visitor.visit_none()
|
||||||
}
|
}
|
||||||
Some(State::Option(true)) => {
|
Some(State::OptionState(true)) => {
|
||||||
visitor.visit_some(self)
|
visitor.visit_some(self)
|
||||||
}
|
}
|
||||||
Some(_) => Err(Error::Syntax),
|
Some(_) => Err(Error::SyntaxError),
|
||||||
None => Err(Error::EndOfStream),
|
None => Err(Error::EndOfStream),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn deserialize_struct<V>(&mut self,
|
fn visit_named_map<V>(&mut self, name: &str, mut visitor: V) -> Result<V::Value, Error>
|
||||||
name: &str,
|
|
||||||
_fields: &'static [&'static str],
|
|
||||||
mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: de::Visitor,
|
where V: de::Visitor,
|
||||||
{
|
{
|
||||||
match self.stack.pop() {
|
match self.stack.pop() {
|
||||||
Some(State::Outer(Outer { inner })) => {
|
Some(State::OuterState(Outer { inner })) => {
|
||||||
if name != "Outer" {
|
if name != "Outer" {
|
||||||
return Err(Error::Syntax);
|
return Err(Error::SyntaxError);
|
||||||
}
|
}
|
||||||
|
|
||||||
self.stack.push(State::Vec(inner));
|
self.stack.push(State::VecState(inner));
|
||||||
self.stack.push(State::Str("inner"));
|
self.stack.push(State::StrState("inner"));
|
||||||
|
|
||||||
visitor.visit_map(OuterMapVisitor {
|
visitor.visit_map(OuterMapVisitor {
|
||||||
de: self,
|
de: self,
|
||||||
state: 0,
|
state: 0,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
Some(State::Inner(Inner { a: (), b, c })) => {
|
Some(State::InnerState(Inner { a: (), b, c })) => {
|
||||||
if name != "Inner" {
|
if name != "Inner" {
|
||||||
return Err(Error::Syntax);
|
return Err(Error::SyntaxError);
|
||||||
}
|
}
|
||||||
|
|
||||||
self.stack.push(State::Map(c));
|
self.stack.push(State::MapState(c));
|
||||||
self.stack.push(State::Str("c"));
|
self.stack.push(State::StrState("c"));
|
||||||
|
|
||||||
self.stack.push(State::Usize(b));
|
self.stack.push(State::UsizeState(b));
|
||||||
self.stack.push(State::Str("b"));
|
self.stack.push(State::StrState("b"));
|
||||||
|
|
||||||
self.stack.push(State::Null);
|
self.stack.push(State::NullState);
|
||||||
self.stack.push(State::Str("a"));
|
self.stack.push(State::StrState("a"));
|
||||||
|
|
||||||
visitor.visit_map(InnerMapVisitor {
|
visitor.visit_map(InnerMapVisitor {
|
||||||
de: self,
|
de: self,
|
||||||
@@ -443,7 +435,7 @@ mod deserializer {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -481,7 +473,7 @@ mod deserializer {
|
|||||||
if self.state == 1 {
|
if self.state == 1 {
|
||||||
Ok(())
|
Ok(())
|
||||||
} else {
|
} else {
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -504,7 +496,7 @@ mod deserializer {
|
|||||||
{
|
{
|
||||||
match self.iter.next() {
|
match self.iter.next() {
|
||||||
Some(value) => {
|
Some(value) => {
|
||||||
self.de.stack.push(State::Inner(value));
|
self.de.stack.push(State::InnerState(value));
|
||||||
Ok(Some(try!(de::Deserialize::deserialize(self.de))))
|
Ok(Some(try!(de::Deserialize::deserialize(self.de))))
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
@@ -515,7 +507,7 @@ mod deserializer {
|
|||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
match self.iter.next() {
|
match self.iter.next() {
|
||||||
Some(_) => Err(Error::Syntax),
|
Some(_) => Err(Error::SyntaxError),
|
||||||
None => Ok(()),
|
None => Ok(()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -557,7 +549,7 @@ mod deserializer {
|
|||||||
if self.state == 3 {
|
if self.state == 3 {
|
||||||
Ok(())
|
Ok(())
|
||||||
} else {
|
} else {
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -580,14 +572,14 @@ mod deserializer {
|
|||||||
{
|
{
|
||||||
match self.iter.next() {
|
match self.iter.next() {
|
||||||
Some((key, Some(value))) => {
|
Some((key, Some(value))) => {
|
||||||
self.de.stack.push(State::Char(value));
|
self.de.stack.push(State::CharState(value));
|
||||||
self.de.stack.push(State::Option(true));
|
self.de.stack.push(State::OptionState(true));
|
||||||
self.de.stack.push(State::String(key));
|
self.de.stack.push(State::StringState(key));
|
||||||
Ok(Some(try!(de::Deserialize::deserialize(self.de))))
|
Ok(Some(try!(de::Deserialize::deserialize(self.de))))
|
||||||
}
|
}
|
||||||
Some((key, None)) => {
|
Some((key, None)) => {
|
||||||
self.de.stack.push(State::Option(false));
|
self.de.stack.push(State::OptionState(false));
|
||||||
self.de.stack.push(State::String(key));
|
self.de.stack.push(State::StringState(key));
|
||||||
Ok(Some(try!(de::Deserialize::deserialize(self.de))))
|
Ok(Some(try!(de::Deserialize::deserialize(self.de))))
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
@@ -604,7 +596,7 @@ mod deserializer {
|
|||||||
|
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
match self.iter.next() {
|
match self.iter.next() {
|
||||||
Some(_) => Err(Error::Syntax),
|
Some(_) => Err(Error::SyntaxError),
|
||||||
None => Ok(()),
|
None => Ok(()),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -619,7 +611,7 @@ mod deserializer {
|
|||||||
fn bench_decoder_0_0(b: &mut Bencher) {
|
fn bench_decoder_0_0(b: &mut Bencher) {
|
||||||
b.iter(|| {
|
b.iter(|| {
|
||||||
let mut map = HashMap::new();
|
let mut map = HashMap::new();
|
||||||
map.insert("abc".to_owned(), Some('c'));
|
map.insert("abc".to_string(), Some('c'));
|
||||||
|
|
||||||
let outer = Outer {
|
let outer = Outer {
|
||||||
inner: vec!(),
|
inner: vec!(),
|
||||||
@@ -658,11 +650,11 @@ fn bench_decoder_1_0(b: &mut Bencher) {
|
|||||||
fn bench_decoder_1_5(b: &mut Bencher) {
|
fn bench_decoder_1_5(b: &mut Bencher) {
|
||||||
b.iter(|| {
|
b.iter(|| {
|
||||||
let mut map = HashMap::new();
|
let mut map = HashMap::new();
|
||||||
map.insert("1".to_owned(), Some('a'));
|
map.insert("1".to_string(), Some('a'));
|
||||||
map.insert("2".to_owned(), None);
|
map.insert("2".to_string(), None);
|
||||||
map.insert("3".to_owned(), Some('b'));
|
map.insert("3".to_string(), Some('b'));
|
||||||
map.insert("4".to_owned(), None);
|
map.insert("4".to_string(), None);
|
||||||
map.insert("5".to_owned(), Some('c'));
|
map.insert("5".to_string(), Some('c'));
|
||||||
|
|
||||||
let outer = Outer {
|
let outer = Outer {
|
||||||
inner: vec!(
|
inner: vec!(
|
||||||
@@ -721,11 +713,11 @@ fn bench_deserializer_1_0(b: &mut Bencher) {
|
|||||||
fn bench_deserializer_1_5(b: &mut Bencher) {
|
fn bench_deserializer_1_5(b: &mut Bencher) {
|
||||||
b.iter(|| {
|
b.iter(|| {
|
||||||
let mut map = HashMap::new();
|
let mut map = HashMap::new();
|
||||||
map.insert("1".to_owned(), Some('a'));
|
map.insert("1".to_string(), Some('a'));
|
||||||
map.insert("2".to_owned(), None);
|
map.insert("2".to_string(), None);
|
||||||
map.insert("3".to_owned(), Some('b'));
|
map.insert("3".to_string(), Some('b'));
|
||||||
map.insert("4".to_owned(), None);
|
map.insert("4".to_string(), None);
|
||||||
map.insert("5".to_owned(), Some('c'));
|
map.insert("5".to_string(), Some('c'));
|
||||||
|
|
||||||
let outer = Outer {
|
let outer = Outer {
|
||||||
inner: vec!(
|
inner: vec!(
|
||||||
@@ -1,6 +1,4 @@
|
|||||||
use std::fmt::Debug;
|
use std::fmt::Debug;
|
||||||
use std::fmt;
|
|
||||||
use std::error;
|
|
||||||
use test::Bencher;
|
use test::Bencher;
|
||||||
|
|
||||||
use rustc_serialize::{Decoder, Decodable};
|
use rustc_serialize::{Decoder, Decodable};
|
||||||
@@ -12,35 +10,20 @@ use serde::de::{Deserializer, Deserialize};
|
|||||||
|
|
||||||
#[derive(PartialEq, Debug)]
|
#[derive(PartialEq, Debug)]
|
||||||
pub enum Error {
|
pub enum Error {
|
||||||
EndOfStream,
|
EndOfStreamError,
|
||||||
Syntax,
|
SyntaxError,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl serde::de::Error for Error {
|
impl serde::de::Error for Error {
|
||||||
fn custom<T: Into<String>>(_: T) -> Error { Error::Syntax }
|
fn syntax_error() -> Error { Error::SyntaxError }
|
||||||
|
|
||||||
fn end_of_stream() -> Error { Error::EndOfStream }
|
fn end_of_stream_error() -> Error { Error::EndOfStreamError }
|
||||||
|
|
||||||
fn unknown_field(_: &str) -> Error { Error::Syntax }
|
fn unknown_field_error(_: &str) -> Error { Error::SyntaxError }
|
||||||
|
|
||||||
fn missing_field(_: &'static str) -> Error { Error::Syntax }
|
fn missing_field_error(_: &'static str) -> Error { Error::SyntaxError }
|
||||||
}
|
}
|
||||||
|
|
||||||
impl fmt::Display for Error {
|
|
||||||
fn fmt(&self, formatter: &mut fmt::Formatter) -> Result<(), fmt::Error> {
|
|
||||||
formatter.write_str(format!("{:?}", self).as_ref())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl error::Error for Error {
|
|
||||||
fn description(&self) -> &str {
|
|
||||||
"Serde Deserialization Error"
|
|
||||||
}
|
|
||||||
|
|
||||||
fn cause(&self) -> Option<&error::Error> {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
//////////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
mod decoder {
|
mod decoder {
|
||||||
@@ -67,104 +50,104 @@ mod decoder {
|
|||||||
impl rustc_serialize::Decoder for UsizeDecoder {
|
impl rustc_serialize::Decoder for UsizeDecoder {
|
||||||
type Error = Error;
|
type Error = Error;
|
||||||
|
|
||||||
fn error(&mut self, _: &str) -> Error { Error::Syntax }
|
fn error(&mut self, _: &str) -> Error { Error::SyntaxError }
|
||||||
|
|
||||||
// Primitive types:
|
// Primitive types:
|
||||||
fn read_nil(&mut self) -> Result<(), Error> { Err(Error::Syntax) }
|
fn read_nil(&mut self) -> Result<(), Error> { Err(Error::SyntaxError) }
|
||||||
#[inline]
|
#[inline]
|
||||||
fn read_usize(&mut self) -> Result<usize, Error> {
|
fn read_usize(&mut self) -> Result<usize, Error> {
|
||||||
match self.iter.next() {
|
match self.iter.next() {
|
||||||
Some(value) => Ok(value),
|
Some(value) => Ok(value),
|
||||||
None => Err(Error::EndOfStream),
|
None => Err(Error::EndOfStreamError),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
fn read_u64(&mut self) -> Result<u64, Error> { Err(Error::Syntax) }
|
fn read_u64(&mut self) -> Result<u64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u32(&mut self) -> Result<u32, Error> { Err(Error::Syntax) }
|
fn read_u32(&mut self) -> Result<u32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u16(&mut self) -> Result<u16, Error> { Err(Error::Syntax) }
|
fn read_u16(&mut self) -> Result<u16, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u8(&mut self) -> Result<u8, Error> { Err(Error::Syntax) }
|
fn read_u8(&mut self) -> Result<u8, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_isize(&mut self) -> Result<isize, Error> { Err(Error::Syntax) }
|
fn read_isize(&mut self) -> Result<isize, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i64(&mut self) -> Result<i64, Error> { Err(Error::Syntax) }
|
fn read_i64(&mut self) -> Result<i64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i32(&mut self) -> Result<i32, Error> { Err(Error::Syntax) }
|
fn read_i32(&mut self) -> Result<i32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i16(&mut self) -> Result<i16, Error> { Err(Error::Syntax) }
|
fn read_i16(&mut self) -> Result<i16, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i8(&mut self) -> Result<i8, Error> { Err(Error::Syntax) }
|
fn read_i8(&mut self) -> Result<i8, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_bool(&mut self) -> Result<bool, Error> { Err(Error::Syntax) }
|
fn read_bool(&mut self) -> Result<bool, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_f64(&mut self) -> Result<f64, Error> { Err(Error::Syntax) }
|
fn read_f64(&mut self) -> Result<f64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_f32(&mut self) -> Result<f32, Error> { Err(Error::Syntax) }
|
fn read_f32(&mut self) -> Result<f32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_char(&mut self) -> Result<char, Error> { Err(Error::Syntax) }
|
fn read_char(&mut self) -> Result<char, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_str(&mut self) -> Result<String, Error> { Err(Error::Syntax) }
|
fn read_str(&mut self) -> Result<String, Error> { Err(Error::SyntaxError) }
|
||||||
|
|
||||||
// Compound types:
|
// Compound types:
|
||||||
fn read_enum<T, F>(&mut self, _name: &str, _f: F) -> Result<T, Error> where
|
fn read_enum<T, F>(&mut self, _name: &str, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
fn read_enum_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_variant_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_enum_variant_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_struct_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
fn read_enum_struct_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_struct_variant_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_enum_struct_variant_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_struct_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_struct_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple<T, F>(&mut self, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple<T, F>(&mut self, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_struct_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_struct_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Specialized types:
|
// Specialized types:
|
||||||
fn read_option<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
fn read_option<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder, bool) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder, bool) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
@@ -184,19 +167,19 @@ mod decoder {
|
|||||||
fn read_map<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
fn read_map<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_map_elt_key<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
fn read_map_elt_key<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_map_elt_val<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
fn read_map_elt_val<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
F: FnOnce(&mut UsizeDecoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -219,105 +202,105 @@ mod decoder {
|
|||||||
impl rustc_serialize::Decoder for U8Decoder {
|
impl rustc_serialize::Decoder for U8Decoder {
|
||||||
type Error = Error;
|
type Error = Error;
|
||||||
|
|
||||||
fn error(&mut self, _: &str) -> Error { Error::Syntax }
|
fn error(&mut self, _: &str) -> Error { Error::SyntaxError }
|
||||||
|
|
||||||
// Primitive types:
|
// Primitive types:
|
||||||
fn read_nil(&mut self) -> Result<(), Error> { Err(Error::Syntax) }
|
fn read_nil(&mut self) -> Result<(), Error> { Err(Error::SyntaxError) }
|
||||||
fn read_usize(&mut self) -> Result<usize, Error> { Err(Error::Syntax) }
|
fn read_usize(&mut self) -> Result<usize, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u64(&mut self) -> Result<u64, Error> { Err(Error::Syntax) }
|
fn read_u64(&mut self) -> Result<u64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u32(&mut self) -> Result<u32, Error> { Err(Error::Syntax) }
|
fn read_u32(&mut self) -> Result<u32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_u16(&mut self) -> Result<u16, Error> { Err(Error::Syntax) }
|
fn read_u16(&mut self) -> Result<u16, Error> { Err(Error::SyntaxError) }
|
||||||
#[inline]
|
#[inline]
|
||||||
fn read_u8(&mut self) -> Result<u8, Error> {
|
fn read_u8(&mut self) -> Result<u8, Error> {
|
||||||
match self.iter.next() {
|
match self.iter.next() {
|
||||||
Some(value) => Ok(value),
|
Some(value) => Ok(value),
|
||||||
None => Err(Error::EndOfStream),
|
None => Err(Error::EndOfStreamError),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#[inline]
|
#[inline]
|
||||||
fn read_isize(&mut self) -> Result<isize, Error> { Err(Error::Syntax) }
|
fn read_isize(&mut self) -> Result<isize, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i64(&mut self) -> Result<i64, Error> { Err(Error::Syntax) }
|
fn read_i64(&mut self) -> Result<i64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i32(&mut self) -> Result<i32, Error> { Err(Error::Syntax) }
|
fn read_i32(&mut self) -> Result<i32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i16(&mut self) -> Result<i16, Error> { Err(Error::Syntax) }
|
fn read_i16(&mut self) -> Result<i16, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_i8(&mut self) -> Result<i8, Error> { Err(Error::Syntax) }
|
fn read_i8(&mut self) -> Result<i8, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_bool(&mut self) -> Result<bool, Error> { Err(Error::Syntax) }
|
fn read_bool(&mut self) -> Result<bool, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_f64(&mut self) -> Result<f64, Error> { Err(Error::Syntax) }
|
fn read_f64(&mut self) -> Result<f64, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_f32(&mut self) -> Result<f32, Error> { Err(Error::Syntax) }
|
fn read_f32(&mut self) -> Result<f32, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_char(&mut self) -> Result<char, Error> { Err(Error::Syntax) }
|
fn read_char(&mut self) -> Result<char, Error> { Err(Error::SyntaxError) }
|
||||||
fn read_str(&mut self) -> Result<String, Error> { Err(Error::Syntax) }
|
fn read_str(&mut self) -> Result<String, Error> { Err(Error::SyntaxError) }
|
||||||
|
|
||||||
// Compound types:
|
// Compound types:
|
||||||
fn read_enum<T, F>(&mut self, _name: &str, _f: F) -> Result<T, Error> where
|
fn read_enum<T, F>(&mut self, _name: &str, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
fn read_enum_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_variant_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_enum_variant_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_struct_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
fn read_enum_struct_variant<T, F>(&mut self, _names: &[&str], _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_enum_struct_variant_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_enum_struct_variant_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_struct_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_struct_field<T, F>(&mut self, _f_name: &str, _f_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple<T, F>(&mut self, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple<T, F>(&mut self, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_struct<T, F>(&mut self, _s_name: &str, _len: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_tuple_struct_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
fn read_tuple_struct_arg<T, F>(&mut self, _a_idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
// Specialized types:
|
// Specialized types:
|
||||||
fn read_option<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
fn read_option<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder, bool) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder, bool) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
@@ -337,19 +320,19 @@ mod decoder {
|
|||||||
fn read_map<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
fn read_map<T, F>(&mut self, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder, usize) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder, usize) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_map_elt_key<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
fn read_map_elt_key<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_map_elt_val<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
fn read_map_elt_val<T, F>(&mut self, _idx: usize, _f: F) -> Result<T, Error> where
|
||||||
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
F: FnOnce(&mut U8Decoder) -> Result<T, Error>,
|
||||||
{
|
{
|
||||||
Err(Error::Syntax)
|
Err(Error::SyntaxError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -366,9 +349,9 @@ mod deserializer {
|
|||||||
|
|
||||||
#[derive(PartialEq, Debug)]
|
#[derive(PartialEq, Debug)]
|
||||||
enum State {
|
enum State {
|
||||||
Start,
|
StartState,
|
||||||
SepOrEnd,
|
SepOrEndState,
|
||||||
End,
|
EndState,
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Deserializer<A> {
|
pub struct Deserializer<A> {
|
||||||
@@ -383,7 +366,7 @@ mod deserializer {
|
|||||||
pub fn new(values: Vec<A>) -> Deserializer<A> {
|
pub fn new(values: Vec<A>) -> Deserializer<A> {
|
||||||
let len = values.len();
|
let len = values.len();
|
||||||
Deserializer {
|
Deserializer {
|
||||||
state: State::Start,
|
state: State::StartState,
|
||||||
iter: values.into_iter(),
|
iter: values.into_iter(),
|
||||||
len: len,
|
len: len,
|
||||||
value: None,
|
value: None,
|
||||||
@@ -395,19 +378,19 @@ mod deserializer {
|
|||||||
type Error = Error;
|
type Error = Error;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn deserialize<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
fn visit<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
where V: de::Visitor,
|
where V: de::Visitor,
|
||||||
{
|
{
|
||||||
match self.state {
|
match self.state {
|
||||||
State::Start => {
|
State::StartState => {
|
||||||
self.state = State::SepOrEnd;
|
self.state = State::SepOrEndState;
|
||||||
visitor.visit_seq(self)
|
visitor.visit_seq(self)
|
||||||
}
|
}
|
||||||
State::SepOrEnd => {
|
State::SepOrEndState => {
|
||||||
visitor.visit_usize(self.value.take().unwrap())
|
visitor.visit_usize(self.value.take().unwrap())
|
||||||
}
|
}
|
||||||
State::End => {
|
State::EndState => {
|
||||||
Err(Error::EndOfStream)
|
Err(Error::EndOfStreamError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -427,7 +410,7 @@ mod deserializer {
|
|||||||
Ok(Some(try!(de::Deserialize::deserialize(self))))
|
Ok(Some(try!(de::Deserialize::deserialize(self))))
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
self.state = State::End;
|
self.state = State::EndState;
|
||||||
Ok(None)
|
Ok(None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -436,9 +419,9 @@ mod deserializer {
|
|||||||
#[inline]
|
#[inline]
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
match self.iter.next() {
|
match self.iter.next() {
|
||||||
Some(_) => Err(Error::Syntax),
|
Some(_) => Err(Error::SyntaxError),
|
||||||
None => {
|
None => {
|
||||||
self.state = State::End;
|
self.state = State::EndState;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -454,19 +437,19 @@ mod deserializer {
|
|||||||
type Error = Error;
|
type Error = Error;
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
fn deserialize<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
fn visit<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
where V: de::Visitor,
|
where V: de::Visitor,
|
||||||
{
|
{
|
||||||
match self.state {
|
match self.state {
|
||||||
State::Start => {
|
State::StartState => {
|
||||||
self.state = State::SepOrEnd;
|
self.state = State::SepOrEndState;
|
||||||
visitor.visit_seq(self)
|
visitor.visit_seq(self)
|
||||||
}
|
}
|
||||||
State::SepOrEnd => {
|
State::SepOrEndState => {
|
||||||
visitor.visit_u8(self.value.take().unwrap())
|
visitor.visit_u8(self.value.take().unwrap())
|
||||||
}
|
}
|
||||||
State::End => {
|
State::EndState => {
|
||||||
Err(Error::EndOfStream)
|
Err(Error::EndOfStreamError)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -486,7 +469,7 @@ mod deserializer {
|
|||||||
Ok(Some(try!(de::Deserialize::deserialize(self))))
|
Ok(Some(try!(de::Deserialize::deserialize(self))))
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
self.state = State::End;
|
self.state = State::EndState;
|
||||||
Ok(None)
|
Ok(None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -495,9 +478,9 @@ mod deserializer {
|
|||||||
#[inline]
|
#[inline]
|
||||||
fn end(&mut self) -> Result<(), Error> {
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
match self.iter.next() {
|
match self.iter.next() {
|
||||||
Some(_) => Err(Error::Syntax),
|
Some(_) => Err(Error::SyntaxError),
|
||||||
None => {
|
None => {
|
||||||
self.state = State::End;
|
self.state = State::EndState;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -521,14 +504,14 @@ fn run_decoder<
|
|||||||
assert_eq!(Ok(value), v);
|
assert_eq!(Ok(value), v);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn run_deserializer<D, T>(mut d: D, value: T)
|
fn run_deserializer<
|
||||||
where D: Deserializer,
|
D: Deserializer<Error=E>,
|
||||||
D::Error: Debug + PartialEq,
|
E: Debug,
|
||||||
T: Clone + PartialEq + Debug + Deserialize
|
T: Clone + PartialEq + Debug + Deserialize
|
||||||
{
|
>(mut d: D, value: T) {
|
||||||
let v = T::deserialize(&mut d);
|
let v: T = Deserialize::deserialize(&mut d).unwrap();
|
||||||
|
|
||||||
assert_eq!(Ok(value), v);
|
assert_eq!(value, v);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[bench]
|
#[bench]
|
||||||
@@ -1,3 +1,4 @@
|
|||||||
|
extern crate syntex;
|
||||||
extern crate serde_codegen;
|
extern crate serde_codegen;
|
||||||
|
|
||||||
use std::env;
|
use std::env;
|
||||||
@@ -12,6 +13,10 @@ fn main() {
|
|||||||
] {
|
] {
|
||||||
let src = Path::new(src);
|
let src = Path::new(src);
|
||||||
let dst = Path::new(&out_dir).join(dst);
|
let dst = Path::new(&out_dir).join(dst);
|
||||||
serde_codegen::expand(&src, &dst).unwrap();
|
|
||||||
|
let mut registry = syntex::Registry::new();
|
||||||
|
|
||||||
|
serde_codegen::register(&mut registry);
|
||||||
|
registry.expand("", &src, &dst).unwrap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,3 @@
|
|||||||
|
extern crate serde;
|
||||||
|
|
||||||
|
include!(concat!(env!("OUT_DIR"), "/test.rs"));
|
||||||
@@ -1,11 +1,11 @@
|
|||||||
extern crate serde;
|
/*
|
||||||
|
|
||||||
#[macro_use]
|
|
||||||
mod macros;
|
|
||||||
|
|
||||||
mod test_annotations;
|
mod test_annotations;
|
||||||
mod test_bytes;
|
mod test_bytes;
|
||||||
mod test_de;
|
mod test_de;
|
||||||
mod test_gen;
|
mod test_json;
|
||||||
|
mod test_json_builder;
|
||||||
|
*/
|
||||||
mod test_macros;
|
mod test_macros;
|
||||||
|
/*
|
||||||
mod test_ser;
|
mod test_ser;
|
||||||
|
*/
|
||||||
@@ -0,0 +1,73 @@
|
|||||||
|
use serde::json;
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
||||||
|
struct Default {
|
||||||
|
a1: i32,
|
||||||
|
#[serde(default)]
|
||||||
|
a2: i32,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
||||||
|
struct Rename {
|
||||||
|
a1: i32,
|
||||||
|
#[serde(rename="a3")]
|
||||||
|
a2: i32,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
||||||
|
struct FormatRename {
|
||||||
|
a1: i32,
|
||||||
|
#[serde(rename(xml= "a4", json="a5"))]
|
||||||
|
a2: i32,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Deserialize, Serialize)]
|
||||||
|
enum SerEnum<A> {
|
||||||
|
Map {
|
||||||
|
a: i8,
|
||||||
|
#[serde(rename(xml= "c", json="d"))]
|
||||||
|
b: A,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_default() {
|
||||||
|
let deserialized_value: Default = json::from_str(&"{\"a1\":1,\"a2\":2}").unwrap();
|
||||||
|
assert_eq!(deserialized_value, Default { a1: 1, a2: 2 });
|
||||||
|
|
||||||
|
let deserialized_value: Default = json::from_str(&"{\"a1\":1}").unwrap();
|
||||||
|
assert_eq!(deserialized_value, Default { a1: 1, a2: 0 });
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_rename() {
|
||||||
|
let value = Rename { a1: 1, a2: 2 };
|
||||||
|
let serialized_value = json::to_string(&value).unwrap();
|
||||||
|
assert_eq!(serialized_value, "{\"a1\":1,\"a3\":2}");
|
||||||
|
|
||||||
|
let deserialized_value: Rename = json::from_str(&serialized_value).unwrap();
|
||||||
|
assert_eq!(value, deserialized_value);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_format_rename() {
|
||||||
|
let value = FormatRename { a1: 1, a2: 2 };
|
||||||
|
let serialized_value = json::to_string(&value).unwrap();
|
||||||
|
assert_eq!(serialized_value, "{\"a1\":1,\"a5\":2}");
|
||||||
|
|
||||||
|
let deserialized_value = json::from_str("{\"a1\":1,\"a5\":2}").unwrap();
|
||||||
|
assert_eq!(value, deserialized_value);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_enum_format_rename() {
|
||||||
|
let s1 = String::new();
|
||||||
|
let value = SerEnum::Map { a: 0i8, b: s1 };
|
||||||
|
let serialized_value = json::to_string(&value).unwrap();
|
||||||
|
let ans = "{\"Map\":{\"a\":0,\"d\":\"\"}}";
|
||||||
|
assert_eq!(serialized_value, ans);
|
||||||
|
|
||||||
|
let deserialized_value = json::from_str(ans).unwrap();
|
||||||
|
assert_eq!(value, deserialized_value);
|
||||||
|
}
|
||||||
@@ -0,0 +1,209 @@
|
|||||||
|
use serde;
|
||||||
|
use serde::Serialize;
|
||||||
|
use serde::bytes::{ByteBuf, Bytes};
|
||||||
|
use serde::json;
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq)]
|
||||||
|
struct Error;
|
||||||
|
|
||||||
|
impl serde::de::Error for Error {
|
||||||
|
fn syntax_error() -> Error { Error }
|
||||||
|
|
||||||
|
fn end_of_stream_error() -> Error { Error }
|
||||||
|
|
||||||
|
fn unknown_field_error(_field: &str) -> Error { Error }
|
||||||
|
|
||||||
|
fn missing_field_error(_field: &'static str) -> Error { Error }
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
struct BytesSerializer {
|
||||||
|
bytes: Vec<u8>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl BytesSerializer {
|
||||||
|
fn new(bytes: Vec<u8>) -> Self {
|
||||||
|
BytesSerializer {
|
||||||
|
bytes: bytes,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl serde::Serializer for BytesSerializer {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit_unit(&mut self) -> Result<(), Error> {
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_bool(&mut self, _v: bool) -> Result<(), Error> {
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_i64(&mut self, _v: i64) -> Result<(), Error> {
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_u64(&mut self, _v: u64) -> Result<(), Error> {
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_f32(&mut self, _v: f32) -> Result<(), Error> {
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_f64(&mut self, _v: f64) -> Result<(), Error> {
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_char(&mut self, _v: char) -> Result<(), Error> {
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_str(&mut self, _v: &str) -> Result<(), Error> {
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_none(&mut self) -> Result<(), Error> {
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_some<V>(&mut self, _value: V) -> Result<(), Error>
|
||||||
|
where V: serde::Serialize,
|
||||||
|
{
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_seq<V>(&mut self, _visitor: V) -> Result<(), Error>
|
||||||
|
where V: serde::ser::SeqVisitor,
|
||||||
|
{
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_seq_elt<T>(&mut self, _value: T) -> Result<(), Error>
|
||||||
|
where T: serde::Serialize
|
||||||
|
{
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_map<V>(&mut self, _visitor: V) -> Result<(), Error>
|
||||||
|
where V: serde::ser::MapVisitor,
|
||||||
|
{
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_map_elt<K, V>(&mut self, _key: K, _value: V) -> Result<(), Error>
|
||||||
|
where K: serde::Serialize,
|
||||||
|
V: serde::Serialize,
|
||||||
|
{
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_bytes(&mut self, bytes: &[u8]) -> Result<(), Error> {
|
||||||
|
assert_eq!(self.bytes, bytes);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
struct BytesDeserializer {
|
||||||
|
bytes: Option<Vec<u8>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl BytesDeserializer {
|
||||||
|
fn new(bytes: Vec<u8>) -> Self {
|
||||||
|
BytesDeserializer {
|
||||||
|
bytes: Some(bytes),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl serde::Deserializer for BytesDeserializer {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit<V>(&mut self, _visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: serde::de::Visitor,
|
||||||
|
{
|
||||||
|
Err(Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_bytes<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: serde::de::Visitor,
|
||||||
|
{
|
||||||
|
visitor.visit_byte_buf(self.bytes.take().unwrap())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_bytes_ser_json() {
|
||||||
|
let buf = vec![];
|
||||||
|
let bytes = Bytes::from(&buf);
|
||||||
|
assert_eq!(json::to_string(&bytes).unwrap(), "[]".to_string());
|
||||||
|
|
||||||
|
let buf = vec![1, 2, 3];
|
||||||
|
let bytes = Bytes::from(&buf);
|
||||||
|
assert_eq!(json::to_string(&bytes).unwrap(), "[1,2,3]".to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_bytes_ser_bytes() {
|
||||||
|
let buf = vec![];
|
||||||
|
let bytes = Bytes::from(&buf);
|
||||||
|
let mut ser = BytesSerializer::new(vec![]);
|
||||||
|
bytes.serialize(&mut ser).unwrap();
|
||||||
|
|
||||||
|
let buf = vec![1, 2, 3];
|
||||||
|
let bytes = Bytes::from(&buf);
|
||||||
|
let mut ser = BytesSerializer::new(vec![1, 2, 3]);
|
||||||
|
bytes.serialize(&mut ser).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_byte_buf_ser_json() {
|
||||||
|
let bytes = ByteBuf::new();
|
||||||
|
assert_eq!(json::to_string(&bytes).unwrap(), "[]".to_string());
|
||||||
|
|
||||||
|
let bytes = ByteBuf::from(vec![1, 2, 3]);
|
||||||
|
assert_eq!(json::to_string(&bytes).unwrap(), "[1,2,3]".to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_byte_buf_ser_bytes() {
|
||||||
|
let bytes = ByteBuf::new();
|
||||||
|
let mut ser = BytesSerializer::new(vec![]);
|
||||||
|
bytes.serialize(&mut ser).unwrap();
|
||||||
|
|
||||||
|
let bytes = ByteBuf::from(vec![1, 2, 3]);
|
||||||
|
let mut ser = BytesSerializer::new(vec![1, 2, 3]);
|
||||||
|
bytes.serialize(&mut ser).unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_byte_buf_de_json() {
|
||||||
|
let bytes = ByteBuf::new();
|
||||||
|
let v: ByteBuf = json::from_str("[]").unwrap();
|
||||||
|
assert_eq!(v, bytes);
|
||||||
|
|
||||||
|
let bytes = ByteBuf::from(vec![1, 2, 3]);
|
||||||
|
let v: ByteBuf = json::from_str("[1, 2, 3]").unwrap();
|
||||||
|
assert_eq!(v, bytes);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_byte_buf_de_bytes() {
|
||||||
|
let mut de = BytesDeserializer::new(vec![]);
|
||||||
|
let bytes = serde::Deserialize::deserialize(&mut de);
|
||||||
|
assert_eq!(bytes, Ok(ByteBuf::new()));
|
||||||
|
|
||||||
|
let mut de = BytesDeserializer::new(vec![1, 2, 3]);
|
||||||
|
let bytes = serde::Deserialize::deserialize(&mut de);
|
||||||
|
assert_eq!(bytes, Ok(ByteBuf::from(vec![1, 2, 3])));
|
||||||
|
}
|
||||||
@@ -0,0 +1,899 @@
|
|||||||
|
use std::collections::{BTreeMap, BTreeSet, HashMap, HashSet};
|
||||||
|
use std::iter;
|
||||||
|
use std::vec;
|
||||||
|
|
||||||
|
use serde::de::{self, Deserialize, Deserializer, Visitor};
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
enum Token {
|
||||||
|
Bool(bool),
|
||||||
|
Isize(isize),
|
||||||
|
I8(i8),
|
||||||
|
I16(i16),
|
||||||
|
I32(i32),
|
||||||
|
I64(i64),
|
||||||
|
Usize(usize),
|
||||||
|
U8(u8),
|
||||||
|
U16(u16),
|
||||||
|
U32(u32),
|
||||||
|
U64(u64),
|
||||||
|
F32(f32),
|
||||||
|
F64(f64),
|
||||||
|
Char(char),
|
||||||
|
Str(&'static str),
|
||||||
|
String(String),
|
||||||
|
|
||||||
|
Option(bool),
|
||||||
|
|
||||||
|
Name(&'static str),
|
||||||
|
|
||||||
|
Unit,
|
||||||
|
|
||||||
|
SeqStart(usize),
|
||||||
|
SeqSep,
|
||||||
|
SeqEnd,
|
||||||
|
|
||||||
|
MapStart(usize),
|
||||||
|
MapSep,
|
||||||
|
MapEnd,
|
||||||
|
|
||||||
|
EnumStart(&'static str),
|
||||||
|
EnumEnd,
|
||||||
|
}
|
||||||
|
|
||||||
|
struct TokenDeserializer {
|
||||||
|
tokens: iter::Peekable<vec::IntoIter<Token>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> TokenDeserializer {
|
||||||
|
fn new(tokens: Vec<Token>) -> TokenDeserializer {
|
||||||
|
TokenDeserializer {
|
||||||
|
tokens: tokens.into_iter().peekable(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, PartialEq, Debug)]
|
||||||
|
enum Error {
|
||||||
|
SyntaxError,
|
||||||
|
EndOfStreamError,
|
||||||
|
UnknownFieldError(String),
|
||||||
|
MissingFieldError(&'static str),
|
||||||
|
InvalidName(&'static str),
|
||||||
|
}
|
||||||
|
|
||||||
|
impl de::Error for Error {
|
||||||
|
fn syntax_error() -> Error { Error::SyntaxError }
|
||||||
|
|
||||||
|
fn end_of_stream_error() -> Error { Error::EndOfStreamError }
|
||||||
|
|
||||||
|
fn unknown_field_error(field: &str) -> Error {
|
||||||
|
Error::UnknownFieldError(field.to_string())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn missing_field_error(field: &'static str) -> Error {
|
||||||
|
Error::MissingFieldError(field)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Deserializer for TokenDeserializer {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: Visitor,
|
||||||
|
{
|
||||||
|
match self.tokens.next() {
|
||||||
|
Some(Token::Bool(v)) => visitor.visit_bool(v),
|
||||||
|
Some(Token::Isize(v)) => visitor.visit_isize(v),
|
||||||
|
Some(Token::I8(v)) => visitor.visit_i8(v),
|
||||||
|
Some(Token::I16(v)) => visitor.visit_i16(v),
|
||||||
|
Some(Token::I32(v)) => visitor.visit_i32(v),
|
||||||
|
Some(Token::I64(v)) => visitor.visit_i64(v),
|
||||||
|
Some(Token::Usize(v)) => visitor.visit_usize(v),
|
||||||
|
Some(Token::U8(v)) => visitor.visit_u8(v),
|
||||||
|
Some(Token::U16(v)) => visitor.visit_u16(v),
|
||||||
|
Some(Token::U32(v)) => visitor.visit_u32(v),
|
||||||
|
Some(Token::U64(v)) => visitor.visit_u64(v),
|
||||||
|
Some(Token::F32(v)) => visitor.visit_f32(v),
|
||||||
|
Some(Token::F64(v)) => visitor.visit_f64(v),
|
||||||
|
Some(Token::Char(v)) => visitor.visit_char(v),
|
||||||
|
Some(Token::Str(v)) => visitor.visit_str(v),
|
||||||
|
Some(Token::String(v)) => visitor.visit_string(v),
|
||||||
|
Some(Token::Option(false)) => visitor.visit_none(),
|
||||||
|
Some(Token::Option(true)) => visitor.visit_some(self),
|
||||||
|
Some(Token::Unit) => visitor.visit_unit(),
|
||||||
|
Some(Token::SeqStart(len)) => {
|
||||||
|
visitor.visit_seq(TokenDeserializerSeqVisitor {
|
||||||
|
de: self,
|
||||||
|
len: len,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
Some(Token::MapStart(len)) => {
|
||||||
|
visitor.visit_map(TokenDeserializerMapVisitor {
|
||||||
|
de: self,
|
||||||
|
len: len,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
Some(Token::Name(_)) => self.visit(visitor),
|
||||||
|
Some(_) => Err(Error::SyntaxError),
|
||||||
|
None => Err(Error::EndOfStreamError),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Hook into `Option` deserializing so we can treat `Unit` as a
|
||||||
|
/// `None`, or a regular value as `Some(value)`.
|
||||||
|
fn visit_option<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: Visitor,
|
||||||
|
{
|
||||||
|
match self.tokens.peek() {
|
||||||
|
Some(&Token::Option(false)) => {
|
||||||
|
self.tokens.next();
|
||||||
|
visitor.visit_none()
|
||||||
|
}
|
||||||
|
Some(&Token::Option(true)) => {
|
||||||
|
self.tokens.next();
|
||||||
|
visitor.visit_some(self)
|
||||||
|
}
|
||||||
|
Some(&Token::Unit) => {
|
||||||
|
self.tokens.next();
|
||||||
|
visitor.visit_none()
|
||||||
|
}
|
||||||
|
Some(_) => visitor.visit_some(self),
|
||||||
|
None => Err(Error::EndOfStreamError),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_enum<V>(&mut self, name: &str, mut visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::EnumVisitor,
|
||||||
|
{
|
||||||
|
match self.tokens.next() {
|
||||||
|
Some(Token::EnumStart(n)) => {
|
||||||
|
if name == n {
|
||||||
|
visitor.visit(TokenDeserializerVariantVisitor {
|
||||||
|
de: self,
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
Err(Error::SyntaxError)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some(_) => Err(Error::SyntaxError),
|
||||||
|
None => Err(Error::EndOfStreamError),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_named_unit<V>(&mut self, name: &str, visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
match self.tokens.peek() {
|
||||||
|
Some(&Token::Name(n)) => {
|
||||||
|
if name == n {
|
||||||
|
self.tokens.next();
|
||||||
|
self.visit_seq(visitor)
|
||||||
|
} else {
|
||||||
|
Err(Error::InvalidName(n))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some(_) => self.visit(visitor),
|
||||||
|
None => Err(Error::EndOfStreamError),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_named_seq<V>(&mut self, name: &str, visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
match self.tokens.peek() {
|
||||||
|
Some(&Token::Name(n)) => {
|
||||||
|
if name == n {
|
||||||
|
self.tokens.next();
|
||||||
|
self.visit_seq(visitor)
|
||||||
|
} else {
|
||||||
|
Err(Error::InvalidName(n))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some(_) => self.visit_seq(visitor),
|
||||||
|
None => Err(Error::EndOfStreamError),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_named_map<V>(&mut self, name: &str, visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
match self.tokens.peek() {
|
||||||
|
Some(&Token::Name(n)) => {
|
||||||
|
if name == n {
|
||||||
|
self.tokens.next();
|
||||||
|
self.visit_map(visitor)
|
||||||
|
} else {
|
||||||
|
Err(Error::InvalidName(n))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some(_) => self.visit_map(visitor),
|
||||||
|
None => Err(Error::EndOfStreamError),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
struct TokenDeserializerSeqVisitor<'a> {
|
||||||
|
de: &'a mut TokenDeserializer,
|
||||||
|
len: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> de::SeqVisitor for TokenDeserializerSeqVisitor<'a> {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit<T>(&mut self) -> Result<Option<T>, Error>
|
||||||
|
where T: Deserialize,
|
||||||
|
{
|
||||||
|
match self.de.tokens.peek() {
|
||||||
|
Some(&Token::SeqSep) => {
|
||||||
|
self.len -= 1;
|
||||||
|
self.de.tokens.next();
|
||||||
|
Ok(Some(try!(Deserialize::deserialize(self.de))))
|
||||||
|
}
|
||||||
|
Some(&Token::SeqEnd) => Ok(None),
|
||||||
|
Some(_) => {
|
||||||
|
Err(Error::SyntaxError)
|
||||||
|
}
|
||||||
|
None => Err(Error::EndOfStreamError),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
|
assert_eq!(self.len, 0);
|
||||||
|
match self.de.tokens.next() {
|
||||||
|
Some(Token::SeqEnd) => Ok(()),
|
||||||
|
Some(_) => Err(Error::SyntaxError),
|
||||||
|
None => Err(Error::EndOfStreamError),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
||||||
|
(self.len, Some(self.len))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
struct TokenDeserializerMapVisitor<'a> {
|
||||||
|
de: &'a mut TokenDeserializer,
|
||||||
|
len: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> de::MapVisitor for TokenDeserializerMapVisitor<'a> {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit_key<K>(&mut self) -> Result<Option<K>, Error>
|
||||||
|
where K: Deserialize,
|
||||||
|
{
|
||||||
|
match self.de.tokens.peek() {
|
||||||
|
Some(&Token::MapSep) => {
|
||||||
|
self.de.tokens.next();
|
||||||
|
self.len -= 1;
|
||||||
|
Ok(Some(try!(Deserialize::deserialize(self.de))))
|
||||||
|
}
|
||||||
|
Some(&Token::MapEnd) => Ok(None),
|
||||||
|
Some(_) => Err(Error::SyntaxError),
|
||||||
|
None => Err(Error::EndOfStreamError),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_value<V>(&mut self) -> Result<V, Error>
|
||||||
|
where V: Deserialize,
|
||||||
|
{
|
||||||
|
Ok(try!(Deserialize::deserialize(self.de)))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn end(&mut self) -> Result<(), Error> {
|
||||||
|
assert_eq!(self.len, 0);
|
||||||
|
match self.de.tokens.next() {
|
||||||
|
Some(Token::MapEnd) => Ok(()),
|
||||||
|
Some(_) => Err(Error::SyntaxError),
|
||||||
|
None => Err(Error::EndOfStreamError),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
||||||
|
(self.len, Some(self.len))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
struct TokenDeserializerVariantVisitor<'a> {
|
||||||
|
de: &'a mut TokenDeserializer,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> de::VariantVisitor for TokenDeserializerVariantVisitor<'a> {
|
||||||
|
type Error = Error;
|
||||||
|
|
||||||
|
fn visit_variant<V>(&mut self) -> Result<V, Error>
|
||||||
|
where V: de::Deserialize,
|
||||||
|
{
|
||||||
|
de::Deserialize::deserialize(self.de)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_unit(&mut self) -> Result<(), Error> {
|
||||||
|
de::Deserialize::deserialize(self.de)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_seq<V>(&mut self, visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
de::Deserializer::visit(self.de, visitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_map<V>(&mut self, visitor: V) -> Result<V::Value, Error>
|
||||||
|
where V: de::Visitor,
|
||||||
|
{
|
||||||
|
de::Deserializer::visit(self.de, visitor)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
#[derive(Copy, Clone, PartialEq, Debug, Deserialize)]
|
||||||
|
struct NamedUnit;
|
||||||
|
|
||||||
|
#[derive(PartialEq, Debug, Deserialize)]
|
||||||
|
struct NamedSeq(i32, i32, i32);
|
||||||
|
|
||||||
|
#[derive(PartialEq, Debug, Deserialize)]
|
||||||
|
struct NamedMap {
|
||||||
|
a: i32,
|
||||||
|
b: i32,
|
||||||
|
c: i32,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq, Debug, Deserialize)]
|
||||||
|
enum Enum {
|
||||||
|
Unit,
|
||||||
|
Seq(i32, i32, i32),
|
||||||
|
Map { a: i32, b: i32, c: i32 }
|
||||||
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
macro_rules! btreeset {
|
||||||
|
() => {
|
||||||
|
BTreeSet::new()
|
||||||
|
};
|
||||||
|
($($value:expr),+) => {
|
||||||
|
{
|
||||||
|
let mut set = BTreeSet::new();
|
||||||
|
$(set.insert($value);)+
|
||||||
|
set
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! btreemap {
|
||||||
|
() => {
|
||||||
|
BTreeMap::new()
|
||||||
|
};
|
||||||
|
($($key:expr => $value:expr),+) => {
|
||||||
|
{
|
||||||
|
let mut map = BTreeMap::new();
|
||||||
|
$(map.insert($key, $value);)+
|
||||||
|
map
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! hashset {
|
||||||
|
() => {
|
||||||
|
HashSet::new()
|
||||||
|
};
|
||||||
|
($($value:expr),+) => {
|
||||||
|
{
|
||||||
|
let mut set = HashSet::new();
|
||||||
|
$(set.insert($value);)+
|
||||||
|
set
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! hashmap {
|
||||||
|
() => {
|
||||||
|
HashMap::new()
|
||||||
|
};
|
||||||
|
($($key:expr => $value:expr),+) => {
|
||||||
|
{
|
||||||
|
let mut map = HashMap::new();
|
||||||
|
$(map.insert($key, $value);)+
|
||||||
|
map
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! declare_test {
|
||||||
|
($name:ident { $($value:expr => $tokens:expr,)+ }) => {
|
||||||
|
#[test]
|
||||||
|
fn $name() {
|
||||||
|
$(
|
||||||
|
let mut de = TokenDeserializer::new($tokens);
|
||||||
|
let value: Result<_, Error> = Deserialize::deserialize(&mut de);
|
||||||
|
assert_eq!(value, Ok($value));
|
||||||
|
)+
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! declare_tests {
|
||||||
|
($($name:ident { $($value:expr => $tokens:expr,)+ })+) => {
|
||||||
|
$(
|
||||||
|
declare_test!($name { $($value => $tokens,)+ });
|
||||||
|
)+
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
declare_tests! {
|
||||||
|
test_bool {
|
||||||
|
true => vec![Token::Bool(true)],
|
||||||
|
false => vec![Token::Bool(false)],
|
||||||
|
}
|
||||||
|
test_isize {
|
||||||
|
0isize => vec![Token::Isize(0)],
|
||||||
|
0isize => vec![Token::I8(0)],
|
||||||
|
0isize => vec![Token::I16(0)],
|
||||||
|
0isize => vec![Token::I32(0)],
|
||||||
|
0isize => vec![Token::I64(0)],
|
||||||
|
0isize => vec![Token::Usize(0)],
|
||||||
|
0isize => vec![Token::U8(0)],
|
||||||
|
0isize => vec![Token::U16(0)],
|
||||||
|
0isize => vec![Token::U32(0)],
|
||||||
|
0isize => vec![Token::U64(0)],
|
||||||
|
0isize => vec![Token::F32(0.)],
|
||||||
|
0isize => vec![Token::F64(0.)],
|
||||||
|
}
|
||||||
|
test_ints {
|
||||||
|
0isize => vec![Token::Isize(0)],
|
||||||
|
0i8 => vec![Token::I8(0)],
|
||||||
|
0i16 => vec![Token::I16(0)],
|
||||||
|
0i32 => vec![Token::I32(0)],
|
||||||
|
0i64 => vec![Token::I64(0)],
|
||||||
|
}
|
||||||
|
test_uints {
|
||||||
|
0usize => vec![Token::Usize(0)],
|
||||||
|
0u8 => vec![Token::U8(0)],
|
||||||
|
0u16 => vec![Token::U16(0)],
|
||||||
|
0u32 => vec![Token::U32(0)],
|
||||||
|
0u64 => vec![Token::U64(0)],
|
||||||
|
}
|
||||||
|
test_floats {
|
||||||
|
0f32 => vec![Token::F32(0.)],
|
||||||
|
0f64 => vec![Token::F64(0.)],
|
||||||
|
}
|
||||||
|
test_char {
|
||||||
|
'a' => vec![Token::Char('a')],
|
||||||
|
'a' => vec![Token::Str("a")],
|
||||||
|
'a' => vec![Token::String("a".to_string())],
|
||||||
|
}
|
||||||
|
test_string {
|
||||||
|
"abc".to_string() => vec![Token::Str("abc")],
|
||||||
|
"abc".to_string() => vec![Token::String("abc".to_string())],
|
||||||
|
"a".to_string() => vec![Token::Char('a')],
|
||||||
|
}
|
||||||
|
test_option {
|
||||||
|
None::<i32> => vec![Token::Unit],
|
||||||
|
None::<i32> => vec![Token::Option(false)],
|
||||||
|
Some(1) => vec![Token::I32(1)],
|
||||||
|
Some(1) => vec![
|
||||||
|
Token::Option(true),
|
||||||
|
Token::I32(1),
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_unit {
|
||||||
|
() => vec![Token::Unit],
|
||||||
|
() => vec![
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
() => vec![
|
||||||
|
Token::Name("Anything"),
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_named_unit {
|
||||||
|
NamedUnit => vec![Token::Unit],
|
||||||
|
NamedUnit => vec![
|
||||||
|
Token::Name("NamedUnit"),
|
||||||
|
Token::Unit,
|
||||||
|
],
|
||||||
|
NamedUnit => vec![
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_named_seq {
|
||||||
|
NamedSeq(1, 2, 3) => vec![
|
||||||
|
Token::SeqStart(3),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
NamedSeq(1, 2, 3) => vec![
|
||||||
|
Token::Name("NamedSeq"),
|
||||||
|
Token::SeqStart(3),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_btreeset {
|
||||||
|
BTreeSet::<isize>::new() => vec![
|
||||||
|
Token::Unit,
|
||||||
|
],
|
||||||
|
BTreeSet::<isize>::new() => vec![
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
btreeset![btreeset![], btreeset![1], btreeset![2, 3]] => vec![
|
||||||
|
Token::SeqStart(3),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::SeqStart(1),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::SeqEnd,
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::SeqStart(2),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::SeqEnd,
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
BTreeSet::<isize>::new() => vec![
|
||||||
|
Token::Name("Anything"),
|
||||||
|
Token::Unit,
|
||||||
|
],
|
||||||
|
BTreeSet::<isize>::new() => vec![
|
||||||
|
Token::Name("Anything"),
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_hashset {
|
||||||
|
HashSet::<isize>::new() => vec![
|
||||||
|
Token::Unit,
|
||||||
|
],
|
||||||
|
HashSet::<isize>::new() => vec![
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
hashset![1, 2, 3] => vec![
|
||||||
|
Token::SeqStart(3),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
HashSet::<isize>::new() => vec![
|
||||||
|
Token::Name("Anything"),
|
||||||
|
Token::Unit,
|
||||||
|
],
|
||||||
|
HashSet::<isize>::new() => vec![
|
||||||
|
Token::Name("Anything"),
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_vec {
|
||||||
|
Vec::<isize>::new() => vec![
|
||||||
|
Token::Unit,
|
||||||
|
],
|
||||||
|
Vec::<isize>::new() => vec![
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
vec![vec![], vec![1], vec![2, 3]] => vec![
|
||||||
|
Token::SeqStart(3),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::SeqStart(1),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::SeqEnd,
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::SeqStart(2),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::SeqEnd,
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
Vec::<isize>::new() => vec![
|
||||||
|
Token::Name("Anything"),
|
||||||
|
Token::Unit,
|
||||||
|
],
|
||||||
|
Vec::<isize>::new() => vec![
|
||||||
|
Token::Name("Anything"),
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_array {
|
||||||
|
[0; 0] => vec![
|
||||||
|
Token::Unit,
|
||||||
|
],
|
||||||
|
[0; 0] => vec![
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
([0; 0], [1], [2, 3]) => vec![
|
||||||
|
Token::SeqStart(3),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::SeqStart(1),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::SeqEnd,
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::SeqStart(2),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::SeqEnd,
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
[0; 0] => vec![
|
||||||
|
Token::Name("Anything"),
|
||||||
|
Token::Unit,
|
||||||
|
],
|
||||||
|
[0; 0] => vec![
|
||||||
|
Token::Name("Anything"),
|
||||||
|
Token::SeqStart(0),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_tuple {
|
||||||
|
(1,) => vec![
|
||||||
|
Token::SeqStart(1),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
(1, 2, 3) => vec![
|
||||||
|
Token::SeqStart(3),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_btreemap {
|
||||||
|
BTreeMap::<isize, isize>::new() => vec![
|
||||||
|
Token::Unit,
|
||||||
|
],
|
||||||
|
BTreeMap::<isize, isize>::new() => vec![
|
||||||
|
Token::MapStart(0),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
btreemap![1 => 2] => vec![
|
||||||
|
Token::MapStart(1),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::I32(2),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
btreemap![1 => 2, 3 => 4] => vec![
|
||||||
|
Token::MapStart(2),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::I32(4),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
btreemap![1 => btreemap![], 2 => btreemap![3 => 4, 5 => 6]] => vec![
|
||||||
|
Token::MapStart(2),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::MapStart(0),
|
||||||
|
Token::MapEnd,
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(2),
|
||||||
|
Token::MapStart(2),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::I32(4),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(5),
|
||||||
|
Token::I32(6),
|
||||||
|
Token::MapEnd,
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
BTreeMap::<isize, isize>::new() => vec![
|
||||||
|
Token::Name("Anything"),
|
||||||
|
Token::Unit,
|
||||||
|
],
|
||||||
|
BTreeMap::<isize, isize>::new() => vec![
|
||||||
|
Token::Name("Anything"),
|
||||||
|
Token::MapStart(0),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_hashmap {
|
||||||
|
HashMap::<isize, isize>::new() => vec![
|
||||||
|
Token::Unit,
|
||||||
|
],
|
||||||
|
HashMap::<isize, isize>::new() => vec![
|
||||||
|
Token::MapStart(0),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
hashmap![1 => 2] => vec![
|
||||||
|
Token::MapStart(1),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::I32(2),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
hashmap![1 => 2, 3 => 4] => vec![
|
||||||
|
Token::MapStart(2),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::I32(4),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
hashmap![1 => hashmap![], 2 => hashmap![3 => 4, 5 => 6]] => vec![
|
||||||
|
Token::MapStart(2),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::MapStart(0),
|
||||||
|
Token::MapEnd,
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(2),
|
||||||
|
Token::MapStart(2),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::I32(4),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(5),
|
||||||
|
Token::I32(6),
|
||||||
|
Token::MapEnd,
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
HashMap::<isize, isize>::new() => vec![
|
||||||
|
Token::Name("Anything"),
|
||||||
|
Token::Unit,
|
||||||
|
],
|
||||||
|
HashMap::<isize, isize>::new() => vec![
|
||||||
|
Token::Name("Anything"),
|
||||||
|
Token::MapStart(0),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_named_map {
|
||||||
|
NamedMap { a: 1, b: 2, c: 3 } => vec![
|
||||||
|
Token::MapStart(3),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("a"),
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("b"),
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("c"),
|
||||||
|
Token::I32(3),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
NamedMap { a: 1, b: 2, c: 3 } => vec![
|
||||||
|
Token::Name("NamedMap"),
|
||||||
|
Token::MapStart(3),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("a"),
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("b"),
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("c"),
|
||||||
|
Token::I32(3),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_enum_unit {
|
||||||
|
Enum::Unit => vec![
|
||||||
|
Token::EnumStart("Enum"),
|
||||||
|
Token::Str("Unit"),
|
||||||
|
Token::Unit,
|
||||||
|
Token::EnumEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_enum_seq {
|
||||||
|
Enum::Seq(1, 2, 3) => vec![
|
||||||
|
Token::EnumStart("Enum"),
|
||||||
|
Token::Str("Seq"),
|
||||||
|
Token::SeqStart(3),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::SeqEnd,
|
||||||
|
Token::EnumEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_enum_map {
|
||||||
|
Enum::Map { a: 1, b: 2, c: 3 } => vec![
|
||||||
|
Token::EnumStart("Enum"),
|
||||||
|
Token::Str("Map"),
|
||||||
|
Token::MapStart(3),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("a"),
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("b"),
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("c"),
|
||||||
|
Token::I32(3),
|
||||||
|
Token::MapEnd,
|
||||||
|
Token::EnumEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
}
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -0,0 +1,50 @@
|
|||||||
|
use std::collections::BTreeMap;
|
||||||
|
|
||||||
|
use serde::json::value::Value;
|
||||||
|
use serde::json::builder::{ArrayBuilder, ObjectBuilder};
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_array_builder() {
|
||||||
|
let value = ArrayBuilder::new().unwrap();
|
||||||
|
assert_eq!(value, Value::Array(Vec::new()));
|
||||||
|
|
||||||
|
let value = ArrayBuilder::new()
|
||||||
|
.push(1)
|
||||||
|
.push(2)
|
||||||
|
.push(3)
|
||||||
|
.unwrap();
|
||||||
|
assert_eq!(value, Value::Array(vec!(Value::U64(1), Value::U64(2), Value::U64(3))));
|
||||||
|
|
||||||
|
let value = ArrayBuilder::new()
|
||||||
|
.push_array(|bld| bld.push(1).push(2).push(3))
|
||||||
|
.unwrap();
|
||||||
|
assert_eq!(value, Value::Array(vec!(Value::Array(vec!(Value::U64(1), Value::U64(2), Value::U64(3))))));
|
||||||
|
|
||||||
|
let value = ArrayBuilder::new()
|
||||||
|
.push_object(|bld|
|
||||||
|
bld
|
||||||
|
.insert("a".to_string(), 1)
|
||||||
|
.insert("b".to_string(), 2))
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let mut map = BTreeMap::new();
|
||||||
|
map.insert("a".to_string(), Value::U64(1));
|
||||||
|
map.insert("b".to_string(), Value::U64(2));
|
||||||
|
assert_eq!(value, Value::Array(vec!(Value::Object(map))));
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_object_builder() {
|
||||||
|
let value = ObjectBuilder::new().unwrap();
|
||||||
|
assert_eq!(value, Value::Object(BTreeMap::new()));
|
||||||
|
|
||||||
|
let value = ObjectBuilder::new()
|
||||||
|
.insert("a".to_string(), 1)
|
||||||
|
.insert("b".to_string(), 2)
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let mut map = BTreeMap::new();
|
||||||
|
map.insert("a".to_string(), Value::U64(1));
|
||||||
|
map.insert("b".to_string(), Value::U64(2));
|
||||||
|
assert_eq!(value, Value::Object(map));
|
||||||
|
}
|
||||||
@@ -0,0 +1,477 @@
|
|||||||
|
use std::collections::BTreeMap;
|
||||||
|
use serde::json::{self, Value};
|
||||||
|
|
||||||
|
macro_rules! btreemap {
|
||||||
|
() => {
|
||||||
|
BTreeMap::new()
|
||||||
|
};
|
||||||
|
($($key:expr => $value:expr),+) => {
|
||||||
|
{
|
||||||
|
let mut map = BTreeMap::new();
|
||||||
|
$(map.insert($key, $value);)+
|
||||||
|
map
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
trait Trait {
|
||||||
|
type Type;
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
||||||
|
struct NamedUnit;
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Serialize)]
|
||||||
|
struct SerNamedTuple<'a, 'b, A: 'a, B: 'b, C>(&'a A, &'b mut B, C);
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Deserialize)]
|
||||||
|
struct DeNamedTuple<A, B, C>(A, B, C);
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Serialize)]
|
||||||
|
struct SerNamedMap<'a, 'b, A: 'a, B: 'b, C> {
|
||||||
|
a: &'a A,
|
||||||
|
b: &'b mut B,
|
||||||
|
c: C,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Deserialize)]
|
||||||
|
struct DeNamedMap<A, B, C> {
|
||||||
|
a: A,
|
||||||
|
b: B,
|
||||||
|
c: C,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Serialize)]
|
||||||
|
enum SerEnum<'a, B: 'a, C: /* Trait + */ 'a, D> where D: /* Trait + */ 'a {
|
||||||
|
Unit,
|
||||||
|
Seq(
|
||||||
|
i8,
|
||||||
|
B,
|
||||||
|
&'a C,
|
||||||
|
//C::Type,
|
||||||
|
&'a mut D,
|
||||||
|
//<D as Trait>::Type,
|
||||||
|
),
|
||||||
|
Map {
|
||||||
|
a: i8,
|
||||||
|
b: B,
|
||||||
|
c: &'a C,
|
||||||
|
//d: C::Type,
|
||||||
|
e: &'a mut D,
|
||||||
|
//f: <D as Trait>::Type,
|
||||||
|
},
|
||||||
|
|
||||||
|
// Make sure we can support more than one variant.
|
||||||
|
_Unit2,
|
||||||
|
_Seq2(
|
||||||
|
i8,
|
||||||
|
B,
|
||||||
|
&'a C,
|
||||||
|
//C::Type,
|
||||||
|
&'a mut D,
|
||||||
|
//<D as Trait>::Type,
|
||||||
|
),
|
||||||
|
_Map2 {
|
||||||
|
a: i8,
|
||||||
|
b: B,
|
||||||
|
c: &'a C,
|
||||||
|
//d: C::Type,
|
||||||
|
e: &'a mut D,
|
||||||
|
//f: <D as Trait>::Type,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Deserialize)]
|
||||||
|
enum DeEnum<B, C: /* Trait */, D> /* where D: Trait */ {
|
||||||
|
Unit,
|
||||||
|
Seq(
|
||||||
|
i8,
|
||||||
|
B,
|
||||||
|
C,
|
||||||
|
//C::Type,
|
||||||
|
D,
|
||||||
|
//<D as Trait>::Type,
|
||||||
|
),
|
||||||
|
Map {
|
||||||
|
a: i8,
|
||||||
|
b: B,
|
||||||
|
c: C,
|
||||||
|
//d: C::Type,
|
||||||
|
e: D,
|
||||||
|
//f: <D as Trait>::Type,
|
||||||
|
},
|
||||||
|
|
||||||
|
// Make sure we can support more than one variant.
|
||||||
|
_Unit2,
|
||||||
|
_Seq2(
|
||||||
|
i8,
|
||||||
|
B,
|
||||||
|
C,
|
||||||
|
//C::Type,
|
||||||
|
D,
|
||||||
|
//<D as Trait>::Type,
|
||||||
|
),
|
||||||
|
_Map2 {
|
||||||
|
a: i8,
|
||||||
|
b: B,
|
||||||
|
c: C,
|
||||||
|
//d: C::Type,
|
||||||
|
e: D,
|
||||||
|
//f: <D as Trait>::Type,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize)]
|
||||||
|
enum Lifetimes<'a> {
|
||||||
|
LifetimeSeq(&'a i32),
|
||||||
|
NoLifetimeSeq(i32),
|
||||||
|
LifetimeMap { a: &'a i32 },
|
||||||
|
NoLifetimeMap { a: i32 },
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_named_unit() {
|
||||||
|
let named_unit = NamedUnit;
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
json::to_string(&named_unit).unwrap(),
|
||||||
|
"null".to_string()
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
json::to_value(&named_unit),
|
||||||
|
Value::Null
|
||||||
|
);
|
||||||
|
|
||||||
|
let v: NamedUnit = json::from_str("null").unwrap();
|
||||||
|
assert_eq!(v, named_unit);
|
||||||
|
|
||||||
|
let v: NamedUnit = json::from_value(Value::Null).unwrap();
|
||||||
|
assert_eq!(v, named_unit);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_ser_named_tuple() {
|
||||||
|
let a = 5;
|
||||||
|
let mut b = 6;
|
||||||
|
let c = 7;
|
||||||
|
let named_tuple = SerNamedTuple(&a, &mut b, c);
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
json::to_string(&named_tuple).unwrap(),
|
||||||
|
"[5,6,7]"
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
json::to_value(&named_tuple),
|
||||||
|
Value::Array(vec![Value::U64(5), Value::U64(6), Value::U64(7)])
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_de_named_tuple() {
|
||||||
|
let v: DeNamedTuple<i32, i32, i32> = json::from_str("[1,2,3]").unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
v,
|
||||||
|
DeNamedTuple(1, 2, 3)
|
||||||
|
);
|
||||||
|
|
||||||
|
let v: Value = json::from_str("[1,2,3]").unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
v,
|
||||||
|
Value::Array(vec![
|
||||||
|
Value::U64(1),
|
||||||
|
Value::U64(2),
|
||||||
|
Value::U64(3),
|
||||||
|
])
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_ser_named_map() {
|
||||||
|
let a = 5;
|
||||||
|
let mut b = 6;
|
||||||
|
let c = 7;
|
||||||
|
let named_map = SerNamedMap {
|
||||||
|
a: &a,
|
||||||
|
b: &mut b,
|
||||||
|
c: c,
|
||||||
|
};
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
json::to_string(&named_map).unwrap(),
|
||||||
|
"{\"a\":5,\"b\":6,\"c\":7}"
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
json::to_value(&named_map),
|
||||||
|
Value::Object(btreemap![
|
||||||
|
"a".to_string() => Value::U64(5),
|
||||||
|
"b".to_string() => Value::U64(6),
|
||||||
|
"c".to_string() => Value::U64(7)
|
||||||
|
])
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_de_named_map() {
|
||||||
|
let v = DeNamedMap {
|
||||||
|
a: 5,
|
||||||
|
b: 6,
|
||||||
|
c: 7,
|
||||||
|
};
|
||||||
|
|
||||||
|
let v2: DeNamedMap<i32, i32, i32> = json::from_str(
|
||||||
|
"{\"a\":5,\"b\":6,\"c\":7}"
|
||||||
|
).unwrap();
|
||||||
|
assert_eq!(v, v2);
|
||||||
|
|
||||||
|
let v2 = json::from_value(Value::Object(btreemap![
|
||||||
|
"a".to_string() => Value::U64(5),
|
||||||
|
"b".to_string() => Value::U64(6),
|
||||||
|
"c".to_string() => Value::U64(7)
|
||||||
|
])).unwrap();
|
||||||
|
assert_eq!(v, v2);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_ser_enum_unit() {
|
||||||
|
assert_eq!(
|
||||||
|
json::to_string(&SerEnum::Unit::<u32, u32, u32>).unwrap(),
|
||||||
|
"{\"Unit\":[]}"
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
json::to_value(&SerEnum::Unit::<u32, u32, u32>),
|
||||||
|
Value::Object(btreemap!(
|
||||||
|
"Unit".to_string() => Value::Array(vec![]))
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_ser_enum_seq() {
|
||||||
|
let a = 1;
|
||||||
|
let b = 2;
|
||||||
|
let c = 3;
|
||||||
|
//let d = 4;
|
||||||
|
let mut e = 5;
|
||||||
|
//let f = 6;
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
json::to_string(&SerEnum::Seq(
|
||||||
|
a,
|
||||||
|
b,
|
||||||
|
&c,
|
||||||
|
//d,
|
||||||
|
&mut e,
|
||||||
|
//f,
|
||||||
|
)).unwrap(),
|
||||||
|
"{\"Seq\":[1,2,3,5]}".to_string()
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
json::to_value(&SerEnum::Seq(
|
||||||
|
a,
|
||||||
|
b,
|
||||||
|
&c,
|
||||||
|
//d,
|
||||||
|
&mut e,
|
||||||
|
//e,
|
||||||
|
)),
|
||||||
|
Value::Object(btreemap!(
|
||||||
|
"Seq".to_string() => Value::Array(vec![
|
||||||
|
Value::U64(1),
|
||||||
|
Value::U64(2),
|
||||||
|
Value::U64(3),
|
||||||
|
//Value::U64(4),
|
||||||
|
Value::U64(5),
|
||||||
|
//Value::U64(6),
|
||||||
|
])
|
||||||
|
))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_ser_enum_map() {
|
||||||
|
let a = 1;
|
||||||
|
let b = 2;
|
||||||
|
let c = 3;
|
||||||
|
//let d = 4;
|
||||||
|
let mut e = 5;
|
||||||
|
//let f = 6;
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
json::to_string(&SerEnum::Map {
|
||||||
|
a: a,
|
||||||
|
b: b,
|
||||||
|
c: &c,
|
||||||
|
//d: d,
|
||||||
|
e: &mut e,
|
||||||
|
//f: f,
|
||||||
|
}).unwrap(),
|
||||||
|
"{\"Map\":{\"a\":1,\"b\":2,\"c\":3,\"e\":5}}".to_string()
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
json::to_value(&SerEnum::Map {
|
||||||
|
a: a,
|
||||||
|
b: b,
|
||||||
|
c: &c,
|
||||||
|
//d: d,
|
||||||
|
e: &mut e,
|
||||||
|
//f: f,
|
||||||
|
}),
|
||||||
|
Value::Object(btreemap!(
|
||||||
|
"Map".to_string() => Value::Object(btreemap![
|
||||||
|
"a".to_string() => Value::U64(1),
|
||||||
|
"b".to_string() => Value::U64(2),
|
||||||
|
"c".to_string() => Value::U64(3),
|
||||||
|
//"d".to_string() => Value::U64(4)
|
||||||
|
"e".to_string() => Value::U64(5)
|
||||||
|
//"f".to_string() => Value::U64(6)
|
||||||
|
])
|
||||||
|
))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_de_enum_unit() {
|
||||||
|
let v: DeEnum<_, _, _> = json::from_str("{\"Unit\":[]}").unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
v,
|
||||||
|
DeEnum::Unit::<u32, u32, u32>
|
||||||
|
);
|
||||||
|
|
||||||
|
let v: DeEnum<_, _, _> = json::from_value(Value::Object(btreemap!(
|
||||||
|
"Unit".to_string() => Value::Array(vec![]))
|
||||||
|
)).unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
v,
|
||||||
|
DeEnum::Unit::<u32, u32, u32>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_de_enum_seq() {
|
||||||
|
let a = 1;
|
||||||
|
let b = 2;
|
||||||
|
let c = 3;
|
||||||
|
//let d = 4;
|
||||||
|
let e = 5;
|
||||||
|
//let f = 6;
|
||||||
|
|
||||||
|
let v: DeEnum<_, _, _> = json::from_str("{\"Seq\":[1,2,3,5]}").unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
v,
|
||||||
|
DeEnum::Seq(
|
||||||
|
a,
|
||||||
|
b,
|
||||||
|
c,
|
||||||
|
//d,
|
||||||
|
e,
|
||||||
|
//f,
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
let v: DeEnum<_, _, _> = json::from_value(Value::Object(btreemap!(
|
||||||
|
"Seq".to_string() => Value::Array(vec![
|
||||||
|
Value::U64(1),
|
||||||
|
Value::U64(2),
|
||||||
|
Value::U64(3),
|
||||||
|
//Value::U64(4),
|
||||||
|
Value::U64(5),
|
||||||
|
//Value::U64(6),
|
||||||
|
])
|
||||||
|
))).unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
v,
|
||||||
|
DeEnum::Seq(
|
||||||
|
a,
|
||||||
|
b,
|
||||||
|
c,
|
||||||
|
//d,
|
||||||
|
e,
|
||||||
|
//e,
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_de_enum_map() {
|
||||||
|
let a = 1;
|
||||||
|
let b = 2;
|
||||||
|
let c = 3;
|
||||||
|
//let d = 4;
|
||||||
|
let e = 5;
|
||||||
|
//let f = 6;
|
||||||
|
|
||||||
|
let v: DeEnum<_, _, _> = json::from_str(
|
||||||
|
"{\"Map\":{\"a\":1,\"b\":2,\"c\":3,\"e\":5}}"
|
||||||
|
).unwrap();
|
||||||
|
assert_eq!(
|
||||||
|
v,
|
||||||
|
DeEnum::Map {
|
||||||
|
a: a,
|
||||||
|
b: b,
|
||||||
|
c: c,
|
||||||
|
//d: d,
|
||||||
|
e: e,
|
||||||
|
//f: f,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
let v: DeEnum<_, _, _> = json::from_value(Value::Object(btreemap!(
|
||||||
|
"Map".to_string() => Value::Object(btreemap![
|
||||||
|
"a".to_string() => Value::U64(1),
|
||||||
|
"b".to_string() => Value::U64(2),
|
||||||
|
"c".to_string() => Value::U64(3),
|
||||||
|
//"d".to_string() => Value::U64(4)
|
||||||
|
"e".to_string() => Value::U64(5)
|
||||||
|
//"f".to_string() => Value::U64(6)
|
||||||
|
])
|
||||||
|
))).unwrap();
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
v,
|
||||||
|
DeEnum::Map {
|
||||||
|
a: a,
|
||||||
|
b: b,
|
||||||
|
c: c,
|
||||||
|
//d: d,
|
||||||
|
e: e,
|
||||||
|
//f: f,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_lifetimes() {
|
||||||
|
let value = 5;
|
||||||
|
let lifetime = Lifetimes::LifetimeSeq(&value);
|
||||||
|
assert_eq!(
|
||||||
|
json::to_string(&lifetime).unwrap(),
|
||||||
|
"{\"LifetimeSeq\":[5]}"
|
||||||
|
);
|
||||||
|
|
||||||
|
let lifetime = Lifetimes::NoLifetimeSeq(5);
|
||||||
|
assert_eq!(
|
||||||
|
json::to_string(&lifetime).unwrap(),
|
||||||
|
"{\"NoLifetimeSeq\":[5]}"
|
||||||
|
);
|
||||||
|
|
||||||
|
let value = 5;
|
||||||
|
let lifetime = Lifetimes::LifetimeMap { a: &value };
|
||||||
|
assert_eq!(
|
||||||
|
json::to_string(&lifetime).unwrap(),
|
||||||
|
"{\"LifetimeMap\":{\"a\":5}}"
|
||||||
|
);
|
||||||
|
|
||||||
|
let lifetime = Lifetimes::NoLifetimeMap { a: 5 };
|
||||||
|
assert_eq!(
|
||||||
|
json::to_string(&lifetime).unwrap(),
|
||||||
|
"{\"NoLifetimeMap\":{\"a\":5}}"
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,540 @@
|
|||||||
|
use std::vec;
|
||||||
|
use std::collections::BTreeMap;
|
||||||
|
|
||||||
|
use serde::ser::{Serialize, Serializer, SeqVisitor, MapVisitor};
|
||||||
|
|
||||||
|
#[derive(Clone, PartialEq, Debug)]
|
||||||
|
pub enum Token<'a> {
|
||||||
|
Bool(bool),
|
||||||
|
Isize(isize),
|
||||||
|
I8(i8),
|
||||||
|
I16(i16),
|
||||||
|
I32(i32),
|
||||||
|
I64(i64),
|
||||||
|
Usize(usize),
|
||||||
|
U8(u8),
|
||||||
|
U16(u16),
|
||||||
|
U32(u32),
|
||||||
|
U64(u64),
|
||||||
|
F32(f32),
|
||||||
|
F64(f64),
|
||||||
|
Char(char),
|
||||||
|
Str(&'a str),
|
||||||
|
|
||||||
|
Option(bool),
|
||||||
|
|
||||||
|
Unit,
|
||||||
|
NamedUnit(&'a str),
|
||||||
|
EnumUnit(&'a str, &'a str),
|
||||||
|
|
||||||
|
SeqStart(Option<usize>),
|
||||||
|
NamedSeqStart(&'a str, Option<usize>),
|
||||||
|
EnumSeqStart(&'a str, &'a str, Option<usize>),
|
||||||
|
SeqSep,
|
||||||
|
SeqEnd,
|
||||||
|
|
||||||
|
MapStart(Option<usize>),
|
||||||
|
NamedMapStart(&'a str, Option<usize>),
|
||||||
|
EnumMapStart(&'a str, &'a str, Option<usize>),
|
||||||
|
MapSep,
|
||||||
|
MapEnd,
|
||||||
|
}
|
||||||
|
|
||||||
|
struct AssertSerializer<'a> {
|
||||||
|
iter: vec::IntoIter<Token<'a>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> AssertSerializer<'a> {
|
||||||
|
fn new(values: Vec<Token<'a>>) -> AssertSerializer {
|
||||||
|
AssertSerializer {
|
||||||
|
iter: values.into_iter(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_sequence<V>(&mut self, mut visitor: V) -> Result<(), ()>
|
||||||
|
where V: SeqVisitor
|
||||||
|
{
|
||||||
|
while let Some(()) = try!(visitor.visit(self)) { }
|
||||||
|
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::SeqEnd));
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_mapping<V>(&mut self, mut visitor: V) -> Result<(), ()>
|
||||||
|
where V: MapVisitor
|
||||||
|
{
|
||||||
|
while let Some(()) = try!(visitor.visit(self)) { }
|
||||||
|
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::MapEnd));
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> Serializer for AssertSerializer<'a> {
|
||||||
|
type Error = ();
|
||||||
|
|
||||||
|
fn visit_unit(&mut self) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::Unit));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_named_unit(&mut self, name: &str) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next().unwrap(), Token::NamedUnit(name));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_enum_unit(&mut self, name: &str, variant: &str) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next().unwrap(), Token::EnumUnit(name, variant));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_bool(&mut self, v: bool) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::Bool(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_isize(&mut self, v: isize) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::Isize(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_i8(&mut self, v: i8) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::I8(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_i16(&mut self, v: i16) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::I16(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_i32(&mut self, v: i32) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::I32(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_i64(&mut self, v: i64) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::I64(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_usize(&mut self, v: usize) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::Usize(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_u8(&mut self, v: u8) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::U8(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_u16(&mut self, v: u16) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::U16(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_u32(&mut self, v: u32) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::U32(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_u64(&mut self, v: u64) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::U64(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_f32(&mut self, v: f32) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::F32(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_f64(&mut self, v: f64) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::F64(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_char(&mut self, v: char) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::Char(v)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_str(&mut self, v: &str) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next().unwrap(), Token::Str(v));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_none(&mut self) -> Result<(), ()> {
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::Option(false)));
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_some<V>(&mut self, value: V) -> Result<(), ()>
|
||||||
|
where V: Serialize,
|
||||||
|
{
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::Option(true)));
|
||||||
|
value.serialize(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
fn visit_seq<V>(&mut self, visitor: V) -> Result<(), ()>
|
||||||
|
where V: SeqVisitor
|
||||||
|
{
|
||||||
|
let len = visitor.len();
|
||||||
|
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::SeqStart(len)));
|
||||||
|
|
||||||
|
self.visit_sequence(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_named_seq<V>(&mut self, name: &str, visitor: V) -> Result<(), ()>
|
||||||
|
where V: SeqVisitor
|
||||||
|
{
|
||||||
|
let len = visitor.len();
|
||||||
|
|
||||||
|
assert_eq!(self.iter.next().unwrap(), Token::NamedSeqStart(name, len));
|
||||||
|
|
||||||
|
self.visit_sequence(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_enum_seq<V>(&mut self,
|
||||||
|
name: &str,
|
||||||
|
variant: &str,
|
||||||
|
visitor: V) -> Result<(), ()>
|
||||||
|
where V: SeqVisitor
|
||||||
|
{
|
||||||
|
let len = visitor.len();
|
||||||
|
|
||||||
|
assert_eq!(self.iter.next().unwrap(), Token::EnumSeqStart(name, variant, len));
|
||||||
|
|
||||||
|
self.visit_sequence(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_seq_elt<T>(&mut self, value: T) -> Result<(), ()>
|
||||||
|
where T: Serialize
|
||||||
|
{
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::SeqSep));
|
||||||
|
value.serialize(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_map<V>(&mut self, visitor: V) -> Result<(), ()>
|
||||||
|
where V: MapVisitor
|
||||||
|
{
|
||||||
|
let len = visitor.len();
|
||||||
|
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::MapStart(len)));
|
||||||
|
|
||||||
|
self.visit_mapping(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_named_map<V>(&mut self, name: &str, visitor: V) -> Result<(), ()>
|
||||||
|
where V: MapVisitor
|
||||||
|
{
|
||||||
|
let len = visitor.len();
|
||||||
|
|
||||||
|
assert_eq!(self.iter.next().unwrap(), Token::NamedMapStart(name, len));
|
||||||
|
|
||||||
|
self.visit_mapping(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_enum_map<V>(&mut self, name: &str, variant: &str, visitor: V) -> Result<(), ()>
|
||||||
|
where V: MapVisitor
|
||||||
|
{
|
||||||
|
let len = visitor.len();
|
||||||
|
|
||||||
|
assert_eq!(self.iter.next().unwrap(), Token::EnumMapStart(name, variant, len));
|
||||||
|
|
||||||
|
self.visit_mapping(visitor)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_map_elt<K, V>(&mut self, key: K, value: V) -> Result<(), ()>
|
||||||
|
where K: Serialize,
|
||||||
|
V: Serialize,
|
||||||
|
{
|
||||||
|
assert_eq!(self.iter.next(), Some(Token::MapSep));
|
||||||
|
|
||||||
|
try!(key.serialize(self));
|
||||||
|
value.serialize(self)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
#[derive(Serialize)]
|
||||||
|
struct NamedUnit;
|
||||||
|
|
||||||
|
#[derive(Serialize)]
|
||||||
|
struct NamedSeq(i32, i32, i32);
|
||||||
|
|
||||||
|
#[derive(Serialize)]
|
||||||
|
struct NamedMap {
|
||||||
|
a: i32,
|
||||||
|
b: i32,
|
||||||
|
c: i32,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Serialize)]
|
||||||
|
enum Enum {
|
||||||
|
Unit,
|
||||||
|
Seq(i32, i32),
|
||||||
|
Map { a: i32, b: i32 },
|
||||||
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
macro_rules! btreemap {
|
||||||
|
() => {
|
||||||
|
BTreeMap::new()
|
||||||
|
};
|
||||||
|
($($key:expr => $value:expr),+) => {
|
||||||
|
{
|
||||||
|
let mut map = BTreeMap::new();
|
||||||
|
$(map.insert($key, $value);)+
|
||||||
|
map
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! declare_test {
|
||||||
|
($name:ident { $($value:expr => $tokens:expr,)+ }) => {
|
||||||
|
#[test]
|
||||||
|
fn $name() {
|
||||||
|
$(
|
||||||
|
let mut ser = AssertSerializer::new($tokens);
|
||||||
|
assert_eq!($value.serialize(&mut ser), Ok(()));
|
||||||
|
)+
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
macro_rules! declare_tests {
|
||||||
|
($($name:ident { $($value:expr => $tokens:expr,)+ })+) => {
|
||||||
|
$(
|
||||||
|
declare_test!($name { $($value => $tokens,)+ });
|
||||||
|
)+
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare_tests! {
|
||||||
|
test_unit {
|
||||||
|
() => vec![Token::Unit],
|
||||||
|
}
|
||||||
|
test_bool {
|
||||||
|
true => vec![Token::Bool(true)],
|
||||||
|
false => vec![Token::Bool(false)],
|
||||||
|
}
|
||||||
|
test_isizes {
|
||||||
|
0isize => vec![Token::Isize(0)],
|
||||||
|
0i8 => vec![Token::I8(0)],
|
||||||
|
0i16 => vec![Token::I16(0)],
|
||||||
|
0i32 => vec![Token::I32(0)],
|
||||||
|
0i64 => vec![Token::I64(0)],
|
||||||
|
}
|
||||||
|
test_usizes {
|
||||||
|
0usize => vec![Token::Usize(0)],
|
||||||
|
0u8 => vec![Token::U8(0)],
|
||||||
|
0u16 => vec![Token::U16(0)],
|
||||||
|
0u32 => vec![Token::U32(0)],
|
||||||
|
0u64 => vec![Token::U64(0)],
|
||||||
|
}
|
||||||
|
test_floats {
|
||||||
|
0f32 => vec![Token::F32(0.)],
|
||||||
|
0f64 => vec![Token::F64(0.)],
|
||||||
|
}
|
||||||
|
test_char {
|
||||||
|
'a' => vec![Token::Char('a')],
|
||||||
|
}
|
||||||
|
test_str {
|
||||||
|
"abc" => vec![Token::Str("abc")],
|
||||||
|
"abc".to_string() => vec![Token::Str("abc")],
|
||||||
|
}
|
||||||
|
test_option {
|
||||||
|
None::<i32> => vec![Token::Option(false)],
|
||||||
|
Some(1) => vec![
|
||||||
|
Token::Option(true),
|
||||||
|
Token::I32(1),
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_slice {
|
||||||
|
&[0][..0] => vec![
|
||||||
|
Token::SeqStart(Some(0)),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
&[1, 2, 3][..] => vec![
|
||||||
|
Token::SeqStart(Some(3)),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_array {
|
||||||
|
[0; 0] => vec![
|
||||||
|
Token::SeqStart(Some(0)),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
[1, 2, 3] => vec![
|
||||||
|
Token::SeqStart(Some(3)),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_vec {
|
||||||
|
Vec::<isize>::new() => vec![
|
||||||
|
Token::SeqStart(Some(0)),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
vec![vec![], vec![1], vec![2, 3]] => vec![
|
||||||
|
Token::SeqStart(Some(3)),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::SeqStart(Some(0)),
|
||||||
|
Token::SeqEnd,
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::SeqStart(Some(1)),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::SeqEnd,
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::SeqStart(Some(2)),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::SeqEnd,
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_tuple {
|
||||||
|
(1,) => vec![
|
||||||
|
Token::SeqStart(Some(1)),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
(1, 2, 3) => vec![
|
||||||
|
Token::SeqStart(Some(3)),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_btreemap {
|
||||||
|
btreemap![1 => 2] => vec![
|
||||||
|
Token::MapStart(Some(1)),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::I32(2),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
btreemap![1 => 2, 3 => 4] => vec![
|
||||||
|
Token::MapStart(Some(2)),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::I32(4),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
btreemap![1 => btreemap![], 2 => btreemap![3 => 4, 5 => 6]] => vec![
|
||||||
|
Token::MapStart(Some(2)),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(1),
|
||||||
|
Token::MapStart(Some(0)),
|
||||||
|
Token::MapEnd,
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(2),
|
||||||
|
Token::MapStart(Some(2)),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::I32(4),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::I32(5),
|
||||||
|
Token::I32(6),
|
||||||
|
Token::MapEnd,
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_named_unit {
|
||||||
|
NamedUnit => vec![Token::NamedUnit("NamedUnit")],
|
||||||
|
}
|
||||||
|
test_named_seq {
|
||||||
|
NamedSeq(1, 2, 3) => vec![
|
||||||
|
Token::NamedSeqStart("NamedSeq", Some(3)),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(3),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_named_map {
|
||||||
|
NamedMap { a: 1, b: 2, c: 3 } => vec![
|
||||||
|
Token::NamedMapStart("NamedMap", Some(3)),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("a"),
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("b"),
|
||||||
|
Token::I32(2),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("c"),
|
||||||
|
Token::I32(3),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
test_enum {
|
||||||
|
Enum::Unit => vec![Token::EnumUnit("Enum", "Unit")],
|
||||||
|
Enum::Seq(1, 2) => vec![
|
||||||
|
Token::EnumSeqStart("Enum", "Seq", Some(2)),
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::SeqSep,
|
||||||
|
Token::I32(2),
|
||||||
|
Token::SeqEnd,
|
||||||
|
],
|
||||||
|
Enum::Map { a: 1, b: 2 } => vec![
|
||||||
|
Token::EnumMapStart("Enum", "Map", Some(2)),
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("a"),
|
||||||
|
Token::I32(1),
|
||||||
|
|
||||||
|
Token::MapSep,
|
||||||
|
Token::Str("b"),
|
||||||
|
Token::I32(2),
|
||||||
|
Token::MapEnd,
|
||||||
|
],
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
[package]
|
|
||||||
name = "serde_testing"
|
|
||||||
version = "0.8.5"
|
|
||||||
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>"]
|
|
||||||
license = "MIT/Apache-2.0"
|
|
||||||
description = "A generic serialization/deserialization framework"
|
|
||||||
homepage = "https://serde.rs"
|
|
||||||
repository = "https://github.com/serde-rs/serde"
|
|
||||||
documentation = "https://docs.serde.rs/serde/"
|
|
||||||
readme = "README.md"
|
|
||||||
keywords = ["serialization"]
|
|
||||||
build = "build.rs"
|
|
||||||
|
|
||||||
[features]
|
|
||||||
unstable-testing = ["clippy", "serde/unstable-testing", "serde_codegen/unstable-testing"]
|
|
||||||
|
|
||||||
[build-dependencies]
|
|
||||||
serde_codegen = { path = "../serde_codegen", features = ["with-syntex"] }
|
|
||||||
|
|
||||||
[dev-dependencies]
|
|
||||||
fnv = "1.0"
|
|
||||||
rustc-serialize = "^0.3.16"
|
|
||||||
serde = { path = "../serde" }
|
|
||||||
serde_test = { path = "../serde_test" }
|
|
||||||
|
|
||||||
[dependencies]
|
|
||||||
clippy = { version = "^0.*", optional = true }
|
|
||||||
|
|
||||||
[[test]]
|
|
||||||
name = "test"
|
|
||||||
path = "tests/test.rs"
|
|
||||||
|
|
||||||
[[bench]]
|
|
||||||
name = "bench"
|
|
||||||
path = "benches/bench.rs"
|
|
||||||
@@ -1,95 +0,0 @@
|
|||||||
#[macro_export]
|
|
||||||
macro_rules! declare_ser_tests {
|
|
||||||
($($name:ident { $($value:expr => $tokens:expr,)+ })+) => {
|
|
||||||
$(
|
|
||||||
#[test]
|
|
||||||
fn $name() {
|
|
||||||
$(
|
|
||||||
assert_ser_tokens(&$value, $tokens);
|
|
||||||
)+
|
|
||||||
}
|
|
||||||
)+
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[macro_export]
|
|
||||||
macro_rules! btreemap {
|
|
||||||
() => {
|
|
||||||
BTreeMap::new()
|
|
||||||
};
|
|
||||||
($($key:expr => $value:expr),+) => {
|
|
||||||
{
|
|
||||||
let mut map = BTreeMap::new();
|
|
||||||
$(map.insert($key, $value);)+
|
|
||||||
map
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! btreeset {
|
|
||||||
() => {
|
|
||||||
BTreeSet::new()
|
|
||||||
};
|
|
||||||
($($value:expr),+) => {
|
|
||||||
{
|
|
||||||
let mut set = BTreeSet::new();
|
|
||||||
$(set.insert($value);)+
|
|
||||||
set
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! btreemap {
|
|
||||||
() => {
|
|
||||||
BTreeMap::new()
|
|
||||||
};
|
|
||||||
($($key:expr => $value:expr),+) => {
|
|
||||||
{
|
|
||||||
let mut map = BTreeMap::new();
|
|
||||||
$(map.insert($key, $value);)+
|
|
||||||
map
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! hashset {
|
|
||||||
() => {
|
|
||||||
HashSet::new()
|
|
||||||
};
|
|
||||||
($($value:expr),+) => {
|
|
||||||
{
|
|
||||||
let mut set = HashSet::new();
|
|
||||||
$(set.insert($value);)+
|
|
||||||
set
|
|
||||||
}
|
|
||||||
};
|
|
||||||
($hasher:ident @ $($value:expr),+) => {
|
|
||||||
{
|
|
||||||
use std::hash::BuildHasherDefault;
|
|
||||||
let mut set = HashSet::with_hasher(BuildHasherDefault::<$hasher>::default());
|
|
||||||
$(set.insert($value);)+
|
|
||||||
set
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! hashmap {
|
|
||||||
() => {
|
|
||||||
HashMap::new()
|
|
||||||
};
|
|
||||||
($($key:expr => $value:expr),+) => {
|
|
||||||
{
|
|
||||||
let mut map = HashMap::new();
|
|
||||||
$(map.insert($key, $value);)+
|
|
||||||
map
|
|
||||||
}
|
|
||||||
};
|
|
||||||
($hasher:ident @ $($key:expr => $value:expr),+) => {
|
|
||||||
{
|
|
||||||
use std::hash::BuildHasherDefault;
|
|
||||||
let mut map = HashMap::with_hasher(BuildHasherDefault::<$hasher>::default());
|
|
||||||
$(map.insert($key, $value);)+
|
|
||||||
map
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
#![cfg_attr(feature = "clippy", feature(plugin))]
|
|
||||||
#![cfg_attr(feature = "clippy", plugin(clippy))]
|
|
||||||
|
|
||||||
include!(concat!(env!("OUT_DIR"), "/test.rs"));
|
|
||||||
@@ -1,976 +0,0 @@
|
|||||||
extern crate serde;
|
|
||||||
use self::serde::{Serialize, Serializer, Deserialize, Deserializer};
|
|
||||||
|
|
||||||
extern crate serde_test;
|
|
||||||
use self::serde_test::{
|
|
||||||
Error,
|
|
||||||
Token,
|
|
||||||
assert_tokens,
|
|
||||||
assert_ser_tokens,
|
|
||||||
assert_de_tokens,
|
|
||||||
assert_de_tokens_error
|
|
||||||
};
|
|
||||||
|
|
||||||
trait MyDefault: Sized {
|
|
||||||
fn my_default() -> Self;
|
|
||||||
}
|
|
||||||
|
|
||||||
trait ShouldSkip: Sized {
|
|
||||||
fn should_skip(&self) -> bool;
|
|
||||||
}
|
|
||||||
|
|
||||||
trait SerializeWith: Sized {
|
|
||||||
fn serialize_with<S>(&self, ser: &mut S) -> Result<(), S::Error>
|
|
||||||
where S: Serializer;
|
|
||||||
}
|
|
||||||
|
|
||||||
trait DeserializeWith: Sized {
|
|
||||||
fn deserialize_with<D>(de: &mut D) -> Result<Self, D::Error>
|
|
||||||
where D: Deserializer;
|
|
||||||
}
|
|
||||||
|
|
||||||
impl MyDefault for i32 {
|
|
||||||
fn my_default() -> Self { 123 }
|
|
||||||
}
|
|
||||||
|
|
||||||
impl ShouldSkip for i32 {
|
|
||||||
fn should_skip(&self) -> bool { *self == 123 }
|
|
||||||
}
|
|
||||||
|
|
||||||
impl SerializeWith for i32 {
|
|
||||||
fn serialize_with<S>(&self, ser: &mut S) -> Result<(), S::Error>
|
|
||||||
where S: Serializer
|
|
||||||
{
|
|
||||||
if *self == 123 {
|
|
||||||
true.serialize(ser)
|
|
||||||
} else {
|
|
||||||
false.serialize(ser)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl DeserializeWith for i32 {
|
|
||||||
fn deserialize_with<D>(de: &mut D) -> Result<Self, D::Error>
|
|
||||||
where D: Deserializer
|
|
||||||
{
|
|
||||||
if try!(Deserialize::deserialize(de)) {
|
|
||||||
Ok(123)
|
|
||||||
} else {
|
|
||||||
Ok(2)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
struct DefaultStruct<A, B, C, D, E>
|
|
||||||
where C: MyDefault,
|
|
||||||
E: MyDefault,
|
|
||||||
{
|
|
||||||
a1: A,
|
|
||||||
#[serde(default)]
|
|
||||||
a2: B,
|
|
||||||
#[serde(default="MyDefault::my_default")]
|
|
||||||
a3: C,
|
|
||||||
#[serde(skip_deserializing)]
|
|
||||||
a4: D,
|
|
||||||
#[serde(skip_deserializing, default="MyDefault::my_default")]
|
|
||||||
a5: E,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_default_struct() {
|
|
||||||
assert_de_tokens(
|
|
||||||
&DefaultStruct { a1: 1, a2: 2, a3: 3, a4: 0, a5: 123 },
|
|
||||||
&[
|
|
||||||
Token::StructStart("DefaultStruct", 3),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a1"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a2"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a3"),
|
|
||||||
Token::I32(3),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a4"),
|
|
||||||
Token::I32(4),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a5"),
|
|
||||||
Token::I32(5),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_de_tokens(
|
|
||||||
&DefaultStruct { a1: 1, a2: 0, a3: 123, a4: 0, a5: 123 },
|
|
||||||
&[
|
|
||||||
Token::StructStart("DefaultStruct", 1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a1"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
enum DefaultEnum<A, B, C, D, E>
|
|
||||||
where C: MyDefault,
|
|
||||||
E: MyDefault
|
|
||||||
{
|
|
||||||
Struct {
|
|
||||||
a1: A,
|
|
||||||
#[serde(default)]
|
|
||||||
a2: B,
|
|
||||||
#[serde(default="MyDefault::my_default")]
|
|
||||||
a3: C,
|
|
||||||
#[serde(skip_deserializing)]
|
|
||||||
a4: D,
|
|
||||||
#[serde(skip_deserializing, default="MyDefault::my_default")]
|
|
||||||
a5: E,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_default_enum() {
|
|
||||||
assert_de_tokens(
|
|
||||||
&DefaultEnum::Struct { a1: 1, a2: 2, a3: 3, a4: 0, a5: 123 },
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("DefaultEnum", "Struct", 5),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a1"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a2"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a3"),
|
|
||||||
Token::I32(3),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a4"),
|
|
||||||
Token::I32(4),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a5"),
|
|
||||||
Token::I32(5),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_de_tokens(
|
|
||||||
&DefaultEnum::Struct { a1: 1, a2: 0, a3: 123, a4: 0, a5: 123 },
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("DefaultEnum", "Struct", 5),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a1"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Does not implement std::default::Default.
|
|
||||||
#[derive(Debug, PartialEq, Deserialize)]
|
|
||||||
struct NoStdDefault(i8);
|
|
||||||
|
|
||||||
impl MyDefault for NoStdDefault {
|
|
||||||
fn my_default() -> Self {
|
|
||||||
NoStdDefault(123)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Deserialize)]
|
|
||||||
struct ContainsNoStdDefault<A: MyDefault> {
|
|
||||||
#[serde(default="MyDefault::my_default")]
|
|
||||||
a: A,
|
|
||||||
}
|
|
||||||
|
|
||||||
// Tests that a struct field does not need to implement std::default::Default if
|
|
||||||
// it is annotated with `default=...`.
|
|
||||||
#[test]
|
|
||||||
fn test_no_std_default() {
|
|
||||||
assert_de_tokens(
|
|
||||||
&ContainsNoStdDefault { a: NoStdDefault(123) },
|
|
||||||
&[
|
|
||||||
Token::StructStart("ContainsNoStdDefault", 1),
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_de_tokens(
|
|
||||||
&ContainsNoStdDefault { a: NoStdDefault(8) },
|
|
||||||
&[
|
|
||||||
Token::StructStart("ContainsNoStdDefault", 1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::StructNewType("NoStdDefault"),
|
|
||||||
Token::I8(8),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Does not implement Deserialize.
|
|
||||||
#[derive(Debug, PartialEq)]
|
|
||||||
struct NotDeserializeStruct(i8);
|
|
||||||
|
|
||||||
impl Default for NotDeserializeStruct {
|
|
||||||
fn default() -> Self {
|
|
||||||
NotDeserializeStruct(123)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl DeserializeWith for NotDeserializeStruct {
|
|
||||||
fn deserialize_with<D>(_: &mut D) -> Result<Self, D::Error>
|
|
||||||
where D: Deserializer
|
|
||||||
{
|
|
||||||
panic!()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Does not implement Deserialize.
|
|
||||||
#[derive(Debug, PartialEq)]
|
|
||||||
enum NotDeserializeEnum { Trouble }
|
|
||||||
|
|
||||||
impl MyDefault for NotDeserializeEnum {
|
|
||||||
fn my_default() -> Self {
|
|
||||||
NotDeserializeEnum::Trouble
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Deserialize)]
|
|
||||||
struct ContainsNotDeserialize<A, B, C: DeserializeWith, E: MyDefault> {
|
|
||||||
#[serde(skip_deserializing)]
|
|
||||||
a: A,
|
|
||||||
#[serde(skip_deserializing, default)]
|
|
||||||
b: B,
|
|
||||||
#[serde(deserialize_with="DeserializeWith::deserialize_with", default)]
|
|
||||||
c: C,
|
|
||||||
#[serde(skip_deserializing, default="MyDefault::my_default")]
|
|
||||||
e: E,
|
|
||||||
}
|
|
||||||
|
|
||||||
// Tests that a struct field does not need to implement Deserialize if it is
|
|
||||||
// annotated with skip_deserializing, whether using the std Default or a
|
|
||||||
// custom default.
|
|
||||||
#[test]
|
|
||||||
fn test_elt_not_deserialize() {
|
|
||||||
assert_de_tokens(
|
|
||||||
&ContainsNotDeserialize {
|
|
||||||
a: NotDeserializeStruct(123),
|
|
||||||
b: NotDeserializeStruct(123),
|
|
||||||
c: NotDeserializeStruct(123),
|
|
||||||
e: NotDeserializeEnum::Trouble,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::StructStart("ContainsNotDeserialize", 3),
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
#[serde(deny_unknown_fields)]
|
|
||||||
struct DenyUnknown {
|
|
||||||
a1: i32,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_ignore_unknown() {
|
|
||||||
// 'Default' allows unknown. Basic smoke test of ignore...
|
|
||||||
assert_de_tokens(
|
|
||||||
&DefaultStruct { a1: 1, a2: 2, a3: 3, a4: 0, a5: 123 },
|
|
||||||
&[
|
|
||||||
Token::StructStart("DefaultStruct", 5),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("whoops1"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a1"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("whoops2"),
|
|
||||||
Token::SeqStart(Some(1)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
Token::SeqEnd,
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a2"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("whoops3"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a3"),
|
|
||||||
Token::I32(3),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_de_tokens_error::<DenyUnknown>(
|
|
||||||
&[
|
|
||||||
Token::StructStart("DenyUnknown", 2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a1"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("whoops"),
|
|
||||||
],
|
|
||||||
Error::UnknownField("whoops".to_owned())
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
#[serde(rename="Superhero")]
|
|
||||||
struct RenameStruct {
|
|
||||||
a1: i32,
|
|
||||||
#[serde(rename="a3")]
|
|
||||||
a2: i32,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
#[serde(rename(serialize="SuperheroSer", deserialize="SuperheroDe"))]
|
|
||||||
struct RenameStructSerializeDeserialize {
|
|
||||||
a1: i32,
|
|
||||||
#[serde(rename(serialize="a4", deserialize="a5"))]
|
|
||||||
a2: i32,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_rename_struct() {
|
|
||||||
assert_tokens(
|
|
||||||
&RenameStruct { a1: 1, a2: 2 },
|
|
||||||
&[
|
|
||||||
Token::StructStart("Superhero", 2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a1"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a3"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_ser_tokens(
|
|
||||||
&RenameStructSerializeDeserialize { a1: 1, a2: 2 },
|
|
||||||
&[
|
|
||||||
Token::StructStart("SuperheroSer", 2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a1"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a4"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_de_tokens(
|
|
||||||
&RenameStructSerializeDeserialize { a1: 1, a2: 2 },
|
|
||||||
&[
|
|
||||||
Token::StructStart("SuperheroDe", 2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a1"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a5"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
#[serde(rename="Superhero")]
|
|
||||||
enum RenameEnum {
|
|
||||||
#[serde(rename="bruce_wayne")]
|
|
||||||
Batman,
|
|
||||||
#[serde(rename="clark_kent")]
|
|
||||||
Superman(i8),
|
|
||||||
#[serde(rename="diana_prince")]
|
|
||||||
WonderWoman(i8, i8),
|
|
||||||
#[serde(rename="barry_allan")]
|
|
||||||
Flash {
|
|
||||||
#[serde(rename="b")]
|
|
||||||
a: i32,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Deserialize, Serialize)]
|
|
||||||
#[serde(rename(serialize="SuperheroSer", deserialize="SuperheroDe"))]
|
|
||||||
enum RenameEnumSerializeDeserialize<A> {
|
|
||||||
#[serde(rename(serialize="dick_grayson", deserialize="jason_todd"))]
|
|
||||||
Robin {
|
|
||||||
a: i8,
|
|
||||||
#[serde(rename(serialize="c"))]
|
|
||||||
#[serde(rename(deserialize="d"))]
|
|
||||||
b: A,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_rename_enum() {
|
|
||||||
assert_tokens(
|
|
||||||
&RenameEnum::Batman,
|
|
||||||
&[
|
|
||||||
Token::EnumUnit("Superhero", "bruce_wayne"),
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_tokens(
|
|
||||||
&RenameEnum::Superman(0),
|
|
||||||
&[
|
|
||||||
Token::EnumNewType("Superhero", "clark_kent"),
|
|
||||||
Token::I8(0),
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_tokens(
|
|
||||||
&RenameEnum::WonderWoman(0, 1),
|
|
||||||
&[
|
|
||||||
Token::EnumSeqStart("Superhero", "diana_prince", 2),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I8(0),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::EnumSeqEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_tokens(
|
|
||||||
&RenameEnum::Flash { a: 1 },
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("Superhero", "barry_allan", 1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_ser_tokens(
|
|
||||||
&RenameEnumSerializeDeserialize::Robin {
|
|
||||||
a: 0,
|
|
||||||
b: String::new(),
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("SuperheroSer", "dick_grayson", 2),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(0),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("c"),
|
|
||||||
Token::Str(""),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_de_tokens(
|
|
||||||
&RenameEnumSerializeDeserialize::Robin {
|
|
||||||
a: 0,
|
|
||||||
b: String::new(),
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("SuperheroDe", "jason_todd", 2),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(0),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("d"),
|
|
||||||
Token::Str(""),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize)]
|
|
||||||
struct SkipSerializingStruct<'a, B, C> where C: ShouldSkip {
|
|
||||||
a: &'a i8,
|
|
||||||
#[serde(skip_serializing)]
|
|
||||||
b: B,
|
|
||||||
#[serde(skip_serializing_if="ShouldSkip::should_skip")]
|
|
||||||
c: C,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_skip_serializing_struct() {
|
|
||||||
let a = 1;
|
|
||||||
assert_ser_tokens(
|
|
||||||
&SkipSerializingStruct {
|
|
||||||
a: &a,
|
|
||||||
b: 2,
|
|
||||||
c: 3,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::StructStart("SkipSerializingStruct", 2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("c"),
|
|
||||||
Token::I32(3),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_ser_tokens(
|
|
||||||
&SkipSerializingStruct {
|
|
||||||
a: &a,
|
|
||||||
b: 2,
|
|
||||||
c: 123,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::StructStart("SkipSerializingStruct", 1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize)]
|
|
||||||
enum SkipSerializingEnum<'a, B, C> where C: ShouldSkip {
|
|
||||||
Struct {
|
|
||||||
a: &'a i8,
|
|
||||||
#[serde(skip_serializing)]
|
|
||||||
_b: B,
|
|
||||||
#[serde(skip_serializing_if="ShouldSkip::should_skip")]
|
|
||||||
c: C,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_skip_serializing_enum() {
|
|
||||||
let a = 1;
|
|
||||||
assert_ser_tokens(
|
|
||||||
&SkipSerializingEnum::Struct {
|
|
||||||
a: &a,
|
|
||||||
_b: 2,
|
|
||||||
c: 3,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("SkipSerializingEnum", "Struct", 2),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("c"),
|
|
||||||
Token::I32(3),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_ser_tokens(
|
|
||||||
&SkipSerializingEnum::Struct {
|
|
||||||
a: &a,
|
|
||||||
_b: 2,
|
|
||||||
c: 123,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("SkipSerializingEnum", "Struct", 1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq)]
|
|
||||||
struct NotSerializeStruct(i8);
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq)]
|
|
||||||
enum NotSerializeEnum { Trouble }
|
|
||||||
|
|
||||||
impl SerializeWith for NotSerializeEnum {
|
|
||||||
fn serialize_with<S>(&self, ser: &mut S) -> Result<(), S::Error>
|
|
||||||
where S: Serializer
|
|
||||||
{
|
|
||||||
"trouble".serialize(ser)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize)]
|
|
||||||
struct ContainsNotSerialize<'a, B, C, D> where B: 'a, D: SerializeWith {
|
|
||||||
a: &'a Option<i8>,
|
|
||||||
#[serde(skip_serializing)]
|
|
||||||
b: &'a B,
|
|
||||||
#[serde(skip_serializing)]
|
|
||||||
c: Option<C>,
|
|
||||||
#[serde(serialize_with="SerializeWith::serialize_with")]
|
|
||||||
d: D,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_elt_not_serialize() {
|
|
||||||
let a = 1;
|
|
||||||
assert_ser_tokens(
|
|
||||||
&ContainsNotSerialize {
|
|
||||||
a: &Some(a),
|
|
||||||
b: &NotSerializeStruct(2),
|
|
||||||
c: Some(NotSerializeEnum::Trouble),
|
|
||||||
d: NotSerializeEnum::Trouble,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::StructStart("ContainsNotSerialize", 2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::Option(true),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("d"),
|
|
||||||
Token::Str("trouble"),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize)]
|
|
||||||
struct SerializeWithStruct<'a, B> where B: SerializeWith {
|
|
||||||
a: &'a i8,
|
|
||||||
#[serde(serialize_with="SerializeWith::serialize_with")]
|
|
||||||
b: B,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_serialize_with_struct() {
|
|
||||||
let a = 1;
|
|
||||||
assert_ser_tokens(
|
|
||||||
&SerializeWithStruct {
|
|
||||||
a: &a,
|
|
||||||
b: 2,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::StructStart("SerializeWithStruct", 2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::Bool(false),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_ser_tokens(
|
|
||||||
&SerializeWithStruct {
|
|
||||||
a: &a,
|
|
||||||
b: 123,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::StructStart("SerializeWithStruct", 2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::Bool(true),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize)]
|
|
||||||
enum SerializeWithEnum<'a, B> where B: SerializeWith {
|
|
||||||
Struct {
|
|
||||||
a: &'a i8,
|
|
||||||
#[serde(serialize_with="SerializeWith::serialize_with")]
|
|
||||||
b: B,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_serialize_with_enum() {
|
|
||||||
let a = 1;
|
|
||||||
assert_ser_tokens(
|
|
||||||
&SerializeWithEnum::Struct {
|
|
||||||
a: &a,
|
|
||||||
b: 2,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("SerializeWithEnum", "Struct", 2),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::Bool(false),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_ser_tokens(
|
|
||||||
&SerializeWithEnum::Struct {
|
|
||||||
a: &a,
|
|
||||||
b: 123,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("SerializeWithEnum", "Struct", 2),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::Bool(true),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Deserialize)]
|
|
||||||
struct DeserializeWithStruct<B> where B: DeserializeWith {
|
|
||||||
a: i8,
|
|
||||||
#[serde(deserialize_with="DeserializeWith::deserialize_with")]
|
|
||||||
b: B,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_deserialize_with_struct() {
|
|
||||||
assert_de_tokens(
|
|
||||||
&DeserializeWithStruct {
|
|
||||||
a: 1,
|
|
||||||
b: 2,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::StructStart("DeserializeWithStruct", 2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::Bool(false),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_de_tokens(
|
|
||||||
&DeserializeWithStruct {
|
|
||||||
a: 1,
|
|
||||||
b: 123,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::StructStart("DeserializeWithStruct", 2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::Bool(true),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Deserialize)]
|
|
||||||
enum DeserializeWithEnum<B> where B: DeserializeWith {
|
|
||||||
Struct {
|
|
||||||
a: i8,
|
|
||||||
#[serde(deserialize_with="DeserializeWith::deserialize_with")]
|
|
||||||
b: B,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_deserialize_with_enum() {
|
|
||||||
assert_de_tokens(
|
|
||||||
&DeserializeWithEnum::Struct {
|
|
||||||
a: 1,
|
|
||||||
b: 2,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("DeserializeWithEnum", "Struct", 2),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::Bool(false),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_de_tokens(
|
|
||||||
&DeserializeWithEnum::Struct {
|
|
||||||
a: 1,
|
|
||||||
b: 123,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("DeserializeWithEnum", "Struct", 2),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::Bool(true),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_missing_renamed_field_struct() {
|
|
||||||
assert_de_tokens_error::<RenameStruct>(
|
|
||||||
&[
|
|
||||||
Token::StructStart("Superhero", 2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a1"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
],
|
|
||||||
Error::MissingField("a3"),
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_de_tokens_error::<RenameStructSerializeDeserialize>(
|
|
||||||
&[
|
|
||||||
Token::StructStart("SuperheroDe", 2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a1"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
],
|
|
||||||
Error::MissingField("a5"),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_missing_renamed_field_enum() {
|
|
||||||
assert_de_tokens_error::<RenameEnum>(
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("Superhero", "barry_allan", 1),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
],
|
|
||||||
Error::MissingField("b"),
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_de_tokens_error::<RenameEnumSerializeDeserialize<i8>>(
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("SuperheroDe", "jason_todd", 2),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(0),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
],
|
|
||||||
Error::MissingField("d"),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Deserialize)]
|
|
||||||
enum InvalidLengthEnum {
|
|
||||||
A(i32, i32, i32),
|
|
||||||
B(#[serde(skip_deserializing)] i32, i32, i32),
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_invalid_length_enum() {
|
|
||||||
assert_de_tokens_error::<InvalidLengthEnum>(
|
|
||||||
&[
|
|
||||||
Token::EnumSeqStart("InvalidLengthEnum", "A", 3),
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::EnumSeqEnd,
|
|
||||||
],
|
|
||||||
Error::InvalidLength(1),
|
|
||||||
);
|
|
||||||
assert_de_tokens_error::<InvalidLengthEnum>(
|
|
||||||
&[
|
|
||||||
Token::EnumSeqStart("InvalidLengthEnum", "B", 3),
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::EnumSeqEnd,
|
|
||||||
],
|
|
||||||
Error::InvalidLength(1),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,460 +0,0 @@
|
|||||||
use std::fmt;
|
|
||||||
use std::error;
|
|
||||||
|
|
||||||
use serde::{Serialize, Serializer, Deserialize, Deserializer};
|
|
||||||
use serde::bytes::{ByteBuf, Bytes};
|
|
||||||
use serde::ser;
|
|
||||||
use serde::de;
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq)]
|
|
||||||
struct Error;
|
|
||||||
|
|
||||||
impl ser::Error for Error {
|
|
||||||
fn custom<T: Into<String>>(_: T) -> Error { Error }
|
|
||||||
}
|
|
||||||
|
|
||||||
impl de::Error for Error {
|
|
||||||
fn custom<T: Into<String>>(_: T) -> Error { Error }
|
|
||||||
|
|
||||||
fn end_of_stream() -> Error { Error }
|
|
||||||
}
|
|
||||||
|
|
||||||
impl fmt::Display for Error {
|
|
||||||
fn fmt(&self, formatter: &mut fmt::Formatter) -> Result<(), fmt::Error> {
|
|
||||||
formatter.write_str(format!("{:?}", self).as_ref())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl error::Error for Error {
|
|
||||||
fn description(&self) -> &str {
|
|
||||||
"Serde Deserialization Error"
|
|
||||||
}
|
|
||||||
|
|
||||||
fn cause(&self) -> Option<&error::Error> {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
struct BytesSerializer {
|
|
||||||
bytes: Vec<u8>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl BytesSerializer {
|
|
||||||
fn new(bytes: Vec<u8>) -> Self {
|
|
||||||
BytesSerializer {
|
|
||||||
bytes: bytes,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Serializer for BytesSerializer {
|
|
||||||
type Error = Error;
|
|
||||||
type SeqState = ();
|
|
||||||
type MapState = ();
|
|
||||||
type TupleState = ();
|
|
||||||
type TupleStructState = ();
|
|
||||||
type TupleVariantState = ();
|
|
||||||
type StructState = ();
|
|
||||||
type StructVariantState = ();
|
|
||||||
|
|
||||||
fn serialize_unit(&mut self) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_unit_struct(&mut self, _name: &'static str) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_unit_variant(&mut self, _: &'static str, _: usize, _: &'static str) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_bool(&mut self, _v: bool) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_isize(&mut self, _v: isize) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_usize(&mut self, _v: usize) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_i8(&mut self, _v: i8) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_u8(&mut self, _v: u8) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_i16(&mut self, _v: i16) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_u16(&mut self, _v: u16) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_i32(&mut self, _v: i32) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_u32(&mut self, _v: u32) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_i64(&mut self, _v: i64) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_u64(&mut self, _v: u64) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_f32(&mut self, _v: f32) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_f64(&mut self, _v: f64) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_char(&mut self, _v: char) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_str(&mut self, _v: &str) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_none(&mut self) -> Result<(), Error> {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_some<V>(&mut self, _value: V) -> Result<(), Error>
|
|
||||||
where V: Serialize,
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_newtype_struct<V>(&mut self, _: &'static str, _value: V) -> Result<(), Error>
|
|
||||||
where V: Serialize,
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_newtype_variant<V>(&mut self, _: &'static str, _: usize, _: &'static str, _value: V) -> Result<(), Error>
|
|
||||||
where V: Serialize,
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_seq(&mut self, _len: Option<usize>) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_seq_fixed_size(&mut self, _len: usize) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_seq_elt<T>(&mut self, _: &mut (), _value: T) -> Result<(), Error>
|
|
||||||
where T: Serialize
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_seq_end(&mut self, _: ()) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple(&mut self, _len: usize) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_elt<T>(&mut self, _: &mut (), _value: T) -> Result<(), Error>
|
|
||||||
where T: Serialize
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_end(&mut self, _: ()) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_struct(&mut self, _: &'static str, _len: usize) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_struct_elt<T>(&mut self, _: &mut (), _value: T) -> Result<(), Error>
|
|
||||||
where T: Serialize
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_struct_end(&mut self, _: ()) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_variant(&mut self, _: &'static str, _: usize, _: &'static str, _len: usize) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_variant_elt<T>(&mut self, _: &mut (), _value: T) -> Result<(), Error>
|
|
||||||
where T: Serialize
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_tuple_variant_end(&mut self, _: ()) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_map(&mut self, _: Option<usize>) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_map_key<T>(&mut self, _: &mut (), _key: T) -> Result<(), Error>
|
|
||||||
where T: Serialize
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_map_value<T>(&mut self, _: &mut (), _value: T) -> Result<(), Error>
|
|
||||||
where T: Serialize
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_map_end(&mut self, _: ()) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_struct(&mut self, _: &'static str, _: usize) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_struct_elt<V>(&mut self, _: &mut (), _key: &'static str, _value: V) -> Result<(), Error>
|
|
||||||
where V: Serialize,
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_struct_end(&mut self, _: ()) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_struct_variant(&mut self, _: &'static str, _: usize, _: &'static str, _: usize) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_struct_variant_elt<V>(&mut self, _: &mut (), _key: &'static str, _value: V) -> Result<(), Error>
|
|
||||||
where V: Serialize,
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_struct_variant_end(&mut self, _: ()) -> Result<(), Error>
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn serialize_bytes(&mut self, bytes: &[u8]) -> Result<(), Error> {
|
|
||||||
assert_eq!(self.bytes, bytes);
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
struct BytesDeserializer {
|
|
||||||
bytes: Option<Vec<u8>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl BytesDeserializer {
|
|
||||||
fn new(bytes: Vec<u8>) -> Self {
|
|
||||||
BytesDeserializer {
|
|
||||||
bytes: Some(bytes),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Deserializer for BytesDeserializer {
|
|
||||||
type Error = Error;
|
|
||||||
|
|
||||||
fn deserialize<V>(&mut self, _visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: de::Visitor,
|
|
||||||
{
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn deserialize_bytes<V>(&mut self, mut visitor: V) -> Result<V::Value, Error>
|
|
||||||
where V: de::Visitor,
|
|
||||||
{
|
|
||||||
visitor.visit_byte_buf(self.bytes.take().unwrap())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn deserialize_seq<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_struct_field<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_map<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_unit<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_ignored_any<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_string<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_str<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_char<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_i64<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_i32<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_i16<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_i8<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_u64<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_u32<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_u16<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_u8<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_f32<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_f64<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_bool<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_usize<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_isize<__V>(&mut self, visitor: __V) -> Result<__V::Value, Self::Error>
|
|
||||||
where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_option<__V>(&mut self, visitor: __V)
|
|
||||||
-> Result<__V::Value, Self::Error> where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_seq_fixed_size<__V>(&mut self, _: usize, visitor: __V)
|
|
||||||
-> Result<__V::Value, Self::Error> where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_unit_struct<__V>(&mut self, _: &str, visitor: __V)
|
|
||||||
-> Result<__V::Value, Self::Error> where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_newtype_struct<__V>(&mut self, _: &str, visitor: __V)
|
|
||||||
-> Result<__V::Value, Self::Error> where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_tuple_struct<__V>(&mut self, _: &str, _: usize, visitor: __V)
|
|
||||||
-> Result<__V::Value, Self::Error> where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_struct<__V>(&mut self, _: &str, _: &[&str], visitor: __V)
|
|
||||||
-> Result<__V::Value, Self::Error> where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_tuple<__V>(&mut self, _: usize, visitor: __V)
|
|
||||||
-> Result<__V::Value, Self::Error> where __V: de::Visitor {
|
|
||||||
self.deserialize(visitor)
|
|
||||||
}
|
|
||||||
fn deserialize_enum<__V>(&mut self, _: &str, _: &[&str], _visitor: __V)
|
|
||||||
-> Result<__V::Value, Self::Error> where __V: de::EnumVisitor {
|
|
||||||
Err(Error)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_bytes_ser_bytes() {
|
|
||||||
let buf = vec![];
|
|
||||||
let bytes = Bytes::from(&buf);
|
|
||||||
let mut ser = BytesSerializer::new(vec![]);
|
|
||||||
bytes.serialize(&mut ser).unwrap();
|
|
||||||
|
|
||||||
let buf = vec![1, 2, 3];
|
|
||||||
let bytes = Bytes::from(&buf);
|
|
||||||
let mut ser = BytesSerializer::new(vec![1, 2, 3]);
|
|
||||||
bytes.serialize(&mut ser).unwrap();
|
|
||||||
}
|
|
||||||
|
|
||||||
///////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_byte_buf_de_bytes() {
|
|
||||||
let mut de = BytesDeserializer::new(vec![]);
|
|
||||||
let bytes = Deserialize::deserialize(&mut de);
|
|
||||||
assert_eq!(bytes, Ok(ByteBuf::new()));
|
|
||||||
|
|
||||||
let mut de = BytesDeserializer::new(vec![1, 2, 3]);
|
|
||||||
let bytes = Deserialize::deserialize(&mut de);
|
|
||||||
assert_eq!(bytes, Ok(ByteBuf::from(vec![1, 2, 3])));
|
|
||||||
}
|
|
||||||
@@ -1,838 +0,0 @@
|
|||||||
use std::collections::{BTreeMap, BTreeSet, HashMap, HashSet};
|
|
||||||
use std::net;
|
|
||||||
use std::path::PathBuf;
|
|
||||||
use std::time::Duration;
|
|
||||||
|
|
||||||
use serde::Deserialize;
|
|
||||||
|
|
||||||
extern crate fnv;
|
|
||||||
use self::fnv::FnvHasher;
|
|
||||||
|
|
||||||
extern crate serde_test;
|
|
||||||
use self::serde_test::{
|
|
||||||
Error,
|
|
||||||
Token,
|
|
||||||
assert_de_tokens,
|
|
||||||
assert_de_tokens_error,
|
|
||||||
};
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#[derive(Copy, Clone, PartialEq, Debug, Deserialize)]
|
|
||||||
struct UnitStruct;
|
|
||||||
|
|
||||||
#[derive(PartialEq, Debug, Deserialize)]
|
|
||||||
struct TupleStruct(i32, i32, i32);
|
|
||||||
|
|
||||||
#[derive(PartialEq, Debug, Deserialize)]
|
|
||||||
struct Struct {
|
|
||||||
a: i32,
|
|
||||||
b: i32,
|
|
||||||
#[serde(skip_deserializing)]
|
|
||||||
c: i32,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(PartialEq, Debug, Deserialize)]
|
|
||||||
enum Enum {
|
|
||||||
Unit,
|
|
||||||
Simple(i32),
|
|
||||||
Seq(i32, i32, i32),
|
|
||||||
Map { a: i32, b: i32, c: i32 }
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
macro_rules! declare_test {
|
|
||||||
($name:ident { $($value:expr => $tokens:expr,)+ }) => {
|
|
||||||
#[test]
|
|
||||||
fn $name() {
|
|
||||||
$(
|
|
||||||
// Test ser/de roundtripping
|
|
||||||
assert_de_tokens(&$value, $tokens);
|
|
||||||
|
|
||||||
// Test that the tokens are ignorable
|
|
||||||
assert_de_tokens_ignore($tokens);
|
|
||||||
)+
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! declare_tests {
|
|
||||||
($($name:ident { $($value:expr => $tokens:expr,)+ })+) => {
|
|
||||||
$(
|
|
||||||
declare_test!($name { $($value => $tokens,)+ });
|
|
||||||
)+
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
macro_rules! declare_error_tests {
|
|
||||||
($($name:ident<$target:ident> { $tokens:expr, $expected:expr, })+) => {
|
|
||||||
$(
|
|
||||||
#[test]
|
|
||||||
fn $name() {
|
|
||||||
assert_de_tokens_error::<$target>($tokens, $expected);
|
|
||||||
}
|
|
||||||
)+
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn assert_de_tokens_ignore(ignorable_tokens: &[Token<'static>]) {
|
|
||||||
#[derive(PartialEq, Debug, Deserialize)]
|
|
||||||
struct IgnoreBase {
|
|
||||||
a: i32,
|
|
||||||
}
|
|
||||||
|
|
||||||
let expected = IgnoreBase{a: 1};
|
|
||||||
|
|
||||||
// Embed the tokens to be ignored in the normal token
|
|
||||||
// stream for an IgnoreBase type
|
|
||||||
let concated_tokens : Vec<Token<'static>> = vec![
|
|
||||||
Token::MapStart(Some(2)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::Str("ignored")
|
|
||||||
]
|
|
||||||
.into_iter()
|
|
||||||
.chain(ignorable_tokens.to_vec().into_iter())
|
|
||||||
.chain(vec![
|
|
||||||
Token::MapEnd,
|
|
||||||
].into_iter())
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let mut de = serde_test::Deserializer::new(concated_tokens.into_iter());
|
|
||||||
let v: Result<IgnoreBase, Error> = Deserialize::deserialize(&mut de);
|
|
||||||
|
|
||||||
// We run this test on every token stream for convenience, but
|
|
||||||
// some token streams don't make sense embedded as a map value,
|
|
||||||
// so we ignore those. SyntaxError is the real sign of trouble.
|
|
||||||
if let Err(Error::UnexpectedToken(_)) = v {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
assert_eq!(v.as_ref(), Ok(&expected));
|
|
||||||
assert_eq!(de.next_token(), None);
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
declare_tests! {
|
|
||||||
test_bool {
|
|
||||||
true => &[Token::Bool(true)],
|
|
||||||
false => &[Token::Bool(false)],
|
|
||||||
}
|
|
||||||
test_isize {
|
|
||||||
0isize => &[Token::Isize(0)],
|
|
||||||
0isize => &[Token::I8(0)],
|
|
||||||
0isize => &[Token::I16(0)],
|
|
||||||
0isize => &[Token::I32(0)],
|
|
||||||
0isize => &[Token::I64(0)],
|
|
||||||
0isize => &[Token::Usize(0)],
|
|
||||||
0isize => &[Token::U8(0)],
|
|
||||||
0isize => &[Token::U16(0)],
|
|
||||||
0isize => &[Token::U32(0)],
|
|
||||||
0isize => &[Token::U64(0)],
|
|
||||||
0isize => &[Token::F32(0.)],
|
|
||||||
0isize => &[Token::F64(0.)],
|
|
||||||
}
|
|
||||||
test_ints {
|
|
||||||
0isize => &[Token::Isize(0)],
|
|
||||||
0i8 => &[Token::I8(0)],
|
|
||||||
0i16 => &[Token::I16(0)],
|
|
||||||
0i32 => &[Token::I32(0)],
|
|
||||||
0i64 => &[Token::I64(0)],
|
|
||||||
}
|
|
||||||
test_uints {
|
|
||||||
0usize => &[Token::Usize(0)],
|
|
||||||
0u8 => &[Token::U8(0)],
|
|
||||||
0u16 => &[Token::U16(0)],
|
|
||||||
0u32 => &[Token::U32(0)],
|
|
||||||
0u64 => &[Token::U64(0)],
|
|
||||||
}
|
|
||||||
test_floats {
|
|
||||||
0f32 => &[Token::F32(0.)],
|
|
||||||
0f64 => &[Token::F64(0.)],
|
|
||||||
}
|
|
||||||
test_char {
|
|
||||||
'a' => &[Token::Char('a')],
|
|
||||||
'a' => &[Token::Str("a")],
|
|
||||||
'a' => &[Token::String("a".to_owned())],
|
|
||||||
}
|
|
||||||
test_string {
|
|
||||||
"abc".to_owned() => &[Token::Str("abc")],
|
|
||||||
"abc".to_owned() => &[Token::String("abc".to_owned())],
|
|
||||||
"a".to_owned() => &[Token::Char('a')],
|
|
||||||
}
|
|
||||||
test_option {
|
|
||||||
None::<i32> => &[Token::Unit],
|
|
||||||
None::<i32> => &[Token::Option(false)],
|
|
||||||
Some(1) => &[Token::I32(1)],
|
|
||||||
Some(1) => &[
|
|
||||||
Token::Option(true),
|
|
||||||
Token::I32(1),
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_result {
|
|
||||||
Ok::<i32, i32>(0) => &[
|
|
||||||
Token::EnumStart("Result"),
|
|
||||||
Token::Str("Ok"),
|
|
||||||
Token::I32(0),
|
|
||||||
],
|
|
||||||
Err::<i32, i32>(1) => &[
|
|
||||||
Token::EnumStart("Result"),
|
|
||||||
Token::Str("Err"),
|
|
||||||
Token::I32(1),
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_unit {
|
|
||||||
() => &[Token::Unit],
|
|
||||||
() => &[
|
|
||||||
Token::SeqStart(Some(0)),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
() => &[
|
|
||||||
Token::SeqStart(None),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
() => &[
|
|
||||||
Token::TupleStructStart("Anything", 0),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_unit_struct {
|
|
||||||
UnitStruct => &[Token::Unit],
|
|
||||||
UnitStruct => &[
|
|
||||||
Token::UnitStruct("UnitStruct"),
|
|
||||||
],
|
|
||||||
UnitStruct => &[
|
|
||||||
Token::SeqStart(Some(0)),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
UnitStruct => &[
|
|
||||||
Token::SeqStart(None),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_tuple_struct {
|
|
||||||
TupleStruct(1, 2, 3) => &[
|
|
||||||
Token::SeqStart(Some(3)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
TupleStruct(1, 2, 3) => &[
|
|
||||||
Token::SeqStart(None),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
TupleStruct(1, 2, 3) => &[
|
|
||||||
Token::TupleStructStart("TupleStruct", 3),
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::TupleStructEnd,
|
|
||||||
],
|
|
||||||
TupleStruct(1, 2, 3) => &[
|
|
||||||
Token::TupleStructStart("TupleStruct", 3),
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::TupleStructEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_btreeset {
|
|
||||||
BTreeSet::<isize>::new() => &[
|
|
||||||
Token::Unit,
|
|
||||||
],
|
|
||||||
BTreeSet::<isize>::new() => &[
|
|
||||||
Token::SeqStart(Some(0)),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
btreeset![btreeset![], btreeset![1], btreeset![2, 3]] => &[
|
|
||||||
Token::SeqStart(Some(3)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqStart(Some(0)),
|
|
||||||
Token::SeqEnd,
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqStart(Some(1)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::SeqEnd,
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqStart(Some(2)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::SeqEnd,
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
BTreeSet::<isize>::new() => &[
|
|
||||||
Token::UnitStruct("Anything"),
|
|
||||||
],
|
|
||||||
BTreeSet::<isize>::new() => &[
|
|
||||||
Token::TupleStructStart("Anything", 0),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_hashset {
|
|
||||||
HashSet::<isize>::new() => &[
|
|
||||||
Token::Unit,
|
|
||||||
],
|
|
||||||
HashSet::<isize>::new() => &[
|
|
||||||
Token::SeqStart(Some(0)),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
hashset![1, 2, 3] => &[
|
|
||||||
Token::SeqStart(Some(3)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
HashSet::<isize>::new() => &[
|
|
||||||
Token::UnitStruct("Anything"),
|
|
||||||
],
|
|
||||||
HashSet::<isize>::new() => &[
|
|
||||||
Token::TupleStructStart("Anything", 0),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
hashset![FnvHasher @ 1, 2, 3] => &[
|
|
||||||
Token::SeqStart(Some(3)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_vec {
|
|
||||||
Vec::<isize>::new() => &[
|
|
||||||
Token::Unit,
|
|
||||||
],
|
|
||||||
Vec::<isize>::new() => &[
|
|
||||||
Token::SeqStart(Some(0)),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
vec![vec![], vec![1], vec![2, 3]] => &[
|
|
||||||
Token::SeqStart(Some(3)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqStart(Some(0)),
|
|
||||||
Token::SeqEnd,
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqStart(Some(1)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::SeqEnd,
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqStart(Some(2)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::SeqEnd,
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
Vec::<isize>::new() => &[
|
|
||||||
Token::UnitStruct("Anything"),
|
|
||||||
],
|
|
||||||
Vec::<isize>::new() => &[
|
|
||||||
Token::TupleStructStart("Anything", 0),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_array {
|
|
||||||
[0; 0] => &[
|
|
||||||
Token::Unit,
|
|
||||||
],
|
|
||||||
[0; 0] => &[
|
|
||||||
Token::SeqStart(Some(0)),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
[0; 0] => &[
|
|
||||||
Token::SeqArrayStart(0),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
([0; 0], [1], [2, 3]) => &[
|
|
||||||
Token::SeqStart(Some(3)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqStart(Some(0)),
|
|
||||||
Token::SeqEnd,
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqStart(Some(1)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::SeqEnd,
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqStart(Some(2)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::SeqEnd,
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
([0; 0], [1], [2, 3]) => &[
|
|
||||||
Token::SeqArrayStart(3),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqArrayStart(0),
|
|
||||||
Token::SeqEnd,
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqArrayStart(1),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::SeqEnd,
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqArrayStart(2),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::SeqEnd,
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
[0; 0] => &[
|
|
||||||
Token::UnitStruct("Anything"),
|
|
||||||
],
|
|
||||||
[0; 0] => &[
|
|
||||||
Token::TupleStructStart("Anything", 0),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_tuple {
|
|
||||||
(1,) => &[
|
|
||||||
Token::SeqStart(Some(1)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
(1, 2, 3) => &[
|
|
||||||
Token::SeqStart(Some(3)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
(1,) => &[
|
|
||||||
Token::TupleStart(1),
|
|
||||||
Token::TupleSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::TupleEnd,
|
|
||||||
],
|
|
||||||
(1, 2, 3) => &[
|
|
||||||
Token::TupleStart(3),
|
|
||||||
Token::TupleSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::TupleSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::TupleSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::TupleEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_btreemap {
|
|
||||||
BTreeMap::<isize, isize>::new() => &[
|
|
||||||
Token::Unit,
|
|
||||||
],
|
|
||||||
BTreeMap::<isize, isize>::new() => &[
|
|
||||||
Token::MapStart(Some(0)),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
btreemap![1 => 2] => &[
|
|
||||||
Token::MapStart(Some(1)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::I32(2),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
btreemap![1 => 2, 3 => 4] => &[
|
|
||||||
Token::MapStart(Some(2)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::I32(4),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
btreemap![1 => btreemap![], 2 => btreemap![3 => 4, 5 => 6]] => &[
|
|
||||||
Token::MapStart(Some(2)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::MapStart(Some(0)),
|
|
||||||
Token::MapEnd,
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(2),
|
|
||||||
Token::MapStart(Some(2)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::I32(4),
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(5),
|
|
||||||
Token::I32(6),
|
|
||||||
Token::MapEnd,
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
BTreeMap::<isize, isize>::new() => &[
|
|
||||||
Token::UnitStruct("Anything"),
|
|
||||||
],
|
|
||||||
BTreeMap::<isize, isize>::new() => &[
|
|
||||||
Token::StructStart("Anything", 0),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_hashmap {
|
|
||||||
HashMap::<isize, isize>::new() => &[
|
|
||||||
Token::Unit,
|
|
||||||
],
|
|
||||||
HashMap::<isize, isize>::new() => &[
|
|
||||||
Token::MapStart(Some(0)),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
hashmap![1 => 2] => &[
|
|
||||||
Token::MapStart(Some(1)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::I32(2),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
hashmap![1 => 2, 3 => 4] => &[
|
|
||||||
Token::MapStart(Some(2)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::I32(4),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
hashmap![1 => hashmap![], 2 => hashmap![3 => 4, 5 => 6]] => &[
|
|
||||||
Token::MapStart(Some(2)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::MapStart(Some(0)),
|
|
||||||
Token::MapEnd,
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(2),
|
|
||||||
Token::MapStart(Some(2)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::I32(4),
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(5),
|
|
||||||
Token::I32(6),
|
|
||||||
Token::MapEnd,
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
HashMap::<isize, isize>::new() => &[
|
|
||||||
Token::UnitStruct("Anything"),
|
|
||||||
],
|
|
||||||
HashMap::<isize, isize>::new() => &[
|
|
||||||
Token::StructStart("Anything", 0),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
hashmap![FnvHasher @ 1 => 2, 3 => 4] => &[
|
|
||||||
Token::MapStart(Some(2)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::I32(4),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_struct {
|
|
||||||
Struct { a: 1, b: 2, c: 0 } => &[
|
|
||||||
Token::MapStart(Some(3)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::I32(2),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
Struct { a: 1, b: 2, c: 0 } => &[
|
|
||||||
Token::StructStart("Struct", 3),
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::I32(2),
|
|
||||||
Token::StructEnd,
|
|
||||||
],
|
|
||||||
Struct { a: 1, b: 2, c: 0 } => &[
|
|
||||||
Token::SeqStart(Some(3)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_struct_with_skip {
|
|
||||||
Struct { a: 1, b: 2, c: 0 } => &[
|
|
||||||
Token::MapStart(Some(3)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::Str("c"),
|
|
||||||
Token::I32(3),
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::Str("d"),
|
|
||||||
Token::I32(4),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
Struct { a: 1, b: 2, c: 0 } => &[
|
|
||||||
Token::StructStart("Struct", 3),
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("c"),
|
|
||||||
Token::I32(3),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("d"),
|
|
||||||
Token::I32(4),
|
|
||||||
Token::StructEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_enum_unit {
|
|
||||||
Enum::Unit => &[
|
|
||||||
Token::EnumUnit("Enum", "Unit"),
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_enum_simple {
|
|
||||||
Enum::Simple(1) => &[
|
|
||||||
Token::EnumNewType("Enum", "Simple"),
|
|
||||||
Token::I32(1),
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_enum_seq {
|
|
||||||
Enum::Seq(1, 2, 3) => &[
|
|
||||||
Token::EnumSeqStart("Enum", "Seq", 3),
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::EnumSeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_enum_map {
|
|
||||||
Enum::Map { a: 1, b: 2, c: 3 } => &[
|
|
||||||
Token::EnumMapStart("Enum", "Map", 3),
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("c"),
|
|
||||||
Token::I32(3),
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_enum_unit_usize {
|
|
||||||
Enum::Unit => &[
|
|
||||||
Token::EnumStart("Enum"),
|
|
||||||
Token::Usize(0),
|
|
||||||
Token::Unit,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_enum_unit_bytes {
|
|
||||||
Enum::Unit => &[
|
|
||||||
Token::EnumStart("Enum"),
|
|
||||||
Token::Bytes(b"Unit"),
|
|
||||||
Token::Unit,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_box {
|
|
||||||
Box::new(0i32) => &[Token::I32(0)],
|
|
||||||
}
|
|
||||||
test_boxed_slice {
|
|
||||||
Box::new([0, 1, 2]) => &[
|
|
||||||
Token::SeqStart(Some(3)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(0),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_duration {
|
|
||||||
Duration::new(1, 2) => &[
|
|
||||||
Token::StructStart("Duration", 2),
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("secs"),
|
|
||||||
Token::U64(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("nanos"),
|
|
||||||
Token::U32(2),
|
|
||||||
Token::StructEnd,
|
|
||||||
],
|
|
||||||
Duration::new(1, 2) => &[
|
|
||||||
Token::SeqStart(Some(2)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I64(1),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I64(2),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_net_ipv4addr {
|
|
||||||
"1.2.3.4".parse::<net::Ipv4Addr>().unwrap() => &[Token::Str("1.2.3.4")],
|
|
||||||
}
|
|
||||||
test_net_ipv6addr {
|
|
||||||
"::1".parse::<net::Ipv6Addr>().unwrap() => &[Token::Str("::1")],
|
|
||||||
}
|
|
||||||
test_net_socketaddr {
|
|
||||||
"1.2.3.4:1234".parse::<net::SocketAddr>().unwrap() => &[Token::Str("1.2.3.4:1234")],
|
|
||||||
"1.2.3.4:1234".parse::<net::SocketAddrV4>().unwrap() => &[Token::Str("1.2.3.4:1234")],
|
|
||||||
"[::1]:1234".parse::<net::SocketAddrV6>().unwrap() => &[Token::Str("[::1]:1234")],
|
|
||||||
}
|
|
||||||
test_path_buf {
|
|
||||||
PathBuf::from("/usr/local/lib") => &[
|
|
||||||
Token::String("/usr/local/lib".to_owned()),
|
|
||||||
],
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "unstable")]
|
|
||||||
#[test]
|
|
||||||
fn test_net_ipaddr() {
|
|
||||||
assert_de_tokens(
|
|
||||||
"1.2.3.4".parse::<net::IpAddr>().unwrap(),
|
|
||||||
&[Token::Str("1.2.3.4")],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
declare_error_tests! {
|
|
||||||
test_unknown_variant<Enum> {
|
|
||||||
&[
|
|
||||||
Token::EnumUnit("Enum", "Foo"),
|
|
||||||
],
|
|
||||||
Error::UnknownVariant("Foo".to_owned()),
|
|
||||||
}
|
|
||||||
test_struct_seq_too_long<Struct> {
|
|
||||||
&[
|
|
||||||
Token::SeqStart(Some(4)),
|
|
||||||
Token::SeqSep, Token::I32(1),
|
|
||||||
Token::SeqSep, Token::I32(2),
|
|
||||||
Token::SeqSep, Token::I32(3),
|
|
||||||
],
|
|
||||||
Error::UnexpectedToken(Token::SeqSep),
|
|
||||||
}
|
|
||||||
test_duplicate_field_struct<Struct> {
|
|
||||||
&[
|
|
||||||
Token::MapStart(Some(3)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
],
|
|
||||||
Error::DuplicateField("a"),
|
|
||||||
}
|
|
||||||
test_duplicate_field_enum<Enum> {
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("Enum", "Map", 3),
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
],
|
|
||||||
Error::DuplicateField("a"),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,226 +0,0 @@
|
|||||||
// These just test that serde_codegen is able to produce code that compiles
|
|
||||||
// successfully when there are a variety of generics and non-(de)serializable
|
|
||||||
// types involved.
|
|
||||||
|
|
||||||
extern crate serde;
|
|
||||||
use self::serde::ser::{Serialize, Serializer};
|
|
||||||
use self::serde::de::{Deserialize, Deserializer};
|
|
||||||
|
|
||||||
use std::borrow::Cow;
|
|
||||||
use std::marker::PhantomData;
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_gen() {
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
struct With<T> {
|
|
||||||
t: T,
|
|
||||||
#[serde(serialize_with="ser_x", deserialize_with="de_x")]
|
|
||||||
x: X,
|
|
||||||
}
|
|
||||||
assert::<With<i32>>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
struct WithRef<'a, T: 'a> {
|
|
||||||
#[serde(skip_deserializing)]
|
|
||||||
t: Option<&'a T>,
|
|
||||||
#[serde(serialize_with="ser_x", deserialize_with="de_x")]
|
|
||||||
x: X,
|
|
||||||
}
|
|
||||||
assert::<WithRef<i32>>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
struct PhantomX {
|
|
||||||
x: PhantomData<X>,
|
|
||||||
}
|
|
||||||
assert::<PhantomX>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
struct PhantomT<T> {
|
|
||||||
t: PhantomData<T>,
|
|
||||||
}
|
|
||||||
assert::<PhantomT<X>>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
struct Bounds<T: Serialize + Deserialize> {
|
|
||||||
t: T,
|
|
||||||
option: Option<T>,
|
|
||||||
boxed: Box<T>,
|
|
||||||
option_boxed: Option<Box<T>>,
|
|
||||||
}
|
|
||||||
assert::<Bounds<i32>>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
struct NoBounds<T> {
|
|
||||||
t: T,
|
|
||||||
option: Option<T>,
|
|
||||||
boxed: Box<T>,
|
|
||||||
option_boxed: Option<Box<T>>,
|
|
||||||
}
|
|
||||||
assert::<NoBounds<i32>>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
enum EnumWith<T> {
|
|
||||||
Unit,
|
|
||||||
Newtype(
|
|
||||||
#[serde(serialize_with="ser_x", deserialize_with="de_x")]
|
|
||||||
X),
|
|
||||||
Tuple(
|
|
||||||
T,
|
|
||||||
#[serde(serialize_with="ser_x", deserialize_with="de_x")]
|
|
||||||
X),
|
|
||||||
Struct {
|
|
||||||
t: T,
|
|
||||||
#[serde(serialize_with="ser_x", deserialize_with="de_x")]
|
|
||||||
x: X },
|
|
||||||
}
|
|
||||||
assert::<EnumWith<i32>>();
|
|
||||||
|
|
||||||
#[derive(Serialize)]
|
|
||||||
struct MultipleRef<'a, 'b, 'c, T> where T: 'c, 'c: 'b, 'b: 'a {
|
|
||||||
t: T,
|
|
||||||
rrrt: &'a &'b &'c T,
|
|
||||||
}
|
|
||||||
assert_ser::<MultipleRef<i32>>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
struct Newtype(
|
|
||||||
#[serde(serialize_with="ser_x", deserialize_with="de_x")]
|
|
||||||
X
|
|
||||||
);
|
|
||||||
assert::<Newtype>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
struct Tuple<T>(
|
|
||||||
T,
|
|
||||||
#[serde(serialize_with="ser_x", deserialize_with="de_x")]
|
|
||||||
X,
|
|
||||||
);
|
|
||||||
assert::<Tuple<i32>>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
enum TreeNode<D> {
|
|
||||||
Split {
|
|
||||||
left: Box<TreeNode<D>>,
|
|
||||||
right: Box<TreeNode<D>>,
|
|
||||||
},
|
|
||||||
Leaf {
|
|
||||||
data: D,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
assert::<TreeNode<i32>>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
struct ListNode<D> {
|
|
||||||
data: D,
|
|
||||||
next: Box<ListNode<D>>,
|
|
||||||
}
|
|
||||||
assert::<ListNode<i32>>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
struct RecursiveA {
|
|
||||||
b: Box<RecursiveB>,
|
|
||||||
}
|
|
||||||
assert::<RecursiveA>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
enum RecursiveB {
|
|
||||||
A(RecursiveA),
|
|
||||||
}
|
|
||||||
assert::<RecursiveB>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
struct RecursiveGenericA<T> {
|
|
||||||
t: T,
|
|
||||||
b: Box<RecursiveGenericB<T>>,
|
|
||||||
}
|
|
||||||
assert::<RecursiveGenericA<i32>>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
enum RecursiveGenericB<T> {
|
|
||||||
T(T),
|
|
||||||
A(RecursiveGenericA<T>),
|
|
||||||
}
|
|
||||||
assert::<RecursiveGenericB<i32>>();
|
|
||||||
|
|
||||||
#[derive(Serialize)]
|
|
||||||
struct OptionStatic<'a> {
|
|
||||||
a: Option<&'a str>,
|
|
||||||
b: Option<&'static str>,
|
|
||||||
}
|
|
||||||
assert_ser::<OptionStatic>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
#[serde(bound="D: SerializeWith + DeserializeWith")]
|
|
||||||
struct WithTraits1<D, E> {
|
|
||||||
#[serde(serialize_with="SerializeWith::serialize_with",
|
|
||||||
deserialize_with="DeserializeWith::deserialize_with")]
|
|
||||||
d: D,
|
|
||||||
#[serde(serialize_with="SerializeWith::serialize_with",
|
|
||||||
deserialize_with="DeserializeWith::deserialize_with",
|
|
||||||
bound="E: SerializeWith + DeserializeWith")]
|
|
||||||
e: E,
|
|
||||||
}
|
|
||||||
assert::<WithTraits1<X, X>>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
#[serde(bound(serialize="D: SerializeWith",
|
|
||||||
deserialize="D: DeserializeWith"))]
|
|
||||||
struct WithTraits2<D, E> {
|
|
||||||
#[serde(serialize_with="SerializeWith::serialize_with",
|
|
||||||
deserialize_with="DeserializeWith::deserialize_with")]
|
|
||||||
d: D,
|
|
||||||
#[serde(serialize_with="SerializeWith::serialize_with",
|
|
||||||
bound(serialize="E: SerializeWith"))]
|
|
||||||
#[serde(deserialize_with="DeserializeWith::deserialize_with",
|
|
||||||
bound(deserialize="E: DeserializeWith"))]
|
|
||||||
e: E,
|
|
||||||
}
|
|
||||||
assert::<WithTraits2<X, X>>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
struct CowStr<'a>(Cow<'a, str>);
|
|
||||||
assert::<CowStr>();
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize)]
|
|
||||||
#[serde(bound(deserialize = "T::Owned: Deserialize"))]
|
|
||||||
struct CowT<'a, T: ?Sized + 'a + ToOwned>(Cow<'a, T>);
|
|
||||||
assert::<CowT<str>>();
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
fn assert<T: Serialize + Deserialize>() {}
|
|
||||||
fn assert_ser<T: Serialize>() {}
|
|
||||||
|
|
||||||
trait SerializeWith {
|
|
||||||
fn serialize_with<S: Serializer>(_: &Self, _: &mut S) -> Result<(), S::Error>;
|
|
||||||
}
|
|
||||||
|
|
||||||
trait DeserializeWith: Sized {
|
|
||||||
fn deserialize_with<D: Deserializer>(_: &mut D) -> Result<Self, D::Error>;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Implements neither Serialize nor Deserialize
|
|
||||||
struct X;
|
|
||||||
|
|
||||||
fn ser_x<S: Serializer>(_: &X, _: &mut S) -> Result<(), S::Error> {
|
|
||||||
unimplemented!()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn de_x<D: Deserializer>(_: &mut D) -> Result<X, D::Error> {
|
|
||||||
unimplemented!()
|
|
||||||
}
|
|
||||||
|
|
||||||
impl SerializeWith for X {
|
|
||||||
fn serialize_with<S: Serializer>(_: &Self, _: &mut S) -> Result<(), S::Error> {
|
|
||||||
unimplemented!()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl DeserializeWith for X {
|
|
||||||
fn deserialize_with<D: Deserializer>(_: &mut D) -> Result<Self, D::Error> {
|
|
||||||
unimplemented!()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,602 +0,0 @@
|
|||||||
extern crate serde_test;
|
|
||||||
use self::serde_test::{
|
|
||||||
Token,
|
|
||||||
assert_tokens,
|
|
||||||
assert_ser_tokens,
|
|
||||||
assert_de_tokens,
|
|
||||||
};
|
|
||||||
|
|
||||||
use std::marker::PhantomData;
|
|
||||||
|
|
||||||
// That tests that the derived Serialize implementation doesn't trigger
|
|
||||||
// any warning about `serializer` not being used, in case of empty enums.
|
|
||||||
#[derive(Serialize)]
|
|
||||||
#[allow(dead_code)]
|
|
||||||
#[deny(unused_variables)]
|
|
||||||
enum Void {}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
struct NamedUnit;
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize)]
|
|
||||||
struct SerNamedTuple<'a, 'b, A: 'a, B: 'b, C>(&'a A, &'b mut B, C);
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Deserialize)]
|
|
||||||
struct DeNamedTuple<A, B, C>(A, B, C);
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize)]
|
|
||||||
struct SerNamedMap<'a, 'b, A: 'a, B: 'b, C> {
|
|
||||||
a: &'a A,
|
|
||||||
b: &'b mut B,
|
|
||||||
c: C,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Deserialize)]
|
|
||||||
struct DeNamedMap<A, B, C> {
|
|
||||||
a: A,
|
|
||||||
b: B,
|
|
||||||
c: C,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize)]
|
|
||||||
enum SerEnum<'a, B: 'a, C: 'a, D> where D: 'a {
|
|
||||||
Unit,
|
|
||||||
Seq(
|
|
||||||
i8,
|
|
||||||
B,
|
|
||||||
&'a C,
|
|
||||||
&'a mut D,
|
|
||||||
),
|
|
||||||
Map {
|
|
||||||
a: i8,
|
|
||||||
b: B,
|
|
||||||
c: &'a C,
|
|
||||||
d: &'a mut D,
|
|
||||||
},
|
|
||||||
|
|
||||||
// Make sure we can support more than one variant.
|
|
||||||
_Unit2,
|
|
||||||
_Seq2(
|
|
||||||
i8,
|
|
||||||
B,
|
|
||||||
&'a C,
|
|
||||||
&'a mut D,
|
|
||||||
),
|
|
||||||
_Map2 {
|
|
||||||
a: i8,
|
|
||||||
b: B,
|
|
||||||
c: &'a C,
|
|
||||||
d: &'a mut D,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
enum DeEnum<B, C, D> {
|
|
||||||
Unit,
|
|
||||||
Seq(
|
|
||||||
i8,
|
|
||||||
B,
|
|
||||||
C,
|
|
||||||
D,
|
|
||||||
),
|
|
||||||
Map {
|
|
||||||
a: i8,
|
|
||||||
b: B,
|
|
||||||
c: C,
|
|
||||||
d: D,
|
|
||||||
},
|
|
||||||
|
|
||||||
// Make sure we can support more than one variant.
|
|
||||||
_Unit2,
|
|
||||||
_Seq2(
|
|
||||||
i8,
|
|
||||||
B,
|
|
||||||
C,
|
|
||||||
D,
|
|
||||||
),
|
|
||||||
_Map2 {
|
|
||||||
a: i8,
|
|
||||||
b: B,
|
|
||||||
c: C,
|
|
||||||
d: D,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Serialize)]
|
|
||||||
enum Lifetimes<'a> {
|
|
||||||
LifetimeSeq(&'a i32),
|
|
||||||
NoLifetimeSeq(i32),
|
|
||||||
LifetimeMap { a: &'a i32 },
|
|
||||||
NoLifetimeMap { a: i32 },
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
pub struct GenericStruct<T> {
|
|
||||||
x: T,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
pub struct GenericNewTypeStruct<T>(T);
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
pub struct GenericTupleStruct<T, U>(T, U);
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
pub enum GenericEnum<T, U> {
|
|
||||||
Unit,
|
|
||||||
NewType(T),
|
|
||||||
Seq(T, U),
|
|
||||||
Map { x: T, y: U },
|
|
||||||
}
|
|
||||||
|
|
||||||
trait AssociatedType {
|
|
||||||
type X;
|
|
||||||
}
|
|
||||||
|
|
||||||
impl AssociatedType for i32 {
|
|
||||||
type X = i32;
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Debug, PartialEq, Serialize, Deserialize)]
|
|
||||||
struct DefaultTyParam<T: AssociatedType<X=i32> = i32> {
|
|
||||||
phantom: PhantomData<T>
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_named_unit() {
|
|
||||||
assert_tokens(
|
|
||||||
&NamedUnit,
|
|
||||||
&[Token::UnitStruct("NamedUnit")]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_ser_named_tuple() {
|
|
||||||
let a = 5;
|
|
||||||
let mut b = 6;
|
|
||||||
let c = 7;
|
|
||||||
assert_ser_tokens(
|
|
||||||
&SerNamedTuple(&a, &mut b, c),
|
|
||||||
&[
|
|
||||||
Token::TupleStructStart("SerNamedTuple", 3),
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(5),
|
|
||||||
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(6),
|
|
||||||
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(7),
|
|
||||||
|
|
||||||
Token::TupleStructEnd,
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_de_named_tuple() {
|
|
||||||
assert_de_tokens(
|
|
||||||
&DeNamedTuple(5, 6, 7),
|
|
||||||
&[
|
|
||||||
Token::SeqStart(Some(3)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(5),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(6),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(7),
|
|
||||||
|
|
||||||
Token::SeqEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_de_tokens(
|
|
||||||
&DeNamedTuple(5, 6, 7),
|
|
||||||
&[
|
|
||||||
Token::TupleStructStart("DeNamedTuple", 3),
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(5),
|
|
||||||
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(6),
|
|
||||||
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(7),
|
|
||||||
|
|
||||||
Token::TupleStructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_ser_named_map() {
|
|
||||||
let a = 5;
|
|
||||||
let mut b = 6;
|
|
||||||
let c = 7;
|
|
||||||
|
|
||||||
assert_ser_tokens(
|
|
||||||
&SerNamedMap {
|
|
||||||
a: &a,
|
|
||||||
b: &mut b,
|
|
||||||
c: c,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::StructStart("SerNamedMap", 3),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(5),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::I32(6),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("c"),
|
|
||||||
Token::I32(7),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_de_named_map() {
|
|
||||||
assert_de_tokens(
|
|
||||||
&DeNamedMap {
|
|
||||||
a: 5,
|
|
||||||
b: 6,
|
|
||||||
c: 7,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::StructStart("DeNamedMap", 3),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(5),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::I32(6),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("c"),
|
|
||||||
Token::I32(7),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_ser_enum_unit() {
|
|
||||||
assert_ser_tokens(
|
|
||||||
&SerEnum::Unit::<u32, u32, u32>,
|
|
||||||
&[
|
|
||||||
Token::EnumUnit("SerEnum", "Unit"),
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_ser_enum_seq() {
|
|
||||||
let a = 1;
|
|
||||||
let b = 2;
|
|
||||||
let c = 3;
|
|
||||||
let mut d = 4;
|
|
||||||
|
|
||||||
assert_ser_tokens(
|
|
||||||
&SerEnum::Seq(
|
|
||||||
a,
|
|
||||||
b,
|
|
||||||
&c,
|
|
||||||
&mut d,
|
|
||||||
),
|
|
||||||
&[
|
|
||||||
Token::EnumSeqStart("SerEnum", "Seq", 4),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I32(4),
|
|
||||||
|
|
||||||
Token::EnumSeqEnd,
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_ser_enum_map() {
|
|
||||||
let a = 1;
|
|
||||||
let b = 2;
|
|
||||||
let c = 3;
|
|
||||||
let mut d = 4;
|
|
||||||
|
|
||||||
assert_ser_tokens(
|
|
||||||
&SerEnum::Map {
|
|
||||||
a: a,
|
|
||||||
b: b,
|
|
||||||
c: &c,
|
|
||||||
d: &mut d,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("SerEnum", "Map", 4),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("c"),
|
|
||||||
Token::I32(3),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("d"),
|
|
||||||
Token::I32(4),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_de_enum_unit() {
|
|
||||||
assert_tokens(
|
|
||||||
&DeEnum::Unit::<u32, u32, u32>,
|
|
||||||
&[
|
|
||||||
Token::EnumUnit("DeEnum", "Unit"),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_de_enum_seq() {
|
|
||||||
let a = 1;
|
|
||||||
let b = 2;
|
|
||||||
let c = 3;
|
|
||||||
let d = 4;
|
|
||||||
|
|
||||||
assert_tokens(
|
|
||||||
&DeEnum::Seq(
|
|
||||||
a,
|
|
||||||
b,
|
|
||||||
c,
|
|
||||||
d,
|
|
||||||
),
|
|
||||||
&[
|
|
||||||
Token::EnumSeqStart("DeEnum", "Seq", 4),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I32(4),
|
|
||||||
|
|
||||||
Token::EnumSeqEnd,
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_de_enum_map() {
|
|
||||||
let a = 1;
|
|
||||||
let b = 2;
|
|
||||||
let c = 3;
|
|
||||||
let d = 4;
|
|
||||||
|
|
||||||
assert_tokens(
|
|
||||||
&DeEnum::Map {
|
|
||||||
a: a,
|
|
||||||
b: b,
|
|
||||||
c: c,
|
|
||||||
d: d,
|
|
||||||
},
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("DeEnum", "Map", 4),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I8(1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("c"),
|
|
||||||
Token::I32(3),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("d"),
|
|
||||||
Token::I32(4),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_lifetimes() {
|
|
||||||
let value = 5;
|
|
||||||
|
|
||||||
assert_ser_tokens(
|
|
||||||
&Lifetimes::LifetimeSeq(&value),
|
|
||||||
&[
|
|
||||||
Token::EnumNewType("Lifetimes", "LifetimeSeq"),
|
|
||||||
Token::I32(5),
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_ser_tokens(
|
|
||||||
&Lifetimes::NoLifetimeSeq(5),
|
|
||||||
&[
|
|
||||||
Token::EnumNewType("Lifetimes", "NoLifetimeSeq"),
|
|
||||||
Token::I32(5),
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_ser_tokens(
|
|
||||||
&Lifetimes::LifetimeMap { a: &value },
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("Lifetimes", "LifetimeMap", 1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(5),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
assert_ser_tokens(
|
|
||||||
&Lifetimes::NoLifetimeMap { a: 5 },
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("Lifetimes", "NoLifetimeMap", 1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(5),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_generic_struct() {
|
|
||||||
assert_tokens(
|
|
||||||
&GenericStruct { x: 5u32 },
|
|
||||||
&[
|
|
||||||
Token::StructStart("GenericStruct", 1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("x"),
|
|
||||||
Token::U32(5),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_generic_newtype_struct() {
|
|
||||||
assert_tokens(
|
|
||||||
&GenericNewTypeStruct(5u32),
|
|
||||||
&[
|
|
||||||
Token::StructNewType("GenericNewTypeStruct"),
|
|
||||||
Token::U32(5),
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_generic_tuple_struct() {
|
|
||||||
assert_tokens(
|
|
||||||
&GenericTupleStruct(5u32, 6u32),
|
|
||||||
&[
|
|
||||||
Token::TupleStructStart("GenericTupleStruct", 2),
|
|
||||||
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::U32(5),
|
|
||||||
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::U32(6),
|
|
||||||
|
|
||||||
Token::TupleStructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_generic_enum_unit() {
|
|
||||||
assert_tokens(
|
|
||||||
&GenericEnum::Unit::<u32, u32>,
|
|
||||||
&[
|
|
||||||
Token::EnumUnit("GenericEnum", "Unit"),
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_generic_enum_newtype() {
|
|
||||||
assert_tokens(
|
|
||||||
&GenericEnum::NewType::<u32, u32>(5),
|
|
||||||
&[
|
|
||||||
Token::EnumNewType("GenericEnum", "NewType"),
|
|
||||||
Token::U32(5),
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_generic_enum_seq() {
|
|
||||||
assert_tokens(
|
|
||||||
&GenericEnum::Seq::<u32, u32>(5, 6),
|
|
||||||
&[
|
|
||||||
Token::EnumSeqStart("GenericEnum", "Seq", 2),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::U32(5),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::U32(6),
|
|
||||||
|
|
||||||
Token::EnumSeqEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_generic_enum_map() {
|
|
||||||
assert_tokens(
|
|
||||||
&GenericEnum::Map::<u32, u32> { x: 5, y: 6 },
|
|
||||||
&[
|
|
||||||
Token::EnumMapStart("GenericEnum", "Map", 2),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("x"),
|
|
||||||
Token::U32(5),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("y"),
|
|
||||||
Token::U32(6),
|
|
||||||
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_default_ty_param() {
|
|
||||||
assert_tokens(
|
|
||||||
&DefaultTyParam::<i32> { phantom: PhantomData },
|
|
||||||
&[
|
|
||||||
Token::StructStart("DefaultTyParam", 1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("phantom"),
|
|
||||||
Token::UnitStruct("PhantomData"),
|
|
||||||
|
|
||||||
Token::StructEnd,
|
|
||||||
]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
@@ -1,390 +0,0 @@
|
|||||||
use std::collections::{BTreeMap, HashMap, HashSet};
|
|
||||||
use std::net;
|
|
||||||
use std::path::{Path, PathBuf};
|
|
||||||
use std::str;
|
|
||||||
use std::time::Duration;
|
|
||||||
|
|
||||||
extern crate serde_test;
|
|
||||||
use self::serde_test::{
|
|
||||||
Error,
|
|
||||||
Token,
|
|
||||||
assert_ser_tokens,
|
|
||||||
assert_ser_tokens_error,
|
|
||||||
};
|
|
||||||
|
|
||||||
extern crate fnv;
|
|
||||||
use self::fnv::FnvHasher;
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
#[derive(Serialize)]
|
|
||||||
struct UnitStruct;
|
|
||||||
|
|
||||||
#[derive(Serialize)]
|
|
||||||
struct TupleStruct(i32, i32, i32);
|
|
||||||
|
|
||||||
#[derive(Serialize)]
|
|
||||||
struct Struct {
|
|
||||||
a: i32,
|
|
||||||
b: i32,
|
|
||||||
c: i32,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Serialize)]
|
|
||||||
enum Enum {
|
|
||||||
Unit,
|
|
||||||
One(i32),
|
|
||||||
Seq(i32, i32),
|
|
||||||
Map { a: i32, b: i32 },
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
declare_ser_tests! {
|
|
||||||
test_unit {
|
|
||||||
() => &[Token::Unit],
|
|
||||||
}
|
|
||||||
test_bool {
|
|
||||||
true => &[Token::Bool(true)],
|
|
||||||
false => &[Token::Bool(false)],
|
|
||||||
}
|
|
||||||
test_isizes {
|
|
||||||
0isize => &[Token::Isize(0)],
|
|
||||||
0i8 => &[Token::I8(0)],
|
|
||||||
0i16 => &[Token::I16(0)],
|
|
||||||
0i32 => &[Token::I32(0)],
|
|
||||||
0i64 => &[Token::I64(0)],
|
|
||||||
}
|
|
||||||
test_usizes {
|
|
||||||
0usize => &[Token::Usize(0)],
|
|
||||||
0u8 => &[Token::U8(0)],
|
|
||||||
0u16 => &[Token::U16(0)],
|
|
||||||
0u32 => &[Token::U32(0)],
|
|
||||||
0u64 => &[Token::U64(0)],
|
|
||||||
}
|
|
||||||
test_floats {
|
|
||||||
0f32 => &[Token::F32(0.)],
|
|
||||||
0f64 => &[Token::F64(0.)],
|
|
||||||
}
|
|
||||||
test_char {
|
|
||||||
'a' => &[Token::Char('a')],
|
|
||||||
}
|
|
||||||
test_str {
|
|
||||||
"abc" => &[Token::Str("abc")],
|
|
||||||
"abc".to_owned() => &[Token::Str("abc")],
|
|
||||||
}
|
|
||||||
test_option {
|
|
||||||
None::<i32> => &[Token::Option(false)],
|
|
||||||
Some(1) => &[
|
|
||||||
Token::Option(true),
|
|
||||||
Token::I32(1),
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_result {
|
|
||||||
Ok::<i32, i32>(0) => &[
|
|
||||||
Token::EnumNewType("Result", "Ok"),
|
|
||||||
Token::I32(0),
|
|
||||||
],
|
|
||||||
Err::<i32, i32>(1) => &[
|
|
||||||
Token::EnumNewType("Result", "Err"),
|
|
||||||
Token::I32(1),
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_slice {
|
|
||||||
&[0][..0] => &[
|
|
||||||
Token::SeqStart(Some(0)),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
&[1, 2, 3][..] => &[
|
|
||||||
Token::SeqStart(Some(3)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_array {
|
|
||||||
[0; 0] => &[
|
|
||||||
Token::SeqArrayStart(0),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
[1, 2, 3] => &[
|
|
||||||
Token::SeqArrayStart(3),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_vec {
|
|
||||||
Vec::<isize>::new() => &[
|
|
||||||
Token::SeqStart(Some(0)),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
vec![vec![], vec![1], vec![2, 3]] => &[
|
|
||||||
Token::SeqStart(Some(3)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqStart(Some(0)),
|
|
||||||
Token::SeqEnd,
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqStart(Some(1)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::SeqEnd,
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::SeqStart(Some(2)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::SeqEnd,
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_hashset {
|
|
||||||
HashSet::<isize>::new() => &[
|
|
||||||
Token::SeqStart(Some(0)),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
hashset![1] => &[
|
|
||||||
Token::SeqStart(Some(1)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
hashset![FnvHasher @ 1] => &[
|
|
||||||
Token::SeqStart(Some(1)),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_tuple {
|
|
||||||
(1,) => &[
|
|
||||||
Token::TupleStart(1),
|
|
||||||
Token::TupleSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::TupleEnd,
|
|
||||||
],
|
|
||||||
(1, 2, 3) => &[
|
|
||||||
Token::TupleStart(3),
|
|
||||||
Token::TupleSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::TupleSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::TupleSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::TupleEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_btreemap {
|
|
||||||
btreemap![1 => 2] => &[
|
|
||||||
Token::MapStart(Some(1)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::I32(2),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
btreemap![1 => 2, 3 => 4] => &[
|
|
||||||
Token::MapStart(Some(2)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::I32(4),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
btreemap![1 => btreemap![], 2 => btreemap![3 => 4, 5 => 6]] => &[
|
|
||||||
Token::MapStart(Some(2)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::MapStart(Some(0)),
|
|
||||||
Token::MapEnd,
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(2),
|
|
||||||
Token::MapStart(Some(2)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::I32(4),
|
|
||||||
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(5),
|
|
||||||
Token::I32(6),
|
|
||||||
Token::MapEnd,
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_hashmap {
|
|
||||||
HashMap::<isize, isize>::new() => &[
|
|
||||||
Token::MapStart(Some(0)),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
hashmap![1 => 2] => &[
|
|
||||||
Token::MapStart(Some(1)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::I32(2),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
hashmap![FnvHasher @ 1 => 2] => &[
|
|
||||||
Token::MapStart(Some(1)),
|
|
||||||
Token::MapSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::I32(2),
|
|
||||||
Token::MapEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_unit_struct {
|
|
||||||
UnitStruct => &[Token::UnitStruct("UnitStruct")],
|
|
||||||
}
|
|
||||||
test_tuple_struct {
|
|
||||||
TupleStruct(1, 2, 3) => &[
|
|
||||||
Token::TupleStructStart("TupleStruct", 3),
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::TupleStructSep,
|
|
||||||
Token::I32(3),
|
|
||||||
Token::TupleStructEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_struct {
|
|
||||||
Struct { a: 1, b: 2, c: 3 } => &[
|
|
||||||
Token::StructStart("Struct", 3),
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::I32(2),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("c"),
|
|
||||||
Token::I32(3),
|
|
||||||
Token::StructEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_enum {
|
|
||||||
Enum::Unit => &[Token::EnumUnit("Enum", "Unit")],
|
|
||||||
Enum::One(42) => &[Token::EnumNewType("Enum", "One"), Token::I32(42)],
|
|
||||||
Enum::Seq(1, 2) => &[
|
|
||||||
Token::EnumSeqStart("Enum", "Seq", 2),
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::EnumSeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
Token::EnumSeqEnd,
|
|
||||||
],
|
|
||||||
Enum::Map { a: 1, b: 2 } => &[
|
|
||||||
Token::EnumMapStart("Enum", "Map", 2),
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("a"),
|
|
||||||
Token::I32(1),
|
|
||||||
|
|
||||||
Token::EnumMapSep,
|
|
||||||
Token::Str("b"),
|
|
||||||
Token::I32(2),
|
|
||||||
Token::EnumMapEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_box {
|
|
||||||
Box::new(0i32) => &[Token::I32(0)],
|
|
||||||
}
|
|
||||||
test_boxed_slice {
|
|
||||||
Box::new([0, 1, 2]) => &[
|
|
||||||
Token::SeqArrayStart(3),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(0),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(1),
|
|
||||||
Token::SeqSep,
|
|
||||||
Token::I32(2),
|
|
||||||
Token::SeqEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_duration {
|
|
||||||
Duration::new(1, 2) => &[
|
|
||||||
Token::StructStart("Duration", 2),
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("secs"),
|
|
||||||
Token::U64(1),
|
|
||||||
|
|
||||||
Token::StructSep,
|
|
||||||
Token::Str("nanos"),
|
|
||||||
Token::U32(2),
|
|
||||||
Token::StructEnd,
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_net_ipv4addr {
|
|
||||||
"1.2.3.4".parse::<net::Ipv4Addr>().unwrap() => &[Token::Str("1.2.3.4")],
|
|
||||||
}
|
|
||||||
test_net_ipv6addr {
|
|
||||||
"::1".parse::<net::Ipv6Addr>().unwrap() => &[Token::Str("::1")],
|
|
||||||
}
|
|
||||||
test_net_socketaddr {
|
|
||||||
"1.2.3.4:1234".parse::<net::SocketAddr>().unwrap() => &[Token::Str("1.2.3.4:1234")],
|
|
||||||
"1.2.3.4:1234".parse::<net::SocketAddrV4>().unwrap() => &[Token::Str("1.2.3.4:1234")],
|
|
||||||
"[::1]:1234".parse::<net::SocketAddrV6>().unwrap() => &[Token::Str("[::1]:1234")],
|
|
||||||
}
|
|
||||||
test_path {
|
|
||||||
Path::new("/usr/local/lib") => &[
|
|
||||||
Token::Str("/usr/local/lib"),
|
|
||||||
],
|
|
||||||
}
|
|
||||||
test_path_buf {
|
|
||||||
PathBuf::from("/usr/local/lib") => &[
|
|
||||||
Token::Str("/usr/local/lib"),
|
|
||||||
],
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(feature = "unstable")]
|
|
||||||
#[test]
|
|
||||||
fn test_net_ipaddr() {
|
|
||||||
assert_ser_tokens(
|
|
||||||
"1.2.3.4".parse::<net::IpAddr>().unwrap(),
|
|
||||||
&[Token::Str("1.2.3.4")],
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_cannot_serialize_paths() {
|
|
||||||
let path = unsafe {
|
|
||||||
str::from_utf8_unchecked(b"Hello \xF0\x90\x80World")
|
|
||||||
};
|
|
||||||
assert_ser_tokens_error(
|
|
||||||
&Path::new(path),
|
|
||||||
&[],
|
|
||||||
Error::InvalidValue("Path contains invalid UTF-8 characters".to_owned()));
|
|
||||||
|
|
||||||
let mut path_buf = PathBuf::new();
|
|
||||||
path_buf.push(path);
|
|
||||||
|
|
||||||
assert_ser_tokens_error(
|
|
||||||
&path_buf,
|
|
||||||
&[],
|
|
||||||
Error::InvalidValue("Path contains invalid UTF-8 characters".to_owned()));
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user