mirror of
https://github.com/pezkuwichain/serde.git
synced 2026-04-29 10:37:55 +00:00
Compare commits
41 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| fccb9499bc | |||
| a139ab2572 | |||
| 1d910a484c | |||
| ee9166ec97 | |||
| b5a9eff32e | |||
| 9441a29663 | |||
| ab6588ef74 | |||
| 1d11f03449 | |||
| e11d01fe1d | |||
| a901f50850 | |||
| c399e9c368 | |||
| 25381be0c9 | |||
| ef2a7c753f | |||
| 99f165b45a | |||
| 2fb5560746 | |||
| bd653ab30c | |||
| b5d68aedaa | |||
| 624879c4c6 | |||
| bd9e9abf35 | |||
| 3e4a23cbd0 | |||
| 6326ceec3f | |||
| 8f4d37c7ec | |||
| 1b8290b318 | |||
| 48193fbccd | |||
| ac8ea72d88 | |||
| f583401284 | |||
| 2d88228b7d | |||
| 0c6a2bbf79 | |||
| a80d830f27 | |||
| 5f3fd9994e | |||
| d6de911855 | |||
| 04af32230e | |||
| 4cb8d079f8 | |||
| 6ab55a1e52 | |||
| acfd19cb46 | |||
| e3058105f0 | |||
| dc200a6450 | |||
| 2c0999a0b9 | |||
| dd460f82a1 | |||
| c3d637f397 | |||
| a803ec1c1f |
@@ -1 +0,0 @@
|
|||||||
test_suite/tests/expand/*.expanded.rs linguist-generated
|
|
||||||
+2
-2
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "serde"
|
name = "serde"
|
||||||
version = "1.0.157" # remember to update html_root_url and serde_derive dependency
|
version = "1.0.163" # remember to update html_root_url and serde_derive dependency
|
||||||
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>", "David Tolnay <dtolnay@gmail.com>"]
|
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>", "David Tolnay <dtolnay@gmail.com>"]
|
||||||
build = "build.rs"
|
build = "build.rs"
|
||||||
categories = ["encoding", "no-std"]
|
categories = ["encoding", "no-std"]
|
||||||
@@ -15,7 +15,7 @@ repository = "https://github.com/serde-rs/serde"
|
|||||||
rust-version = "1.19"
|
rust-version = "1.19"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
serde_derive = { version = "=1.0.157", optional = true, path = "../serde_derive" }
|
serde_derive = { version = "=1.0.163", optional = true, path = "../serde_derive" }
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
serde_derive = { version = "1.0", path = "../serde_derive" }
|
serde_derive = { version = "1.0", path = "../serde_derive" }
|
||||||
|
|||||||
@@ -994,7 +994,8 @@ seq_impl!(
|
|||||||
HashSet::clear,
|
HashSet::clear,
|
||||||
HashSet::with_capacity_and_hasher(size_hint::cautious(seq.size_hint()), S::default()),
|
HashSet::with_capacity_and_hasher(size_hint::cautious(seq.size_hint()), S::default()),
|
||||||
HashSet::reserve,
|
HashSet::reserve,
|
||||||
HashSet::insert);
|
HashSet::insert
|
||||||
|
);
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
#[cfg(any(feature = "std", feature = "alloc"))]
|
||||||
seq_impl!(
|
seq_impl!(
|
||||||
@@ -1409,16 +1410,14 @@ macro_rules! map_impl {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
#[cfg(any(feature = "std", feature = "alloc"))]
|
||||||
map_impl!(
|
map_impl!(BTreeMap<K: Ord, V>, map, BTreeMap::new());
|
||||||
BTreeMap<K: Ord, V>,
|
|
||||||
map,
|
|
||||||
BTreeMap::new());
|
|
||||||
|
|
||||||
#[cfg(feature = "std")]
|
#[cfg(feature = "std")]
|
||||||
map_impl!(
|
map_impl!(
|
||||||
HashMap<K: Eq + Hash, V, S: BuildHasher + Default>,
|
HashMap<K: Eq + Hash, V, S: BuildHasher + Default>,
|
||||||
map,
|
map,
|
||||||
HashMap::with_capacity_and_hasher(size_hint::cautious(map.size_hint()), S::default()));
|
HashMap::with_capacity_and_hasher(size_hint::cautious(map.size_hint()), S::default())
|
||||||
|
);
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
|||||||
+1
-1
@@ -93,7 +93,7 @@
|
|||||||
////////////////////////////////////////////////////////////////////////////////
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
// Serde types in rustdoc of other crates get linked to here.
|
// Serde types in rustdoc of other crates get linked to here.
|
||||||
#![doc(html_root_url = "https://docs.rs/serde/1.0.157")]
|
#![doc(html_root_url = "https://docs.rs/serde/1.0.163")]
|
||||||
// Support using Serde without the standard library!
|
// Support using Serde without the standard library!
|
||||||
#![cfg_attr(not(feature = "std"), no_std)]
|
#![cfg_attr(not(feature = "std"), no_std)]
|
||||||
// Unstable functionality only if the user asks for it. For tracking and
|
// Unstable functionality only if the user asks for it. For tracking and
|
||||||
|
|||||||
+48
-98
@@ -982,9 +982,16 @@ mod content {
|
|||||||
where
|
where
|
||||||
E: de::Error,
|
E: de::Error,
|
||||||
{
|
{
|
||||||
if field == self.tag {
|
self.visit_bytes(field.as_bytes())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn visit_bytes<E>(self, field: &[u8]) -> Result<Self::Value, E>
|
||||||
|
where
|
||||||
|
E: de::Error,
|
||||||
|
{
|
||||||
|
if field == self.tag.as_bytes() {
|
||||||
Ok(TagContentOtherField::Tag)
|
Ok(TagContentOtherField::Tag)
|
||||||
} else if field == self.content {
|
} else if field == self.content.as_bytes() {
|
||||||
Ok(TagContentOtherField::Content)
|
Ok(TagContentOtherField::Content)
|
||||||
} else {
|
} else {
|
||||||
Ok(TagContentOtherField::Other)
|
Ok(TagContentOtherField::Other)
|
||||||
@@ -2731,11 +2738,7 @@ where
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
visitor.visit_map(FlatInternallyTaggedAccess {
|
self.deserialize_map(visitor)
|
||||||
iter: self.0.iter_mut(),
|
|
||||||
pending: None,
|
|
||||||
_marker: PhantomData,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn deserialize_enum<V>(
|
fn deserialize_enum<V>(
|
||||||
@@ -2747,17 +2750,8 @@ where
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
for item in self.0.iter_mut() {
|
for entry in self.0.iter_mut() {
|
||||||
// items in the vector are nulled out when used. So we can only use
|
if let Some((key, value)) = flat_map_take_entry(entry, variants) {
|
||||||
// an item if it's still filled in and if the field is one we care
|
|
||||||
// about.
|
|
||||||
let use_item = match *item {
|
|
||||||
None => false,
|
|
||||||
Some((ref c, _)) => c.as_str().map_or(false, |x| variants.contains(&x)),
|
|
||||||
};
|
|
||||||
|
|
||||||
if use_item {
|
|
||||||
let (key, value) = item.take().unwrap();
|
|
||||||
return visitor.visit_enum(EnumDeserializer::new(key, Some(value)));
|
return visitor.visit_enum(EnumDeserializer::new(key, Some(value)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -2772,7 +2766,11 @@ where
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
visitor.visit_map(FlatMapAccess::new(self.0.iter()))
|
visitor.visit_map(FlatMapAccess {
|
||||||
|
iter: self.0.iter(),
|
||||||
|
pending_content: None,
|
||||||
|
_marker: PhantomData,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn deserialize_struct<V>(
|
fn deserialize_struct<V>(
|
||||||
@@ -2784,7 +2782,12 @@ where
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
visitor.visit_map(FlatStructAccess::new(self.0.iter_mut(), fields))
|
visitor.visit_map(FlatStructAccess {
|
||||||
|
iter: self.0.iter_mut(),
|
||||||
|
pending_content: None,
|
||||||
|
fields: fields,
|
||||||
|
_marker: PhantomData,
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn deserialize_newtype_struct<V>(self, _name: &str, visitor: V) -> Result<V::Value, Self::Error>
|
fn deserialize_newtype_struct<V>(self, _name: &str, visitor: V) -> Result<V::Value, Self::Error>
|
||||||
@@ -2838,25 +2841,12 @@ where
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
#[cfg(any(feature = "std", feature = "alloc"))]
|
||||||
pub struct FlatMapAccess<'a, 'de: 'a, E> {
|
struct FlatMapAccess<'a, 'de: 'a, E> {
|
||||||
iter: slice::Iter<'a, Option<(Content<'de>, Content<'de>)>>,
|
iter: slice::Iter<'a, Option<(Content<'de>, Content<'de>)>>,
|
||||||
pending_content: Option<&'a Content<'de>>,
|
pending_content: Option<&'a Content<'de>>,
|
||||||
_marker: PhantomData<E>,
|
_marker: PhantomData<E>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
|
||||||
impl<'a, 'de, E> FlatMapAccess<'a, 'de, E> {
|
|
||||||
fn new(
|
|
||||||
iter: slice::Iter<'a, Option<(Content<'de>, Content<'de>)>>,
|
|
||||||
) -> FlatMapAccess<'a, 'de, E> {
|
|
||||||
FlatMapAccess {
|
|
||||||
iter: iter,
|
|
||||||
pending_content: None,
|
|
||||||
_marker: PhantomData,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
#[cfg(any(feature = "std", feature = "alloc"))]
|
||||||
impl<'a, 'de, E> MapAccess<'de> for FlatMapAccess<'a, 'de, E>
|
impl<'a, 'de, E> MapAccess<'de> for FlatMapAccess<'a, 'de, E>
|
||||||
where
|
where
|
||||||
@@ -2871,6 +2861,10 @@ where
|
|||||||
for item in &mut self.iter {
|
for item in &mut self.iter {
|
||||||
// Items in the vector are nulled out when used by a struct.
|
// Items in the vector are nulled out when used by a struct.
|
||||||
if let Some((ref key, ref content)) = *item {
|
if let Some((ref key, ref content)) = *item {
|
||||||
|
// Do not take(), instead borrow this entry. The internally tagged
|
||||||
|
// enum does its own buffering so we can't tell whether this entry
|
||||||
|
// is going to be consumed. Borrowing here leaves the entry
|
||||||
|
// available for later flattened fields.
|
||||||
self.pending_content = Some(content);
|
self.pending_content = Some(content);
|
||||||
return seed.deserialize(ContentRefDeserializer::new(key)).map(Some);
|
return seed.deserialize(ContentRefDeserializer::new(key)).map(Some);
|
||||||
}
|
}
|
||||||
@@ -2890,28 +2884,13 @@ where
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
#[cfg(any(feature = "std", feature = "alloc"))]
|
||||||
pub struct FlatStructAccess<'a, 'de: 'a, E> {
|
struct FlatStructAccess<'a, 'de: 'a, E> {
|
||||||
iter: slice::IterMut<'a, Option<(Content<'de>, Content<'de>)>>,
|
iter: slice::IterMut<'a, Option<(Content<'de>, Content<'de>)>>,
|
||||||
pending_content: Option<Content<'de>>,
|
pending_content: Option<Content<'de>>,
|
||||||
fields: &'static [&'static str],
|
fields: &'static [&'static str],
|
||||||
_marker: PhantomData<E>,
|
_marker: PhantomData<E>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
|
||||||
impl<'a, 'de, E> FlatStructAccess<'a, 'de, E> {
|
|
||||||
fn new(
|
|
||||||
iter: slice::IterMut<'a, Option<(Content<'de>, Content<'de>)>>,
|
|
||||||
fields: &'static [&'static str],
|
|
||||||
) -> FlatStructAccess<'a, 'de, E> {
|
|
||||||
FlatStructAccess {
|
|
||||||
iter: iter,
|
|
||||||
pending_content: None,
|
|
||||||
fields: fields,
|
|
||||||
_marker: PhantomData,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
#[cfg(any(feature = "std", feature = "alloc"))]
|
||||||
impl<'a, 'de, E> MapAccess<'de> for FlatStructAccess<'a, 'de, E>
|
impl<'a, 'de, E> MapAccess<'de> for FlatStructAccess<'a, 'de, E>
|
||||||
where
|
where
|
||||||
@@ -2923,17 +2902,8 @@ where
|
|||||||
where
|
where
|
||||||
T: DeserializeSeed<'de>,
|
T: DeserializeSeed<'de>,
|
||||||
{
|
{
|
||||||
while let Some(item) = self.iter.next() {
|
for entry in self.iter.by_ref() {
|
||||||
// items in the vector are nulled out when used. So we can only use
|
if let Some((key, content)) = flat_map_take_entry(entry, self.fields) {
|
||||||
// an item if it's still filled in and if the field is one we care
|
|
||||||
// about. In case we do not know which fields we want, we take them all.
|
|
||||||
let use_item = match *item {
|
|
||||||
None => false,
|
|
||||||
Some((ref c, _)) => c.as_str().map_or(false, |key| self.fields.contains(&key)),
|
|
||||||
};
|
|
||||||
|
|
||||||
if use_item {
|
|
||||||
let (key, content) = item.take().unwrap();
|
|
||||||
self.pending_content = Some(content);
|
self.pending_content = Some(content);
|
||||||
return seed.deserialize(ContentDeserializer::new(key)).map(Some);
|
return seed.deserialize(ContentDeserializer::new(key)).map(Some);
|
||||||
}
|
}
|
||||||
@@ -2952,44 +2922,24 @@ where
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Claims one key-value pair from a FlatMapDeserializer's field buffer if the
|
||||||
|
/// field name matches any of the recognized ones.
|
||||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
#[cfg(any(feature = "std", feature = "alloc"))]
|
||||||
pub struct FlatInternallyTaggedAccess<'a, 'de: 'a, E> {
|
fn flat_map_take_entry<'de>(
|
||||||
iter: slice::IterMut<'a, Option<(Content<'de>, Content<'de>)>>,
|
entry: &mut Option<(Content<'de>, Content<'de>)>,
|
||||||
pending: Option<&'a Content<'de>>,
|
recognized: &[&str],
|
||||||
_marker: PhantomData<E>,
|
) -> Option<(Content<'de>, Content<'de>)> {
|
||||||
}
|
// Entries in the FlatMapDeserializer buffer are nulled out as they get
|
||||||
|
// claimed for deserialization. We only use an entry if it is still present
|
||||||
|
// and if the field is one recognized by the current data structure.
|
||||||
|
let is_recognized = match entry {
|
||||||
|
None => false,
|
||||||
|
Some((k, _v)) => k.as_str().map_or(false, |name| recognized.contains(&name)),
|
||||||
|
};
|
||||||
|
|
||||||
#[cfg(any(feature = "std", feature = "alloc"))]
|
if is_recognized {
|
||||||
impl<'a, 'de, E> MapAccess<'de> for FlatInternallyTaggedAccess<'a, 'de, E>
|
entry.take()
|
||||||
where
|
} else {
|
||||||
E: Error,
|
None
|
||||||
{
|
|
||||||
type Error = E;
|
|
||||||
|
|
||||||
fn next_key_seed<T>(&mut self, seed: T) -> Result<Option<T::Value>, Self::Error>
|
|
||||||
where
|
|
||||||
T: DeserializeSeed<'de>,
|
|
||||||
{
|
|
||||||
for item in &mut self.iter {
|
|
||||||
if let Some((ref key, ref content)) = *item {
|
|
||||||
// Do not take(), instead borrow this entry. The internally tagged
|
|
||||||
// enum does its own buffering so we can't tell whether this entry
|
|
||||||
// is going to be consumed. Borrowing here leaves the entry
|
|
||||||
// available for later flattened fields.
|
|
||||||
self.pending = Some(content);
|
|
||||||
return seed.deserialize(ContentRefDeserializer::new(key)).map(Some);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Ok(None)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn next_value_seed<T>(&mut self, seed: T) -> Result<T::Value, Self::Error>
|
|
||||||
where
|
|
||||||
T: DeserializeSeed<'de>,
|
|
||||||
{
|
|
||||||
match self.pending.take() {
|
|
||||||
Some(value) => seed.deserialize(ContentRefDeserializer::new(value)),
|
|
||||||
None => panic!("value is missing"),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "serde_derive"
|
name = "serde_derive"
|
||||||
version = "1.0.157" # remember to update html_root_url
|
version = "1.0.163" # remember to update html_root_url
|
||||||
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>", "David Tolnay <dtolnay@gmail.com>"]
|
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>", "David Tolnay <dtolnay@gmail.com>"]
|
||||||
categories = ["no-std"]
|
categories = ["no-std"]
|
||||||
description = "Macros 1.1 implementation of #[derive(Serialize, Deserialize)]"
|
description = "Macros 1.1 implementation of #[derive(Serialize, Deserialize)]"
|
||||||
@@ -24,7 +24,7 @@ proc-macro = true
|
|||||||
[dependencies]
|
[dependencies]
|
||||||
proc-macro2 = "1.0"
|
proc-macro2 = "1.0"
|
||||||
quote = "1.0"
|
quote = "1.0"
|
||||||
syn = "2.0"
|
syn = "2.0.3"
|
||||||
|
|
||||||
[dev-dependencies]
|
[dev-dependencies]
|
||||||
serde = { version = "1.0", path = "../serde" }
|
serde = { version = "1.0", path = "../serde" }
|
||||||
|
|||||||
@@ -1,38 +0,0 @@
|
|||||||
use std::env;
|
|
||||||
use std::process::Command;
|
|
||||||
use std::str;
|
|
||||||
|
|
||||||
// The rustc-cfg strings below are *not* public API. Please let us know by
|
|
||||||
// opening a GitHub issue if your build environment requires some way to enable
|
|
||||||
// these cfgs other than by executing our build script.
|
|
||||||
fn main() {
|
|
||||||
println!("cargo:rerun-if-changed=build.rs");
|
|
||||||
|
|
||||||
let minor = match rustc_minor_version() {
|
|
||||||
Some(minor) => minor,
|
|
||||||
None => return,
|
|
||||||
};
|
|
||||||
|
|
||||||
// Underscore const names stabilized in Rust 1.37:
|
|
||||||
// https://blog.rust-lang.org/2019/08/15/Rust-1.37.0.html#using-unnamed-const-items-for-macros
|
|
||||||
if minor < 37 {
|
|
||||||
println!("cargo:rustc-cfg=no_underscore_consts");
|
|
||||||
}
|
|
||||||
|
|
||||||
// The ptr::addr_of! macro stabilized in Rust 1.51:
|
|
||||||
// https://blog.rust-lang.org/2021/03/25/Rust-1.51.0.html#stabilized-apis
|
|
||||||
if minor < 51 {
|
|
||||||
println!("cargo:rustc-cfg=no_ptr_addr_of");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn rustc_minor_version() -> Option<u32> {
|
|
||||||
let rustc = env::var_os("RUSTC")?;
|
|
||||||
let output = Command::new(rustc).arg("--version").output().ok()?;
|
|
||||||
let version = str::from_utf8(&output.stdout).ok()?;
|
|
||||||
let mut pieces = version.split('.');
|
|
||||||
if pieces.next() != Some("rustc 1") {
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
pieces.next()?.parse().ok()
|
|
||||||
}
|
|
||||||
+42
-31
@@ -69,8 +69,6 @@ pub fn expand_derive_deserialize(
|
|||||||
|
|
||||||
Ok(dummy::wrap_in_const(
|
Ok(dummy::wrap_in_const(
|
||||||
cont.attrs.custom_serde_path(),
|
cont.attrs.custom_serde_path(),
|
||||||
"DESERIALIZE",
|
|
||||||
ident,
|
|
||||||
impl_block,
|
impl_block,
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
@@ -419,6 +417,7 @@ fn deserialize_unit_struct(params: &Parameters, cattrs: &attr::Container) -> Fra
|
|||||||
let expecting = cattrs.expecting().unwrap_or(&expecting);
|
let expecting = cattrs.expecting().unwrap_or(&expecting);
|
||||||
|
|
||||||
quote_block! {
|
quote_block! {
|
||||||
|
#[doc(hidden)]
|
||||||
struct __Visitor;
|
struct __Visitor;
|
||||||
|
|
||||||
impl<'de> _serde::de::Visitor<'de> for __Visitor {
|
impl<'de> _serde::de::Visitor<'de> for __Visitor {
|
||||||
@@ -515,6 +514,7 @@ fn deserialize_tuple(
|
|||||||
};
|
};
|
||||||
|
|
||||||
quote_block! {
|
quote_block! {
|
||||||
|
#[doc(hidden)]
|
||||||
struct __Visitor #de_impl_generics #where_clause {
|
struct __Visitor #de_impl_generics #where_clause {
|
||||||
marker: _serde::__private::PhantomData<#this_type #ty_generics>,
|
marker: _serde::__private::PhantomData<#this_type #ty_generics>,
|
||||||
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
||||||
@@ -605,6 +605,7 @@ fn deserialize_tuple_in_place(
|
|||||||
let place_life = place_lifetime();
|
let place_life = place_lifetime();
|
||||||
|
|
||||||
quote_block! {
|
quote_block! {
|
||||||
|
#[doc(hidden)]
|
||||||
struct __Visitor #in_place_impl_generics #where_clause {
|
struct __Visitor #in_place_impl_generics #where_clause {
|
||||||
place: &#place_life mut #this_type #ty_generics,
|
place: &#place_life mut #this_type #ty_generics,
|
||||||
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
||||||
@@ -1020,6 +1021,7 @@ fn deserialize_struct(
|
|||||||
quote_block! {
|
quote_block! {
|
||||||
#field_visitor
|
#field_visitor
|
||||||
|
|
||||||
|
#[doc(hidden)]
|
||||||
struct __Visitor #de_impl_generics #where_clause {
|
struct __Visitor #de_impl_generics #where_clause {
|
||||||
marker: _serde::__private::PhantomData<#this_type #ty_generics>,
|
marker: _serde::__private::PhantomData<#this_type #ty_generics>,
|
||||||
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
||||||
@@ -1132,6 +1134,7 @@ fn deserialize_struct_in_place(
|
|||||||
Some(quote_block! {
|
Some(quote_block! {
|
||||||
#field_visitor
|
#field_visitor
|
||||||
|
|
||||||
|
#[doc(hidden)]
|
||||||
struct __Visitor #in_place_impl_generics #where_clause {
|
struct __Visitor #in_place_impl_generics #where_clause {
|
||||||
place: &#place_life mut #this_type #ty_generics,
|
place: &#place_life mut #this_type #ty_generics,
|
||||||
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
||||||
@@ -1203,6 +1206,7 @@ fn prepare_enum_variant_enum(
|
|||||||
let variants_stmt = {
|
let variants_stmt = {
|
||||||
let variant_names = variant_names_idents.iter().map(|(name, _, _)| name);
|
let variant_names = variant_names_idents.iter().map(|(name, _, _)| name);
|
||||||
quote! {
|
quote! {
|
||||||
|
#[doc(hidden)]
|
||||||
const VARIANTS: &'static [&'static str] = &[ #(#variant_names),* ];
|
const VARIANTS: &'static [&'static str] = &[ #(#variant_names),* ];
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@@ -1275,6 +1279,7 @@ fn deserialize_externally_tagged_enum(
|
|||||||
quote_block! {
|
quote_block! {
|
||||||
#variant_visitor
|
#variant_visitor
|
||||||
|
|
||||||
|
#[doc(hidden)]
|
||||||
struct __Visitor #de_impl_generics #where_clause {
|
struct __Visitor #de_impl_generics #where_clause {
|
||||||
marker: _serde::__private::PhantomData<#this_type #ty_generics>,
|
marker: _serde::__private::PhantomData<#this_type #ty_generics>,
|
||||||
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
||||||
@@ -1522,6 +1527,7 @@ fn deserialize_adjacently_tagged_enum(
|
|||||||
|
|
||||||
#variants_stmt
|
#variants_stmt
|
||||||
|
|
||||||
|
#[doc(hidden)]
|
||||||
struct __Seed #de_impl_generics #where_clause {
|
struct __Seed #de_impl_generics #where_clause {
|
||||||
field: __Field,
|
field: __Field,
|
||||||
marker: _serde::__private::PhantomData<#this_type #ty_generics>,
|
marker: _serde::__private::PhantomData<#this_type #ty_generics>,
|
||||||
@@ -1541,6 +1547,7 @@ fn deserialize_adjacently_tagged_enum(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[doc(hidden)]
|
||||||
struct __Visitor #de_impl_generics #where_clause {
|
struct __Visitor #de_impl_generics #where_clause {
|
||||||
marker: _serde::__private::PhantomData<#this_type #ty_generics>,
|
marker: _serde::__private::PhantomData<#this_type #ty_generics>,
|
||||||
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
||||||
@@ -1643,6 +1650,7 @@ fn deserialize_adjacently_tagged_enum(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[doc(hidden)]
|
||||||
const FIELDS: &'static [&'static str] = &[#tag, #content];
|
const FIELDS: &'static [&'static str] = &[#tag, #content];
|
||||||
_serde::Deserializer::deserialize_struct(
|
_serde::Deserializer::deserialize_struct(
|
||||||
__deserializer,
|
__deserializer,
|
||||||
@@ -1954,11 +1962,13 @@ fn deserialize_generated_identifier(
|
|||||||
|
|
||||||
quote_block! {
|
quote_block! {
|
||||||
#[allow(non_camel_case_types)]
|
#[allow(non_camel_case_types)]
|
||||||
|
#[doc(hidden)]
|
||||||
enum __Field #lifetime {
|
enum __Field #lifetime {
|
||||||
#(#field_idents,)*
|
#(#field_idents,)*
|
||||||
#ignore_variant
|
#ignore_variant
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[doc(hidden)]
|
||||||
struct __FieldVisitor;
|
struct __FieldVisitor;
|
||||||
|
|
||||||
impl<'de> _serde::de::Visitor<'de> for __FieldVisitor {
|
impl<'de> _serde::de::Visitor<'de> for __FieldVisitor {
|
||||||
@@ -2046,11 +2056,13 @@ fn deserialize_custom_identifier(
|
|||||||
None
|
None
|
||||||
} else if is_variant {
|
} else if is_variant {
|
||||||
let variants = quote! {
|
let variants = quote! {
|
||||||
|
#[doc(hidden)]
|
||||||
const VARIANTS: &'static [&'static str] = &[ #(#names),* ];
|
const VARIANTS: &'static [&'static str] = &[ #(#names),* ];
|
||||||
};
|
};
|
||||||
Some(variants)
|
Some(variants)
|
||||||
} else {
|
} else {
|
||||||
let fields = quote! {
|
let fields = quote! {
|
||||||
|
#[doc(hidden)]
|
||||||
const FIELDS: &'static [&'static str] = &[ #(#names),* ];
|
const FIELDS: &'static [&'static str] = &[ #(#names),* ];
|
||||||
};
|
};
|
||||||
Some(fields)
|
Some(fields)
|
||||||
@@ -2072,6 +2084,7 @@ fn deserialize_custom_identifier(
|
|||||||
quote_block! {
|
quote_block! {
|
||||||
#names_const
|
#names_const
|
||||||
|
|
||||||
|
#[doc(hidden)]
|
||||||
struct __FieldVisitor #de_impl_generics #where_clause {
|
struct __FieldVisitor #de_impl_generics #where_clause {
|
||||||
marker: _serde::__private::PhantomData<#this_type #ty_generics>,
|
marker: _serde::__private::PhantomData<#this_type #ty_generics>,
|
||||||
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
lifetime: _serde::__private::PhantomData<&#delife ()>,
|
||||||
@@ -2406,6 +2419,7 @@ fn deserialize_struct_as_struct_visitor(
|
|||||||
.flat_map(|(_, _, aliases)| aliases);
|
.flat_map(|(_, _, aliases)| aliases);
|
||||||
|
|
||||||
quote_block! {
|
quote_block! {
|
||||||
|
#[doc(hidden)]
|
||||||
const FIELDS: &'static [&'static str] = &[ #(#field_names),* ];
|
const FIELDS: &'static [&'static str] = &[ #(#field_names),* ];
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@@ -2684,6 +2698,7 @@ fn deserialize_struct_as_struct_in_place_visitor(
|
|||||||
let fields_stmt = {
|
let fields_stmt = {
|
||||||
let field_names = field_names_idents.iter().map(|(name, _, _)| name);
|
let field_names = field_names_idents.iter().map(|(name, _, _)| name);
|
||||||
quote_block! {
|
quote_block! {
|
||||||
|
#[doc(hidden)]
|
||||||
const FIELDS: &'static [&'static str] = &[ #(#field_names),* ];
|
const FIELDS: &'static [&'static str] = &[ #(#field_names),* ];
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@@ -2864,6 +2879,7 @@ fn wrap_deserialize_with(
|
|||||||
let delife = params.borrowed.de_lifetime();
|
let delife = params.borrowed.de_lifetime();
|
||||||
|
|
||||||
let wrapper = quote! {
|
let wrapper = quote! {
|
||||||
|
#[doc(hidden)]
|
||||||
struct __DeserializeWith #de_impl_generics #where_clause {
|
struct __DeserializeWith #de_impl_generics #where_clause {
|
||||||
value: #value_ty,
|
value: #value_ty,
|
||||||
phantom: _serde::__private::PhantomData<#this_type #ty_generics>,
|
phantom: _serde::__private::PhantomData<#this_type #ty_generics>,
|
||||||
@@ -3068,23 +3084,31 @@ struct DeTypeGenerics<'a>(&'a Parameters);
|
|||||||
#[cfg(feature = "deserialize_in_place")]
|
#[cfg(feature = "deserialize_in_place")]
|
||||||
struct InPlaceTypeGenerics<'a>(&'a Parameters);
|
struct InPlaceTypeGenerics<'a>(&'a Parameters);
|
||||||
|
|
||||||
|
fn de_type_generics_to_tokens(
|
||||||
|
mut generics: syn::Generics,
|
||||||
|
borrowed: &BorrowedLifetimes,
|
||||||
|
tokens: &mut TokenStream,
|
||||||
|
) {
|
||||||
|
if borrowed.de_lifetime_param().is_some() {
|
||||||
|
let def = syn::LifetimeParam {
|
||||||
|
attrs: Vec::new(),
|
||||||
|
lifetime: syn::Lifetime::new("'de", Span::call_site()),
|
||||||
|
colon_token: None,
|
||||||
|
bounds: Punctuated::new(),
|
||||||
|
};
|
||||||
|
// Prepend 'de lifetime to list of generics
|
||||||
|
generics.params = Some(syn::GenericParam::Lifetime(def))
|
||||||
|
.into_iter()
|
||||||
|
.chain(generics.params)
|
||||||
|
.collect();
|
||||||
|
}
|
||||||
|
let (_, ty_generics, _) = generics.split_for_impl();
|
||||||
|
ty_generics.to_tokens(tokens);
|
||||||
|
}
|
||||||
|
|
||||||
impl<'a> ToTokens for DeTypeGenerics<'a> {
|
impl<'a> ToTokens for DeTypeGenerics<'a> {
|
||||||
fn to_tokens(&self, tokens: &mut TokenStream) {
|
fn to_tokens(&self, tokens: &mut TokenStream) {
|
||||||
let mut generics = self.0.generics.clone();
|
de_type_generics_to_tokens(self.0.generics.clone(), &self.0.borrowed, tokens);
|
||||||
if self.0.borrowed.de_lifetime_param().is_some() {
|
|
||||||
let def = syn::LifetimeParam {
|
|
||||||
attrs: Vec::new(),
|
|
||||||
lifetime: syn::Lifetime::new("'de", Span::call_site()),
|
|
||||||
colon_token: None,
|
|
||||||
bounds: Punctuated::new(),
|
|
||||||
};
|
|
||||||
generics.params = Some(syn::GenericParam::Lifetime(def))
|
|
||||||
.into_iter()
|
|
||||||
.chain(generics.params)
|
|
||||||
.collect();
|
|
||||||
}
|
|
||||||
let (_, ty_generics, _) = generics.split_for_impl();
|
|
||||||
ty_generics.to_tokens(tokens);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -3097,20 +3121,7 @@ impl<'a> ToTokens for InPlaceTypeGenerics<'a> {
|
|||||||
.chain(generics.params)
|
.chain(generics.params)
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
if self.0.borrowed.de_lifetime_param().is_some() {
|
de_type_generics_to_tokens(generics, &self.0.borrowed, tokens);
|
||||||
let def = syn::LifetimeParam {
|
|
||||||
attrs: Vec::new(),
|
|
||||||
lifetime: syn::Lifetime::new("'de", Span::call_site()),
|
|
||||||
colon_token: None,
|
|
||||||
bounds: Punctuated::new(),
|
|
||||||
};
|
|
||||||
generics.params = Some(syn::GenericParam::Lifetime(def))
|
|
||||||
.into_iter()
|
|
||||||
.chain(generics.params)
|
|
||||||
.collect();
|
|
||||||
}
|
|
||||||
let (_, ty_generics, _) = generics.split_for_impl();
|
|
||||||
ty_generics.to_tokens(tokens);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,23 +1,11 @@
|
|||||||
use proc_macro2::{Ident, TokenStream};
|
use proc_macro2::TokenStream;
|
||||||
use quote::format_ident;
|
|
||||||
|
|
||||||
use syn;
|
use syn;
|
||||||
use try;
|
use try;
|
||||||
|
|
||||||
pub fn wrap_in_const(
|
pub fn wrap_in_const(serde_path: Option<&syn::Path>, code: TokenStream) -> TokenStream {
|
||||||
serde_path: Option<&syn::Path>,
|
|
||||||
trait_: &str,
|
|
||||||
ty: &Ident,
|
|
||||||
code: TokenStream,
|
|
||||||
) -> TokenStream {
|
|
||||||
let try_replacement = try::replacement();
|
let try_replacement = try::replacement();
|
||||||
|
|
||||||
let dummy_const = if cfg!(no_underscore_consts) {
|
|
||||||
format_ident!("_IMPL_{}_FOR_{}", trait_, unraw(ty))
|
|
||||||
} else {
|
|
||||||
format_ident!("_")
|
|
||||||
};
|
|
||||||
|
|
||||||
let use_serde = match serde_path {
|
let use_serde = match serde_path {
|
||||||
Some(path) => quote! {
|
Some(path) => quote! {
|
||||||
use #path as _serde;
|
use #path as _serde;
|
||||||
@@ -31,14 +19,10 @@ pub fn wrap_in_const(
|
|||||||
quote! {
|
quote! {
|
||||||
#[doc(hidden)]
|
#[doc(hidden)]
|
||||||
#[allow(non_upper_case_globals, unused_attributes, unused_qualifications)]
|
#[allow(non_upper_case_globals, unused_attributes, unused_qualifications)]
|
||||||
const #dummy_const: () = {
|
const _: () = {
|
||||||
#use_serde
|
#use_serde
|
||||||
#try_replacement
|
#try_replacement
|
||||||
#code
|
#code
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn unraw(ident: &Ident) -> String {
|
|
||||||
ident.to_string().trim_start_matches("r#").to_owned()
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -307,6 +307,12 @@ impl Container {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let syn::Meta::List(meta) = &attr.meta {
|
||||||
|
if meta.tokens.is_empty() {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if let Err(err) = attr.parse_nested_meta(|meta| {
|
if let Err(err) = attr.parse_nested_meta(|meta| {
|
||||||
if meta.path == RENAME {
|
if meta.path == RENAME {
|
||||||
// #[serde(rename = "foo")]
|
// #[serde(rename = "foo")]
|
||||||
@@ -762,6 +768,12 @@ impl Variant {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let syn::Meta::List(meta) = &attr.meta {
|
||||||
|
if meta.tokens.is_empty() {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if let Err(err) = attr.parse_nested_meta(|meta| {
|
if let Err(err) = attr.parse_nested_meta(|meta| {
|
||||||
if meta.path == RENAME {
|
if meta.path == RENAME {
|
||||||
// #[serde(rename = "foo")]
|
// #[serde(rename = "foo")]
|
||||||
@@ -1033,6 +1045,12 @@ impl Field {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if let syn::Meta::List(meta) = &attr.meta {
|
||||||
|
if meta.tokens.is_empty() {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if let Err(err) = attr.parse_nested_meta(|meta| {
|
if let Err(err) = attr.parse_nested_meta(|meta| {
|
||||||
if meta.path == RENAME {
|
if meta.path == RENAME {
|
||||||
// #[serde(rename = "foo")]
|
// #[serde(rename = "foo")]
|
||||||
@@ -1381,21 +1399,26 @@ fn get_lit_str2(
|
|||||||
meta_item_name: Symbol,
|
meta_item_name: Symbol,
|
||||||
meta: &ParseNestedMeta,
|
meta: &ParseNestedMeta,
|
||||||
) -> syn::Result<Option<syn::LitStr>> {
|
) -> syn::Result<Option<syn::LitStr>> {
|
||||||
match meta.value()?.parse()? {
|
let expr: syn::Expr = meta.value()?.parse()?;
|
||||||
syn::Expr::Lit(syn::ExprLit {
|
let mut value = &expr;
|
||||||
lit: syn::Lit::Str(lit),
|
while let syn::Expr::Group(e) = value {
|
||||||
..
|
value = &e.expr;
|
||||||
}) => Ok(Some(lit)),
|
}
|
||||||
expr => {
|
if let syn::Expr::Lit(syn::ExprLit {
|
||||||
cx.error_spanned_by(
|
lit: syn::Lit::Str(lit),
|
||||||
expr,
|
..
|
||||||
format!(
|
}) = value
|
||||||
"expected serde {} attribute to be a string: `{} = \"...\"`",
|
{
|
||||||
attr_name, meta_item_name
|
Ok(Some(lit.clone()))
|
||||||
),
|
} else {
|
||||||
);
|
cx.error_spanned_by(
|
||||||
Ok(None)
|
expr,
|
||||||
}
|
format!(
|
||||||
|
"expected serde {} attribute to be a string: `{} = \"...\"`",
|
||||||
|
attr_name, meta_item_name
|
||||||
|
),
|
||||||
|
);
|
||||||
|
Ok(None)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -3,8 +3,8 @@ use internals::attr::{Identifier, TagType};
|
|||||||
use internals::{ungroup, Ctxt, Derive};
|
use internals::{ungroup, Ctxt, Derive};
|
||||||
use syn::{Member, Type};
|
use syn::{Member, Type};
|
||||||
|
|
||||||
/// Cross-cutting checks that require looking at more than a single attrs
|
// Cross-cutting checks that require looking at more than a single attrs object.
|
||||||
/// object. Simpler checks should happen when parsing and building the attrs.
|
// Simpler checks should happen when parsing and building the attrs.
|
||||||
pub fn check(cx: &Ctxt, cont: &mut Container, derive: Derive) {
|
pub fn check(cx: &Ctxt, cont: &mut Container, derive: Derive) {
|
||||||
check_remote_generic(cx, cont);
|
check_remote_generic(cx, cont);
|
||||||
check_getter(cx, cont);
|
check_getter(cx, cont);
|
||||||
@@ -17,18 +17,18 @@ pub fn check(cx: &Ctxt, cont: &mut Container, derive: Derive) {
|
|||||||
check_from_and_try_from(cx, cont);
|
check_from_and_try_from(cx, cont);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Remote derive definition type must have either all of the generics of the
|
// Remote derive definition type must have either all of the generics of the
|
||||||
/// remote type:
|
// remote type:
|
||||||
///
|
//
|
||||||
/// #[serde(remote = "Generic")]
|
// #[serde(remote = "Generic")]
|
||||||
/// struct Generic<T> {…}
|
// struct Generic<T> {…}
|
||||||
///
|
//
|
||||||
/// or none of them, i.e. defining impls for one concrete instantiation of the
|
// or none of them, i.e. defining impls for one concrete instantiation of the
|
||||||
/// remote type only:
|
// remote type only:
|
||||||
///
|
//
|
||||||
/// #[serde(remote = "Generic<T>")]
|
// #[serde(remote = "Generic<T>")]
|
||||||
/// struct ConcreteDef {…}
|
// struct ConcreteDef {…}
|
||||||
///
|
//
|
||||||
fn check_remote_generic(cx: &Ctxt, cont: &Container) {
|
fn check_remote_generic(cx: &Ctxt, cont: &Container) {
|
||||||
if let Some(remote) = cont.attrs.remote() {
|
if let Some(remote) = cont.attrs.remote() {
|
||||||
let local_has_generic = !cont.generics.params.is_empty();
|
let local_has_generic = !cont.generics.params.is_empty();
|
||||||
@@ -39,8 +39,8 @@ fn check_remote_generic(cx: &Ctxt, cont: &Container) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Getters are only allowed inside structs (not enums) with the `remote`
|
// Getters are only allowed inside structs (not enums) with the `remote`
|
||||||
/// attribute.
|
// attribute.
|
||||||
fn check_getter(cx: &Ctxt, cont: &Container) {
|
fn check_getter(cx: &Ctxt, cont: &Container) {
|
||||||
match cont.data {
|
match cont.data {
|
||||||
Data::Enum(_) => {
|
Data::Enum(_) => {
|
||||||
@@ -62,7 +62,7 @@ fn check_getter(cx: &Ctxt, cont: &Container) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Flattening has some restrictions we can test.
|
// Flattening has some restrictions we can test.
|
||||||
fn check_flatten(cx: &Ctxt, cont: &Container) {
|
fn check_flatten(cx: &Ctxt, cont: &Container) {
|
||||||
match &cont.data {
|
match &cont.data {
|
||||||
Data::Enum(variants) => {
|
Data::Enum(variants) => {
|
||||||
@@ -101,12 +101,12 @@ fn check_flatten_field(cx: &Ctxt, style: Style, field: &Field) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The `other` attribute must be used at most once and it must be the last
|
// The `other` attribute must be used at most once and it must be the last
|
||||||
/// variant of an enum.
|
// variant of an enum.
|
||||||
///
|
//
|
||||||
/// Inside a `variant_identifier` all variants must be unit variants. Inside a
|
// Inside a `variant_identifier` all variants must be unit variants. Inside a
|
||||||
/// `field_identifier` all but possibly one variant must be unit variants. The
|
// `field_identifier` all but possibly one variant must be unit variants. The
|
||||||
/// last variant may be a newtype variant which is an implicit "other" case.
|
// last variant may be a newtype variant which is an implicit "other" case.
|
||||||
fn check_identifier(cx: &Ctxt, cont: &Container) {
|
fn check_identifier(cx: &Ctxt, cont: &Container) {
|
||||||
let variants = match &cont.data {
|
let variants = match &cont.data {
|
||||||
Data::Enum(variants) => variants,
|
Data::Enum(variants) => variants,
|
||||||
@@ -189,8 +189,8 @@ fn check_identifier(cx: &Ctxt, cont: &Container) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Skip-(de)serializing attributes are not allowed on variants marked
|
// Skip-(de)serializing attributes are not allowed on variants marked
|
||||||
/// (de)serialize_with.
|
// (de)serialize_with.
|
||||||
fn check_variant_skip_attrs(cx: &Ctxt, cont: &Container) {
|
fn check_variant_skip_attrs(cx: &Ctxt, cont: &Container) {
|
||||||
let variants = match &cont.data {
|
let variants = match &cont.data {
|
||||||
Data::Enum(variants) => variants,
|
Data::Enum(variants) => variants,
|
||||||
@@ -264,10 +264,9 @@ fn check_variant_skip_attrs(cx: &Ctxt, cont: &Container) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The tag of an internally-tagged struct variant must not be
|
// The tag of an internally-tagged struct variant must not be the same as either
|
||||||
/// the same as either one of its fields, as this would result in
|
// one of its fields, as this would result in duplicate keys in the serialized
|
||||||
/// duplicate keys in the serialized output and/or ambiguity in
|
// output and/or ambiguity in the to-be-deserialized input.
|
||||||
/// the to-be-deserialized input.
|
|
||||||
fn check_internal_tag_field_name_conflict(cx: &Ctxt, cont: &Container) {
|
fn check_internal_tag_field_name_conflict(cx: &Ctxt, cont: &Container) {
|
||||||
let variants = match &cont.data {
|
let variants = match &cont.data {
|
||||||
Data::Enum(variants) => variants,
|
Data::Enum(variants) => variants,
|
||||||
@@ -313,8 +312,8 @@ fn check_internal_tag_field_name_conflict(cx: &Ctxt, cont: &Container) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// In the case of adjacently-tagged enums, the type and the
|
// In the case of adjacently-tagged enums, the type and the contents tag must
|
||||||
/// contents tag must differ, for the same reason.
|
// differ, for the same reason.
|
||||||
fn check_adjacent_tag_conflict(cx: &Ctxt, cont: &Container) {
|
fn check_adjacent_tag_conflict(cx: &Ctxt, cont: &Container) {
|
||||||
let (type_tag, content_tag) = match cont.attrs.tag() {
|
let (type_tag, content_tag) = match cont.attrs.tag() {
|
||||||
TagType::Adjacent { tag, content } => (tag, content),
|
TagType::Adjacent { tag, content } => (tag, content),
|
||||||
@@ -332,7 +331,7 @@ fn check_adjacent_tag_conflict(cx: &Ctxt, cont: &Container) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Enums and unit structs cannot be transparent.
|
// Enums and unit structs cannot be transparent.
|
||||||
fn check_transparent(cx: &Ctxt, cont: &mut Container, derive: Derive) {
|
fn check_transparent(cx: &Ctxt, cont: &mut Container, derive: Derive) {
|
||||||
if !cont.attrs.transparent() {
|
if !cont.attrs.transparent() {
|
||||||
return;
|
return;
|
||||||
|
|||||||
@@ -13,7 +13,7 @@
|
|||||||
//!
|
//!
|
||||||
//! [https://serde.rs/derive.html]: https://serde.rs/derive.html
|
//! [https://serde.rs/derive.html]: https://serde.rs/derive.html
|
||||||
|
|
||||||
#![doc(html_root_url = "https://docs.rs/serde_derive/1.0.157")]
|
#![doc(html_root_url = "https://docs.rs/serde_derive/1.0.163")]
|
||||||
#![allow(unknown_lints, bare_trait_objects)]
|
#![allow(unknown_lints, bare_trait_objects)]
|
||||||
// Ignored clippy lints
|
// Ignored clippy lints
|
||||||
#![allow(
|
#![allow(
|
||||||
|
|||||||
@@ -97,29 +97,14 @@ fn pretend_fields_used_struct_packed(cont: &Container, fields: &[Field]) -> Toke
|
|||||||
|
|
||||||
let members = fields.iter().map(|field| &field.member).collect::<Vec<_>>();
|
let members = fields.iter().map(|field| &field.member).collect::<Vec<_>>();
|
||||||
|
|
||||||
#[cfg(not(no_ptr_addr_of))]
|
quote! {
|
||||||
{
|
match _serde::__private::None::<&#type_ident #ty_generics> {
|
||||||
quote! {
|
_serde::__private::Some(__v @ #type_ident { #(#members: _),* }) => {
|
||||||
match _serde::__private::None::<&#type_ident #ty_generics> {
|
#(
|
||||||
_serde::__private::Some(__v @ #type_ident { #(#members: _),* }) => {
|
let _ = _serde::__private::ptr::addr_of!(__v.#members);
|
||||||
#(
|
)*
|
||||||
let _ = _serde::__private::ptr::addr_of!(__v.#members);
|
|
||||||
)*
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(no_ptr_addr_of)]
|
|
||||||
{
|
|
||||||
let placeholders = (0usize..).map(|i| format_ident!("__v{}", i));
|
|
||||||
|
|
||||||
quote! {
|
|
||||||
match _serde::__private::None::<#type_ident #ty_generics> {
|
|
||||||
_serde::__private::Some(#type_ident { #(#members: #placeholders),* }) => {}
|
|
||||||
_ => {}
|
|
||||||
}
|
}
|
||||||
|
_ => {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -59,8 +59,6 @@ pub fn expand_derive_serialize(
|
|||||||
|
|
||||||
Ok(dummy::wrap_in_const(
|
Ok(dummy::wrap_in_const(
|
||||||
cont.attrs.custom_serde_path(),
|
cont.attrs.custom_serde_path(),
|
||||||
"SERIALIZE",
|
|
||||||
ident,
|
|
||||||
impl_block,
|
impl_block,
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
@@ -719,6 +717,7 @@ fn serialize_adjacently_tagged_variant(
|
|||||||
let (wrapper_impl_generics, wrapper_ty_generics, _) = wrapper_generics.split_for_impl();
|
let (wrapper_impl_generics, wrapper_ty_generics, _) = wrapper_generics.split_for_impl();
|
||||||
|
|
||||||
quote_block! {
|
quote_block! {
|
||||||
|
#[doc(hidden)]
|
||||||
struct __AdjacentlyTagged #wrapper_generics #where_clause {
|
struct __AdjacentlyTagged #wrapper_generics #where_clause {
|
||||||
data: (#(&'__a #fields_ty,)*),
|
data: (#(&'__a #fields_ty,)*),
|
||||||
phantom: _serde::__private::PhantomData<#this_type #ty_generics>,
|
phantom: _serde::__private::PhantomData<#this_type #ty_generics>,
|
||||||
@@ -982,6 +981,7 @@ fn serialize_struct_variant_with_flatten(
|
|||||||
let (wrapper_impl_generics, wrapper_ty_generics, _) = wrapper_generics.split_for_impl();
|
let (wrapper_impl_generics, wrapper_ty_generics, _) = wrapper_generics.split_for_impl();
|
||||||
|
|
||||||
quote_block! {
|
quote_block! {
|
||||||
|
#[doc(hidden)]
|
||||||
struct __EnumFlatten #wrapper_generics #where_clause {
|
struct __EnumFlatten #wrapper_generics #where_clause {
|
||||||
data: (#(&'__a #fields_ty,)*),
|
data: (#(&'__a #fields_ty,)*),
|
||||||
phantom: _serde::__private::PhantomData<#this_type #ty_generics>,
|
phantom: _serde::__private::PhantomData<#this_type #ty_generics>,
|
||||||
@@ -1212,6 +1212,7 @@ fn wrap_serialize_with(
|
|||||||
});
|
});
|
||||||
|
|
||||||
quote!({
|
quote!({
|
||||||
|
#[doc(hidden)]
|
||||||
struct __SerializeWith #wrapper_impl_generics #where_clause {
|
struct __SerializeWith #wrapper_impl_generics #where_clause {
|
||||||
values: (#(&'__a #field_tys, )*),
|
values: (#(&'__a #field_tys, )*),
|
||||||
phantom: _serde::__private::PhantomData<#this_type #ty_generics>,
|
phantom: _serde::__private::PhantomData<#this_type #ty_generics>,
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "serde_derive_internals"
|
name = "serde_derive_internals"
|
||||||
version = "0.26.0" # remember to update html_root_url
|
version = "0.27.0" # remember to update html_root_url
|
||||||
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>", "David Tolnay <dtolnay@gmail.com>"]
|
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>", "David Tolnay <dtolnay@gmail.com>"]
|
||||||
description = "AST representation used by Serde derive macros. Unstable."
|
description = "AST representation used by Serde derive macros. Unstable."
|
||||||
documentation = "https://docs.rs/serde_derive_internals"
|
documentation = "https://docs.rs/serde_derive_internals"
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
#![doc(html_root_url = "https://docs.rs/serde_derive_internals/0.26.0")]
|
#![doc(html_root_url = "https://docs.rs/serde_derive_internals/0.27.0")]
|
||||||
#![allow(unknown_lints, bare_trait_objects)]
|
#![allow(unknown_lints, bare_trait_objects)]
|
||||||
// Ignored clippy lints
|
// Ignored clippy lints
|
||||||
#![allow(
|
#![allow(
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "serde_test"
|
name = "serde_test"
|
||||||
version = "1.0.157" # remember to update html_root_url
|
version = "1.0.163" # remember to update html_root_url
|
||||||
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>", "David Tolnay <dtolnay@gmail.com>"]
|
authors = ["Erick Tryzelaar <erick.tryzelaar@gmail.com>", "David Tolnay <dtolnay@gmail.com>"]
|
||||||
build = "build.rs"
|
build = "build.rs"
|
||||||
categories = ["development-tools::testing"]
|
categories = ["development-tools::testing"]
|
||||||
|
|||||||
+83
-95
@@ -12,32 +12,29 @@ pub struct Deserializer<'de> {
|
|||||||
tokens: &'de [Token],
|
tokens: &'de [Token],
|
||||||
}
|
}
|
||||||
|
|
||||||
macro_rules! assert_next_token {
|
fn assert_next_token(de: &mut Deserializer, expected: Token) -> Result<(), Error> {
|
||||||
($de:expr, $expected:expr) => {
|
match de.next_token_opt() {
|
||||||
match $de.next_token_opt() {
|
Some(token) if token == expected => Ok(()),
|
||||||
Some(token) if token == $expected => {}
|
Some(other) => Err(de::Error::custom(format!(
|
||||||
Some(other) => panic!(
|
"expected Token::{} but deserialization wants Token::{}",
|
||||||
"expected Token::{} but deserialization wants Token::{}",
|
other, expected,
|
||||||
other, $expected
|
))),
|
||||||
),
|
None => Err(de::Error::custom(format!(
|
||||||
None => panic!(
|
"end of tokens but deserialization wants Token::{}",
|
||||||
"end of tokens but deserialization wants Token::{}",
|
expected,
|
||||||
$expected
|
))),
|
||||||
),
|
}
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
macro_rules! unexpected {
|
fn unexpected(token: Token) -> Error {
|
||||||
($token:expr) => {
|
de::Error::custom(format!(
|
||||||
panic!("deserialization did not expect this token: {}", $token)
|
"deserialization did not expect this token: {}",
|
||||||
};
|
token,
|
||||||
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
macro_rules! end_of_tokens {
|
fn end_of_tokens() -> Error {
|
||||||
() => {
|
de::Error::custom("ran out of tokens to deserialize")
|
||||||
panic!("ran out of tokens to deserialize")
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'de> Deserializer<'de> {
|
impl<'de> Deserializer<'de> {
|
||||||
@@ -49,11 +46,8 @@ impl<'de> Deserializer<'de> {
|
|||||||
self.tokens.first().cloned()
|
self.tokens.first().cloned()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn peek_token(&self) -> Token {
|
fn peek_token(&self) -> Result<Token, Error> {
|
||||||
match self.peek_token_opt() {
|
self.peek_token_opt().ok_or_else(end_of_tokens)
|
||||||
Some(token) => token,
|
|
||||||
None => end_of_tokens!(),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn next_token_opt(&mut self) -> Option<Token> {
|
pub fn next_token_opt(&mut self) -> Option<Token> {
|
||||||
@@ -66,14 +60,10 @@ impl<'de> Deserializer<'de> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn next_token(&mut self) -> Token {
|
fn next_token(&mut self) -> Result<Token, Error> {
|
||||||
match self.tokens.split_first() {
|
let (&first, rest) = self.tokens.split_first().ok_or_else(end_of_tokens)?;
|
||||||
Some((&first, rest)) => {
|
self.tokens = rest;
|
||||||
self.tokens = rest;
|
Ok(first)
|
||||||
first
|
|
||||||
}
|
|
||||||
None => end_of_tokens!(),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn remaining(&self) -> usize {
|
pub fn remaining(&self) -> usize {
|
||||||
@@ -94,7 +84,7 @@ impl<'de> Deserializer<'de> {
|
|||||||
len: len,
|
len: len,
|
||||||
end: end,
|
end: end,
|
||||||
})?;
|
})?;
|
||||||
assert_next_token!(self, end);
|
assert_next_token(self, end)?;
|
||||||
Ok(value)
|
Ok(value)
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -112,7 +102,7 @@ impl<'de> Deserializer<'de> {
|
|||||||
len: len,
|
len: len,
|
||||||
end: end,
|
end: end,
|
||||||
})?;
|
})?;
|
||||||
assert_next_token!(self, end);
|
assert_next_token(self, end)?;
|
||||||
Ok(value)
|
Ok(value)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -129,7 +119,7 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
let token = self.next_token();
|
let token = self.next_token()?;
|
||||||
match token {
|
match token {
|
||||||
Token::Bool(v) => visitor.visit_bool(v),
|
Token::Bool(v) => visitor.visit_bool(v),
|
||||||
Token::I8(v) => visitor.visit_i8(v),
|
Token::I8(v) => visitor.visit_i8(v),
|
||||||
@@ -161,50 +151,50 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
|
|||||||
Token::Map { len } => self.visit_map(len, Token::MapEnd, visitor),
|
Token::Map { len } => self.visit_map(len, Token::MapEnd, visitor),
|
||||||
Token::Struct { len, .. } => self.visit_map(Some(len), Token::StructEnd, visitor),
|
Token::Struct { len, .. } => self.visit_map(Some(len), Token::StructEnd, visitor),
|
||||||
Token::Enum { .. } => {
|
Token::Enum { .. } => {
|
||||||
let variant = self.next_token();
|
let variant = self.next_token()?;
|
||||||
let next = self.peek_token();
|
let next = self.peek_token()?;
|
||||||
match (variant, next) {
|
match (variant, next) {
|
||||||
(Token::Str(variant), Token::Unit) => {
|
(Token::Str(variant), Token::Unit) => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_str(variant)
|
visitor.visit_str(variant)
|
||||||
}
|
}
|
||||||
(Token::BorrowedStr(variant), Token::Unit) => {
|
(Token::BorrowedStr(variant), Token::Unit) => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_borrowed_str(variant)
|
visitor.visit_borrowed_str(variant)
|
||||||
}
|
}
|
||||||
(Token::String(variant), Token::Unit) => {
|
(Token::String(variant), Token::Unit) => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_string(variant.to_string())
|
visitor.visit_string(variant.to_string())
|
||||||
}
|
}
|
||||||
(Token::Bytes(variant), Token::Unit) => {
|
(Token::Bytes(variant), Token::Unit) => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_bytes(variant)
|
visitor.visit_bytes(variant)
|
||||||
}
|
}
|
||||||
(Token::BorrowedBytes(variant), Token::Unit) => {
|
(Token::BorrowedBytes(variant), Token::Unit) => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_borrowed_bytes(variant)
|
visitor.visit_borrowed_bytes(variant)
|
||||||
}
|
}
|
||||||
(Token::ByteBuf(variant), Token::Unit) => {
|
(Token::ByteBuf(variant), Token::Unit) => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_byte_buf(variant.to_vec())
|
visitor.visit_byte_buf(variant.to_vec())
|
||||||
}
|
}
|
||||||
(Token::U8(variant), Token::Unit) => {
|
(Token::U8(variant), Token::Unit) => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_u8(variant)
|
visitor.visit_u8(variant)
|
||||||
}
|
}
|
||||||
(Token::U16(variant), Token::Unit) => {
|
(Token::U16(variant), Token::Unit) => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_u16(variant)
|
visitor.visit_u16(variant)
|
||||||
}
|
}
|
||||||
(Token::U32(variant), Token::Unit) => {
|
(Token::U32(variant), Token::Unit) => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_u32(variant)
|
visitor.visit_u32(variant)
|
||||||
}
|
}
|
||||||
(Token::U64(variant), Token::Unit) => {
|
(Token::U64(variant), Token::Unit) => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_u64(variant)
|
visitor.visit_u64(variant)
|
||||||
}
|
}
|
||||||
(variant, Token::Unit) => unexpected!(variant),
|
(variant, Token::Unit) => Err(unexpected(variant)),
|
||||||
(variant, _) => {
|
(variant, _) => {
|
||||||
visitor.visit_map(EnumMapVisitor::new(self, variant, EnumFormat::Any))
|
visitor.visit_map(EnumMapVisitor::new(self, variant, EnumFormat::Any))
|
||||||
}
|
}
|
||||||
@@ -232,9 +222,7 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
|
|||||||
| Token::MapEnd
|
| Token::MapEnd
|
||||||
| Token::StructEnd
|
| Token::StructEnd
|
||||||
| Token::TupleVariantEnd
|
| Token::TupleVariantEnd
|
||||||
| Token::StructVariantEnd => {
|
| Token::StructVariantEnd => Err(unexpected(token)),
|
||||||
unexpected!(token);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -242,13 +230,13 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
match self.peek_token() {
|
match self.peek_token()? {
|
||||||
Token::Unit | Token::None => {
|
Token::Unit | Token::None => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_none()
|
visitor.visit_none()
|
||||||
}
|
}
|
||||||
Token::Some => {
|
Token::Some => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_some(self)
|
visitor.visit_some(self)
|
||||||
}
|
}
|
||||||
_ => self.deserialize_any(visitor),
|
_ => self.deserialize_any(visitor),
|
||||||
@@ -264,9 +252,9 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
match self.peek_token() {
|
match self.peek_token()? {
|
||||||
Token::Enum { name: n } if name == n => {
|
Token::Enum { name: n } if name == n => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
|
|
||||||
visitor.visit_enum(DeserializerEnumVisitor { de: self })
|
visitor.visit_enum(DeserializerEnumVisitor { de: self })
|
||||||
}
|
}
|
||||||
@@ -286,9 +274,9 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
match self.peek_token() {
|
match self.peek_token()? {
|
||||||
Token::UnitStruct { .. } => {
|
Token::UnitStruct { .. } => {
|
||||||
assert_next_token!(self, Token::UnitStruct { name: name });
|
assert_next_token(self, Token::UnitStruct { name: name })?;
|
||||||
visitor.visit_unit()
|
visitor.visit_unit()
|
||||||
}
|
}
|
||||||
_ => self.deserialize_any(visitor),
|
_ => self.deserialize_any(visitor),
|
||||||
@@ -303,9 +291,9 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
match self.peek_token() {
|
match self.peek_token()? {
|
||||||
Token::NewtypeStruct { .. } => {
|
Token::NewtypeStruct { .. } => {
|
||||||
assert_next_token!(self, Token::NewtypeStruct { name: name });
|
assert_next_token(self, Token::NewtypeStruct { name: name })?;
|
||||||
visitor.visit_newtype_struct(self)
|
visitor.visit_newtype_struct(self)
|
||||||
}
|
}
|
||||||
_ => self.deserialize_any(visitor),
|
_ => self.deserialize_any(visitor),
|
||||||
@@ -316,21 +304,21 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
match self.peek_token() {
|
match self.peek_token()? {
|
||||||
Token::Unit | Token::UnitStruct { .. } => {
|
Token::Unit | Token::UnitStruct { .. } => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_unit()
|
visitor.visit_unit()
|
||||||
}
|
}
|
||||||
Token::Seq { .. } => {
|
Token::Seq { .. } => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
self.visit_seq(Some(len), Token::SeqEnd, visitor)
|
self.visit_seq(Some(len), Token::SeqEnd, visitor)
|
||||||
}
|
}
|
||||||
Token::Tuple { .. } => {
|
Token::Tuple { .. } => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
self.visit_seq(Some(len), Token::TupleEnd, visitor)
|
self.visit_seq(Some(len), Token::TupleEnd, visitor)
|
||||||
}
|
}
|
||||||
Token::TupleStruct { .. } => {
|
Token::TupleStruct { .. } => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
self.visit_seq(Some(len), Token::TupleStructEnd, visitor)
|
self.visit_seq(Some(len), Token::TupleStructEnd, visitor)
|
||||||
}
|
}
|
||||||
_ => self.deserialize_any(visitor),
|
_ => self.deserialize_any(visitor),
|
||||||
@@ -346,25 +334,25 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
match self.peek_token() {
|
match self.peek_token()? {
|
||||||
Token::Unit => {
|
Token::Unit => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
visitor.visit_unit()
|
visitor.visit_unit()
|
||||||
}
|
}
|
||||||
Token::UnitStruct { .. } => {
|
Token::UnitStruct { .. } => {
|
||||||
assert_next_token!(self, Token::UnitStruct { name: name });
|
assert_next_token(self, Token::UnitStruct { name: name })?;
|
||||||
visitor.visit_unit()
|
visitor.visit_unit()
|
||||||
}
|
}
|
||||||
Token::Seq { .. } => {
|
Token::Seq { .. } => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
self.visit_seq(Some(len), Token::SeqEnd, visitor)
|
self.visit_seq(Some(len), Token::SeqEnd, visitor)
|
||||||
}
|
}
|
||||||
Token::Tuple { .. } => {
|
Token::Tuple { .. } => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
self.visit_seq(Some(len), Token::TupleEnd, visitor)
|
self.visit_seq(Some(len), Token::TupleEnd, visitor)
|
||||||
}
|
}
|
||||||
Token::TupleStruct { len: n, .. } => {
|
Token::TupleStruct { len: n, .. } => {
|
||||||
assert_next_token!(self, Token::TupleStruct { name: name, len: n });
|
assert_next_token(self, Token::TupleStruct { name: name, len: n })?;
|
||||||
self.visit_seq(Some(len), Token::TupleStructEnd, visitor)
|
self.visit_seq(Some(len), Token::TupleStructEnd, visitor)
|
||||||
}
|
}
|
||||||
_ => self.deserialize_any(visitor),
|
_ => self.deserialize_any(visitor),
|
||||||
@@ -380,13 +368,13 @@ impl<'de, 'a> de::Deserializer<'de> for &'a mut Deserializer<'de> {
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
match self.peek_token() {
|
match self.peek_token()? {
|
||||||
Token::Struct { len: n, .. } => {
|
Token::Struct { len: n, .. } => {
|
||||||
assert_next_token!(self, Token::Struct { name: name, len: n });
|
assert_next_token(self, Token::Struct { name: name, len: n })?;
|
||||||
self.visit_map(Some(fields.len()), Token::StructEnd, visitor)
|
self.visit_map(Some(fields.len()), Token::StructEnd, visitor)
|
||||||
}
|
}
|
||||||
Token::Map { .. } => {
|
Token::Map { .. } => {
|
||||||
self.next_token();
|
self.next_token()?;
|
||||||
self.visit_map(Some(fields.len()), Token::MapEnd, visitor)
|
self.visit_map(Some(fields.len()), Token::MapEnd, visitor)
|
||||||
}
|
}
|
||||||
_ => self.deserialize_any(visitor),
|
_ => self.deserialize_any(visitor),
|
||||||
@@ -476,7 +464,7 @@ impl<'de, 'a> EnumAccess<'de> for DeserializerEnumVisitor<'a, 'de> {
|
|||||||
where
|
where
|
||||||
V: DeserializeSeed<'de>,
|
V: DeserializeSeed<'de>,
|
||||||
{
|
{
|
||||||
match self.de.peek_token() {
|
match self.de.peek_token()? {
|
||||||
Token::UnitVariant { variant: v, .. }
|
Token::UnitVariant { variant: v, .. }
|
||||||
| Token::NewtypeVariant { variant: v, .. }
|
| Token::NewtypeVariant { variant: v, .. }
|
||||||
| Token::TupleVariant { variant: v, .. }
|
| Token::TupleVariant { variant: v, .. }
|
||||||
@@ -497,9 +485,9 @@ impl<'de, 'a> VariantAccess<'de> for DeserializerEnumVisitor<'a, 'de> {
|
|||||||
type Error = Error;
|
type Error = Error;
|
||||||
|
|
||||||
fn unit_variant(self) -> Result<(), Error> {
|
fn unit_variant(self) -> Result<(), Error> {
|
||||||
match self.de.peek_token() {
|
match self.de.peek_token()? {
|
||||||
Token::UnitVariant { .. } => {
|
Token::UnitVariant { .. } => {
|
||||||
self.de.next_token();
|
self.de.next_token()?;
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
_ => Deserialize::deserialize(self.de),
|
_ => Deserialize::deserialize(self.de),
|
||||||
@@ -510,9 +498,9 @@ impl<'de, 'a> VariantAccess<'de> for DeserializerEnumVisitor<'a, 'de> {
|
|||||||
where
|
where
|
||||||
T: DeserializeSeed<'de>,
|
T: DeserializeSeed<'de>,
|
||||||
{
|
{
|
||||||
match self.de.peek_token() {
|
match self.de.peek_token()? {
|
||||||
Token::NewtypeVariant { .. } => {
|
Token::NewtypeVariant { .. } => {
|
||||||
self.de.next_token();
|
self.de.next_token()?;
|
||||||
seed.deserialize(self.de)
|
seed.deserialize(self.de)
|
||||||
}
|
}
|
||||||
_ => seed.deserialize(self.de),
|
_ => seed.deserialize(self.de),
|
||||||
@@ -523,26 +511,26 @@ impl<'de, 'a> VariantAccess<'de> for DeserializerEnumVisitor<'a, 'de> {
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
match self.de.peek_token() {
|
match self.de.peek_token()? {
|
||||||
Token::TupleVariant { len: enum_len, .. } => {
|
Token::TupleVariant { len: enum_len, .. } => {
|
||||||
let token = self.de.next_token();
|
let token = self.de.next_token()?;
|
||||||
|
|
||||||
if len == enum_len {
|
if len == enum_len {
|
||||||
self.de
|
self.de
|
||||||
.visit_seq(Some(len), Token::TupleVariantEnd, visitor)
|
.visit_seq(Some(len), Token::TupleVariantEnd, visitor)
|
||||||
} else {
|
} else {
|
||||||
unexpected!(token);
|
Err(unexpected(token))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Token::Seq {
|
Token::Seq {
|
||||||
len: Some(enum_len),
|
len: Some(enum_len),
|
||||||
} => {
|
} => {
|
||||||
let token = self.de.next_token();
|
let token = self.de.next_token()?;
|
||||||
|
|
||||||
if len == enum_len {
|
if len == enum_len {
|
||||||
self.de.visit_seq(Some(len), Token::SeqEnd, visitor)
|
self.de.visit_seq(Some(len), Token::SeqEnd, visitor)
|
||||||
} else {
|
} else {
|
||||||
unexpected!(token);
|
Err(unexpected(token))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_ => de::Deserializer::deserialize_any(self.de, visitor),
|
_ => de::Deserializer::deserialize_any(self.de, visitor),
|
||||||
@@ -557,27 +545,27 @@ impl<'de, 'a> VariantAccess<'de> for DeserializerEnumVisitor<'a, 'de> {
|
|||||||
where
|
where
|
||||||
V: Visitor<'de>,
|
V: Visitor<'de>,
|
||||||
{
|
{
|
||||||
match self.de.peek_token() {
|
match self.de.peek_token()? {
|
||||||
Token::StructVariant { len: enum_len, .. } => {
|
Token::StructVariant { len: enum_len, .. } => {
|
||||||
let token = self.de.next_token();
|
let token = self.de.next_token()?;
|
||||||
|
|
||||||
if fields.len() == enum_len {
|
if fields.len() == enum_len {
|
||||||
self.de
|
self.de
|
||||||
.visit_map(Some(fields.len()), Token::StructVariantEnd, visitor)
|
.visit_map(Some(fields.len()), Token::StructVariantEnd, visitor)
|
||||||
} else {
|
} else {
|
||||||
unexpected!(token);
|
Err(unexpected(token))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Token::Map {
|
Token::Map {
|
||||||
len: Some(enum_len),
|
len: Some(enum_len),
|
||||||
} => {
|
} => {
|
||||||
let token = self.de.next_token();
|
let token = self.de.next_token()?;
|
||||||
|
|
||||||
if fields.len() == enum_len {
|
if fields.len() == enum_len {
|
||||||
self.de
|
self.de
|
||||||
.visit_map(Some(fields.len()), Token::MapEnd, visitor)
|
.visit_map(Some(fields.len()), Token::MapEnd, visitor)
|
||||||
} else {
|
} else {
|
||||||
unexpected!(token);
|
Err(unexpected(token))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
_ => de::Deserializer::deserialize_any(self.de, visitor),
|
_ => de::Deserializer::deserialize_any(self.de, visitor),
|
||||||
@@ -622,7 +610,7 @@ impl<'de, 'a> MapAccess<'de> for EnumMapVisitor<'a, 'de> {
|
|||||||
.deserialize(BytesDeserializer { value: variant })
|
.deserialize(BytesDeserializer { value: variant })
|
||||||
.map(Some),
|
.map(Some),
|
||||||
Some(Token::U32(variant)) => seed.deserialize(variant.into_deserializer()).map(Some),
|
Some(Token::U32(variant)) => seed.deserialize(variant.into_deserializer()).map(Some),
|
||||||
Some(other) => unexpected!(other),
|
Some(other) => Err(unexpected(other)),
|
||||||
None => Ok(None),
|
None => Ok(None),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -641,7 +629,7 @@ impl<'de, 'a> MapAccess<'de> for EnumMapVisitor<'a, 'de> {
|
|||||||
};
|
};
|
||||||
seed.deserialize(SeqAccessDeserializer::new(visitor))?
|
seed.deserialize(SeqAccessDeserializer::new(visitor))?
|
||||||
};
|
};
|
||||||
assert_next_token!(self.de, Token::TupleVariantEnd);
|
assert_next_token(self.de, Token::TupleVariantEnd)?;
|
||||||
Ok(value)
|
Ok(value)
|
||||||
}
|
}
|
||||||
EnumFormat::Map => {
|
EnumFormat::Map => {
|
||||||
@@ -653,7 +641,7 @@ impl<'de, 'a> MapAccess<'de> for EnumMapVisitor<'a, 'de> {
|
|||||||
};
|
};
|
||||||
seed.deserialize(MapAccessDeserializer::new(visitor))?
|
seed.deserialize(MapAccessDeserializer::new(visitor))?
|
||||||
};
|
};
|
||||||
assert_next_token!(self.de, Token::StructVariantEnd);
|
assert_next_token(self.de, Token::StructVariantEnd)?;
|
||||||
Ok(value)
|
Ok(value)
|
||||||
}
|
}
|
||||||
EnumFormat::Any => seed.deserialize(&mut *self.de),
|
EnumFormat::Any => seed.deserialize(&mut *self.de),
|
||||||
|
|||||||
@@ -140,7 +140,7 @@
|
|||||||
//! # }
|
//! # }
|
||||||
//! ```
|
//! ```
|
||||||
|
|
||||||
#![doc(html_root_url = "https://docs.rs/serde_test/1.0.157")]
|
#![doc(html_root_url = "https://docs.rs/serde_test/1.0.163")]
|
||||||
#![cfg_attr(feature = "cargo-clippy", allow(renamed_and_removed_lints))]
|
#![cfg_attr(feature = "cargo-clippy", allow(renamed_and_removed_lints))]
|
||||||
// Ignored clippy lints
|
// Ignored clippy lints
|
||||||
#![cfg_attr(feature = "cargo-clippy", allow(float_cmp, needless_doctest_main))]
|
#![cfg_attr(feature = "cargo-clippy", allow(float_cmp, needless_doctest_main))]
|
||||||
@@ -182,7 +182,3 @@ pub use assert::{
|
|||||||
pub use token::Token;
|
pub use token::Token;
|
||||||
|
|
||||||
pub use configure::{Compact, Configure, Readable};
|
pub use configure::{Compact, Configure, Readable};
|
||||||
|
|
||||||
// Not public API.
|
|
||||||
#[doc(hidden)]
|
|
||||||
pub use de::Deserializer;
|
|
||||||
|
|||||||
@@ -63,14 +63,12 @@ macro_rules! assert_next_token {
|
|||||||
($ser:expr, $actual:expr, $pat:pat, $guard:expr) => {
|
($ser:expr, $actual:expr, $pat:pat, $guard:expr) => {
|
||||||
match $ser.next_token() {
|
match $ser.next_token() {
|
||||||
Some($pat) if $guard => {}
|
Some($pat) if $guard => {}
|
||||||
Some(expected) => {
|
Some(expected) => return Err(ser::Error::custom(
|
||||||
panic!("expected Token::{} but serialized as {}",
|
format!("expected Token::{} but serialized as {}", expected, $actual)
|
||||||
expected, $actual);
|
)),
|
||||||
}
|
None => return Err(ser::Error::custom(
|
||||||
None => {
|
format!("expected end of tokens, but {} was serialized", $actual)
|
||||||
panic!("expected end of tokens, but {} was serialized",
|
)),
|
||||||
$actual);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,11 @@
|
|||||||
|
use serde::Deserialize;
|
||||||
|
|
||||||
|
macro_rules! bug {
|
||||||
|
($serde_path:literal) => {
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
#[serde(crate = $serde_path)]
|
||||||
|
pub struct Struct;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
bug!("serde");
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
use serde_derive::Serialize;
|
||||||
|
|
||||||
|
#[derive(Serialize)]
|
||||||
|
#[serde()]
|
||||||
|
pub struct S;
|
||||||
@@ -2317,6 +2317,53 @@ fn test_internally_tagged_enum_new_type_with_unit() {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_adjacently_tagged_enum_bytes() {
|
||||||
|
#[derive(Serialize, Deserialize, PartialEq, Debug)]
|
||||||
|
#[serde(tag = "t", content = "c")]
|
||||||
|
enum Data {
|
||||||
|
A { a: i32 },
|
||||||
|
}
|
||||||
|
|
||||||
|
let data = Data::A { a: 0 };
|
||||||
|
|
||||||
|
assert_tokens(
|
||||||
|
&data,
|
||||||
|
&[
|
||||||
|
Token::Struct {
|
||||||
|
name: "Data",
|
||||||
|
len: 2,
|
||||||
|
},
|
||||||
|
Token::Str("t"),
|
||||||
|
Token::Str("A"),
|
||||||
|
Token::Str("c"),
|
||||||
|
Token::Struct { name: "A", len: 1 },
|
||||||
|
Token::Str("a"),
|
||||||
|
Token::I32(0),
|
||||||
|
Token::StructEnd,
|
||||||
|
Token::StructEnd,
|
||||||
|
],
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_de_tokens(
|
||||||
|
&data,
|
||||||
|
&[
|
||||||
|
Token::Struct {
|
||||||
|
name: "Data",
|
||||||
|
len: 2,
|
||||||
|
},
|
||||||
|
Token::Bytes(b"t"),
|
||||||
|
Token::Str("A"),
|
||||||
|
Token::Bytes(b"c"),
|
||||||
|
Token::Struct { name: "A", len: 1 },
|
||||||
|
Token::Str("a"),
|
||||||
|
Token::I32(0),
|
||||||
|
Token::StructEnd,
|
||||||
|
Token::StructEnd,
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_adjacently_tagged_enum_containing_flatten() {
|
fn test_adjacently_tagged_enum_containing_flatten() {
|
||||||
#[derive(Serialize, Deserialize, PartialEq, Debug)]
|
#[derive(Serialize, Deserialize, PartialEq, Debug)]
|
||||||
|
|||||||
@@ -4,6 +4,8 @@
|
|||||||
clippy::used_underscore_binding
|
clippy::used_underscore_binding
|
||||||
)]
|
)]
|
||||||
|
|
||||||
|
use serde::de::value::{BorrowedStrDeserializer, MapDeserializer};
|
||||||
|
use serde::de::IntoDeserializer;
|
||||||
use serde::{Deserialize, Deserializer};
|
use serde::{Deserialize, Deserializer};
|
||||||
use serde_test::{assert_de_tokens, assert_de_tokens_error, Token};
|
use serde_test::{assert_de_tokens, assert_de_tokens_error, Token};
|
||||||
|
|
||||||
@@ -130,20 +132,22 @@ fn test_cow() {
|
|||||||
borrowed: Cow<'b, str>,
|
borrowed: Cow<'b, str>,
|
||||||
}
|
}
|
||||||
|
|
||||||
let tokens = &[
|
struct BorrowedStr(&'static str);
|
||||||
Token::Struct {
|
|
||||||
name: "Cows",
|
|
||||||
len: 2,
|
|
||||||
},
|
|
||||||
Token::Str("copied"),
|
|
||||||
Token::BorrowedStr("copied"),
|
|
||||||
Token::Str("borrowed"),
|
|
||||||
Token::BorrowedStr("borrowed"),
|
|
||||||
Token::StructEnd,
|
|
||||||
];
|
|
||||||
|
|
||||||
let mut de = serde_test::Deserializer::new(tokens);
|
impl<'de> IntoDeserializer<'de> for BorrowedStr {
|
||||||
let cows = Cows::deserialize(&mut de).unwrap();
|
type Deserializer = BorrowedStrDeserializer<'de, serde::de::value::Error>;
|
||||||
|
|
||||||
|
fn into_deserializer(self) -> Self::Deserializer {
|
||||||
|
BorrowedStrDeserializer::new(self.0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let de = MapDeserializer::new(IntoIterator::into_iter([
|
||||||
|
("copied", BorrowedStr("copied")),
|
||||||
|
("borrowed", BorrowedStr("borrowed")),
|
||||||
|
]));
|
||||||
|
|
||||||
|
let cows = Cows::deserialize(de).unwrap();
|
||||||
|
|
||||||
match cows.copied {
|
match cows.copied {
|
||||||
Cow::Owned(ref s) if s == "copied" => {}
|
Cow::Owned(ref s) if s == "copied" => {}
|
||||||
|
|||||||
+17
-23
@@ -33,7 +33,7 @@ use std::time::{Duration, UNIX_EPOCH};
|
|||||||
use std::sync::atomic::{AtomicI64, AtomicU64};
|
use std::sync::atomic::{AtomicI64, AtomicU64};
|
||||||
|
|
||||||
use fnv::FnvHasher;
|
use fnv::FnvHasher;
|
||||||
use serde::de::DeserializeOwned;
|
use serde::de::{DeserializeOwned, IntoDeserializer};
|
||||||
use serde::{Deserialize, Deserializer};
|
use serde::{Deserialize, Deserializer};
|
||||||
use serde_test::{assert_de_tokens, Configure, Token};
|
use serde_test::{assert_de_tokens, Configure, Token};
|
||||||
|
|
||||||
@@ -202,9 +202,8 @@ fn assert_de_tokens_ignore(ignorable_tokens: &[Token]) {
|
|||||||
.chain(vec![Token::MapEnd].into_iter())
|
.chain(vec![Token::MapEnd].into_iter())
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
let mut de = serde_test::Deserializer::new(&concated_tokens);
|
let expected = IgnoreBase { a: 1 };
|
||||||
let base = IgnoreBase::deserialize(&mut de).unwrap();
|
assert_de_tokens(&expected, &concated_tokens);
|
||||||
assert_eq!(base, IgnoreBase { a: 1 });
|
|
||||||
}
|
}
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////
|
//////////////////////////////////////////////////////////////////////////
|
||||||
@@ -2245,39 +2244,34 @@ fn test_cstr() {
|
|||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_atomics() {
|
fn test_atomics() {
|
||||||
fn test<L, A, T>(load: L, val: T, token: Token)
|
fn test<L, A, T>(load: L, val: T)
|
||||||
where
|
where
|
||||||
L: Fn(&A, Ordering) -> T,
|
L: Fn(&A, Ordering) -> T,
|
||||||
A: DeserializeOwned,
|
A: DeserializeOwned,
|
||||||
T: PartialEq + Debug,
|
T: PartialEq + Debug + Copy + for<'de> IntoDeserializer<'de>,
|
||||||
{
|
{
|
||||||
let tokens = &[token];
|
match A::deserialize(val.into_deserializer()) {
|
||||||
let mut de = serde_test::Deserializer::new(tokens);
|
|
||||||
match A::deserialize(&mut de) {
|
|
||||||
Ok(v) => {
|
Ok(v) => {
|
||||||
let loaded = load(&v, Ordering::Relaxed);
|
let loaded = load(&v, Ordering::Relaxed);
|
||||||
assert_eq!(val, loaded);
|
assert_eq!(val, loaded);
|
||||||
}
|
}
|
||||||
Err(e) => panic!("tokens failed to deserialize: {}", e),
|
Err(e) => panic!("tokens failed to deserialize: {}", e),
|
||||||
};
|
|
||||||
if de.remaining() > 0 {
|
|
||||||
panic!("{} remaining tokens", de.remaining());
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
test(AtomicBool::load, true, Token::Bool(true));
|
test(AtomicBool::load, true);
|
||||||
test(AtomicI8::load, -127, Token::I8(-127i8));
|
test(AtomicI8::load, -127i8);
|
||||||
test(AtomicI16::load, -510, Token::I16(-510i16));
|
test(AtomicI16::load, -510i16);
|
||||||
test(AtomicI32::load, -131072, Token::I32(-131072i32));
|
test(AtomicI32::load, -131072i32);
|
||||||
test(AtomicIsize::load, -131072isize, Token::I32(-131072));
|
test(AtomicIsize::load, -131072isize);
|
||||||
test(AtomicU8::load, 127, Token::U8(127u8));
|
test(AtomicU8::load, 127u8);
|
||||||
test(AtomicU16::load, 510u16, Token::U16(510u16));
|
test(AtomicU16::load, 510u16);
|
||||||
test(AtomicU32::load, 131072u32, Token::U32(131072u32));
|
test(AtomicU32::load, 131072u32);
|
||||||
test(AtomicUsize::load, 131072usize, Token::U32(131072));
|
test(AtomicUsize::load, 131072usize);
|
||||||
|
|
||||||
#[cfg(target_arch = "x86_64")]
|
#[cfg(target_arch = "x86_64")]
|
||||||
{
|
{
|
||||||
test(AtomicI64::load, -8589934592, Token::I64(-8589934592));
|
test(AtomicI64::load, -8589934592i64);
|
||||||
test(AtomicU64::load, 8589934592u64, Token::U64(8589934592));
|
test(AtomicU64::load, 8589934592u64);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user