use std::mem::size_of;
use odbc_sys::{Date, Time, Timestamp};
use crate::{Bit, DataType};
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
pub enum BufferDesc {
Binary {
length: usize,
},
Text {
max_str_len: usize,
},
WText {
max_str_len: usize,
},
F64 {
nullable: bool,
},
F32 {
nullable: bool,
},
Date {
nullable: bool,
},
Time {
nullable: bool,
},
Timestamp {
nullable: bool,
},
I8 {
nullable: bool,
},
I16 {
nullable: bool,
},
I32 {
nullable: bool,
},
I64 {
nullable: bool,
},
U8 {
nullable: bool,
},
Bit {
nullable: bool,
},
}
impl BufferDesc {
pub fn from_data_type(data_type: DataType, nullable: bool) -> Option<Self> {
let buffer_desc = match data_type {
DataType::Numeric { precision, scale }
| DataType::Decimal { precision, scale } if scale == 0 && precision < 3 => BufferDesc::I8 { nullable },
DataType::Numeric { precision, scale }
| DataType::Decimal { precision, scale } if scale == 0 && precision < 10 => BufferDesc::I32 { nullable },
DataType::Numeric { precision, scale }
| DataType::Decimal { precision, scale } if scale == 0 && precision < 19 => BufferDesc::I64 { nullable },
DataType::Integer => BufferDesc::I32 { nullable },
DataType::SmallInt => BufferDesc::I16 { nullable },
DataType::Float { precision: 0..=24 } | DataType::Real => BufferDesc::F32 { nullable },
DataType::Float { precision: 25..=53 } |DataType::Double => BufferDesc::F64 { nullable },
DataType::Date => BufferDesc::Date { nullable },
DataType::Time { precision: 0 } => BufferDesc::Time { nullable },
DataType::Timestamp { precision: _ } => BufferDesc::Timestamp { nullable },
DataType::BigInt => BufferDesc::I64 { nullable },
DataType::TinyInt => BufferDesc::I8 { nullable },
DataType::Bit => BufferDesc::Bit { nullable },
DataType::Varbinary { length }
| DataType::Binary { length }
| DataType::LongVarbinary { length } => BufferDesc::Binary { length },
DataType::Varchar { length }
| DataType::WVarchar { length }
| DataType::WChar {length }
| DataType::Char { length }
| DataType::LongVarchar { length } => BufferDesc::Text { max_str_len : length },
| DataType::Numeric { precision: _, scale: _ }
| DataType::Decimal { precision: _, scale: _ }
| DataType::Time { precision: _ } => BufferDesc::Text { max_str_len: data_type.display_size().unwrap() },
DataType::Unknown
| DataType::Float { precision: _ }
| DataType::Other { data_type: _, column_size: _, decimal_digits: _ } => return None,
};
Some(buffer_desc)
}
pub fn bytes_per_row(&self) -> usize {
let size_indicator = |nullable: bool| if nullable { size_of::<isize>() } else { 0 };
match *self {
BufferDesc::Binary { length } => length + size_indicator(true),
BufferDesc::Text { max_str_len } => max_str_len + 1 + size_indicator(true),
BufferDesc::WText { max_str_len } => (max_str_len + 1) * 2 + size_indicator(true),
BufferDesc::F64 { nullable } => size_of::<f64>() + size_indicator(nullable),
BufferDesc::F32 { nullable } => size_of::<f32>() + size_indicator(nullable),
BufferDesc::Date { nullable } => size_of::<Date>() + size_indicator(nullable),
BufferDesc::Time { nullable } => size_of::<Time>() + size_indicator(nullable),
BufferDesc::Timestamp { nullable } => size_of::<Timestamp>() + size_indicator(nullable),
BufferDesc::I8 { nullable } => size_of::<i8>() + size_indicator(nullable),
BufferDesc::I16 { nullable } => size_of::<i16>() + size_indicator(nullable),
BufferDesc::I32 { nullable } => size_of::<i32>() + size_indicator(nullable),
BufferDesc::I64 { nullable } => size_of::<i64>() + size_indicator(nullable),
BufferDesc::U8 { nullable } => size_of::<u8>() + size_indicator(nullable),
BufferDesc::Bit { nullable } => size_of::<Bit>() + size_indicator(nullable),
}
}
}
#[derive(Clone, Copy, Debug)]
pub struct BufferDescription {
pub nullable: bool,
pub kind: BufferKind,
}
impl BufferDescription {
pub fn bytes_per_row(&self) -> usize {
let conv: BufferDesc = (*self).into();
conv.bytes_per_row()
}
}
impl From<BufferDescription> for BufferDesc {
fn from(source: BufferDescription) -> Self {
let nullable = source.nullable;
match source.kind {
BufferKind::Binary { length } => BufferDesc::Binary { length },
BufferKind::Text { max_str_len } => BufferDesc::Text { max_str_len },
BufferKind::WText { max_str_len } => BufferDesc::WText { max_str_len },
BufferKind::F64 => BufferDesc::F64 { nullable },
BufferKind::F32 => BufferDesc::F32 { nullable },
BufferKind::Date => BufferDesc::Date { nullable },
BufferKind::Time => BufferDesc::Time { nullable },
BufferKind::Timestamp => BufferDesc::Timestamp { nullable },
BufferKind::I8 => BufferDesc::I8 { nullable },
BufferKind::I16 => BufferDesc::I16 { nullable },
BufferKind::I32 => BufferDesc::I32 { nullable },
BufferKind::I64 => BufferDesc::I64 { nullable },
BufferKind::U8 => BufferDesc::U8 { nullable },
BufferKind::Bit => BufferDesc::Bit { nullable },
}
}
}
#[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 {
#[deprecated = "Use BufferDesc::from_data_type instead"]
pub fn from_data_type(data_type: DataType) -> Option<Self> {
let buffer_kind = match data_type {
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 { precision: 0..=24 } | DataType::Real => BufferKind::F32,
DataType::Float { precision: 25..=53 } |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 }
| DataType::LongVarbinary { length } => BufferKind::Binary { length },
DataType::Varchar { length }
| DataType::WVarchar { length }
| DataType::WChar {length }
| DataType::Char { length }
| DataType::LongVarchar { 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() },
DataType::Unknown
| DataType::Float { precision: _ }
| DataType::Other { data_type: _, column_size: _, decimal_digits: _ } => return None,
};
Some(buffer_kind)
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
#[cfg(target_pointer_width = "64")] fn bytes_per_row() {
let bpr = |kind, nullable| BufferDescription { nullable, kind }.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));
}
}