mirror of
https://github.com/pezkuwichain/serde.git
synced 2026-06-18 02:41:03 +00:00
Add test with generic deserialize_with function
This commit is contained in:
+23
-8
@@ -1788,7 +1788,7 @@ fn deserialize_untagged_variant(
|
|||||||
deserializer: TokenStream,
|
deserializer: TokenStream,
|
||||||
) -> Fragment {
|
) -> Fragment {
|
||||||
if let Some(path) = variant.attrs.deserialize_with() {
|
if let Some(path) = variant.attrs.deserialize_with() {
|
||||||
let unwrap_fn = unwrap_to_variant_closure(params, variant, quote!(__wrap));
|
let unwrap_fn = unwrap_to_variant_closure(params, variant, false);
|
||||||
return quote_block! {
|
return quote_block! {
|
||||||
_serde::__private::Result::map(#path(#deserializer), #unwrap_fn)
|
_serde::__private::Result::map(#path(#deserializer), #unwrap_fn)
|
||||||
};
|
};
|
||||||
@@ -2894,7 +2894,7 @@ fn wrap_deserialize_variant_with(
|
|||||||
let (wrapper, wrapper_ty) =
|
let (wrapper, wrapper_ty) =
|
||||||
wrap_deserialize_with(params, "e!((#(#field_tys),*)), deserialize_with);
|
wrap_deserialize_with(params, "e!((#(#field_tys),*)), deserialize_with);
|
||||||
|
|
||||||
let unwrap_fn = unwrap_to_variant_closure(params, variant, quote!(__wrap.value));
|
let unwrap_fn = unwrap_to_variant_closure(params, variant, true);
|
||||||
|
|
||||||
(wrapper, wrapper_ty, unwrap_fn)
|
(wrapper, wrapper_ty, unwrap_fn)
|
||||||
}
|
}
|
||||||
@@ -2903,11 +2903,26 @@ fn wrap_deserialize_variant_with(
|
|||||||
fn unwrap_to_variant_closure(
|
fn unwrap_to_variant_closure(
|
||||||
params: &Parameters,
|
params: &Parameters,
|
||||||
variant: &Variant,
|
variant: &Variant,
|
||||||
wrapper: TokenStream,
|
with_wrapper: bool,
|
||||||
) -> TokenStream {
|
) -> TokenStream {
|
||||||
let this = ¶ms.this;
|
let this = ¶ms.this;
|
||||||
let variant_ident = &variant.ident;
|
let variant_ident = &variant.ident;
|
||||||
|
|
||||||
|
let (arg, wrapper) = if with_wrapper {
|
||||||
|
(
|
||||||
|
quote!{ __wrap },
|
||||||
|
quote!{ __wrap.value },
|
||||||
|
)
|
||||||
|
} else {
|
||||||
|
let field_tys = variant.fields.iter().map(|field| field.ty);
|
||||||
|
(
|
||||||
|
// When additional DeserializeWith wrapper is not used, Rust
|
||||||
|
// is not able to infer types, so we help him
|
||||||
|
quote!{ __wrap: (#(#field_tys),*) },
|
||||||
|
quote!{ __wrap },
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
let field_access = (0..variant.fields.len()).map(|n| {
|
let field_access = (0..variant.fields.len()).map(|n| {
|
||||||
Member::Unnamed(Index {
|
Member::Unnamed(Index {
|
||||||
index: n as u32,
|
index: n as u32,
|
||||||
@@ -2919,23 +2934,23 @@ fn unwrap_to_variant_closure(
|
|||||||
Style::Struct if variant.fields.len() == 1 => {
|
Style::Struct if variant.fields.len() == 1 => {
|
||||||
let member = &variant.fields[0].member;
|
let member = &variant.fields[0].member;
|
||||||
quote! {
|
quote! {
|
||||||
|__wrap| #this::#variant_ident { #member: #wrapper }
|
|#arg| #this::#variant_ident { #member: #wrapper }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Style::Struct => {
|
Style::Struct => {
|
||||||
let members = variant.fields.iter().map(|field| &field.member);
|
let members = variant.fields.iter().map(|field| &field.member);
|
||||||
quote! {
|
quote! {
|
||||||
|__wrap| #this::#variant_ident { #(#members: #wrapper.#field_access),* }
|
|#arg| #this::#variant_ident { #(#members: #wrapper.#field_access),* }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Style::Tuple => quote! {
|
Style::Tuple => quote! {
|
||||||
|__wrap| #this::#variant_ident(#(#wrapper.#field_access),*)
|
|#arg| #this::#variant_ident(#(#wrapper.#field_access),*)
|
||||||
},
|
},
|
||||||
Style::Newtype => quote! {
|
Style::Newtype => quote! {
|
||||||
|__wrap| #this::#variant_ident(#wrapper)
|
|#arg| #this::#variant_ident(#wrapper)
|
||||||
},
|
},
|
||||||
Style::Unit => quote! {
|
Style::Unit => quote! {
|
||||||
|__wrap| #this::#variant_ident
|
|#arg| #this::#variant_ident
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -821,3 +821,26 @@ where
|
|||||||
{
|
{
|
||||||
vec.first().serialize(serializer)
|
vec.first().serialize(serializer)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
#[derive(Debug, PartialEq, Deserialize)]
|
||||||
|
#[serde(tag = "tag")]
|
||||||
|
enum InternallyTagged {
|
||||||
|
#[serde(deserialize_with = "deserialize_generic")]
|
||||||
|
Unit,
|
||||||
|
|
||||||
|
#[serde(deserialize_with = "deserialize_generic")]
|
||||||
|
Newtype(i32),
|
||||||
|
|
||||||
|
#[serde(deserialize_with = "deserialize_generic")]
|
||||||
|
Struct { f1: String, f2: u8 },
|
||||||
|
}
|
||||||
|
|
||||||
|
fn deserialize_generic<'de, T, D>(deserializer: D) -> StdResult<T, D::Error>
|
||||||
|
where
|
||||||
|
T: Deserialize<'de>,
|
||||||
|
D: Deserializer<'de>,
|
||||||
|
{
|
||||||
|
T::deserialize(deserializer)
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user