tags:

views:

1282

answers:

5

I'm defining a user element with XSD. For this example, a user has a name, email and one or more nationalities. I've tried:

<xs:all>
  <xs:element name="name" blabla />
  <xs:element name="email" blabla />
  <xs:element name="nationality" minOccurs="1" maxOccurs="unbounded" />
</xs:all>

However, that is illegal. Apparently elements inside an "All" can only occur one time (or not at all). I could fix this by changing the All to a Sequence, but then people would have to enter the properties in the exact order, which I actually don't care about.

Is there a combination of these two available? Not according to http://www.w3schools.com/Schema/schema_complex_indicators.asp, but maybe it's hidden (or my inexperienced eyes don't see it).

By intuition, I also tried:

<xs:all>
  <xs:element name="name" blabla />
  <xs:element name="email" blabla />
  <xs:sequence>
    <xs:element name="nationality" minOccurs="1" maxOccurs="unbounded" />
  </xs:sequence>
</xs:all>

But that's unfortunately invalid.


Here is the current, real, piece of XSD:

  <!-- user -->
  <xs:complexType name="user">
    <xs:sequence>
      <xs:element name="firstname" type="xs:string" minOccurs="1" maxOccurs="1" />
      <xs:element name="appendix" type="xs:string" minOccurs="0" maxOccurs="1" />
      <xs:element name="lastname" type="xs:string" minOccurs="1" maxOccurs="1" />
      <xs:element name="address" type="xs:string" minOccurs="1" maxOccurs="1" />
      <xs:element name="zipcode" type="xs:string" minOccurs="1" maxOccurs="1" />
      <xs:element name="city" type="xs:string" minOccurs="1" maxOccurs="1"/>
      <xs:element name="username" type="xs:string" minOccurs="1" maxOccurs="1"/>
      <xs:element name="email" type="xs:string" minOccurs="1" maxOccurs="1"/>
   <xs:element name="country" type="country" minOccurs="1" maxOccurs="1"/>
   <xs:element name="nationality" type="xs:string" minOccurs="1" maxOccurs="unbounded"/>
    </xs:sequence>
  </xs:complexType>
+4  A: 

Could you just turn your "nationality" thingie into its own complexType and then use that new complex type inside your xs:all?

<xs:complexType name="NationalityType">
  <xs:sequence>   
    <xs:element name="nationality" minOccurs="1" maxOccurs="unbounded" />
  </xs:sequence>
</xs:complexType>

<xs:all>
  <xs:element name="name" blabla />
  <xs:element name="email" blabla />
  <xs:element name="nationalities" type="NationalityType" />
</xs:all>

I don't have anything at hand to test this, so this is really just off the top of my head..... give it a try!

EDIT: tested it by now - it works, the only minor price to pay is that your XML will have to look something like this:

<....>
  <email>......</email>
  <nationalities>
    <nationality>ABC</nationality>
    <nationality>CDE</nationality>
  </nationalities>
  <name>.....</name>
</.....>

So you get an extra node that will contain the arbitrary long list of <nationality> items.

Marc

marc_s
A: 

Or, since "USER" will be set up with multiple child elements, why not set it up as a complex type? Something like this should work.

<xs:complexType>
  <xs:sequence>
    <xs:element name="Name" type="xs:string" />
    <xs:element name="Password" type="xs:string" />
    <xs:element minOccurs="1" maxOccurs="unbounded" name="Nationality" type="xs:string" />
  </xs:sequence>
</xs:complexType>
AllenG
It already is a complex type. I've added the real XSD to the question, so you can see it too :)
Bart van Heukelom
A: 

xs:choice wont work? If not, just wrap that in a sequence or vice versa.

leppie
Huh? How would that help? Please give an example.
Jeremy Stein
A: 

Your code should be valid in XSD 1.1. For XSD 1.0 you have to find a workaround.

jelovirt
A: 

I think what you're looking for would go against the intent of XML. It would seems strange to have a valid XML fragment like this:

<user>
  <nationality/>
  <name/>
  <nationality/>
  <email/>
  <nationality/>
</user>

It sounds like your asking for something like what marc_s proposed.

Jeremy Stein