xs: clé, pourquoi la validation passe lorsque la valeur clé n'est pas membre de référence clé?
-
30-09-2019 - |
Question
Je suis intéressé par la définition d'une contrainte de clé dans mon XSD. Je crois comprendre que l'utilisation xs:key
devrait limiter la valeur utilisée à un membre d'une référence liste des valeurs.
Supposons que nous utilisons l'échantillon Xsd,
<xs:schema
xmlns:xs="http://www.w3.org/2001/XMLSchema"
targetNamespace="namespace1"
xmlns:r="namespace1"
elementFormDefault="qualified">
<xs:element name="root">
<xs:complexType>
<xs:sequence>
<xs:element name="A" type="r:A" maxOccurs="unbounded">
<xs:keyref name="dummy" refer="r:pNumKey">
<xs:selector xpath="part"/>
<xs:field xpath="@ref-number"/>
</xs:keyref>
</xs:element>
<xs:element name="B" type="r:B"/>
</xs:sequence>
</xs:complexType>
<xs:key name="pNumKey">
<xs:selector xpath="r:B/r:part"/>
<xs:field xpath="@key-number"/>
</xs:key>
</xs:element>
<xs:complexType name="A">
<xs:sequence>
<xs:element name="part" maxOccurs="unbounded">
<xs:complexType>
<xs:simpleContent>
<xs:extension base="xs:string">
<xs:attribute name="ref-number" type="xs:integer"/>
</xs:extension>
</xs:simpleContent>
</xs:complexType>
</xs:element>
</xs:sequence>
</xs:complexType>
<xs:complexType name="B">
<xs:sequence>
<xs:element name="part" maxOccurs="unbounded">
<xs:complexType>
<xs:simpleContent>
<xs:extension base="xs:string">
<xs:attribute name="key-number" type="xs:integer"/>
</xs:extension>
</xs:simpleContent>
</xs:complexType>
</xs:element>
</xs:sequence>
</xs:complexType>
</xs:schema>
échantillon XML,
<root xmlns="namespace1">
<A>
<!--
if the ref-number is not equal to one of the key-number,
the validation will give error
-->
<part ref-number="1"/>
</A>
<A>
<!--
if the ref-number is not equal to one of the key-number,
the validation will give error
-->
<part ref-number="2"/>
</A>
<B>
<part key-number="1"/>
<part key-number="2"/>
<part key-number="3"/>
</B>
</root>
et, disons, une simple validation
[TestMethod]
public void Test_Schema()
{
string schemaFileName = @"sampleSchema.xsd";
string xmlFileName = @"sampleXml.xml";
XmlReaderSettings settings = new XmlReaderSettings
{
ValidationType = ValidationType.Schema,
ValidationFlags =
XmlSchemaValidationFlags.ProcessInlineSchema |
XmlSchemaValidationFlags.ProcessSchemaLocation |
XmlSchemaValidationFlags.ReportValidationWarnings,
};
settings.Schemas.Add (schema);
settings.ValidationEventHandler +=
(o, e) => { throw new Exception("CRASH"); };
XmlSchema schema =
XmlSchema.Read (
File.OpenText (schemaFileName),
(o, e) => { throw new Exception ("BOOM"); });
XmlReader reader = XmlReader.Create (xmlFileName, settings);
while (reader.Read ()) { }
}
comment est-il, la validation réussit toujours quand j'utilise de mauvaises valeurs pour A/part[@ref-number]
?
<root xmlns="namespace1">
<A>
<!-- doesn't go CRASH BOOM bang! why not? :( -->
<part ref-number="5"/>
</A>
<B>
<part key-number="1"/>
<part key-number="2"/>
<part key-number="3"/>
</B>
</root>
Quelqu'un est-il un ou tous Xsd, Xml, ou la validation ci-dessus incorrect? Ou ai-je mal compris le but recherché de xs:key
?
La solution
Comme d'habitude, après une bonne nuit de repos et un nouveau regard, repéré pas moins de 2 erreurs dans cet exercice.
- erreur d'abord, la validation de contraintes d'identité est un processus explicite, induite par ensemble
XmlSchemaValidationFlags.ProcessIdentityConstraints
parXmlReaderSettings.ValidationFlag
et - seconde erreur, Exemple msdn contient une erreur dans le schéma,
<xs:selector xpath="part"/>
devrait lire<xs:selector xpath="r:part"/>
.
échantillon de travail complet est comme suit,
<xs:schema
xmlns:xs="http://www.w3.org/2001/XMLSchema"
targetNamespace="namespace1"
xmlns:r="namespace1"
elementFormDefault="qualified">
<xs:element name="root">
<xs:complexType>
<xs:sequence>
<xs:element name="A" type="r:A" maxOccurs="unbounded">
<xs:keyref name="dummy" refer="r:pNumKey">
<!-- without 'r:' below, key was not recognized, boo-urns msdn! -->
<xs:selector xpath="r:part"/>
<xs:field xpath="@ref-number"/>
</xs:keyref>
</xs:element>
<xs:element name="B" type="r:B"/>
</xs:sequence>
</xs:complexType>
<xs:key name="pNumKey">
<xs:selector xpath="r:B/r:part"/>
<xs:field xpath="@key-number"/>
</xs:key>
</xs:element>
<xs:complexType name="A">
<xs:sequence>
<xs:element name="part" maxOccurs="unbounded">
<xs:complexType>
<xs:simpleContent>
<xs:extension base="xs:string">
<xs:attribute name="ref-number" type="xs:integer"/>
</xs:extension>
</xs:simpleContent>
</xs:complexType>
</xs:element>
</xs:sequence>
</xs:complexType>
<xs:complexType name="B">
<xs:sequence>
<xs:element name="part" maxOccurs="unbounded">
<xs:complexType>
<xs:simpleContent>
<xs:extension base="xs:string">
<xs:attribute name="key-number" type="xs:integer"/>
</xs:extension>
</xs:simpleContent>
</xs:complexType>
</xs:element>
</xs:sequence>
</xs:complexType>
</xs:schema>
échantillon XML
<root xmlns="namespace1">
<A>
<!-- goes CRASH BOOM bang! failure for the win! -->
<part ref-number="5"/>
</A>
<B>
<part key-number="1"/>
<part key-number="2"/>
<part key-number="3"/>
</B>
</root>
validateur simple,
[TestMethod]
public void Test_Schema()
{
string schemaFileName = @"sampleSchema.xsd";
string xmlFileName = @"sampleXml.xml";
XmlSchema schema =
XmlSchema.Read(
File.OpenText(schemaFileName),
(o, e) => { throw new Exception("BOOM"); });
XmlReaderSettings settings = new XmlReaderSettings
{
ValidationType = ValidationType.Schema,
ValidationFlags =
XmlSchemaValidationFlags.ProcessInlineSchema |
XmlSchemaValidationFlags.ProcessSchemaLocation |
XmlSchemaValidationFlags.ReportValidationWarnings |
// d'oh! explicit flag for processing identity constraints!
XmlSchemaValidationFlags.ProcessIdentityConstraints,
};
settings.Schemas.Add(schema);
settings.ValidationEventHandler +=
(o, e) => { throw new Exception("CRASH"); };
XmlReader reader = XmlReader.Create(xmlFileName, settings);
while (reader.Read()) { }
}
Autres conseils
Je suppose que vous devez utiliser ValidationEventHandler:
settings.ValidationEventHandler += new ValidationEventHandler (ValidationCallBack);
private static void ValidationCallBack (object sender, ValidationEventArgs args) {}