-
Notifications
You must be signed in to change notification settings - Fork 8
Expand file tree
/
Copy pathserde_string.rs
More file actions
122 lines (106 loc) · 3.52 KB
/
serde_string.rs
File metadata and controls
122 lines (106 loc) · 3.52 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
use serde::{de::Error, Deserializer, ser::{SerializeSeq, Serializer}, Deserialize};
use serde_json;
pub fn serialize<S>(value: &i64, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
serializer.serialize_str(&value.to_string())
}
pub fn serialize_vec<S>(values: &Vec<i64>, serializer: S) -> Result<S::Ok, S::Error>
where
S: Serializer,
{
let mut seq = serializer.serialize_seq(Some(values.len()))?;
for v in values.iter() {
seq.serialize_element(&v.to_string())?;
}
seq.end()
}
pub fn deserialize<'de, D>(deserializer: D) -> Result<i64, D::Error>
where
D: Deserializer<'de>,
{
match serde_json::Value::deserialize(deserializer)? {
serde_json::Value::String(s) => s.parse::<i64>().map_err(Error::custom),
serde_json::Value::Number(n) => n
.as_i64()
.ok_or_else(|| Error::custom("number out of range for i64")),
_ => Err(Error::custom("expected string or number")),
}
}
pub fn deserialize_vec<'de, D>(deserializer: D) -> Result<Vec<i64>, D::Error>
where
D: Deserializer<'de>,
{
let strings: Vec<String> = Vec::<String>::deserialize(deserializer)?;
strings
.into_iter()
.map(|s| s.parse::<i64>().map_err(Error::custom))
.collect()
}
pub fn deserialize_option_vec<'de, D>(deserializer: D) -> Result<Option<Vec<i64>>, D::Error>
where
D: Deserializer<'de>,
{
// Use a custom visitor to handle the deserialization
// This allows us to handle both string arrays and number arrays
deserializer.deserialize_option(OptionVecVisitor)
}
/// Deserializes Option<Vec<i64>> but converts None to empty Vec<i64>
/// This allows Question to accept null roles in input but always have Vec<i64> internally
pub fn deserialize_option_vec_to_vec<'de, D>(deserializer: D) -> Result<Vec<i64>, D::Error>
where
D: Deserializer<'de>,
{
let option = deserialize_option_vec(deserializer)?;
Ok(option.unwrap_or_default())
}
struct OptionVecVisitor;
impl<'de> serde::de::Visitor<'de> for OptionVecVisitor {
type Value = Option<Vec<i64>>;
fn expecting(&self, formatter: &mut std::fmt::Formatter) -> std::fmt::Result {
formatter.write_str("null or an array of strings or numbers")
}
fn visit_none<E>(self) -> Result<Self::Value, E>
where
E: Error,
{
Ok(None)
}
fn visit_unit<E>(self) -> Result<Self::Value, E>
where
E: Error,
{
Ok(None)
}
fn visit_some<D>(self, deserializer: D) -> Result<Self::Value, D::Error>
where
D: Deserializer<'de>,
{
deserializer.deserialize_seq(VecVisitor).map(Some)
}
}
struct VecVisitor;
impl<'de> serde::de::Visitor<'de> for VecVisitor {
type Value = Vec<i64>;
fn expecting(&self, formatter: &mut std::fmt::Formatter) -> std::fmt::Result {
formatter.write_str("an array of strings or numbers")
}
fn visit_seq<A>(self, mut seq: A) -> Result<Self::Value, A::Error>
where
A: serde::de::SeqAccess<'de>,
{
let mut vec = Vec::new();
while let Some(elem) = seq.next_element::<serde_json::Value>()? {
let parsed = match elem {
serde_json::Value::String(s) => s.parse::<i64>().map_err(Error::custom)?,
serde_json::Value::Number(n) => {
n.as_i64().ok_or_else(|| Error::custom("number out of range for i64"))?
}
_ => return Err(Error::custom("expected string or number in roles array")),
};
vec.push(parsed);
}
Ok(vec)
}
}