vortex_array/arrays/varbinview/compute/
cast.rs1use std::sync::Arc;
5
6use vortex_error::VortexResult;
7
8use crate::ArrayRef;
9use crate::ExecutionCtx;
10use crate::IntoArray;
11use crate::array::ArrayView;
12use crate::arrays::VarBinView;
13use crate::arrays::VarBinViewArray;
14use crate::dtype::DType;
15use crate::scalar_fn::fns::cast::CastKernel;
16use crate::scalar_fn::fns::cast::CastReduce;
17use crate::validity::Validity;
18
19fn build_with_validity(
20 array: ArrayView<'_, VarBinView>,
21 new_dtype: DType,
22 new_validity: Validity,
23) -> ArrayRef {
24 unsafe {
26 VarBinViewArray::new_handle_unchecked(
27 array.views_handle().clone(),
28 Arc::clone(array.data_buffers()),
29 new_dtype,
30 new_validity,
31 )
32 .into_array()
33 }
34}
35
36impl CastReduce for VarBinView {
37 fn cast(array: ArrayView<'_, VarBinView>, dtype: &DType) -> VortexResult<Option<ArrayRef>> {
38 if !array.dtype().eq_ignore_nullability(dtype) {
39 return Ok(None);
40 }
41
42 let new_nullability = dtype.nullability();
43 let Some(new_validity) = array
44 .validity()?
45 .trivially_cast_nullability(new_nullability, array.len())?
46 else {
47 return Ok(None);
48 };
49 let new_dtype = array.dtype().with_nullability(new_nullability);
50 Ok(Some(build_with_validity(array, new_dtype, new_validity)))
51 }
52}
53
54impl CastKernel for VarBinView {
55 fn cast(
56 array: ArrayView<'_, VarBinView>,
57 dtype: &DType,
58 ctx: &mut ExecutionCtx,
59 ) -> VortexResult<Option<ArrayRef>> {
60 if !array.dtype().eq_ignore_nullability(dtype) {
61 return Ok(None);
62 }
63
64 let new_nullability = dtype.nullability();
65 let new_validity = array
66 .validity()?
67 .cast_nullability(new_nullability, array.len(), ctx)?;
68 let new_dtype = array.dtype().with_nullability(new_nullability);
69 Ok(Some(build_with_validity(array, new_dtype, new_validity)))
70 }
71}
72
73#[cfg(test)]
74mod tests {
75 use std::sync::LazyLock;
76
77 use rstest::rstest;
78 use vortex_session::VortexSession;
79
80 use crate::Canonical;
81 use crate::IntoArray;
82 use crate::VortexSessionExecute;
83 use crate::arrays::VarBinViewArray;
84 use crate::builtins::ArrayBuiltins;
85 use crate::compute::conformance::cast::test_cast_conformance;
86 use crate::dtype::DType;
87 use crate::dtype::Nullability;
88
89 static SESSION: LazyLock<VortexSession> = LazyLock::new(crate::array_session);
90
91 #[rstest]
92 #[case(
93 DType::Utf8(Nullability::Nullable),
94 DType::Utf8(Nullability::NonNullable)
95 )]
96 #[case(
97 DType::Binary(Nullability::Nullable),
98 DType::Binary(Nullability::NonNullable)
99 )]
100 #[case(
101 DType::Utf8(Nullability::NonNullable),
102 DType::Utf8(Nullability::Nullable)
103 )]
104 #[case(
105 DType::Binary(Nullability::NonNullable),
106 DType::Binary(Nullability::Nullable)
107 )]
108 fn try_cast_varbin_nullable(#[case] source: DType, #[case] target: DType) {
109 let varbin = VarBinViewArray::from_iter(vec![Some("a"), Some("b"), Some("c")], source);
110
111 let res = varbin.into_array().cast(target.clone());
112 assert_eq!(res.unwrap().dtype(), &target);
113 }
114
115 #[rstest]
116 #[case(DType::Utf8(Nullability::Nullable))]
117 #[case(DType::Binary(Nullability::Nullable))]
118 fn try_cast_varbin_fail(#[case] source: DType) {
119 let non_nullable_source = source.as_nonnullable();
121 let varbin = VarBinViewArray::from_iter(vec![Some("a"), Some("b"), None], source);
122 let mut ctx = SESSION.create_execution_ctx();
123 let result = varbin
124 .into_array()
125 .cast(non_nullable_source)
126 .and_then(|a| a.execute::<Canonical>(&mut ctx).map(|c| c.into_array()));
127 assert!(result.is_err(), "Expected error, got: {result:?}");
128 }
129
130 #[rstest]
131 #[case(VarBinViewArray::from_iter(vec![Some("hello"), Some("world"), Some("test")], DType::Utf8(Nullability::NonNullable)))]
132 #[case(VarBinViewArray::from_iter(vec![Some("hello"), None, Some("world")], DType::Utf8(Nullability::Nullable)))]
133 #[case(VarBinViewArray::from_iter(vec![Some(b"binary".as_slice()), Some(b"data".as_slice())], DType::Binary(Nullability::NonNullable)))]
134 #[case(VarBinViewArray::from_iter(vec![Some(b"test".as_slice()), None], DType::Binary(Nullability::Nullable)))]
135 #[case(VarBinViewArray::from_iter(vec![Some("single")], DType::Utf8(Nullability::NonNullable)))]
136 #[case(VarBinViewArray::from_iter(vec![Some("very long string that exceeds the inline size to test view functionality with multiple buffers")], DType::Utf8(Nullability::NonNullable)))]
137 fn test_cast_varbinview_conformance(#[case] array: VarBinViewArray) {
138 test_cast_conformance(&array.into_array());
139 }
140}