use crate::array::*;
use crate::bitmap::Bitmap;
use crate::datatypes::PhysicalType;
fn validity_size(validity: Option<&Bitmap>) -> usize {
validity.as_ref().map(|b| b.as_slice().0.len()).unwrap_or(0)
}
macro_rules! dyn_binary {
($array:expr, $ty:ty, $o:ty) => {{
let array = $array.as_any().downcast_ref::<$ty>().unwrap();
let offsets = array.offsets().buffer();
let values_start = offsets[0] as usize;
let values_end = offsets[offsets.len() - 1] as usize;
values_end - values_start
+ offsets.len() * std::mem::size_of::<$o>()
+ validity_size(array.validity())
}};
}
pub fn estimated_bytes_size(array: &dyn Array) -> usize {
use PhysicalType::*;
match array.data_type().to_physical_type() {
Null => 0,
Boolean => {
let array = array.as_any().downcast_ref::<BooleanArray>().unwrap();
array.values().as_slice().0.len() + validity_size(array.validity())
}
Primitive(primitive) => with_match_primitive_type!(primitive, |$T| {
let array = array
.as_any()
.downcast_ref::<PrimitiveArray<$T>>()
.unwrap();
array.values().len() * std::mem::size_of::<$T>() + validity_size(array.validity())
}),
Binary => dyn_binary!(array, BinaryArray<i32>, i32),
FixedSizeBinary => {
let array = array
.as_any()
.downcast_ref::<FixedSizeBinaryArray>()
.unwrap();
array.values().len() + validity_size(array.validity())
}
LargeBinary => dyn_binary!(array, BinaryArray<i64>, i64),
Utf8 => dyn_binary!(array, Utf8Array<i32>, i32),
LargeUtf8 => dyn_binary!(array, Utf8Array<i64>, i64),
List => {
let array = array.as_any().downcast_ref::<ListArray<i32>>().unwrap();
estimated_bytes_size(array.values().as_ref())
+ array.offsets().len() * std::mem::size_of::<i32>()
+ validity_size(array.validity())
}
FixedSizeList => {
let array = array.as_any().downcast_ref::<FixedSizeListArray>().unwrap();
estimated_bytes_size(array.values().as_ref()) + validity_size(array.validity())
}
LargeList => {
let array = array.as_any().downcast_ref::<ListArray<i64>>().unwrap();
estimated_bytes_size(array.values().as_ref())
+ array.offsets().len() * std::mem::size_of::<i64>()
+ validity_size(array.validity())
}
Struct => {
let array = array.as_any().downcast_ref::<StructArray>().unwrap();
array
.values()
.iter()
.map(|x| x.as_ref())
.map(estimated_bytes_size)
.sum::<usize>()
+ validity_size(array.validity())
}
Union => {
let array = array.as_any().downcast_ref::<UnionArray>().unwrap();
let types = array.types().len() * std::mem::size_of::<i8>();
let offsets = array
.offsets()
.as_ref()
.map(|x| x.len() * std::mem::size_of::<i32>())
.unwrap_or_default();
let fields = array
.fields()
.iter()
.map(|x| x.as_ref())
.map(estimated_bytes_size)
.sum::<usize>();
types + offsets + fields
}
Dictionary(key_type) => match_integer_type!(key_type, |$T| {
let array = array
.as_any()
.downcast_ref::<DictionaryArray<$T>>()
.unwrap();
estimated_bytes_size(array.keys()) + estimated_bytes_size(array.values().as_ref())
}),
Map => {
let array = array.as_any().downcast_ref::<MapArray>().unwrap();
let offsets = array.offsets().len() * std::mem::size_of::<i32>();
offsets + estimated_bytes_size(array.field().as_ref()) + validity_size(array.validity())
}
}
}