use std::mem::size_of;
use odbc_sys::{Date, Time, Timestamp};
use crate::{Bit, DataType};
#[derive(Clone, Copy, Debug)]
pub struct BufferDescription {
pub nullable: bool,
pub kind: BufferKind,
}
impl BufferDescription {
pub fn bytes_per_row(&self) -> usize {
let indicator = size_of::<isize>();
let opt_indicator = if self.nullable { indicator } else { 0 };
match self.kind {
BufferKind::Binary { length } => length + indicator,
BufferKind::Text { max_str_len } => max_str_len + 1 + indicator,
BufferKind::WText { max_str_len } => (max_str_len + 1) * 2 + indicator,
BufferKind::F64 => size_of::<f64>() + opt_indicator,
BufferKind::F32 => size_of::<f32>() + opt_indicator,
BufferKind::Date => size_of::<Date>() + opt_indicator,
BufferKind::Time => size_of::<Time>() + opt_indicator,
BufferKind::Timestamp => size_of::<Timestamp>() + opt_indicator,
BufferKind::I8 => size_of::<i8>() + opt_indicator,
BufferKind::I16 => size_of::<i16>() + opt_indicator,
BufferKind::I32 => size_of::<i32>() + opt_indicator,
BufferKind::I64 => size_of::<i64>() + opt_indicator,
BufferKind::U8 => size_of::<u8>() + opt_indicator,
BufferKind::Bit => size_of::<Bit>() + opt_indicator,
}
}
}
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
pub enum BufferKind {
Binary {
length: usize,
},
Text {
max_str_len: usize,
},
WText {
max_str_len: usize,
},
F64,
F32,
Date,
Time,
Timestamp,
I8,
I16,
I32,
I64,
U8,
Bit,
}
impl BufferKind {
pub fn from_data_type(data_type: DataType) -> Option<Self> {
let buffer_kind = match data_type {
DataType::Unknown
| DataType::Other { data_type: _, column_size: _, decimal_digits: _ } => return None,
DataType::Numeric { precision, scale }
| DataType::Decimal { precision, scale } if scale == 0 && precision < 3 => BufferKind::I8,
DataType::Numeric { precision, scale }
| DataType::Decimal { precision, scale } if scale == 0 && precision < 10 => BufferKind::I32,
DataType::Numeric { precision, scale }
| DataType::Decimal { precision, scale } if scale == 0 && precision < 19 => BufferKind::I64,
DataType::Integer => BufferKind::I32,
DataType::SmallInt => BufferKind::I16,
DataType::Float | DataType::Real => BufferKind::F32,
DataType::Double => BufferKind::F64,
DataType::Date => BufferKind::Date,
DataType::Time { precision: 0 } => BufferKind::Time,
DataType::Timestamp { precision: _ } => BufferKind::Timestamp,
DataType::BigInt => BufferKind::I64,
DataType::TinyInt => BufferKind::I8,
DataType::Bit => BufferKind::Bit,
DataType::Varbinary { length }
| DataType::Binary { length } => BufferKind::Binary { length },
DataType::Varchar { length }
| DataType::WVarchar { length }
| DataType::WChar {length }
| DataType::Char { length } => BufferKind::Text { max_str_len : length },
| DataType::Numeric { precision: _, scale: _ }
| DataType::Decimal { precision: _, scale: _ }
| DataType::Time { precision: _ } => BufferKind::Text { max_str_len: data_type.display_size().unwrap() },
};
Some(buffer_kind)
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
#[cfg(target_pointer_width = "64")] fn bytes_per_row() {
let bpr = |kind, nullable| BufferDescription { kind, nullable }.bytes_per_row();
assert_eq!(5 + 8, bpr(BufferKind::Binary { length: 5 }, false));
assert_eq!(5 + 1 + 8, bpr(BufferKind::Text { max_str_len: 5 }, false));
assert_eq!(10 + 2 + 8, bpr(BufferKind::WText { max_str_len: 5 }, false));
assert_eq!(6, bpr(BufferKind::Date, false));
assert_eq!(6, bpr(BufferKind::Time, false));
assert_eq!(16, bpr(BufferKind::Timestamp, false));
assert_eq!(1, bpr(BufferKind::Bit, false));
assert_eq!(1 + 8, bpr(BufferKind::Bit, true));
assert_eq!(4, bpr(BufferKind::F32, false));
assert_eq!(8, bpr(BufferKind::F64, false));
assert_eq!(1, bpr(BufferKind::I8, false));
assert_eq!(2, bpr(BufferKind::I16, false));
assert_eq!(4, bpr(BufferKind::I32, false));
assert_eq!(8, bpr(BufferKind::I64, false));
assert_eq!(1, bpr(BufferKind::U8, false));
}
}