From 9392350b71c4df36894080d3bac5366ed3b3db87 Mon Sep 17 00:00:00 2001 From: morph <82043364+morph-dev@users.noreply.github.com> Date: Fri, 20 Oct 2023 13:18:19 +0300 Subject: [PATCH 1/2] fix: Make NodeId deserializable from serde_json::Value fix: #62 --- src/node_id.rs | 22 +++++++++++++++++++--- 1 file changed, 19 insertions(+), 3 deletions(-) diff --git a/src/node_id.rs b/src/node_id.rs index 61f7aa6..831d4d3 100644 --- a/src/node_id.rs +++ b/src/node_id.rs @@ -122,8 +122,8 @@ mod serde_hex_prfx { T: hex::FromHex, ::Error: std::fmt::Display, { - let raw: &[u8] = serde::Deserialize::deserialize(deserializer)?; - let src = raw.strip_prefix(b"0x").unwrap_or(raw); + let raw: String = serde::Deserialize::deserialize(deserializer)?; + let src = raw.strip_prefix("0x").unwrap_or(&raw); hex::FromHex::from_hex(src).map_err(serde::de::Error::custom) } } @@ -143,12 +143,28 @@ mod tests { #[cfg(feature = "serde")] #[test] - fn test_serde() { + fn test_serde_str() { let node = NodeId::random(); let json_string = serde_json::to_string(&node).unwrap(); assert_eq!(node, serde_json::from_str::(&json_string).unwrap()); } + #[cfg(feature = "serde")] + #[test] + fn test_serde_slice() { + let node = NodeId::random(); + let json_bytes = serde_json::to_vec(&node).unwrap(); + assert_eq!(node, serde_json::from_slice::(&json_bytes).unwrap()); + } + + #[cfg(feature = "serde")] + #[test] + fn test_serde_value() { + let node = NodeId::random(); + let value = serde_json::to_value(&node).unwrap(); + assert_eq!(node, serde_json::from_value::(value).unwrap()); + } + #[cfg(feature = "serde")] #[test] fn test_serde_0x() { From a06a2ed1d5cd09345e91a55e4544aea58b1eb87d Mon Sep 17 00:00:00 2001 From: Diva M Date: Sun, 22 Oct 2023 11:01:09 -0500 Subject: [PATCH 2/2] use owned strings only when necessary --- src/node_id.rs | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/src/node_id.rs b/src/node_id.rs index 831d4d3..7e3b5f8 100644 --- a/src/node_id.rs +++ b/src/node_id.rs @@ -107,6 +107,7 @@ impl std::fmt::Debug for NodeId { /// Serialize with the 0x prefix. #[cfg(feature = "serde")] mod serde_hex_prfx { + pub fn serialize + hex::ToHex, S: serde::Serializer>( data: &T, serializer: S, @@ -122,7 +123,15 @@ mod serde_hex_prfx { T: hex::FromHex, ::Error: std::fmt::Display, { - let raw: String = serde::Deserialize::deserialize(deserializer)?; + /// Helper struct to obtain a owned string when necessary (using [`serde_json`], for + /// example) or a borrowed string with the appropriate lifetime (most the time). + // NOTE: see https://github.com/serde-rs/serde/issues/1413#issuecomment-494892266 and + // https://github.com/sigp/enr/issues/62 + #[derive(serde::Deserialize)] + struct CowNodeId<'a>(#[serde(borrow)] std::borrow::Cow<'a, str>); + + let CowNodeId::<'de>(raw) = serde::Deserialize::deserialize(deserializer)?; + let src = raw.strip_prefix("0x").unwrap_or(&raw); hex::FromHex::from_hex(src).map_err(serde::de::Error::custom) }