From 487540c3d7e9320188e78324c31176ebbd9ef434 Mon Sep 17 00:00:00 2001 From: ZENOTME Date: Tue, 17 Oct 2023 20:48:58 +0800 Subject: [PATCH] AVRO-3886: serialize attribute in schema to support custom logical type --- lang/rust/avro/src/schema.rs | 73 ++++++++++++++++++++++++++++++++---- 1 file changed, 66 insertions(+), 7 deletions(-) diff --git a/lang/rust/avro/src/schema.rs b/lang/rust/avro/src/schema.rs index 0f49a53080d..e321342a227 100644 --- a/lang/rust/avro/src/schema.rs +++ b/lang/rust/avro/src/schema.rs @@ -1620,9 +1620,8 @@ impl Parser { } } - let name = Name::parse(complex, enclosing_namespace)?; - let fully_qualified_name = name.clone(); - let aliases = fix_aliases_namespace(complex.aliases(), &name.namespace); + let fully_qualified_name = Name::parse(complex, enclosing_namespace)?; + let aliases = fix_aliases_namespace(complex.aliases(), &fully_qualified_name.namespace); let symbols: Vec = symbols_opt .and_then(|v| v.as_array()) @@ -1753,9 +1752,8 @@ impl Parser { None => Err(Error::GetFixedSizeField), }?; - let name = Name::parse(complex, enclosing_namespace)?; - let fully_qualified_name = name.clone(); - let aliases = fix_aliases_namespace(complex.aliases(), &name.namespace); + let fully_qualified_name = Name::parse(complex, enclosing_namespace)?; + let aliases = fix_aliases_namespace(complex.aliases(), &fully_qualified_name.namespace); let schema = Schema::Fixed(FixedSchema { name: fully_qualified_name.clone(), @@ -1845,6 +1843,7 @@ impl Serialize for Schema { ref aliases, ref doc, ref fields, + ref attributes, .. }) => { let mut map = serializer.serialize_map(None)?; @@ -1860,12 +1859,16 @@ impl Serialize for Schema { map.serialize_entry("aliases", aliases)?; } map.serialize_entry("fields", fields)?; + for attr in attributes { + map.serialize_entry(attr.0, attr.1)?; + } map.end() } Schema::Enum(EnumSchema { ref name, ref symbols, ref aliases, + ref attributes, .. }) => { let mut map = serializer.serialize_map(None)?; @@ -1879,6 +1882,9 @@ impl Serialize for Schema { if let Some(ref aliases) = aliases { map.serialize_entry("aliases", aliases)?; } + for attr in attributes { + map.serialize_entry(attr.0, attr.1)?; + } map.end() } Schema::Fixed(FixedSchema { @@ -1886,7 +1892,7 @@ impl Serialize for Schema { ref doc, ref size, ref aliases, - .. + ref attributes, }) => { let mut map = serializer.serialize_map(None)?; map.serialize_entry("type", "fixed")?; @@ -1902,6 +1908,10 @@ impl Serialize for Schema { if let Some(ref aliases) = aliases { map.serialize_entry("aliases", aliases)?; } + + for attr in attributes { + map.serialize_entry(attr.0, attr.1)?; + } map.end() } Schema::Decimal(DecimalSchema { @@ -6125,4 +6135,53 @@ mod tests { Ok(()) } + + #[test] + fn avro_3886_serialize_attributes() -> TestResult { + // Test serialize enum attributes + let schema = Schema::Enum(EnumSchema { + name: Name::new("a")?, + aliases: None, + doc: None, + symbols: vec![], + default: None, + attributes: BTreeMap::from([("logicalType".into(), "time-millis".into())]), + }); + let serialized = serde_json::to_string(&schema)?; + assert_eq!( + r#"{"type":"enum","name":"a","symbols":[],"logicalType":"time-millis"}"#, + &serialized + ); + + // Test serialize fixed custom_attributes + let schema = Schema::Fixed(FixedSchema { + name: Name::new("a")?, + aliases: None, + doc: None, + size: 1, + attributes: BTreeMap::from([("logicalType".into(), "time-millis".into())]), + }); + let serialized = serde_json::to_string(&schema)?; + assert_eq!( + r#"{"type":"fixed","name":"a","size":1,"logicalType":"time-millis"}"#, + &serialized + ); + + // Test serialize record custom_attributes + let schema = Schema::Record(RecordSchema { + name: Name::new("a")?, + aliases: None, + doc: None, + fields: vec![], + lookup: BTreeMap::new(), + attributes: BTreeMap::from([("logicalType".into(), "time-millis".into())]), + }); + let serialized = serde_json::to_string(&schema)?; + assert_eq!( + r#"{"type":"record","name":"a","fields":[],"logicalType":"time-millis"}"#, + &serialized + ); + + Ok(()) + } }