use crate::{bitmap::Bitmap, buffer::Buffer, datatypes::DataType, error::ArrowError};
use super::Array;
mod ffi;
pub(super) mod fmt;
mod iterator;
mod mutable;
pub use mutable::*;
#[derive(Clone)]
pub struct FixedSizeBinaryArray {
size: usize, data_type: DataType,
values: Buffer<u8>,
validity: Option<Bitmap>,
}
impl FixedSizeBinaryArray {
pub fn try_new(
data_type: DataType,
values: Buffer<u8>,
validity: Option<Bitmap>,
) -> Result<Self, ArrowError> {
let size = Self::maybe_get_size(&data_type)?;
if values.len() % size != 0 {
return Err(ArrowError::oos(format!(
"values (of len {}) must be a multiple of size ({}) in FixedSizeBinaryArray.",
values.len(),
size
)));
}
let len = values.len() / size;
if validity
.as_ref()
.map_or(false, |validity| validity.len() != len)
{
return Err(ArrowError::oos(
"validity mask length must be equal to the number of values divided by size",
));
}
Ok(Self {
size,
data_type,
values,
validity,
})
}
pub fn new(data_type: DataType, values: Buffer<u8>, validity: Option<Bitmap>) -> Self {
Self::try_new(data_type, values, validity).unwrap()
}
pub fn from_data(data_type: DataType, values: Buffer<u8>, validity: Option<Bitmap>) -> Self {
Self::new(data_type, values, validity)
}
pub fn new_empty(data_type: DataType) -> Self {
Self::new(data_type, Buffer::new(), None)
}
pub fn new_null(data_type: DataType, length: usize) -> Self {
let size = Self::maybe_get_size(&data_type).unwrap();
Self::new(
data_type,
Buffer::new_zeroed(length * size),
Some(Bitmap::new_zeroed(length)),
)
}
}
impl FixedSizeBinaryArray {
#[must_use]
pub fn slice(&self, offset: usize, length: usize) -> Self {
assert!(
offset + length <= self.len(),
"the offset of the new Buffer cannot exceed the existing length"
);
unsafe { self.slice_unchecked(offset, length) }
}
#[must_use]
pub unsafe fn slice_unchecked(&self, offset: usize, length: usize) -> Self {
let validity = self
.validity
.clone()
.map(|x| x.slice_unchecked(offset, length));
let values = self
.values
.clone()
.slice_unchecked(offset * self.size as usize, length * self.size as usize);
Self {
data_type: self.data_type.clone(),
size: self.size,
values,
validity,
}
}
#[must_use]
pub fn with_validity(&self, validity: Option<Bitmap>) -> Self {
if matches!(&validity, Some(bitmap) if bitmap.len() != self.len()) {
panic!("validity should be as least as large as the array")
}
let mut arr = self.clone();
arr.validity = validity;
arr
}
}
impl FixedSizeBinaryArray {
#[inline]
pub fn len(&self) -> usize {
self.values.len() / self.size as usize
}
#[inline]
pub fn validity(&self) -> Option<&Bitmap> {
self.validity.as_ref()
}
pub fn values(&self) -> &Buffer<u8> {
&self.values
}
#[inline]
pub fn value(&self, i: usize) -> &[u8] {
assert!(i < self.len());
unsafe { self.value_unchecked(i) }
}
#[inline]
pub unsafe fn value_unchecked(&self, i: usize) -> &[u8] {
self.values
.get_unchecked(i * self.size..(i + 1) * self.size)
}
#[inline]
pub fn to(self, data_type: DataType) -> Self {
match (
data_type.to_logical_type(),
self.data_type().to_logical_type(),
) {
(DataType::FixedSizeBinary(size_a), DataType::FixedSizeBinary(size_b))
if size_a == size_b => {}
_ => panic!("Wrong DataType"),
}
Self {
size: self.size,
data_type,
values: self.values,
validity: self.validity,
}
}
pub fn size(&self) -> usize {
self.size
}
}
impl FixedSizeBinaryArray {
pub(crate) fn maybe_get_size(data_type: &DataType) -> Result<usize, ArrowError> {
match data_type.to_logical_type() {
DataType::FixedSizeBinary(size) => Ok(*size),
_ => Err(ArrowError::oos(
"FixedSizeBinaryArray expects DataType::FixedSizeBinary",
)),
}
}
pub(crate) fn get_size(data_type: &DataType) -> usize {
Self::maybe_get_size(data_type).unwrap()
}
}
impl Array for FixedSizeBinaryArray {
#[inline]
fn as_any(&self) -> &dyn std::any::Any {
self
}
#[inline]
fn len(&self) -> usize {
self.len()
}
#[inline]
fn data_type(&self) -> &DataType {
&self.data_type
}
fn validity(&self) -> Option<&Bitmap> {
self.validity.as_ref()
}
fn slice(&self, offset: usize, length: usize) -> Box<dyn Array> {
Box::new(self.slice(offset, length))
}
unsafe fn slice_unchecked(&self, offset: usize, length: usize) -> Box<dyn Array> {
Box::new(self.slice_unchecked(offset, length))
}
fn with_validity(&self, validity: Option<Bitmap>) -> Box<dyn Array> {
Box::new(self.with_validity(validity))
}
}
impl FixedSizeBinaryArray {
pub fn try_from_iter<P: AsRef<[u8]>, I: IntoIterator<Item = Option<P>>>(
iter: I,
size: usize,
) -> Result<Self, ArrowError> {
MutableFixedSizeBinaryArray::try_from_iter(iter, size).map(|x| x.into())
}
pub fn from_iter<P: AsRef<[u8]>, I: IntoIterator<Item = Option<P>>>(
iter: I,
size: usize,
) -> Self {
MutableFixedSizeBinaryArray::try_from_iter(iter, size)
.unwrap()
.into()
}
}
pub trait FixedSizeBinaryValues {
fn values(&self) -> &[u8];
fn size(&self) -> usize;
}
impl FixedSizeBinaryValues for FixedSizeBinaryArray {
#[inline]
fn values(&self) -> &[u8] {
&self.values
}
#[inline]
fn size(&self) -> usize {
self.size as usize
}
}