From 8898a076b152c80cb0daf2af90f6f02659845ab6 Mon Sep 17 00:00:00 2001 From: Linus Kirkwood Date: Tue, 22 Aug 2023 02:27:38 +1000 Subject: [PATCH 1/3] Added support for all element. --- xsd-parser/src/parser/all.rs | 39 ++++++++++++++++++++++++++++ xsd-parser/src/parser/element.rs | 1 + xsd-parser/src/parser/mod.rs | 1 + xsd-parser/src/parser/node_parser.rs | 2 ++ 4 files changed, 43 insertions(+) create mode 100644 xsd-parser/src/parser/all.rs diff --git a/xsd-parser/src/parser/all.rs b/xsd-parser/src/parser/all.rs new file mode 100644 index 00000000..dbbd42b0 --- /dev/null +++ b/xsd-parser/src/parser/all.rs @@ -0,0 +1,39 @@ +use std::cell::RefCell; + +use roxmltree::Node; + +use crate::parser::node_parser::parse_node; +use crate::parser::types::{RsEntity, Struct, StructField, TypeModifier}; +use crate::parser::utils::{enum_to_field, get_documentation, get_parent_name}; +use crate::parser::xsd_elements::{ElementType, XsdNode}; + +pub fn parse_all(node: &Node, parent: &Node) -> RsEntity { + let name = get_parent_name(node); + RsEntity::Struct(Struct { + name: name.into(), + comment: get_documentation(parent), + subtypes: vec![], + fields: RefCell::new(elements_to_fields(node, name)), + ..Default::default() + }) +} + +fn elements_to_fields(choice: &Node, parent_name: &str) -> Vec { + choice + .children() + .filter(|n| n.is_element() && n.xsd_type() != ElementType::Annotation) + .map(|n| match parse_node(&n, choice) { + RsEntity::StructField(mut sf) => { + if sf.type_name.ends_with(parent_name) { + sf.type_modifiers.push(TypeModifier::Recursive) + } + sf + } + RsEntity::Enum(mut en) => { + en.name = format!("{}Choice", parent_name); + enum_to_field(en) + } + _ => unreachable!("\nError: {:?}\n{:?}", n, parse_node(&n, choice)), + }) + .collect() +} diff --git a/xsd-parser/src/parser/element.rs b/xsd-parser/src/parser/element.rs index 69794ab4..df840312 100644 --- a/xsd-parser/src/parser/element.rs +++ b/xsd-parser/src/parser/element.rs @@ -14,6 +14,7 @@ const SUPPORTED_CONTENT_TYPES: [ElementType; 2] = pub fn parse_element(node: &Node, parent: &Node) -> RsEntity { match parent.xsd_type() { ElementType::Schema => parse_global_element(node), + ElementType::All => parse_field_of_sequence(node, parent), ElementType::Sequence => parse_field_of_sequence(node, parent), ElementType::Choice => parse_case_of_choice(node), _ => element_default(node), diff --git a/xsd-parser/src/parser/mod.rs b/xsd-parser/src/parser/mod.rs index e4192187..e7215ad8 100644 --- a/xsd-parser/src/parser/mod.rs +++ b/xsd-parser/src/parser/mod.rs @@ -1,4 +1,5 @@ mod any; +mod all; mod any_attribute; mod attribute; mod attribute_group; diff --git a/xsd-parser/src/parser/node_parser.rs b/xsd-parser/src/parser/node_parser.rs index f1ae89cf..eab87b25 100644 --- a/xsd-parser/src/parser/node_parser.rs +++ b/xsd-parser/src/parser/node_parser.rs @@ -1,5 +1,6 @@ use roxmltree::Node; +use crate::parser::all::parse_all; use crate::parser::any::parse_any; use crate::parser::any_attribute::parse_any_attribute; use crate::parser::attribute::parse_attribute; @@ -23,6 +24,7 @@ pub fn parse_node(node: &Node, parent: &Node) -> RsEntity { use ElementType::*; match node.xsd_type() { + All => parse_all(node, parent), Any => parse_any(node), AnyAttribute => parse_any_attribute(node), Attribute => parse_attribute(node, parent), From c8d797dacf31f0e44d2c05a04ef1f95e49e00a6b Mon Sep 17 00:00:00 2001 From: Linus Kirkwood Date: Mon, 18 Mar 2024 17:41:03 +1100 Subject: [PATCH 2/3] Added tests for xs:all element --- xsd-parser/tests/all/example.xml | 8 ++++++++ xsd-parser/tests/all/expected.rs | 19 +++++++++++++++++ xsd-parser/tests/all/input.xsd | 16 +++++++++++++++ xsd-parser/tests/all/mod.rs | 35 ++++++++++++++++++++++++++++++++ xsd-parser/tests/mod.rs | 1 + 5 files changed, 79 insertions(+) create mode 100644 xsd-parser/tests/all/example.xml create mode 100644 xsd-parser/tests/all/expected.rs create mode 100644 xsd-parser/tests/all/input.xsd create mode 100644 xsd-parser/tests/all/mod.rs diff --git a/xsd-parser/tests/all/example.xml b/xsd-parser/tests/all/example.xml new file mode 100644 index 00000000..8b7cee2e --- /dev/null +++ b/xsd-parser/tests/all/example.xml @@ -0,0 +1,8 @@ + + + 111 + 222 + 333 + 444 + 555 + diff --git a/xsd-parser/tests/all/expected.rs b/xsd-parser/tests/all/expected.rs new file mode 100644 index 00000000..4534a17a --- /dev/null +++ b/xsd-parser/tests/all/expected.rs @@ -0,0 +1,19 @@ +#[derive(Default, PartialEq, Debug, YaSerialize, YaDeserialize)] +#[yaserde(prefix = "tns", namespace = "tns: http://example.com")] +pub struct FooType { + #[yaserde(prefix = "tns", rename = "Once")] + pub once: i32, + + #[yaserde(prefix = "tns", rename = "Optional")] + pub optional: Option, + + #[yaserde(prefix = "tns", rename = "OnceSpecify")] + pub once_specify: i32, + + #[yaserde(prefix = "tns", rename = "TwiceOrMore")] + pub twice_or_more: Vec, +} + +impl Validate for FooType {} + +// pub type Foo = FooType; diff --git a/xsd-parser/tests/all/input.xsd b/xsd-parser/tests/all/input.xsd new file mode 100644 index 00000000..287b105c --- /dev/null +++ b/xsd-parser/tests/all/input.xsd @@ -0,0 +1,16 @@ + + + + + + + + + + + + + diff --git a/xsd-parser/tests/all/mod.rs b/xsd-parser/tests/all/mod.rs new file mode 100644 index 00000000..dd87aa9e --- /dev/null +++ b/xsd-parser/tests/all/mod.rs @@ -0,0 +1,35 @@ +use super::utils; + +#[test] +fn deserialization_works() { + mod expected { + use xsd_parser::generator::validator::Validate; + use yaserde_derive::{YaDeserialize, YaSerialize}; + + include!("expected.rs"); + } + + let ser = include_str!("example.xml"); + + let de: expected::FooType = yaserde::de::from_str(ser).unwrap(); + + assert_eq!( + de, + expected::FooType { + once: 222, + optional: None, + once_specify: 444, + twice_or_more: vec![111, 333, 555] + } + ); +} + +#[test] +fn generator_does_not_panic() { + println!("{}", utils::generate(include_str!("input.xsd"))) +} + +#[test] +fn generator_output_has_correct_ast() { + utils::ast_test(include_str!("input.xsd"), include_str!("expected.rs")); +} diff --git a/xsd-parser/tests/mod.rs b/xsd-parser/tests/mod.rs index 6daeaa9f..0b33f849 100644 --- a/xsd-parser/tests/mod.rs +++ b/xsd-parser/tests/mod.rs @@ -1,5 +1,6 @@ #[macro_use] mod utils; +mod all; mod any; mod choice; mod complex_type; From 008e0a38afc33750a81912d853593632b294cd03 Mon Sep 17 00:00:00 2001 From: Linus Kirkwood Date: Tue, 19 Mar 2024 12:01:21 +1100 Subject: [PATCH 3/3] Formatted with rustfmt --- xsd-parser/src/parser/mod.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/xsd-parser/src/parser/mod.rs b/xsd-parser/src/parser/mod.rs index e7215ad8..0138dc3d 100644 --- a/xsd-parser/src/parser/mod.rs +++ b/xsd-parser/src/parser/mod.rs @@ -1,5 +1,5 @@ -mod any; mod all; +mod any; mod any_attribute; mod attribute; mod attribute_group;