xs: key, warum ist vorbei Validierung, wenn Schlüsselwert nicht Mitglied der Schlüsselreferenz ist?
-
30-09-2019 - |
Frage
Ich bin daran interessiert, eine Schlüsselbedingung in meiner Xsd zu definieren. Es ist mein Verständnis, dass xs:key
verwenden, sollten Sie den Wert auf ein Mitglied eines referenzierten verwendet beschränken Werteliste.
Unter der Annahme, wir die Probe Xsd verwenden,
<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>
XML-Beispiel,
<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>
und, sagen wir, einige einfache Validierung
[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 ()) { }
}
, wie es ist, Validierung erfolgreich ist immer noch, wenn ich schlechte Werte für A/part[@ref-number]
verwenden?
<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>
Sie können eine oder alle Xsd, XML oder Validierung oben nicht richtig? Oder habe ich den beabsichtigten Zweck xs:key
falsch verstanden?
Lösung
Wie üblich, nach einer erholsamen Nacht und einem frischen Look, entdeckte nicht weniger als 2 Fehler in dieser Übung.
- Erster Fehler, Identität Validierung Einschränkungen ist ein explizites Verfahren, induziert über
XmlSchemaValidationFlags.ProcessIdentityConstraints
Satz vonXmlReaderSettings.ValidationFlag
und - zweiter Fehler, Msdn Probe enthält einen Fehler im Schema,
<xs:selector xpath="part"/>
sollte<xs:selector xpath="r:part"/>
lesen.
Vollarbeitsprobe wird wie folgt
<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>
XML-Beispiel
<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>
einfache Prüfung
[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()) { }
}
Andere Tipps
Ich denke, man sollte Validation verwenden:
settings.ValidationEventHandler += new ValidationEventHandler (ValidationCallBack);
private static void ValidationCallBack (object sender, ValidationEventArgs args) {}