xso-proc: add support for string literals for namespaces

Makes it easier to use ad-hoc.
This commit is contained in:
Jonas Schäfer 2024-06-22 09:30:45 +02:00
parent 4d1166b66d
commit 2efef5ceeb
3 changed files with 46 additions and 13 deletions

View file

@ -169,3 +169,17 @@ fn name_path_roundtrip() {
}; };
roundtrip_full::<NamePath>("<bar xmlns='urn:example:ns1'/>"); roundtrip_full::<NamePath>("<bar xmlns='urn:example:ns1'/>");
} }
#[derive(FromXml, IntoXml, PartialEq, Debug, Clone)]
#[xml(namespace = "urn:example:ns2", name = "baz")]
struct NamespaceLit;
#[test]
fn namespace_lit_roundtrip() {
#[allow(unused_imports)]
use std::{
option::Option::{None, Some},
result::Result::{Err, Ok},
};
roundtrip_full::<NamespaceLit>("<baz xmlns='urn:example:ns2'/>");
}

View file

@ -17,11 +17,34 @@ use syn::{spanned::Spanned, *};
use rxml_validation::NcName; use rxml_validation::NcName;
/// Type alias for a `#[xml(namespace = ..)]` attribute. /// Value for the `#[xml(namespace = ..)]` attribute.
/// #[derive(Debug)]
/// This may, in the future, be replaced by an enum supporting multiple pub(crate) enum NamespaceRef {
/// ways to specify a namespace. /// The XML namespace is specified as a string literal.
pub(crate) type NamespaceRef = Path; LitStr(LitStr),
/// The XML namespace is specified as a path.
Path(Path),
}
impl syn::parse::Parse for NamespaceRef {
fn parse(input: syn::parse::ParseStream<'_>) -> Result<Self> {
if input.peek(syn::LitStr) {
Ok(Self::LitStr(input.parse()?))
} else {
Ok(Self::Path(input.parse()?))
}
}
}
impl quote::ToTokens for NamespaceRef {
fn to_tokens(&self, tokens: &mut TokenStream) {
match self {
Self::LitStr(ref lit) => lit.to_tokens(tokens),
Self::Path(ref path) => path.to_tokens(tokens),
}
}
}
/// Value for the `#[xml(name = .. )]` attribute. /// Value for the `#[xml(name = .. )]` attribute.
#[derive(Debug)] #[derive(Debug)]

View file

@ -7,10 +7,8 @@ to [`macro@IntoXml`].
```rust ```rust
# use xso::FromXml; # use xso::FromXml;
static MY_NAMESPACE: &str = "urn:example";
#[derive(FromXml, Debug, PartialEq)] #[derive(FromXml, Debug, PartialEq)]
#[xml(namespace = MY_NAMESPACE, name = "foo")] #[xml(namespace = "urn:example", name = "foo")]
struct Foo; struct Foo;
let foo: Foo = xso::from_bytes(b"<foo xmlns='urn:example'/>").unwrap(); let foo: Foo = xso::from_bytes(b"<foo xmlns='urn:example'/>").unwrap();
@ -31,7 +29,7 @@ All key-value pairs interpreted by these derive macros must be wrapped in a
| Key | Value type | Description | | Key | Value type | Description |
| --- | --- | --- | | --- | --- | --- |
| `namespace` | *path* | The path to a `&'static str` which holds the XML namespace to match. | | `namespace` | *string literal* or *path* | The XML element namespace to match. If it is a *path*, it must point at a `&'static str`. |
| `name` | *string literal* or *path* | The XML element name to match. If it is a *path*, it must point at a `&'static NcNameStr`. | | `name` | *string literal* or *path* | The XML element name to match. If it is a *path*, it must point at a `&'static NcNameStr`. |
Note that the `name` value must be a valid XML element name, without colons. Note that the `name` value must be a valid XML element name, without colons.
@ -40,9 +38,8 @@ and cannot be overridden. The following will thus not compile:
```compile_fail ```compile_fail
# use xso::FromXml; # use xso::FromXml;
# static MY_NAMESPACE: &'static str = "urn:example";
#[derive(FromXml, Debug, PartialEq)] #[derive(FromXml, Debug, PartialEq)]
#[xml(namespace = MY_NAMESPACE, name = "fnord:foo")] // colon not allowed #[xml(namespace = "urn:example", name = "fnord:foo")] // colon not allowed
struct Foo; struct Foo;
``` ```
@ -53,9 +50,8 @@ work:
```compile_fail ```compile_fail
# use xso::FromXml; # use xso::FromXml;
# static MY_NAMESPACE: &str = "urn:example";
#[derive(FromXml, Debug, PartialEq)] #[derive(FromXml, Debug, PartialEq)]
#[xml(namespace = MY_NAMESPACE, name = "foo")] #[xml(namespace = "urn:example", name = "foo")]
struct Foo { struct Foo {
some_field: String, some_field: String,
} }