1use std::any::Any;
5
6use vortex_buffer::BufferMut;
7use vortex_error::VortexExpect;
8use vortex_error::VortexResult;
9use vortex_error::vortex_ensure;
10use vortex_error::vortex_err;
11use vortex_error::vortex_panic;
12use vortex_mask::Mask;
13
14use crate::ArrayRef;
15use crate::IntoArray;
16use crate::ToCanonical;
17use crate::arrays::DecimalArray;
18use crate::builders::ArrayBuilder;
19use crate::builders::DEFAULT_BUILDER_CAPACITY;
20use crate::builders::LazyBitBufferBuilder;
21use crate::canonical::Canonical;
22use crate::dtype::BigCast;
23use crate::dtype::DType;
24use crate::dtype::DecimalDType;
25use crate::dtype::NativeDecimalType;
26use crate::dtype::Nullability;
27use crate::dtype::i256;
28use crate::match_each_decimal_value;
29use crate::match_each_decimal_value_type;
30use crate::scalar::DecimalValue;
31use crate::scalar::Scalar;
32
33pub struct DecimalBuilder {
39 dtype: DType,
40 values: DecimalBuffer,
41 nulls: LazyBitBufferBuilder,
42}
43
44enum DecimalBuffer {
50 I8(BufferMut<i8>),
51 I16(BufferMut<i16>),
52 I32(BufferMut<i32>),
53 I64(BufferMut<i64>),
54 I128(BufferMut<i128>),
55 I256(BufferMut<i256>),
56}
57
58macro_rules! delegate_fn {
59 ($self:expr, | $tname:ident, $buffer:ident | $body:block) => {{
60 #[allow(unused)]
61 match $self {
62 DecimalBuffer::I8(buffer) => {
63 type $tname = i8;
64 let $buffer = buffer;
65 $body
66 }
67 DecimalBuffer::I16(buffer) => {
68 type $tname = i16;
69 let $buffer = buffer;
70 $body
71 }
72 DecimalBuffer::I32(buffer) => {
73 type $tname = i32;
74 let $buffer = buffer;
75 $body
76 }
77 DecimalBuffer::I64(buffer) => {
78 type $tname = i64;
79 let $buffer = buffer;
80 $body
81 }
82 DecimalBuffer::I128(buffer) => {
83 type $tname = i128;
84 let $buffer = buffer;
85 $body
86 }
87 DecimalBuffer::I256(buffer) => {
88 type $tname = i256;
89 let $buffer = buffer;
90 $body
91 }
92 }
93 }};
94}
95
96impl DecimalBuilder {
97 pub fn new<T: NativeDecimalType>(decimal: DecimalDType, nullability: Nullability) -> Self {
99 Self::with_capacity::<T>(DEFAULT_BUILDER_CAPACITY, decimal, nullability)
100 }
101
102 pub fn with_capacity<T: NativeDecimalType>(
104 capacity: usize,
105 decimal: DecimalDType,
106 nullability: Nullability,
107 ) -> Self {
108 Self {
109 dtype: DType::Decimal(decimal, nullability),
110 values: match_each_decimal_value_type!(T::DECIMAL_TYPE, |D| {
111 DecimalBuffer::from(BufferMut::<D>::with_capacity(capacity))
112 }),
113 nulls: LazyBitBufferBuilder::new(capacity),
114 }
115 }
116
117 pub fn append_value<V: NativeDecimalType>(&mut self, value: V) {
119 self.values.push(value);
120 self.nulls.append_non_null();
121 }
122
123 pub fn append_n_values<V: NativeDecimalType>(&mut self, value: V, n: usize) {
125 self.values.push_n(value, n);
126 self.nulls.append_n_non_nulls(n);
127 }
128
129 pub fn finish_into_decimal(&mut self) -> DecimalArray {
131 let validity = self.nulls.finish_with_nullability(self.dtype.nullability());
132
133 let decimal_dtype = *self.decimal_dtype();
134
135 delegate_fn!(std::mem::take(&mut self.values), |T, values| {
136 DecimalArray::new::<T>(values.freeze(), decimal_dtype, validity)
137 })
138 }
139
140 pub fn decimal_dtype(&self) -> &DecimalDType {
142 let DType::Decimal(decimal_dtype, _) = &self.dtype else {
143 vortex_panic!("`DecimalBuilder` somehow had dtype {}", self.dtype);
144 };
145
146 decimal_dtype
147 }
148}
149
150impl ArrayBuilder for DecimalBuilder {
151 fn as_any(&self) -> &dyn Any {
152 self
153 }
154
155 fn as_any_mut(&mut self) -> &mut dyn Any {
156 self
157 }
158
159 fn dtype(&self) -> &DType {
160 &self.dtype
161 }
162
163 fn len(&self) -> usize {
164 self.values.len()
165 }
166
167 fn append_zeros(&mut self, n: usize) {
168 self.values.push_n(0, n);
169 self.nulls.append_n_non_nulls(n);
170 }
171
172 unsafe fn append_nulls_unchecked(&mut self, n: usize) {
173 self.values.push_n(0, n);
174 self.nulls.append_n_nulls(n);
175 }
176
177 fn append_scalar(&mut self, scalar: &Scalar) -> VortexResult<()> {
178 vortex_ensure!(
179 scalar.dtype() == self.dtype(),
180 "DecimalBuilder expected scalar with dtype {}, got {}",
181 self.dtype(),
182 scalar.dtype()
183 );
184
185 match scalar.as_decimal().decimal_value() {
186 None => self.append_null(),
187 Some(v) => match_each_decimal_value!(v, |dec_val| {
188 self.append_value(dec_val);
189 }),
190 }
191
192 Ok(())
193 }
194
195 unsafe fn extend_from_array_unchecked(&mut self, array: &ArrayRef) {
196 let decimal_array = array.to_decimal();
197
198 match_each_decimal_value_type!(decimal_array.values_type(), |D| {
199 self.values
202 .extend(decimal_array.buffer::<D>().iter().copied());
203 });
204
205 self.nulls.append_validity_mask(
206 decimal_array
207 .validity_mask()
208 .vortex_expect("validity_mask in extend_from_array_unchecked"),
209 );
210 }
211
212 fn reserve_exact(&mut self, additional: usize) {
213 self.values.reserve(additional);
214 self.nulls.reserve_exact(additional);
215 }
216
217 unsafe fn set_validity_unchecked(&mut self, validity: Mask) {
218 self.nulls = LazyBitBufferBuilder::new(validity.len());
219 self.nulls.append_validity_mask(validity);
220 }
221
222 fn finish(&mut self) -> ArrayRef {
223 self.finish_into_decimal().into_array()
224 }
225
226 fn finish_into_canonical(&mut self) -> Canonical {
227 Canonical::Decimal(self.finish_into_decimal())
228 }
229}
230
231impl DecimalBuffer {
232 fn push<V: NativeDecimalType>(&mut self, value: V) {
233 delegate_fn!(self, |T, buffer| {
234 buffer.push(
235 <T as BigCast>::from(value)
236 .ok_or_else(|| {
237 vortex_err!(
238 "decimal conversion failure {:?}, type: {:?} to {:?}",
239 value,
240 V::DECIMAL_TYPE,
241 T::DECIMAL_TYPE,
242 )
243 })
244 .vortex_expect("operation should succeed in builder"),
245 )
246 });
247 }
248
249 fn push_n<V: NativeDecimalType>(&mut self, value: V, n: usize) {
250 delegate_fn!(self, |T, buffer| {
251 buffer.push_n(
252 <T as BigCast>::from(value).vortex_expect("decimal conversion failure"),
253 n,
254 )
255 });
256 }
257
258 fn reserve(&mut self, additional: usize) {
259 delegate_fn!(self, |T, buffer| { buffer.reserve(additional) })
260 }
261
262 fn len(&self) -> usize {
263 delegate_fn!(self, |T, buffer| { buffer.len() })
264 }
265
266 pub fn extend<I, V: NativeDecimalType>(&mut self, iter: I)
267 where
268 I: Iterator<Item = V>,
269 {
270 delegate_fn!(self, |T, buffer| {
271 buffer.extend(
272 iter.map(|x| <T as BigCast>::from(x).vortex_expect("decimal conversion failure")),
273 )
274 })
275 }
276}
277
278macro_rules! impl_from_buffer {
279 ($T:ty, $variant:ident) => {
280 impl From<BufferMut<$T>> for DecimalBuffer {
281 fn from(buffer: BufferMut<$T>) -> Self {
282 Self::$variant(buffer)
283 }
284 }
285 };
286}
287
288impl_from_buffer!(i8, I8);
289impl_from_buffer!(i16, I16);
290impl_from_buffer!(i32, I32);
291impl_from_buffer!(i64, I64);
292impl_from_buffer!(i128, I128);
293impl_from_buffer!(i256, I256);
294
295impl Default for DecimalBuffer {
296 fn default() -> Self {
297 Self::I8(BufferMut::<i8>::empty())
298 }
299}
300
301#[cfg(test)]
302mod tests {
303 use crate::assert_arrays_eq;
304 use crate::builders::ArrayBuilder;
305 use crate::builders::DecimalBuilder;
306 use crate::builders::decimal::DecimalArray;
307 use crate::dtype::DecimalDType;
308
309 #[test]
310 fn test_mixed_extend() {
311 let values = 42i8;
312
313 let mut i8s = DecimalBuilder::new::<i8>(DecimalDType::new(2, 1), false.into());
314 for v in 0..values {
315 i8s.append_value(v);
316 }
317 let i8s = i8s.finish();
318
319 let mut i128s = DecimalBuilder::new::<i128>(DecimalDType::new(2, 1), false.into());
320 i128s.extend_from_array(&i8s);
321 let i128s = i128s.finish();
322
323 for i in 0..i8s.len() {
324 assert_eq!(i8s.scalar_at(i).unwrap(), i128s.scalar_at(i).unwrap());
325 }
326 }
327
328 #[test]
329 fn test_append_scalar() {
330 use crate::scalar::Scalar;
331
332 let mut builder = DecimalBuilder::new::<i64>(DecimalDType::new(10, 2), true.into());
334 builder.append_value(1234i64);
335 builder.append_value(5678i64);
336 builder.append_null();
337
338 let array = builder.finish();
339 let expected = DecimalArray::from_option_iter(
340 [Some(1234i64), Some(5678), None],
341 DecimalDType::new(10, 2),
342 );
343 assert_arrays_eq!(&array, &expected);
344
345 let mut builder2 = DecimalBuilder::new::<i64>(DecimalDType::new(10, 2), true.into());
347 for i in 0..array.len() {
348 let scalar = array.scalar_at(i).unwrap();
349 builder2.append_scalar(&scalar).unwrap();
350 }
351
352 let array2 = builder2.finish();
353 assert_arrays_eq!(&array2, &array);
354
355 let mut builder = DecimalBuilder::new::<i64>(DecimalDType::new(10, 2), false.into());
357 let wrong_scalar = Scalar::from(true);
358 assert!(builder.append_scalar(&wrong_scalar).is_err());
359 }
360}