risinglight 0.2.0

An OLAP database system for educational purpose
Documentation
// Copyright 2022 RisingLight Project Authors. Licensed under Apache-2.0.

use std::hash::Hash;

use bytes::{Buf, BufMut};
use ordered_float::OrderedFloat;
use rust_decimal::Decimal;

use crate::array::{
    Array, BlobArray, BoolArray, DateArray, DecimalArray, F64Array, I32Array, I64Array,
    IntervalArray, Utf8Array,
};
use crate::types::{BlobRef, Date, Interval, F64};

/// Encode a primitive value into fixed-width buffer
pub trait PrimitiveFixedWidthEncode:
    Copy + Clone + 'static + Send + Sync + PartialEq + Hash + Eq
{
    /// Width of each element
    const WIDTH: usize;
    const DEFAULT_VALUE: &'static Self;

    type ArrayType: Array<Item = Self>;

    /// Encode current primitive data to the end of an `Vec<u8>`.
    fn encode(&self, buffer: &mut impl BufMut);

    /// Decode a data from a bytes array.
    fn decode(buffer: &mut impl Buf) -> Self;
}

impl PrimitiveFixedWidthEncode for bool {
    const WIDTH: usize = std::mem::size_of::<u8>();
    const DEFAULT_VALUE: &'static bool = &false;
    type ArrayType = BoolArray;

    fn encode(&self, buffer: &mut impl BufMut) {
        buffer.put_u8(*self as u8)
    }

    fn decode(buffer: &mut impl Buf) -> Self {
        buffer.get_u8() != 0
    }
}

impl PrimitiveFixedWidthEncode for i32 {
    const WIDTH: usize = std::mem::size_of::<i32>();
    const DEFAULT_VALUE: &'static i32 = &0;

    type ArrayType = I32Array;

    fn encode(&self, buffer: &mut impl BufMut) {
        buffer.put_i32_le(*self);
    }

    fn decode(buffer: &mut impl Buf) -> Self {
        buffer.get_i32_le()
    }
}

impl PrimitiveFixedWidthEncode for i64 {
    const WIDTH: usize = std::mem::size_of::<i64>();
    const DEFAULT_VALUE: &'static i64 = &0;

    type ArrayType = I64Array;

    fn encode(&self, buffer: &mut impl BufMut) {
        buffer.put_i64_le(*self);
    }

    fn decode(buffer: &mut impl Buf) -> Self {
        buffer.get_i64_le()
    }
}

impl PrimitiveFixedWidthEncode for F64 {
    const WIDTH: usize = std::mem::size_of::<F64>();
    const DEFAULT_VALUE: &'static F64 = &OrderedFloat(0.0);

    type ArrayType = F64Array;

    fn encode(&self, buffer: &mut impl BufMut) {
        buffer.put_f64_le(self.0);
    }

    fn decode(buffer: &mut impl Buf) -> Self {
        buffer.get_f64_le().into()
    }
}

impl PrimitiveFixedWidthEncode for Decimal {
    const WIDTH: usize = std::mem::size_of::<Decimal>();
    const DEFAULT_VALUE: &'static Self = &Decimal::from_parts(0, 0, 0, false, 0);

    type ArrayType = DecimalArray;

    fn encode(&self, buffer: &mut impl BufMut) {
        buffer.put_u128_le(u128::from_le_bytes(self.serialize()))
    }

    fn decode(buffer: &mut impl Buf) -> Self {
        Decimal::deserialize(buffer.get_u128_le().to_le_bytes())
    }
}

impl PrimitiveFixedWidthEncode for Date {
    const WIDTH: usize = std::mem::size_of::<i32>();
    const DEFAULT_VALUE: &'static Self = &Date::new(0);

    type ArrayType = DateArray;

    fn encode(&self, buffer: &mut impl BufMut) {
        buffer.put_i32(self.get_inner());
    }

    fn decode(buffer: &mut impl Buf) -> Self {
        Date::new(buffer.get_i32())
    }
}

impl PrimitiveFixedWidthEncode for Interval {
    const WIDTH: usize = std::mem::size_of::<i32>() + std::mem::size_of::<i32>();
    const DEFAULT_VALUE: &'static Self = &Interval::from_days(0);

    type ArrayType = IntervalArray;

    fn encode(&self, buffer: &mut impl BufMut) {
        buffer.put_i32(self.num_months());
        buffer.put_i32(self.days());
    }

    fn decode(buffer: &mut impl Buf) -> Self {
        let months = buffer.get_i32();
        let days = buffer.get_i32();
        Interval::from_md(months, days)
    }
}

pub trait BlobEncode {
    type ArrayType: Array<Item = Self>;

    /// Returns the length (in bytes) of the blob slice.
    fn len(&self) -> usize;

    /// Converts a slice of bytes to a blob slice.
    fn from_byte_slice(bytes: &[u8]) -> &Self;

    fn to_byte_slice(&self) -> &[u8];
}

impl BlobEncode for BlobRef {
    type ArrayType = BlobArray;

    fn len(&self) -> usize {
        self.as_ref().len()
    }

    fn from_byte_slice(bytes: &[u8]) -> &Self {
        BlobRef::new(bytes)
    }

    fn to_byte_slice(&self) -> &[u8] {
        self.as_ref()
    }
}

impl BlobEncode for str {
    type ArrayType = Utf8Array;

    fn len(&self) -> usize {
        self.len()
    }

    fn from_byte_slice(bytes: &[u8]) -> &Self {
        std::str::from_utf8(bytes).unwrap()
    }

    fn to_byte_slice(&self) -> &[u8] {
        self.as_bytes()
    }
}