use base64::prelude::*;
use quick_xml::{
events::{BytesStart, Event},
name::QName,
Reader, Writer,
};
use serde::forward_to_deserialize_any;
use std::convert::TryInto;
use crate::error::{DecodingError, EncodingError};
use crate::util::{ReaderExt, WriterExt};
use crate::{Error, Result};
use super::{MapDeserializer, MapSerializer};
use super::{SeqDeserializer, SeqSerializer};
#[doc(hidden)]
pub struct Deserializer<'a, 'r> {
pub(crate) reader: &'a mut Reader<&'r [u8]>,
}
impl<'a, 'r> Deserializer<'a, 'r> {
pub fn new(reader: &'a mut Reader<&'r [u8]>) -> Result<Self> {
let ret = Deserializer { reader };
Ok(ret)
}
}
impl<'de, 'a, 'r> serde::Deserializer<'de> for Deserializer<'a, 'r> {
type Error = Error;
#[allow(clippy::cognitive_complexity)]
fn deserialize_any<V>(self, visitor: V) -> Result<V::Value>
where
V: serde::de::Visitor<'de>,
{
let ret = match self.reader.read_event() {
Ok(Event::Text(e)) => visitor.visit_str::<Self::Error>(
e.unescape()
.map_err(DecodingError::from)?.as_ref(),
)?,
Ok(Event::End(ref e)) if e.name() == QName(b"value") => return visitor.visit_str(""),
Ok(Event::Start(ref e)) => match e.name() {
QName(b"int") | QName(b"i4") | QName(b"i8") => {
let text = self
.reader
.read_text(e.name())
.map_err(DecodingError::from)?;
let val: i64 = text.parse().map_err(DecodingError::from)?;
if let Ok(val) = val.try_into() {
visitor.visit_i8::<Self::Error>(val)?
} else if let Ok(val) = val.try_into() {
visitor.visit_i16::<Self::Error>(val)?
} else if let Ok(val) = val.try_into() {
visitor.visit_i32::<Self::Error>(val)?
} else {
visitor.visit_i64::<Self::Error>(val)?
}
}
QName(b"boolean") => {
let text = self
.reader
.read_text(e.name())
.map_err(DecodingError::from)?;
match text.as_ref() {
"1" => visitor.visit_bool::<Self::Error>(true),
"0" => visitor.visit_bool::<Self::Error>(false),
_ => return Err(DecodingError::BooleanDecodeError(text.into_owned()).into()),
}?
}
QName(b"string") => {
visitor.visit_str::<Self::Error>(
self.reader
.read_text(e.name())
.map_err(DecodingError::from)?.as_ref(),
)?
}
QName(b"double") => {
let text = self
.reader
.read_text(e.name())
.map_err(DecodingError::from)?;
visitor.visit_f64::<Self::Error>(text.parse().map_err(DecodingError::from)?)?
}
QName(b"dateTime.iso8601") => {
visitor.visit_str::<Self::Error>(
self.reader
.read_text(e.name())
.map_err(DecodingError::from)?.as_ref(),
)?
}
QName(b"base64") => {
let text = self
.reader
.read_text(e.name())
.map_err(DecodingError::from)?;
visitor.visit_byte_buf::<Self::Error>(
BASE64_STANDARD.decode(text.as_ref()).map_err(DecodingError::from)?,
)?
}
QName(b"struct") => visitor.visit_map(MapDeserializer::new(self.reader, b"struct"))?,
QName(b"array") => {
visitor.visit_seq(SeqDeserializer::new(self.reader, QName(b"data"), Some(QName(b"array")))?)?
}
QName(b"nil") => {
self.reader
.read_to_end(e.name())
.map_err(DecodingError::from)?;
visitor.visit_unit::<Self::Error>()?
}
_ => {
return Err(DecodingError::UnexpectedTag(
String::from_utf8_lossy(e.name().into_inner()).into(),
"one of int|i4|i8|boolean|string|double|dateTime.iso8601|base64|struct|array|nil"
.into(),
)
.into())
}
},
Ok(Event::Eof) => {
return Err(DecodingError::UnexpectedEOF(
"one of int|i4|i8|boolean|string|double|dateTime.iso8601|base64|struct|array|nil"
.into(),
)
.into())
}
Ok(_) => {
return Err(DecodingError::UnexpectedEvent(
"one of int|i4|i8|boolean|string|double|dateTime.iso8601|base64|struct|array|nil"
.into(),
)
.into())
}
Err(e) => return Err(DecodingError::from(e).into()),
};
self.reader
.read_to_end(QName(b"value"))
.map_err(DecodingError::from)?;
Ok(ret)
}
fn deserialize_option<V>(self, visitor: V) -> Result<V::Value>
where
V: serde::de::Visitor<'de>,
{
let mut reader = self.reader.clone();
if let Ok(Event::Start(ref e)) = reader.read_event() {
if e.name() == QName(b"nil") {
reader.read_to_end(e.name()).map_err(DecodingError::from)?;
*self.reader = reader;
return visitor.visit_none::<Self::Error>();
}
}
visitor.visit_some(self)
}
forward_to_deserialize_any!(
bool i8 i16 i32 i64 u8 u16 u32 u64 f32 f64 char str string bytes
byte_buf unit unit_struct newtype_struct seq tuple
tuple_struct map struct enum identifier ignored_any
);
}
#[doc(hidden)]
pub struct Serializer<'a, W>
where
W: std::io::Write,
{
writer: &'a mut Writer<W>,
}
impl<'a, W> Serializer<'a, W>
where
W: std::io::Write,
{
pub fn new(writer: &'a mut Writer<W>) -> Self {
Serializer { writer }
}
}
impl<'a, W> serde::Serializer for Serializer<'a, W>
where
W: std::io::Write,
{
type Ok = ();
type Error = Error;
type SerializeSeq = SeqSerializer<'a, W>;
type SerializeTuple = SeqSerializer<'a, W>;
type SerializeTupleStruct = SeqSerializer<'a, W>;
type SerializeTupleVariant = SeqSerializer<'a, W>;
type SerializeMap = MapSerializer<'a, W>;
type SerializeStruct = MapSerializer<'a, W>;
type SerializeStructVariant = MapSerializer<'a, W>;
fn serialize_bool(self, v: bool) -> Result<Self::Ok> {
self.writer.write_start_tag("value")?;
self.writer
.write_safe_tag("boolean", if v { "1" } else { "0" })?;
self.writer.write_end_tag("value")?;
Ok(())
}
fn serialize_i8(self, v: i8) -> Result<Self::Ok> {
self.serialize_i64(v as i64)
}
fn serialize_i16(self, v: i16) -> Result<Self::Ok> {
self.serialize_i64(v as i64)
}
fn serialize_i32(self, v: i32) -> Result<Self::Ok> {
self.serialize_i64(v as i64)
}
fn serialize_i64(self, v: i64) -> Result<Self::Ok> {
self.writer.write_start_tag("value")?;
self.writer.write_safe_tag("int", &v.to_string())?;
self.writer.write_end_tag("value")?;
Ok(())
}
fn serialize_u8(self, v: u8) -> Result<Self::Ok> {
self.serialize_u64(v as u64)
}
fn serialize_u16(self, v: u16) -> Result<Self::Ok> {
self.serialize_u64(v as u64)
}
fn serialize_u32(self, v: u32) -> Result<Self::Ok> {
self.serialize_u64(v as u64)
}
fn serialize_u64(self, v: u64) -> Result<Self::Ok> {
self.writer.write_start_tag("value")?;
self.writer.write_safe_tag("int", &v.to_string())?;
self.writer.write_end_tag("value")?;
Ok(())
}
fn serialize_f32(self, v: f32) -> Result<Self::Ok> {
self.serialize_f64(v as f64)
}
fn serialize_f64(self, v: f64) -> Result<Self::Ok> {
self.writer.write_start_tag("value")?;
self.writer.write_safe_tag("double", &v.to_string())?;
self.writer.write_end_tag("value")?;
Ok(())
}
fn serialize_char(self, v: char) -> Result<Self::Ok> {
self.writer.write_start_tag("value")?;
self.writer.write_tag("string", &v.to_string())?;
self.writer.write_end_tag("value")?;
Ok(())
}
fn serialize_str(self, v: &str) -> Result<Self::Ok> {
self.writer.write_start_tag("value")?;
self.writer.write_tag("string", v)?;
self.writer.write_end_tag("value")?;
Ok(())
}
fn serialize_bytes(self, v: &[u8]) -> Result<Self::Ok> {
self.writer.write_start_tag("value")?;
self.writer
.write_safe_tag("base64", &BASE64_STANDARD.encode(v))?;
self.writer.write_end_tag("value")?;
Ok(())
}
fn serialize_none(self) -> Result<Self::Ok> {
self.serialize_unit()
}
fn serialize_some<T>(self, v: &T) -> Result<Self::Ok>
where
T: ?Sized + serde::Serialize,
{
v.serialize(self)
}
fn serialize_unit(self) -> Result<Self::Ok> {
self.writer.write_start_tag("value")?;
self.writer
.write_event(Event::Empty(BytesStart::new("nil")))
.map_err(EncodingError::from)?;
self.writer.write_end_tag("value")?;
Ok(())
}
fn serialize_unit_struct(self, _name: &'static str) -> Result<Self::Ok> {
self.serialize_unit()
}
fn serialize_unit_variant(
self,
_name: &'static str,
_variant_index: u32,
_variant: &'static str,
) -> Result<Self::Ok> {
self.serialize_unit()
}
fn serialize_newtype_struct<T>(self, _name: &'static str, value: &T) -> Result<Self::Ok>
where
T: ?Sized + serde::Serialize,
{
value.serialize(self)
}
fn serialize_newtype_variant<T>(
self,
_name: &'static str,
_variant_index: u32,
_variant: &'static str,
_value: &T,
) -> Result<Self::Ok>
where
T: ?Sized + serde::Serialize,
{
unimplemented!();
}
fn serialize_seq(self, _len: Option<usize>) -> Result<Self::SerializeSeq> {
Self::SerializeSeq::new(self.writer)
}
fn serialize_tuple(self, len: usize) -> Result<Self::SerializeTuple> {
self.serialize_seq(Some(len))
}
fn serialize_tuple_struct(
self,
_name: &'static str,
len: usize,
) -> Result<Self::SerializeTupleStruct> {
self.serialize_seq(Some(len))
}
fn serialize_tuple_variant(
self,
_name: &'static str,
_variant_index: u32,
_variant: &'static str,
len: usize,
) -> Result<Self::SerializeTupleVariant> {
self.serialize_seq(Some(len))
}
fn serialize_map(self, _len: Option<usize>) -> Result<Self::SerializeMap> {
Self::SerializeMap::new(self.writer)
}
fn serialize_struct(self, _name: &'static str, len: usize) -> Result<Self::SerializeStruct> {
self.serialize_map(Some(len))
}
fn serialize_struct_variant(
self,
_name: &'static str,
_variant_index: u32,
_variant: &'static str,
len: usize,
) -> Result<Self::SerializeStructVariant> {
self.serialize_map(Some(len))
}
}
#[doc(hidden)]
#[allow(dead_code)]
pub fn from_str<'a, T>(val: &'a str) -> Result<T>
where
T: serde::de::Deserialize<'a>,
{
let mut reader = Reader::from_str(val);
reader.expand_empty_elements(true);
reader.trim_text(true);
reader.expect_tag(QName(b"value"))?;
let deserializer = Deserializer::new(&mut reader)?;
T::deserialize(deserializer)
}
#[doc(hidden)]
#[allow(dead_code)]
pub fn to_string<T>(val: &T) -> Result<String>
where
T: ?Sized + serde::Serialize,
{
let mut writer = Writer::new(Vec::new());
let ser = Serializer::new(&mut writer);
val.serialize(ser)?;
Ok(String::from_utf8(writer.into_inner()).map_err(DecodingError::from)?)
}
#[cfg(test)]
mod tests {
use super::{from_str, to_string};
use serde::{Deserialize, Serialize};
#[derive(Debug, PartialEq, Serialize, Deserialize)]
struct Test {
hello: String,
}
#[test]
fn test_from_str() {
let x: i32 = from_str("<value><int>42</int></value>").unwrap();
assert_eq!(x, 42);
let x: bool = from_str("<value><boolean>1</boolean></value>").unwrap();
assert_eq!(x, true);
let x: Vec<i32> = from_str("<value><array><data><value><int>1</int></value><value><int>2</int></value><value><int>3</int></value></data></array></value>").unwrap();
assert_eq!(x, vec![1, 2, 3]);
let x: Test = from_str("<value><struct><member><name>hello</name><value><string>world</string></value></member></struct></value>").unwrap();
assert_eq!(
x,
Test {
hello: "world".to_string()
}
);
let x: Option<String> = from_str("<value><nil/></value>").unwrap();
assert_eq!(x, None);
let x: Option<String> = from_str("<value>hello world</value>").unwrap();
assert_eq!(x, Some("hello world".to_string()));
}
#[test]
fn test_to_string() {
assert_eq!(&to_string(&42).unwrap(), "<value><int>42</int></value>");
assert_eq!(
&to_string(&true).unwrap(),
"<value><boolean>1</boolean></value>"
);
assert_eq!(
&to_string(&vec![1, 2, 3]).unwrap(),
"<value><array><data><value><int>1</int></value><value><int>2</int></value><value><int>3</int></value></data></array></value>"
);
assert_eq!(
&to_string(&Test {
hello: "world".to_string()
}).unwrap(),
"<value><struct><member><name>hello</name><value><string>world</string></value></member></struct></value>",
);
assert_eq!(
&to_string(&Some("hello world".to_string())).unwrap(),
"<value><string>hello world</string></value>",
);
assert_eq!(
&to_string(&None::<String>).unwrap(),
"<value><nil/></value>",
);
}
}