- Sponsor
-
Notifications
You must be signed in to change notification settings - Fork 812
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Implement #[serde(other)] on enum variant #1382
Conversation
serde_derive/src/de.rs
Outdated
@@ -1850,6 +1866,10 @@ fn deserialize_generated_identifier( | |||
let ignore_variant = quote!(__other(_serde::private::de::Content<'de>),); | |||
let fallthrough = quote!(_serde::export::Ok(__Field::__other(__value))); | |||
(Some(ignore_variant), Some(fallthrough)) | |||
} else if other_idx.is_some() { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This should be written as else if let Some(other_idx) = other_idx
to avoid the unwrap below.
serde_derive/src/internals/check.rs
Outdated
if i < variants.len() - 1 { | ||
cx.error("#[serde(other)] must be the last variant"); | ||
} | ||
} | ||
|
||
// Variant with `other` attribute must be a unit variant. | ||
(_, Identifier::Field, true) => { | ||
// TODO: Only in field_identifier ? | ||
(_, Identifier::Field, true, _) => { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What currently happens if you write serde(other)
on a variant that is not unit?
Here are two cases that I would expect to work but do not. Do you think these would be possible to support? #[derive(Deserialize, PartialEq, Debug)]
enum E {
A,
#[serde(other)]
Other(String),
}
fn main() {
assert_eq!(
serde_json::from_str::<E>(r#" "foo" "#).unwrap(),
E::Other("foo".to_owned()),
);
} #[derive(Deserialize, PartialEq, Debug)]
struct Artifact {}
#[derive(Deserialize, PartialEq, Debug)]
enum Message {
CompilerArtifact(Artifact),
#[serde(other)]
Unknown,
}
fn main() {
assert_eq!(
serde_json::from_str::<Message>(r#" {"CompilerMessage":[]} "#).unwrap(),
Message::Unknown,
);
} |
🤔 this is a bit of a puzzle. The first one IMO doesn't make sense: it's deserializing an externally tagged enum, so we expect something in the form The second one should work though, so I probably messed up somewhere! (I mostly tested on internally tagged enums for my use-case ^^'). I'll take a look. |
I would expect the first one to work by analogy with how |
Ah, so this is quite different from how I originally envisioned this to work. I thought the data contained in the So here's a proposal: #[derive(Deserialize)]
enum Test {
A,
#[serde(other)]
B(#[serde(other_tag)] String, serde_json::Value)
} The field equipped with Then, we'd get The upside is that we can keep parsing data even in the Unknown case (maybe we can still get some useful data that we know are guaranteed to always be present regardless of the enum ?) while having a consistent way to access the variant name. Would this work? |
I guess my confusion is in when you would ever want to capture the data of an unknown variant but not its variant name. I would expect the default least-effort way of using |
Alright, here's my reasoning: As a user, I want the ability to access both the data and the tag in a simple way. Because of this, I want Other to behave like the other enum variant, because that's the intuitive thing. Here's an example: we have an API that sends requests following this schema: {
"action": "some_action",
"version": "1.0.0",
"field_of_some_action": "1",
"field_of_some_action2": "2",
} (this is an internally tagged enum). The API contract says So, how would I parse this API? #[derive(Deserialize, PartialEq, Debug)]
#[serde(tag = "action")]
enum E {
Action1, // whatever
#[serde(other)]
UnknownAction { version: String },
}
fn main() {
assert_eq!(
serde_json::from_str::<E>(r#" "{ "action": "unknown", "version": "2.0.0" }" "#).unwrap(),
E::UnknownAction { version: "2.0.0" }),
);
} Now, obviously, in many/most cases, we'd still want the ability to access the tag. For this case, we need to put it somewhere in that
The first felt the most intuitive to me. With this case, in order to access the #[derive(Deserialize, PartialEq, Debug)]
#[serde(tag = "action")]
enum E {
Action1, // whatever
#[serde(other)]
UnknownAction { #[serde(other_tag)] action: String, version: String },
}
fn main() {
assert_eq!(
serde_json::from_str::<E>(r#" "{ "action": "unknown", "version": "2.0.0" }" "#).unwrap(),
E::UnknownAction { action: "unknown", version: "2.0.0" }),
);
} I'm still trying to figure out if this is even implementable though. Maybe I could just enforce that the Other is a unit struct for now, and then we can discuss what the proper course of action is for accessing the tag/data in a separate issue and follow-up PR ? |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks, this looks like the right behavior for unit variants.
Does this implement the following use-case: Data shape: {
"errors": ["ErrorKind1", "ErrorKind2", ...]
} So we get 0 or more errors. Rust code #[derive(Deserialize)]
struct Result {
errors: Vec<ErrorKind>
}
enum ErrorKind {
ErrorKind1,
ErrorKind2,
// ...
#[serde(other)]
Unrecognised(String),
} That is, I receive a list of errors, and most of the time I can strongly type them, but on occasion I don't recognize the string, and so fall back to just storing the string value. This seems to be different to the use-case discussed in the PR. |
The pr only allows the #[serde(other)] variant to be unit-style. A future pr could extend it to allow recovering the tag. From my testing, however, there are several problems that don't have am obvious solution:
|
A conservative approach would be to just reject the parse if the discriminant type and the extra type don't match, and just support numbers and strings. Then the functionality you mention in your 2 points could be added incrementally, whilst retaining backwards compatibility. |
I'll have a go at implementing this and see what it looks like. |
Guess this has gotten stalled? |
This gets me started, thanks. I'm using EDIT: Using an |
What if I have a tag I want but also want to have a blanket |
Part of #912.
It might require more complex testing WRT non-unit
other
fields?Untagged enums are not supported. I'm not quite sure the checks are correct, so if someone could make sure they make sense (they might be a bit too lax).