citadel_proto 0.13.0

Networking library for the Citadel Protocol
Documentation
//! Bytes Codec Implementation for Citadel Protocol
//!
//! This module provides a basic bytes codec implementation for raw data transmission in the Citadel Protocol.
//! The codec handles efficient encoding and decoding of raw bytes with configurable buffer capacity.
//!
//! # Features
//! - Configurable buffer capacity for memory efficiency
//! - Zero-copy split operations for optimal performance
//! - Automatic buffer resizing when capacity is insufficient
//! - Implementation of tokio_util's Encoder and Decoder traits
//!
//! # Important Notes
//! - The codec maintains a minimum buffer size defined by `CODEC_MIN_BUFFER`
//! - Buffer capacity is automatically increased if it falls below the minimum
//! - The codec performs best with buffer sizes optimized for your use case
//!
//! # Related Components
//! - `citadel_io::tokio_util::codec` - Provides the core codec traits
//! - `crate::constants` - Defines important buffer size constants
//! - Network transport layers that utilize this codec for raw byte transmission

use std::io;

use bytes::BufMut;
use bytes::{Bytes, BytesMut};
use citadel_io::tokio_util::codec::{Decoder, Encoder};

use crate::constants::CODEC_MIN_BUFFER;

#[derive(Copy, Clone, Debug, Eq, PartialEq, Ord, PartialOrd, Hash, Default)]
pub struct BytesCodec(usize);

impl BytesCodec {
    /// Creates a new `BytesCodec` for shipping around raw bytes.
    pub fn new(buffer_capacity: usize) -> BytesCodec {
        BytesCodec(buffer_capacity)
    }
}

impl Decoder for BytesCodec {
    type Item = BytesMut;
    type Error = io::Error;

    fn decode(&mut self, buf: &mut BytesMut) -> Result<Option<BytesMut>, io::Error> {
        if buf.capacity() < CODEC_MIN_BUFFER {
            buf.reserve(self.0 - buf.capacity());
        }
        if !buf.is_empty() {
            let len = buf.len();
            let ret = buf.split_to(len);
            Ok(Some(ret))
        } else {
            Ok(None)
        }
    }
}

impl Encoder<Bytes> for BytesCodec {
    type Error = io::Error;

    fn encode(&mut self, data: Bytes, buf: &mut BytesMut) -> Result<(), io::Error> {
        buf.reserve(data.len());
        buf.put(data);
        Ok(())
    }
}

#[cfg(test)]
mod tests {
    use crate::constants::CODEC_BUFFER_CAPACITY;
    use crate::proto::codec::BytesCodec;
    use bytes::{BufMut, Bytes, BytesMut};
    use citadel_io::tokio_util::codec::{Decoder, Encoder};

    #[test]
    fn test_bytes_codec() {
        citadel_logging::setup_log();

        let mut codec = BytesCodec::new(CODEC_BUFFER_CAPACITY);
        let mut buf = BytesMut::new();

        let mut ret = BytesMut::new();

        for x in 0..(CODEC_BUFFER_CAPACITY * 2) {
            let val = (x % 255) as u8;
            ret.put_u8(val);

            let slice = &[val];
            let data = Bytes::copy_from_slice(slice as &[u8]);
            codec.encode(data, &mut buf).unwrap();
        }

        if let Ok(Some(data)) = codec.decode(&mut buf) {
            assert_eq!(data, ret);
            assert!(codec.decode(&mut buf).unwrap().is_none());
        } else {
            panic!("Failed test");
        }
    }
}