aldrin-core 0.13.0

Shared core components of Aldrin, a message bus for service-oriented RPC and interprocess communication.
Documentation
use super::message_ops::Sealed;
use super::{
    Message, MessageDeserializeError, MessageKind, MessageOps, MessageSerializeError,
    MessageSerializer, MessageWithValueDeserializer,
};
use crate::tags::{self, PrimaryTag, Tag};
use crate::{
    Deserialize, DeserializeError, DeserializePrimary, Deserializer, Serialize, SerializeError,
    SerializePrimary, SerializedValue, SerializedValueSlice, Serializer,
};
use bytes::BytesMut;
use num_enum::{IntoPrimitive, TryFromPrimitive};

#[derive(Debug, Clone, PartialEq, Eq, Default)]
#[cfg_attr(feature = "fuzzing", derive(arbitrary::Arbitrary))]
pub struct ConnectReplyData {
    pub user: Option<SerializedValue>,
}

impl ConnectReplyData {
    pub fn new() -> Self {
        Self::default()
    }

    pub fn serialize_user_as<T: Tag>(
        &mut self,
        user: impl Serialize<T>,
    ) -> Result<&mut Self, SerializeError> {
        self.user = SerializedValue::serialize_as(user).map(Some)?;
        Ok(self)
    }

    pub fn serialize_user(
        &mut self,
        user: impl SerializePrimary,
    ) -> Result<&mut Self, SerializeError> {
        self.serialize_user_as(user)
    }

    pub fn deserialize_user_as<T: Tag, U: Deserialize<T>>(
        &self,
    ) -> Option<Result<U, DeserializeError>> {
        self.user
            .as_deref()
            .map(SerializedValueSlice::deserialize_as)
    }

    pub fn deserialize_user<T: DeserializePrimary>(&self) -> Option<Result<T, DeserializeError>> {
        self.deserialize_user_as()
    }
}

impl Tag for ConnectReplyData {}

impl PrimaryTag for ConnectReplyData {
    type Tag = Self;
}

#[derive(IntoPrimitive, TryFromPrimitive)]
#[repr(u32)]
enum ConnectReplyDataField {
    User = 0,
}

impl Serialize<Self> for ConnectReplyData {
    fn serialize(self, serializer: Serializer) -> Result<(), SerializeError> {
        serializer.serialize(&self)
    }
}

impl Serialize<ConnectReplyData> for &ConnectReplyData {
    fn serialize(self, serializer: Serializer) -> Result<(), SerializeError> {
        let mut serializer = serializer.serialize_struct2()?;

        serializer.serialize_if_some::<tags::Option<tags::Value>>(
            ConnectReplyDataField::User,
            &self.user,
        )?;

        serializer.finish()
    }
}

impl Deserialize<Self> for ConnectReplyData {
    fn deserialize(deserializer: Deserializer) -> Result<Self, DeserializeError> {
        let mut deserializer = deserializer.deserialize_struct()?;

        let mut user = None;

        while let Some(deserializer) = deserializer.deserialize()? {
            match deserializer.try_id() {
                Ok(ConnectReplyDataField::User) => user = deserializer.deserialize()?,
                Err(_) => deserializer.skip()?,
            }
        }

        deserializer.finish(Self { user })
    }
}

#[derive(Debug, Copy, Clone, PartialEq, Eq, IntoPrimitive, TryFromPrimitive)]
#[repr(u8)]
enum ConnectReplyKind {
    Ok = 0,
    Rejected = 1,
    IncompatibleVersion = 2,
}

#[derive(Debug, Copy, Clone, PartialEq, Eq)]
#[cfg_attr(feature = "fuzzing", derive(arbitrary::Arbitrary))]
pub enum ConnectResult {
    Ok(u32),
    Rejected,
    IncompatibleVersion,
}

#[derive(Debug, Clone, PartialEq, Eq)]
#[cfg_attr(feature = "fuzzing", derive(arbitrary::Arbitrary))]
pub struct ConnectReply2 {
    pub result: ConnectResult,
    pub value: SerializedValue,
}

impl MessageOps for ConnectReply2 {
    fn kind(&self) -> MessageKind {
        MessageKind::ConnectReply2
    }

    fn serialize_message(self) -> Result<BytesMut, MessageSerializeError> {
        let mut serializer = MessageSerializer::with_value(self.value, MessageKind::ConnectReply2)?;

        match self.result {
            ConnectResult::Ok(version) => {
                serializer.put_discriminant_u8(ConnectReplyKind::Ok);
                serializer.put_varint_u32_le(version);
                serializer.finish()
            }

            ConnectResult::Rejected => {
                serializer.put_discriminant_u8(ConnectReplyKind::Rejected);
                serializer.finish()
            }

            ConnectResult::IncompatibleVersion => {
                serializer.put_discriminant_u8(ConnectReplyKind::IncompatibleVersion);
                serializer.finish()
            }
        }
    }

    fn deserialize_message(buf: BytesMut) -> Result<Self, MessageDeserializeError> {
        let mut deserializer = MessageWithValueDeserializer::new(buf, MessageKind::ConnectReply2)?;

        match deserializer.try_get_discriminant_u8()? {
            ConnectReplyKind::Ok => {
                let version = deserializer.try_get_varint_u32_le()?;
                let value = deserializer.finish()?;

                Ok(Self {
                    result: ConnectResult::Ok(version),
                    value,
                })
            }

            ConnectReplyKind::Rejected => {
                let value = deserializer.finish()?;

                Ok(Self {
                    result: ConnectResult::Rejected,
                    value,
                })
            }

            ConnectReplyKind::IncompatibleVersion => {
                let value = deserializer.finish()?;

                Ok(Self {
                    result: ConnectResult::IncompatibleVersion,
                    value,
                })
            }
        }
    }

    fn value(&self) -> Option<&SerializedValueSlice> {
        Some(&self.value)
    }

    fn value_mut(&mut self) -> Option<&mut SerializedValue> {
        Some(&mut self.value)
    }
}

impl Sealed for ConnectReply2 {}

impl From<ConnectReply2> for Message {
    fn from(msg: ConnectReply2) -> Self {
        Self::ConnectReply2(msg)
    }
}

#[cfg(test)]
mod test {
    use super::super::test::{assert_deserialize_eq_with_value, assert_serialize_eq};
    use super::super::Message;
    use super::{ConnectReply2, ConnectReplyData, ConnectResult};
    use crate::SerializedValue;

    #[test]
    fn ok() {
        let serialized = [13, 0, 0, 0, 47, 2, 0, 0, 0, 65, 0, 0, 1];
        let value = ConnectReplyData::new();

        let msg = ConnectReply2 {
            result: ConnectResult::Ok(1),
            value: SerializedValue::serialize(&value).unwrap(),
        };
        assert_serialize_eq(&msg, serialized);
        assert_deserialize_eq_with_value(&msg, serialized, &value);

        let msg = Message::ConnectReply2(msg);
        assert_serialize_eq(&msg, serialized);
        assert_deserialize_eq_with_value(&msg, serialized, &value);
    }

    #[test]
    fn rejected() {
        let serialized = [12, 0, 0, 0, 47, 2, 0, 0, 0, 65, 0, 1];
        let value = ConnectReplyData::new();

        let msg = ConnectReply2 {
            result: ConnectResult::Rejected,
            value: SerializedValue::serialize(&value).unwrap(),
        };
        assert_serialize_eq(&msg, serialized);
        assert_deserialize_eq_with_value(&msg, serialized, &value);

        let msg = Message::ConnectReply2(msg);
        assert_serialize_eq(&msg, serialized);
        assert_deserialize_eq_with_value(&msg, serialized, &value);
    }

    #[test]
    fn incompatible_version() {
        let serialized = [12, 0, 0, 0, 47, 2, 0, 0, 0, 65, 0, 2];
        let value = ConnectReplyData::new();

        let msg = ConnectReply2 {
            result: ConnectResult::IncompatibleVersion,
            value: SerializedValue::serialize(&value).unwrap(),
        };
        assert_serialize_eq(&msg, serialized);
        assert_deserialize_eq_with_value(&msg, serialized, &value);

        let msg = Message::ConnectReply2(msg);
        assert_serialize_eq(&msg, serialized);
        assert_deserialize_eq_with_value(&msg, serialized, &value);
    }
}