use std::{
fmt,
sync::{atomic::AtomicUsize, Mutex},
};
use bytes::Bytes;
use engineioxide::Str;
use serde::{de::Visitor, ser::Impossible, Deserialize, Serialize};
use crate::{packet::Packet, Value};
#[derive(Debug, Default)]
pub struct ParserState {
pub partial_bin_packet: Mutex<Option<Packet>>,
pub incoming_binary_cnt: AtomicUsize,
}
pub trait Parse: Default + Copy {
type EncodeError: std::error::Error;
type DecodeError: std::error::Error;
fn encode(self, packet: Packet) -> Value;
fn decode_str(
self,
state: &ParserState,
data: Str,
) -> Result<Packet, ParseError<Self::DecodeError>>;
fn decode_bin(
self,
state: &ParserState,
bin: Bytes,
) -> Result<Packet, ParseError<Self::DecodeError>>;
fn encode_value<T: ?Sized + Serialize>(
self,
data: &T,
event: Option<&str>,
) -> Result<Value, Self::EncodeError>;
fn decode_value<'de, T: Deserialize<'de>>(
self,
value: &'de mut Value,
with_event: bool,
) -> Result<T, Self::DecodeError>;
fn decode_default<'de, T: Deserialize<'de>>(
self,
value: Option<&'de Value>,
) -> Result<T, Self::DecodeError>;
fn encode_default<T: ?Sized + Serialize>(self, data: &T) -> Result<Value, Self::EncodeError>;
fn read_event(self, value: &Value) -> Result<&str, Self::DecodeError>;
}
#[derive(thiserror::Error, Debug)]
pub enum ParseError<E: std::error::Error> {
#[error("invalid packet type")]
InvalidPacketType,
#[error("invalid ack id")]
InvalidAckId,
#[error("invalid event name")]
InvalidEventName,
#[error("invalid data")]
InvalidData,
#[error("invalid namespace")]
InvalidNamespace,
#[error("invalid attachments")]
InvalidAttachments,
#[error(
"received unexpected binary data. Make sure you are using the same parser on both ends."
)]
UnexpectedBinaryPacket,
#[error(
"received unexpected string data. Make sure you are using the same parser on both ends."
)]
UnexpectedStringPacket,
#[error("needs more binary data for packet completion")]
NeedsMoreBinaryData,
#[error("parser error: {0:?}")]
ParserError(#[from] E),
}
impl<E: std::error::Error> ParseError<E> {
pub fn wrap_err<E1: std::error::Error>(self, f: impl FnOnce(E) -> E1) -> ParseError<E1> {
match self {
Self::ParserError(e) => ParseError::ParserError(f(e)),
ParseError::InvalidPacketType => ParseError::InvalidPacketType,
ParseError::InvalidAckId => ParseError::InvalidAckId,
ParseError::InvalidEventName => ParseError::InvalidEventName,
ParseError::InvalidData => ParseError::InvalidData,
ParseError::InvalidNamespace => ParseError::InvalidNamespace,
ParseError::InvalidAttachments => ParseError::InvalidAttachments,
ParseError::UnexpectedBinaryPacket => ParseError::UnexpectedBinaryPacket,
ParseError::UnexpectedStringPacket => ParseError::UnexpectedStringPacket,
ParseError::NeedsMoreBinaryData => ParseError::NeedsMoreBinaryData,
}
}
}
pub struct FirstElement<T>(std::marker::PhantomData<T>);
impl<T> Default for FirstElement<T> {
fn default() -> Self {
Self(Default::default())
}
}
impl<'de, T: serde::Deserialize<'de>> serde::de::Visitor<'de> for FirstElement<T> {
type Value = T;
fn expecting(&self, formatter: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
write!(formatter, "a sequence in which we care about first element",)
}
fn visit_seq<A>(self, mut seq: A) -> Result<Self::Value, A::Error>
where
A: serde::de::SeqAccess<'de>,
{
use serde::de::Error;
let data = seq
.next_element::<T>()?
.ok_or(A::Error::custom("first element not found"));
while seq.next_element::<serde::de::IgnoredAny>()?.is_some() {}
data
}
}
impl<'de, T: serde::Deserialize<'de>> serde::de::DeserializeSeed<'de> for FirstElement<T> {
type Value = T;
fn deserialize<D>(self, deserializer: D) -> Result<Self::Value, D::Error>
where
D: serde::Deserializer<'de>,
{
deserializer.deserialize_seq(self)
}
}
struct IsTupleSerde;
#[derive(Debug)]
struct IsTupleSerdeError(bool);
impl fmt::Display for IsTupleSerdeError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "IsTupleSerializerError: {}", self.0)
}
}
impl std::error::Error for IsTupleSerdeError {}
impl serde::ser::Error for IsTupleSerdeError {
fn custom<T: fmt::Display>(_msg: T) -> Self {
IsTupleSerdeError(false)
}
}
impl serde::de::Error for IsTupleSerdeError {
fn custom<T: fmt::Display>(_msg: T) -> Self {
IsTupleSerdeError(false)
}
}
impl<'de> serde::Deserializer<'de> for IsTupleSerde {
type Error = IsTupleSerdeError;
serde::forward_to_deserialize_any! {
bool i8 i16 i32 i64 i128 u8 u16 u32 u64 u128 f32 f64 char str
string unit unit_struct seq map
struct enum identifier ignored_any bytes byte_buf option
}
fn deserialize_any<V: Visitor<'de>>(self, _visitor: V) -> Result<V::Value, Self::Error> {
Err(IsTupleSerdeError(false))
}
fn deserialize_tuple<V: Visitor<'de>>(
self,
_len: usize,
_visitor: V,
) -> Result<V::Value, Self::Error> {
Err(IsTupleSerdeError(true))
}
fn deserialize_tuple_struct<V: Visitor<'de>>(
self,
_name: &'static str,
_len: usize,
_visitor: V,
) -> Result<V::Value, Self::Error> {
Err(IsTupleSerdeError(true))
}
fn deserialize_newtype_struct<V: Visitor<'de>>(
self,
_name: &'static str,
_visitor: V,
) -> Result<V::Value, Self::Error> {
Err(IsTupleSerdeError(true))
}
}
impl serde::Serializer for IsTupleSerde {
type Ok = bool;
type Error = IsTupleSerdeError;
type SerializeSeq = Impossible<bool, IsTupleSerdeError>;
type SerializeTuple = Impossible<bool, IsTupleSerdeError>;
type SerializeTupleStruct = Impossible<bool, IsTupleSerdeError>;
type SerializeTupleVariant = Impossible<bool, IsTupleSerdeError>;
type SerializeMap = Impossible<bool, IsTupleSerdeError>;
type SerializeStruct = Impossible<bool, IsTupleSerdeError>;
type SerializeStructVariant = Impossible<bool, IsTupleSerdeError>;
fn serialize_bool(self, _v: bool) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_i8(self, _v: i8) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_i16(self, _v: i16) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_i32(self, _v: i32) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_i64(self, _v: i64) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_u8(self, _v: u8) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_u16(self, _v: u16) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_u32(self, _v: u32) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_u64(self, _v: u64) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_f32(self, _v: f32) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_f64(self, _v: f64) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_char(self, _v: char) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_str(self, _v: &str) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_bytes(self, _v: &[u8]) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_none(self) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_some<T>(self, _value: &T) -> Result<Self::Ok, Self::Error>
where
T: ?Sized + serde::Serialize,
{
Ok(false)
}
fn serialize_unit(self) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_unit_struct(self, _name: &'static str) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_unit_variant(
self,
_name: &'static str,
_variant_index: u32,
_variant: &'static str,
) -> Result<Self::Ok, Self::Error> {
Ok(false)
}
fn serialize_newtype_struct<T>(
self,
_name: &'static str,
_value: &T,
) -> Result<Self::Ok, Self::Error>
where
T: ?Sized + serde::Serialize,
{
Ok(true)
}
fn serialize_newtype_variant<T>(
self,
_name: &'static str,
_variant_index: u32,
_variant: &'static str,
_value: &T,
) -> Result<Self::Ok, Self::Error>
where
T: ?Sized + serde::Serialize,
{
Ok(false)
}
fn serialize_seq(self, _len: Option<usize>) -> Result<Self::SerializeSeq, Self::Error> {
Err(IsTupleSerdeError(false))
}
fn serialize_tuple(self, _len: usize) -> Result<Self::SerializeTuple, Self::Error> {
Err(IsTupleSerdeError(true))
}
fn serialize_tuple_struct(
self,
_name: &'static str,
_len: usize,
) -> Result<Self::SerializeTupleStruct, Self::Error> {
Err(IsTupleSerdeError(true))
}
fn serialize_tuple_variant(
self,
_name: &'static str,
_variant_index: u32,
_variant: &'static str,
_len: usize,
) -> Result<Self::SerializeTupleVariant, Self::Error> {
Err(IsTupleSerdeError(false))
}
fn serialize_map(self, _len: Option<usize>) -> Result<Self::SerializeMap, Self::Error> {
Err(IsTupleSerdeError(false))
}
fn serialize_struct(
self,
_name: &'static str,
_len: usize,
) -> Result<Self::SerializeStruct, Self::Error> {
Err(IsTupleSerdeError(false))
}
fn serialize_struct_variant(
self,
_name: &'static str,
_variant_index: u32,
_variant: &'static str,
_len: usize,
) -> Result<Self::SerializeStructVariant, Self::Error> {
Err(IsTupleSerdeError(false))
}
}
pub fn is_ser_tuple<T: ?Sized + serde::Serialize>(value: &T) -> bool {
match value.serialize(IsTupleSerde) {
Ok(v) | Err(IsTupleSerdeError(v)) => v,
}
}
pub fn is_de_tuple<'de, T: serde::Deserialize<'de>>() -> bool {
match T::deserialize(IsTupleSerde) {
Ok(_) => unreachable!(),
Err(IsTupleSerdeError(v)) => v,
}
}
#[cfg(test)]
mod test {
use super::*;
use serde::{Deserialize, Serialize};
#[test]
fn is_tuple() {
assert!(is_ser_tuple(&(1, 2, 3)));
assert!(is_de_tuple::<(usize, usize, usize)>());
assert!(is_ser_tuple(&[1, 2, 3]));
assert!(is_de_tuple::<[usize; 3]>());
#[derive(Serialize, Deserialize)]
struct TupleStruct<'a>(&'a str);
assert!(is_ser_tuple(&TupleStruct("test")));
assert!(is_de_tuple::<TupleStruct<'_>>());
assert!(!is_ser_tuple(&vec![1, 2, 3]));
assert!(!is_de_tuple::<Vec<usize>>());
#[derive(Serialize, Deserialize)]
struct UnitStruct;
assert!(!is_ser_tuple(&UnitStruct));
assert!(!is_de_tuple::<UnitStruct>());
}
}