lance_encoding/encodings/physical/
fixed_size_binary.rsuse std::sync::Arc;
use arrow_buffer::ScalarBuffer;
use arrow_schema::DataType;
use futures::{future::BoxFuture, FutureExt};
use lance_core::Result;
use crate::{
buffer::LanceBuffer,
data::{BlockInfo, DataBlock, FixedWidthDataBlock, VariableWidthBlock},
decoder::{PageScheduler, PrimitivePageDecoder},
encoder::{ArrayEncoder, EncodedArray},
format::ProtobufUtils,
EncodingsIo,
};
#[derive(Debug)]
pub struct FixedSizeBinaryPageScheduler {
bytes_scheduler: Box<dyn PageScheduler>,
byte_width: u32,
bytes_per_offset: u32,
}
impl FixedSizeBinaryPageScheduler {
pub fn new(
bytes_scheduler: Box<dyn PageScheduler>,
byte_width: u32,
bytes_per_offset: u32,
) -> Self {
Self {
bytes_scheduler,
byte_width,
bytes_per_offset,
}
}
}
impl PageScheduler for FixedSizeBinaryPageScheduler {
fn schedule_ranges(
&self,
ranges: &[std::ops::Range<u64>],
scheduler: &Arc<dyn EncodingsIo>,
top_level_row: u64,
) -> BoxFuture<'static, Result<Box<dyn PrimitivePageDecoder>>> {
let expanded_ranges = ranges
.iter()
.map(|range| {
(range.start * self.byte_width as u64)..(range.end * self.byte_width as u64)
})
.collect::<Vec<_>>();
let bytes_page_decoder =
self.bytes_scheduler
.schedule_ranges(&expanded_ranges, scheduler, top_level_row);
let byte_width = self.byte_width as u64;
let bytes_per_offset = self.bytes_per_offset;
async move {
let bytes_decoder = bytes_page_decoder.await?;
Ok(Box::new(FixedSizeBinaryDecoder {
bytes_decoder,
byte_width,
bytes_per_offset,
}) as Box<dyn PrimitivePageDecoder>)
}
.boxed()
}
}
pub struct FixedSizeBinaryDecoder {
bytes_decoder: Box<dyn PrimitivePageDecoder>,
byte_width: u64,
bytes_per_offset: u32,
}
impl PrimitivePageDecoder for FixedSizeBinaryDecoder {
fn decode(&self, rows_to_skip: u64, num_rows: u64) -> Result<DataBlock> {
let rows_to_skip = rows_to_skip * self.byte_width;
let num_bytes = num_rows * self.byte_width;
let bytes = self.bytes_decoder.decode(rows_to_skip, num_bytes)?;
let bytes = bytes.as_fixed_width().unwrap();
debug_assert_eq!(bytes.bits_per_value, self.byte_width * 8);
let offsets_buffer = match self.bytes_per_offset {
8 => {
let offsets_vec = (0..(num_rows + 1))
.map(|i| i * self.byte_width)
.collect::<Vec<_>>();
ScalarBuffer::from(offsets_vec).into_inner()
}
4 => {
let offsets_vec = (0..(num_rows as u32 + 1))
.map(|i| i * self.byte_width as u32)
.collect::<Vec<_>>();
ScalarBuffer::from(offsets_vec).into_inner()
}
_ => panic!("Unsupported offsets type"),
};
let string_data = DataBlock::VariableWidth(VariableWidthBlock {
bits_per_offset: (self.bytes_per_offset * 8) as u8,
data: bytes.data,
num_values: num_rows,
offsets: LanceBuffer::from(offsets_buffer),
block_info: BlockInfo::new(),
});
Ok(string_data)
}
}
#[derive(Debug)]
pub struct FixedSizeBinaryEncoder {
bytes_encoder: Box<dyn ArrayEncoder>,
byte_width: usize,
}
impl FixedSizeBinaryEncoder {
pub fn new(bytes_encoder: Box<dyn ArrayEncoder>, byte_width: usize) -> Self {
Self {
bytes_encoder,
byte_width,
}
}
}
impl ArrayEncoder for FixedSizeBinaryEncoder {
fn encode(
&self,
data: DataBlock,
_data_type: &DataType,
buffer_index: &mut u32,
) -> Result<EncodedArray> {
let bytes_data = data.as_variable_width().unwrap();
let fixed_data = DataBlock::FixedWidth(FixedWidthDataBlock {
bits_per_value: 8 * self.byte_width as u64,
data: bytes_data.data,
num_values: bytes_data.num_values,
block_info: BlockInfo::new(),
});
let encoded_data = self.bytes_encoder.encode(
fixed_data,
&DataType::FixedSizeBinary(self.byte_width as i32),
buffer_index,
)?;
let encoding =
ProtobufUtils::fixed_size_binary(encoded_data.encoding, self.byte_width as u32);
Ok(EncodedArray {
data: encoded_data.data,
encoding,
})
}
}
#[cfg(test)]
mod tests {
use std::{collections::HashMap, sync::Arc};
use arrow::array::LargeStringBuilder;
use arrow_array::{Array, ArrayRef, FixedSizeBinaryArray, LargeStringArray, StringArray};
use arrow_buffer::Buffer;
use arrow_data::ArrayData;
use arrow_schema::{DataType, Field};
use crate::data::{DataBlock, FixedWidthDataBlock};
use crate::decoder::PrimitivePageDecoder;
use crate::encodings::physical::fixed_size_binary::FixedSizeBinaryDecoder;
use crate::{
testing::{check_round_trip_encoding_of_data, check_round_trip_encoding_random, TestCases},
version::LanceFileVersion,
};
#[test_log::test(tokio::test)]
async fn test_fixed_size_utf8_binary() {
let field = Field::new("", DataType::Utf8, false);
check_round_trip_encoding_random(field, LanceFileVersion::V2_0).await;
}
#[test_log::test(tokio::test)]
async fn test_fixed_size_binary() {
let field = Field::new("", DataType::Binary, false);
check_round_trip_encoding_random(field, LanceFileVersion::V2_0).await;
}
#[test_log::test(tokio::test)]
async fn test_fixed_size_large_binary() {
let field = Field::new("", DataType::LargeBinary, true);
check_round_trip_encoding_random(field, LanceFileVersion::V2_0).await;
}
#[test_log::test(tokio::test)]
async fn test_fixed_size_large_utf8() {
let field = Field::new("", DataType::LargeUtf8, true);
check_round_trip_encoding_random(field, LanceFileVersion::V2_0).await;
}
#[test_log::test(tokio::test)]
async fn test_simple_fixed_size_utf8() {
let string_array = StringArray::from(vec![
Some("abc"),
Some("def"),
Some("ghi"),
Some("jkl"),
Some("mno"),
]);
let test_cases = TestCases::default()
.with_range(0..2)
.with_range(0..3)
.with_range(1..3)
.with_indices(vec![0, 1, 3, 4]);
check_round_trip_encoding_of_data(
vec![Arc::new(string_array)],
&test_cases,
HashMap::new(),
)
.await;
}
#[test_log::test(tokio::test)]
async fn test_simple_fixed_size_with_nulls_utf8() {
let string_array =
LargeStringArray::from(vec![Some("abc"), None, Some("ghi"), None, Some("mno")]);
let test_cases = TestCases::default()
.with_range(0..2)
.with_range(0..3)
.with_range(1..3)
.with_indices(vec![0, 1, 3, 4]);
check_round_trip_encoding_of_data(
vec![Arc::new(string_array)],
&test_cases,
HashMap::new(),
)
.await;
}
#[test_log::test(tokio::test)]
async fn test_fixed_size_sliced_utf8() {
let string_array = StringArray::from(vec![Some("abc"), Some("def"), None, Some("fgh")]);
let string_array = string_array.slice(1, 3);
let test_cases = TestCases::default()
.with_range(0..1)
.with_range(0..2)
.with_range(1..2);
check_round_trip_encoding_of_data(
vec![Arc::new(string_array)],
&test_cases,
HashMap::new(),
)
.await;
}
#[test_log::test(tokio::test)]
async fn test_fixed_size_empty_strings() {
let string_array = Arc::new(StringArray::from(vec![Some(""), None, Some("")]));
let test_cases = TestCases::default().with_range(0..2).with_indices(vec![1]);
check_round_trip_encoding_of_data(vec![string_array.clone()], &test_cases, HashMap::new())
.await;
let test_cases = test_cases.with_batch_size(1);
check_round_trip_encoding_of_data(vec![string_array], &test_cases, HashMap::new()).await;
}
#[test_log::test(tokio::test)]
#[ignore] async fn test_jumbo_string() {
let mut string_builder = LargeStringBuilder::new();
let giant_string = String::from_iter((0..(1024 * 1024)).map(|_| '0'));
for _ in 0..5000 {
string_builder.append_option(Some(&giant_string));
}
let giant_array = Arc::new(string_builder.finish()) as ArrayRef;
let arrs = vec![giant_array];
let test_cases = TestCases::default().without_validation();
check_round_trip_encoding_of_data(arrs, &test_cases, HashMap::new()).await;
}
struct FixedWidthCloningPageDecoder {
data_block: FixedWidthDataBlock,
}
impl PrimitivePageDecoder for FixedWidthCloningPageDecoder {
fn decode(
&self,
_rows_to_skip: u64,
_num_rows: u64,
) -> lance_core::error::Result<DataBlock> {
Ok(DataBlock::FixedWidth(FixedWidthDataBlock {
data: self.data_block.data.deep_copy(),
bits_per_value: self.data_block.bits_per_value,
num_values: self.data_block.num_values,
block_info: self.data_block.block_info.clone(),
}))
}
}
#[test]
fn test_fixed_size_binary_decoder() {
let values: [u8; 6] = *b"aaabbb";
let num_values = 2u64;
let byte_width = 3;
let array_data = ArrayData::builder(DataType::FixedSizeBinary(byte_width))
.len(num_values as usize)
.add_buffer(Buffer::from(&values[..]))
.build()
.unwrap();
let fixed_size_binary_array = FixedSizeBinaryArray::from(array_data);
let arrays = vec![Arc::new(fixed_size_binary_array) as ArrayRef];
let fixed_width_data_block = DataBlock::from_arrays(&arrays, num_values);
assert_eq!(fixed_width_data_block.name(), "FixedWidth");
let bytes_decoder = FixedWidthCloningPageDecoder {
data_block: fixed_width_data_block.as_fixed_width().unwrap(),
};
let decoder = FixedSizeBinaryDecoder {
bytes_decoder: Box::new(bytes_decoder),
byte_width: byte_width as u64,
bytes_per_offset: 4, };
let decoded_binary = decoder.decode(0, num_values).unwrap();
let maybe_data = decoded_binary.into_arrow(DataType::Utf8, true);
assert!(maybe_data.is_ok());
let data = maybe_data.unwrap();
let string_array = StringArray::from(data);
assert_eq!(string_array.len(), num_values as usize);
assert_eq!(string_array.value(0), "aaa");
assert_eq!(string_array.value(1), "bbb");
}
}