xs: ключ, почему проверка проходит, когда значение ключа не является частью ссылки на ключ?
Я заинтересован в определении ключевого ограничения в моем Xsd. Насколько я понимаю, использование xs:key
должен ограничивать значение, используемое членом ссылочного списка значений.
Предполагая, что мы используем образец 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>
образец 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>
и, скажем, некоторая простая проверка
[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 ()) { }
}
как это, проверка все еще успешно, когда я использую плохие значения для 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>
Является ли какой-либо один или все Xsd, Xml или проверка, указанные выше, неверными? Или я неправильно понял предназначение xs:key
?
2 ответа
Решение
Как обычно, после хорошего ночного отдыха и свежего взгляда, в этом упражнении замечено не менее 2 ошибок.
- Первая ошибка, проверка идентичности ограничений является явным процессом, вызванным через
XmlSchemaValidationFlags.ProcessIdentityConstraints
установить с помощьюXmlReaderSettings.ValidationFlag
, а также - вторая ошибка, образец msdn содержит ошибку в схеме,
<xs:selector xpath="part"/>
должен прочесть<xs:selector xpath="r:part"/>
,
Полный рабочий образец выглядит следующим образом:
<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
<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>
простой валидатор
[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()) { }
}
Я думаю, вы должны использовать ValidationEventHandler:
settings.ValidationEventHandler += new ValidationEventHandler (ValidationCallBack);
private static void ValidationCallBack (object sender, ValidationEventArgs args) {}