Skip to main content

parquet_variant/builder/
object.rs

1// Licensed to the Apache Software Foundation (ASF) under one
2// or more contributor license agreements.  See the NOTICE file
3// distributed with this work for additional information
4// regarding copyright ownership.  The ASF licenses this file
5// to you under the Apache License, Version 2.0 (the
6// "License"); you may not use this file except in compliance
7// with the License.  You may obtain a copy of the License at
8//
9//   http://www.apache.org/licenses/LICENSE-2.0
10//
11// Unless required by applicable law or agreed to in writing,
12// software distributed under the License is distributed on an
13// "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
14// KIND, either express or implied.  See the License for the
15// specific language governing permissions and limitations
16// under the License.
17use crate::builder::list::ListBuilder;
18use crate::builder::metadata::MetadataBuilder;
19use crate::decoder::VariantBasicType;
20use crate::{
21    BASIC_TYPE_BITS, BuilderSpecificState, ParentState, ValueBuilder, Variant, VariantBuilderExt,
22    int_size,
23};
24use arrow_schema::ArrowError;
25use indexmap::IndexMap;
26
27fn object_header<const LARGE_BIT: u8, const ID_SIZE: u8, const OFFSET_SIZE: u8>() -> u8 {
28    (LARGE_BIT << (BASIC_TYPE_BITS + 4))
29        | ((ID_SIZE - 1) << (BASIC_TYPE_BITS + 2))
30        | ((OFFSET_SIZE - 1) << BASIC_TYPE_BITS)
31        | VariantBasicType::Object as u8
32}
33
34struct ObjectHeaderWriter<const OFFSET_SIZE: u8, const ID_SIZE: u8>();
35
36impl<const OFFSET_SIZE: u8, const ID_SIZE: u8> ObjectHeaderWriter<OFFSET_SIZE, ID_SIZE> {
37    fn write(
38        dst: &mut Vec<u8>,
39        num_fields: usize,
40        field_ids: impl Iterator<Item = u32>,
41        offsets: impl Iterator<Item = usize>,
42        data_size: usize,
43    ) {
44        let is_large = num_fields > u8::MAX as usize;
45        // num_fields will consume 4 bytes when it is larger than u8::MAX
46        if is_large {
47            dst.push(object_header::<1, { ID_SIZE }, { OFFSET_SIZE }>());
48            append_packed_u32::<4>(dst, num_fields);
49        } else {
50            dst.push(object_header::<0, { ID_SIZE }, { OFFSET_SIZE }>());
51            append_packed_u32::<1>(dst, num_fields);
52        }
53
54        for id in field_ids {
55            append_packed_u32::<ID_SIZE>(dst, id as usize);
56        }
57
58        for off in offsets {
59            append_packed_u32::<OFFSET_SIZE>(dst, off);
60        }
61
62        append_packed_u32::<OFFSET_SIZE>(dst, data_size);
63    }
64}
65
66#[inline(always)]
67fn append_packed_u32<const SIZE: u8>(dest: &mut Vec<u8>, value: usize) {
68    dest.extend_from_slice(&value.to_le_bytes()[..SIZE as usize]);
69}
70
71/// A builder for creating [`Variant::Object`] values.
72///
73/// See the examples on [`VariantBuilder`] for usage.
74///
75/// [`VariantBuilder`]: crate::VariantBuilder
76#[derive(Debug)]
77pub struct ObjectBuilder<'a, S: BuilderSpecificState> {
78    parent_state: ParentState<'a, S>,
79    pub(crate) fields: IndexMap<u32, usize>, // (field_id, offset)
80    validate_unique_fields: bool,
81}
82
83impl<'a, S: BuilderSpecificState> ObjectBuilder<'a, S> {
84    /// Creates a new object builder, nested on top of the given parent state.
85    pub fn new(parent_state: ParentState<'a, S>, validate_unique_fields: bool) -> Self {
86        Self {
87            parent_state,
88            fields: IndexMap::new(),
89            validate_unique_fields,
90        }
91    }
92
93    /// Add a field with key and value to the object
94    ///
95    /// # See Also
96    /// - [`ObjectBuilder::try_insert`] for a fallible version.
97    /// - [`ObjectBuilder::with_field`] for a builder-style API.
98    ///
99    /// # Panics
100    ///
101    /// This method will panic if the variant contains duplicate field names in objects
102    /// when validation is enabled. For a fallible version, use [`ObjectBuilder::try_insert`]
103    pub fn insert<'m, 'd, T: Into<Variant<'m, 'd>>>(&mut self, key: &str, value: T) {
104        let (state, _) = self.parent_state(key).unwrap();
105        ValueBuilder::append_variant(state, value.into())
106    }
107
108    /// Add a field with key and value to the object
109    ///
110    /// # See Also
111    /// - [`ObjectBuilder::insert`] for an infallible version that panics
112    /// - [`ObjectBuilder::try_with_field`] for a builder-style API.
113    ///
114    /// # Note
115    /// Attempting to insert a duplicate field name produces an error if unique field
116    /// validation is enabled. Otherwise, the new value overwrites the previous field mapping
117    /// without erasing the old value, resulting in a larger variant
118    pub fn try_insert<'m, 'd, T: Into<Variant<'m, 'd>>>(
119        &mut self,
120        key: &str,
121        value: T,
122    ) -> Result<(), ArrowError> {
123        let (state, _) = self.parent_state(key)?;
124        ValueBuilder::try_append_variant(state, value.into())
125    }
126
127    /// Add a field with key and value to the object by copying raw bytes when possible.
128    ///
129    /// For objects and lists, this directly copies their underlying byte representation instead of
130    /// performing a logical copy, and without touching the metadata builder. For other variant
131    /// types, this falls back to the standard append behavior.
132    ///
133    /// The caller must ensure that the metadata dictionary is already built and correct for
134    /// any objects or lists being appended, but the value's new field name is handled normally.
135    ///
136    /// # Panics
137    ///
138    /// This method will panic if the variant contains duplicate field names in objects
139    /// when validation is enabled. For a fallible version, use [`ObjectBuilder::try_insert_bytes`]
140    pub fn insert_bytes<'m, 'd>(&mut self, key: &str, value: impl Into<Variant<'m, 'd>>) {
141        self.try_insert_bytes(key, value).unwrap()
142    }
143
144    /// Add a field with key and value to the object by copying raw bytes when possible.
145    ///
146    /// For objects and lists, this directly copies their underlying byte representation instead of
147    /// performing a logical copy, and without touching the metadata builder. For other variant
148    /// types, this falls back to the standard append behavior.
149    ///
150    /// The caller must ensure that the metadata dictionary is already built and correct for
151    /// any objects or lists being appended, but the value's new field name is handled normally.
152    ///
153    /// # Note
154    /// When inserting duplicate keys, the new value overwrites the previous mapping,
155    /// but the old value remains in the buffer, resulting in a larger variant
156    pub fn try_insert_bytes<'m, 'd>(
157        &mut self,
158        key: &str,
159        value: impl Into<Variant<'m, 'd>>,
160    ) -> Result<(), ArrowError> {
161        let (state, _) = self.parent_state(key)?;
162        ValueBuilder::append_variant_bytes(state, value.into());
163        Ok(())
164    }
165
166    /// Builder style API for adding a field with key and value to the object
167    ///
168    /// Same as [`ObjectBuilder::insert`], but returns `self` for chaining.
169    pub fn with_field<'m, 'd, T: Into<Variant<'m, 'd>>>(mut self, key: &str, value: T) -> Self {
170        self.insert(key, value);
171        self
172    }
173
174    /// Builder style API for adding a field with key and value to the object
175    ///
176    /// Same as [`ObjectBuilder::try_insert`], but returns `self` for chaining.
177    pub fn try_with_field<'m, 'd, T: Into<Variant<'m, 'd>>>(
178        mut self,
179        key: &str,
180        value: T,
181    ) -> Result<Self, ArrowError> {
182        self.try_insert(key, value)?;
183        Ok(self)
184    }
185
186    /// Enables validation for unique field keys when inserting into this object.
187    ///
188    /// When this is enabled, calling [`ObjectBuilder::finish`] will return an error
189    /// if any duplicate field keys were added using [`ObjectBuilder::insert`].
190    pub fn with_validate_unique_fields(mut self, validate_unique_fields: bool) -> Self {
191        self.validate_unique_fields = validate_unique_fields;
192        self
193    }
194
195    // Returns validate_unique_fields because we can no longer reference self once this method returns.
196    fn parent_state<'b>(
197        &'b mut self,
198        field_name: &str,
199    ) -> Result<(ParentState<'b, ObjectState<'b>>, bool), ArrowError> {
200        let validate_unique_fields = self.validate_unique_fields;
201        let state = ParentState::try_object(
202            self.parent_state.value_builder,
203            self.parent_state.metadata_builder,
204            &mut self.fields,
205            self.parent_state.saved_value_builder_offset,
206            field_name,
207            validate_unique_fields,
208        )?;
209        Ok((state, validate_unique_fields))
210    }
211
212    /// Returns an object builder that can be used to append a new (nested) object to this object.
213    ///
214    /// Panics if the proposed key was a duplicate
215    ///
216    /// WARNING: The builder will have no effect unless/until [`ObjectBuilder::finish`] is called.
217    pub fn new_object<'b>(&'b mut self, key: &'b str) -> ObjectBuilder<'b, ObjectState<'b>> {
218        self.try_new_object(key).unwrap()
219    }
220
221    /// Returns an object builder that can be used to append a new (nested) object to this object.
222    ///
223    /// Fails if the proposed key was a duplicate
224    ///
225    /// WARNING: The builder will have no effect unless/until [`ObjectBuilder::finish`] is called.
226    pub fn try_new_object<'b>(
227        &'b mut self,
228        key: &str,
229    ) -> Result<ObjectBuilder<'b, ObjectState<'b>>, ArrowError> {
230        let (parent_state, validate_unique_fields) = self.parent_state(key)?;
231        Ok(ObjectBuilder::new(parent_state, validate_unique_fields))
232    }
233
234    /// Returns a list builder that can be used to append a new (nested) list to this object.
235    ///
236    /// Panics if the proposed key was a duplicate
237    ///
238    /// WARNING: The builder will have no effect unless/until [`ListBuilder::finish`] is called.
239    pub fn new_list<'b>(&'b mut self, key: &str) -> ListBuilder<'b, ObjectState<'b>> {
240        self.try_new_list(key).unwrap()
241    }
242
243    /// Returns a list builder that can be used to append a new (nested) list to this object.
244    ///
245    /// Fails if the proposed key was a duplicate
246    ///
247    /// WARNING: The builder will have no effect unless/until [`ListBuilder::finish`] is called.
248    pub fn try_new_list<'b>(
249        &'b mut self,
250        key: &str,
251    ) -> Result<ListBuilder<'b, ObjectState<'b>>, ArrowError> {
252        let (parent_state, validate_unique_fields) = self.parent_state(key)?;
253        Ok(ListBuilder::new(parent_state, validate_unique_fields))
254    }
255
256    /// Finalizes this object and appends it to its parent, which otherwise remains unmodified.
257    pub fn finish(mut self) {
258        let metadata_builder = self.parent_state.metadata_builder();
259
260        self.fields.sort_by(|&field_a_id, _, &field_b_id, _| {
261            let field_a_name = metadata_builder.field_name(field_a_id as usize);
262            let field_b_name = metadata_builder.field_name(field_b_id as usize);
263            field_a_name.cmp(field_b_name)
264        });
265
266        let max_id = self.fields.iter().map(|(i, _)| *i).max().unwrap_or(0);
267        let id_size = int_size(max_id as usize);
268
269        let starting_offset = self.parent_state.saved_value_builder_offset;
270        let value_builder = self.parent_state.value_builder();
271        let current_offset = value_builder.offset();
272        // Current object starts from `object_start_offset`
273        let data_size = current_offset - starting_offset;
274        let offset_size = int_size(data_size);
275
276        let num_fields = self.fields.len();
277        let is_large = num_fields > u8::MAX as usize;
278
279        let header_size = 1 + // header byte
280            (if is_large { 4 } else { 1 }) + // num_fields
281            (num_fields * id_size as usize) + // field IDs
282            ((num_fields + 1) * offset_size as usize); // field offsets + data_size
283
284        let mut bytes_to_splice = Vec::with_capacity(header_size);
285
286        macro_rules! write_header {
287            ($offset_size:expr, $id_size:expr) => {
288                ObjectHeaderWriter::<{ $offset_size as u8 }, { $id_size as u8 }>::write(
289                    &mut bytes_to_splice,
290                    num_fields,
291                    self.fields.keys().copied(),
292                    self.fields.values().copied(),
293                    data_size,
294                )
295            };
296        }
297
298        use crate::decoder::OffsetSizeBytes::*;
299        match (offset_size, id_size) {
300            (One, One) => write_header!(One, One),
301            (One, Two) => write_header!(One, Two),
302            (One, Three) => write_header!(One, Three),
303            (One, Four) => write_header!(One, Four),
304            (Two, One) => write_header!(Two, One),
305            (Two, Two) => write_header!(Two, Two),
306            (Two, Three) => write_header!(Two, Three),
307            (Two, Four) => write_header!(Two, Four),
308            (Three, One) => write_header!(Three, One),
309            (Three, Two) => write_header!(Three, Two),
310            (Three, Three) => write_header!(Three, Three),
311            (Three, Four) => write_header!(Three, Four),
312            (Four, One) => write_header!(Four, One),
313            (Four, Two) => write_header!(Four, Two),
314            (Four, Three) => write_header!(Four, Three),
315            (Four, Four) => write_header!(Four, Four),
316        }
317
318        // Shift existing data to make room for the header
319        value_builder
320            .inner_mut()
321            .splice(starting_offset..starting_offset, bytes_to_splice);
322
323        self.parent_state.finish();
324    }
325}
326
327impl<'a, 'm, 'v, S, K, V> Extend<(K, V)> for ObjectBuilder<'a, S>
328where
329    S: BuilderSpecificState,
330    K: AsRef<str>,
331    V: Into<Variant<'m, 'v>>,
332{
333    fn extend<T: IntoIterator<Item = (K, V)>>(&mut self, iter: T) {
334        for (key, value) in iter.into_iter() {
335            self.insert(key.as_ref(), value);
336        }
337    }
338}
339
340/// Internal state for object building
341#[derive(Debug)]
342pub struct ObjectState<'a> {
343    fields: &'a mut IndexMap<u32, usize>,
344    saved_fields_size: usize,
345}
346
347// `ObjectBuilder::finish()` eagerly updates the field offsets, which we should rollback on failure.
348impl BuilderSpecificState for ObjectState<'_> {
349    fn rollback(&mut self) {
350        self.fields.truncate(self.saved_fields_size);
351    }
352}
353
354impl<'a> ParentState<'a, ObjectState<'a>> {
355    /// Creates a new instance suitable for an [`ObjectBuilder`]. The value and metadata builder state
356    /// is checkpointed and will roll back on drop, unless [`Self::finish`] is called. The new
357    /// field's name and offset are also captured eagerly and will also roll back if not finished.
358    ///
359    /// The call fails if the field name is invalid (e.g. because it duplicates an existing field).
360    pub fn try_object(
361        value_builder: &'a mut ValueBuilder,
362        metadata_builder: &'a mut dyn MetadataBuilder,
363        fields: &'a mut IndexMap<u32, usize>,
364        saved_parent_value_builder_offset: usize,
365        field_name: &str,
366        validate_unique_fields: bool,
367    ) -> Result<Self, ArrowError> {
368        // The saved_parent_buffer_offset is the buffer size as of when the parent builder was
369        // constructed. The saved_buffer_offset is the buffer size as of now (when a child builder
370        // is created). The variant field_offset entry for this field is their difference.
371        let saved_value_builder_offset = value_builder.offset();
372        let saved_fields_size = fields.len();
373        let saved_metadata_builder_dict_size = metadata_builder.num_field_names();
374        let field_id = metadata_builder.try_upsert_field_name(field_name)?;
375        let field_start = saved_value_builder_offset - saved_parent_value_builder_offset;
376        if fields.insert(field_id, field_start).is_some() && validate_unique_fields {
377            return Err(ArrowError::InvalidArgumentError(format!(
378                "Duplicate field name: {field_name}"
379            )));
380        }
381
382        let builder_state = ObjectState {
383            fields,
384            saved_fields_size,
385        };
386        Ok(Self {
387            saved_metadata_builder_dict_size,
388            saved_value_builder_offset,
389            value_builder,
390            metadata_builder,
391            builder_state,
392            finished: false,
393        })
394    }
395}
396
397/// A [`VariantBuilderExt`] that inserts a new field into a variant object.
398pub struct ObjectFieldBuilder<'o, 'v, 's, S: BuilderSpecificState> {
399    key: &'s str,
400    builder: &'o mut ObjectBuilder<'v, S>,
401}
402
403impl<'o, 'v, 's, S: BuilderSpecificState> ObjectFieldBuilder<'o, 'v, 's, S> {
404    pub fn new(key: &'s str, builder: &'o mut ObjectBuilder<'v, S>) -> Self {
405        Self { key, builder }
406    }
407}
408
409impl<S: BuilderSpecificState> VariantBuilderExt for ObjectFieldBuilder<'_, '_, '_, S> {
410    type State<'a>
411        = ObjectState<'a>
412    where
413        Self: 'a;
414
415    /// A NULL object field is interpreted as missing, so nothing gets inserted at all.
416    fn append_null(&mut self) {}
417    fn append_value<'m, 'v>(&mut self, value: impl Into<Variant<'m, 'v>>) {
418        self.builder.insert(self.key, value);
419    }
420
421    fn try_new_list(&mut self) -> Result<ListBuilder<'_, Self::State<'_>>, ArrowError> {
422        self.builder.try_new_list(self.key)
423    }
424
425    fn try_new_object(&mut self) -> Result<ObjectBuilder<'_, Self::State<'_>>, ArrowError> {
426        self.builder.try_new_object(self.key)
427    }
428}
429
430#[cfg(test)]
431mod tests {
432    use crate::{
433        ParentState, ValueBuilder, Variant, VariantBuilder, VariantMetadata,
434        WritableMetadataBuilder,
435        builder::{metadata::ReadOnlyMetadataBuilder, object::ObjectBuilder},
436        decoder::VariantBasicType,
437    };
438
439    #[test]
440    fn test_object() {
441        let mut builder = VariantBuilder::new();
442
443        builder
444            .new_object()
445            .with_field("name", "John")
446            .with_field("age", 42i8)
447            .finish();
448
449        let (metadata, value) = builder.finish();
450        assert!(!metadata.is_empty());
451        assert!(!value.is_empty());
452    }
453
454    #[test]
455    fn test_object_field_ordering() {
456        let mut builder = VariantBuilder::new();
457
458        builder
459            .new_object()
460            .with_field("zebra", "stripes")
461            .with_field("apple", "red")
462            .with_field("banana", "yellow")
463            .finish();
464
465        let (_, value) = builder.finish();
466
467        let header = value[0];
468        assert_eq!(header & 0x03, VariantBasicType::Object as u8);
469
470        let field_count = value[1] as usize;
471        assert_eq!(field_count, 3);
472
473        // Get field IDs from the object header
474        let field_ids: Vec<u8> = value[2..5].to_vec();
475
476        // apple(1), banana(2), zebra(0)
477        assert_eq!(field_ids, vec![1, 2, 0]);
478    }
479
480    #[test]
481    fn test_duplicate_fields_in_object() {
482        let mut builder = VariantBuilder::new();
483        builder
484            .new_object()
485            .with_field("name", "Ron Artest")
486            .with_field("name", "Metta World Peace") // Duplicate field
487            .finish();
488
489        let (metadata, value) = builder.finish();
490        let variant = Variant::try_new(&metadata, &value).unwrap();
491
492        let obj = variant.as_object().unwrap();
493        assert_eq!(obj.len(), 1);
494        assert_eq!(obj.field(0).unwrap(), Variant::from("Metta World Peace"));
495
496        assert_eq!(
497            vec![("name", Variant::from("Metta World Peace"))],
498            obj.iter().collect::<Vec<_>>()
499        );
500    }
501
502    #[test]
503    fn test_read_only_metadata_builder() {
504        // First create some metadata with a few field names
505        let mut default_builder = WritableMetadataBuilder::default();
506        default_builder.upsert_field_name("name");
507        default_builder.upsert_field_name("age");
508        default_builder.upsert_field_name("active");
509        default_builder.finish();
510        let metadata_bytes = default_builder.into_inner();
511
512        // Use the metadata to build new variant values
513        let metadata = VariantMetadata::try_new(&metadata_bytes).unwrap();
514        let mut metadata_builder = ReadOnlyMetadataBuilder::new(&metadata);
515        let mut value_builder = ValueBuilder::new();
516
517        {
518            let state = ParentState::variant(&mut value_builder, &mut metadata_builder);
519            let mut obj = ObjectBuilder::new(state, false);
520
521            // These should succeed because the fields exist in the metadata
522            obj.insert("name", "Alice");
523            obj.insert("age", 30i8);
524            obj.insert("active", true);
525            obj.finish();
526        }
527
528        let value = value_builder.into_inner();
529
530        // Verify the variant was built correctly
531        let variant = Variant::try_new(&metadata_bytes, &value).unwrap();
532        let obj = variant.as_object().unwrap();
533        assert_eq!(obj.get("name"), Some(Variant::from("Alice")));
534        assert_eq!(obj.get("age"), Some(Variant::Int8(30)));
535        assert_eq!(obj.get("active"), Some(Variant::from(true)));
536    }
537
538    // matthew
539    #[test]
540    fn test_append_object() {
541        let (m1, v1) = make_object();
542        let variant = Variant::new(&m1, &v1);
543
544        let mut builder = VariantBuilder::new().with_metadata(VariantMetadata::new(&m1));
545
546        builder.append_value(variant.clone());
547
548        let (metadata, value) = builder.finish();
549        assert_eq!(variant, Variant::new(&metadata, &value));
550    }
551
552    /// make an object variant with field names in reverse lexicographical order
553    fn make_object() -> (Vec<u8>, Vec<u8>) {
554        let mut builder = VariantBuilder::new();
555
556        let mut obj = builder.new_object();
557
558        obj.insert("b", true);
559        obj.insert("a", false);
560        obj.finish();
561        builder.finish()
562    }
563
564    #[test]
565    fn test_append_nested_object() {
566        let (m1, v1) = make_nested_object();
567        let variant = Variant::new(&m1, &v1);
568
569        // because we can guarantee metadata is validated through the builder
570        let mut builder = VariantBuilder::new().with_metadata(VariantMetadata::new(&m1));
571        builder.append_value(variant.clone());
572
573        let (metadata, value) = builder.finish();
574        let result_variant = Variant::new(&metadata, &value);
575
576        assert_eq!(variant, result_variant);
577    }
578
579    /// make a nested object variant
580    fn make_nested_object() -> (Vec<u8>, Vec<u8>) {
581        let mut builder = VariantBuilder::new();
582
583        {
584            let mut outer_obj = builder.new_object();
585
586            {
587                let mut inner_obj = outer_obj.new_object("b");
588                inner_obj.insert("a", "inner_value");
589                inner_obj.finish();
590            }
591
592            outer_obj.finish();
593        }
594
595        builder.finish()
596    }
597
598    #[test]
599    fn test_nested_object() {
600        /*
601        {
602            "c": {
603                "b": "a"
604            }
605        }
606
607        */
608
609        let mut builder = VariantBuilder::new();
610        {
611            let mut outer_object_builder = builder.new_object();
612            {
613                let mut inner_object_builder = outer_object_builder.new_object("c");
614                inner_object_builder.insert("b", "a");
615                inner_object_builder.finish();
616            }
617
618            outer_object_builder.finish();
619        }
620
621        let (metadata, value) = builder.finish();
622        let variant = Variant::try_new(&metadata, &value).unwrap();
623        let outer_object = variant.as_object().unwrap();
624
625        assert_eq!(outer_object.len(), 1);
626        assert_eq!(outer_object.field_name(0).unwrap(), "c");
627
628        let inner_object_variant = outer_object.field(0).unwrap();
629        let inner_object = inner_object_variant.as_object().unwrap();
630
631        assert_eq!(inner_object.len(), 1);
632        assert_eq!(inner_object.field_name(0).unwrap(), "b");
633        assert_eq!(inner_object.field(0).unwrap(), Variant::from("a"));
634    }
635
636    #[test]
637    fn test_nested_object_with_duplicate_field_names_per_object() {
638        /*
639        {
640            "c": {
641                "b": false,
642                "c": "a"
643            },
644            "b": false,
645        }
646
647        */
648
649        let mut builder = VariantBuilder::new();
650        {
651            let mut outer_object_builder = builder.new_object();
652            {
653                let mut inner_object_builder = outer_object_builder.new_object("c");
654                inner_object_builder.insert("b", false);
655                inner_object_builder.insert("c", "a");
656
657                inner_object_builder.finish();
658            }
659
660            outer_object_builder.insert("b", false);
661            outer_object_builder.finish();
662        }
663
664        let (metadata, value) = builder.finish();
665        let variant = Variant::try_new(&metadata, &value).unwrap();
666        let outer_object = variant.as_object().unwrap();
667
668        assert_eq!(outer_object.len(), 2);
669        assert_eq!(outer_object.field_name(0).unwrap(), "b");
670
671        let inner_object_variant = outer_object.field(1).unwrap();
672        let inner_object = inner_object_variant.as_object().unwrap();
673
674        assert_eq!(inner_object.len(), 2);
675        assert_eq!(inner_object.field_name(0).unwrap(), "b");
676        assert_eq!(inner_object.field(0).unwrap(), Variant::from(false));
677        assert_eq!(inner_object.field_name(1).unwrap(), "c");
678        assert_eq!(inner_object.field(1).unwrap(), Variant::from("a"));
679    }
680
681    #[test]
682    fn test_nested_object_with_heterogeneous_fields() {
683        /*
684        {
685            "a": false,
686            "c": {
687                "b": "a",
688                "c": {
689                   "aa": "bb",
690                },
691                "d": {
692                    "cc": "dd"
693                }
694            },
695            "b": true,
696            "d": {
697               "e": 1,
698               "f": [1, true],
699               "g": ["tree", false],
700            }
701        }
702        */
703
704        let mut builder = VariantBuilder::new();
705        {
706            let mut outer_object_builder = builder.new_object();
707
708            outer_object_builder.insert("a", false);
709
710            {
711                let mut inner_object_builder = outer_object_builder.new_object("c");
712                inner_object_builder.insert("b", "a");
713
714                {
715                    let mut inner_inner_object_builder = inner_object_builder.new_object("c");
716                    inner_inner_object_builder.insert("aa", "bb");
717                    inner_inner_object_builder.finish();
718                }
719
720                {
721                    let mut inner_inner_object_builder = inner_object_builder.new_object("d");
722                    inner_inner_object_builder.insert("cc", "dd");
723                    inner_inner_object_builder.finish();
724                }
725                inner_object_builder.finish();
726            }
727
728            outer_object_builder.insert("b", true);
729
730            {
731                let mut inner_object_builder = outer_object_builder.new_object("d");
732                inner_object_builder.insert("e", 1);
733                {
734                    let mut inner_list_builder = inner_object_builder.new_list("f");
735                    inner_list_builder.append_value(1);
736                    inner_list_builder.append_value(true);
737
738                    inner_list_builder.finish();
739                }
740
741                {
742                    let mut inner_list_builder = inner_object_builder.new_list("g");
743                    inner_list_builder.append_value("tree");
744                    inner_list_builder.append_value(false);
745
746                    inner_list_builder.finish();
747                }
748
749                inner_object_builder.finish();
750            }
751
752            outer_object_builder.finish();
753        }
754
755        let (metadata, value) = builder.finish();
756
757        // note, object fields are now sorted lexigraphically by field name
758        /*
759         {
760            "a": false,
761            "b": true,
762            "c": {
763                "b": "a",
764                "c": {
765                   "aa": "bb",
766                },
767                "d": {
768                    "cc": "dd"
769                }
770            },
771            "d": {
772               "e": 1,
773               "f": [1, true],
774               "g": ["tree", false],
775            }
776        }
777        */
778
779        let variant = Variant::try_new(&metadata, &value).unwrap();
780        let outer_object = variant.as_object().unwrap();
781
782        assert_eq!(outer_object.len(), 4);
783
784        assert_eq!(outer_object.field_name(0).unwrap(), "a");
785        assert_eq!(outer_object.field(0).unwrap(), Variant::from(false));
786
787        assert_eq!(outer_object.field_name(2).unwrap(), "c");
788
789        let inner_object_variant = outer_object.field(2).unwrap();
790        let inner_object = inner_object_variant.as_object().unwrap();
791
792        assert_eq!(inner_object.len(), 3);
793        assert_eq!(inner_object.field_name(0).unwrap(), "b");
794        assert_eq!(inner_object.field(0).unwrap(), Variant::from("a"));
795
796        let inner_iner_object_variant_c = inner_object.field(1).unwrap();
797        let inner_inner_object_c = inner_iner_object_variant_c.as_object().unwrap();
798        assert_eq!(inner_inner_object_c.len(), 1);
799        assert_eq!(inner_inner_object_c.field_name(0).unwrap(), "aa");
800        assert_eq!(inner_inner_object_c.field(0).unwrap(), Variant::from("bb"));
801
802        let inner_iner_object_variant_d = inner_object.field(2).unwrap();
803        let inner_inner_object_d = inner_iner_object_variant_d.as_object().unwrap();
804        assert_eq!(inner_inner_object_d.len(), 1);
805        assert_eq!(inner_inner_object_d.field_name(0).unwrap(), "cc");
806        assert_eq!(inner_inner_object_d.field(0).unwrap(), Variant::from("dd"));
807
808        assert_eq!(outer_object.field_name(1).unwrap(), "b");
809        assert_eq!(outer_object.field(1).unwrap(), Variant::from(true));
810
811        let out_object_variant_d = outer_object.field(3).unwrap();
812        let out_object_d = out_object_variant_d.as_object().unwrap();
813        assert_eq!(out_object_d.len(), 3);
814        assert_eq!("e", out_object_d.field_name(0).unwrap());
815        assert_eq!(Variant::from(1), out_object_d.field(0).unwrap());
816        assert_eq!("f", out_object_d.field_name(1).unwrap());
817
818        let first_inner_list_variant_f = out_object_d.field(1).unwrap();
819        let first_inner_list_f = first_inner_list_variant_f.as_list().unwrap();
820        assert_eq!(2, first_inner_list_f.len());
821        assert_eq!(Variant::from(1), first_inner_list_f.get(0).unwrap());
822        assert_eq!(Variant::from(true), first_inner_list_f.get(1).unwrap());
823
824        let second_inner_list_variant_g = out_object_d.field(2).unwrap();
825        let second_inner_list_g = second_inner_list_variant_g.as_list().unwrap();
826        assert_eq!(2, second_inner_list_g.len());
827        assert_eq!(Variant::from("tree"), second_inner_list_g.get(0).unwrap());
828        assert_eq!(Variant::from(false), second_inner_list_g.get(1).unwrap());
829    }
830
831    #[test]
832    fn test_object_without_unique_field_validation() {
833        let mut builder = VariantBuilder::new();
834
835        // Root object with duplicates
836        let mut obj = builder.new_object();
837        obj.insert("a", 1);
838        obj.insert("a", 2);
839        obj.finish();
840
841        // Deeply nested list structure with duplicates
842        let mut builder = VariantBuilder::new();
843        let mut outer_list = builder.new_list();
844        let mut inner_list = outer_list.new_list();
845        let mut nested_obj = inner_list.new_object();
846        nested_obj.insert("x", 1);
847        nested_obj.insert("x", 2);
848        nested_obj.new_list("x").with_value(3).finish();
849        nested_obj.new_object("x").with_field("y", 4).finish();
850        nested_obj.finish();
851        inner_list.finish();
852        outer_list.finish();
853
854        // Verify the nested object is built correctly -- the nested object "x" should have "won"
855        let (metadata, value) = builder.finish();
856        let variant = Variant::try_new(&metadata, &value).unwrap();
857        let outer_element = variant.get_list_element(0).unwrap();
858        let inner_element = outer_element.get_list_element(0).unwrap();
859        let outer_field = inner_element.get_object_field("x").unwrap();
860        let inner_field = outer_field.get_object_field("y").unwrap();
861        assert_eq!(inner_field, Variant::from(4));
862    }
863
864    #[test]
865    fn test_object_with_unique_field_validation() {
866        let mut builder = VariantBuilder::new().with_validate_unique_fields(true);
867
868        // Root-level object with duplicates
869        let result = builder
870            .new_object()
871            .with_field("a", 1)
872            .with_field("b", 2)
873            .try_with_field("a", 3);
874        assert_eq!(
875            result.unwrap_err().to_string(),
876            "Invalid argument error: Duplicate field name: a"
877        );
878
879        // Deeply nested list -> list -> object with duplicate
880        let mut outer_list = builder.new_list();
881        let mut inner_list = outer_list.new_list();
882        let mut object = inner_list.new_object().with_field("x", 1);
883        let nested_result = object.try_insert("x", 2);
884        assert_eq!(
885            nested_result.unwrap_err().to_string(),
886            "Invalid argument error: Duplicate field name: x"
887        );
888        let nested_result = object.try_new_list("x");
889        assert_eq!(
890            nested_result.unwrap_err().to_string(),
891            "Invalid argument error: Duplicate field name: x"
892        );
893
894        let nested_result = object.try_new_object("x");
895        assert_eq!(
896            nested_result.unwrap_err().to_string(),
897            "Invalid argument error: Duplicate field name: x"
898        );
899
900        drop(object);
901        inner_list.finish();
902        outer_list.finish();
903
904        // Valid object should succeed (fresh builder — one top-level value per VariantBuilder)
905        let mut builder = VariantBuilder::new().with_validate_unique_fields(true);
906        let mut list = builder.new_list();
907        let mut valid_obj = list.new_object();
908        valid_obj.insert("m", 1);
909        valid_obj.insert("n", 2);
910
911        valid_obj.finish();
912        list.finish();
913    }
914}