XSD ключ / Keyref начинающий вопрос

Я пытаюсь реализовать очень простое ограничение схемы XML.

Атрибут idref для элементов типа должен иметь только значение, соответствующее атрибуту id хотя бы для одного элемента .

Если это не имеет для вас никакого смысла, тогда, пожалуйста, просто посмотрите на приведенный ниже пример XML-документа, я думаю, что он действительно объясняет это лучше, чем моя попытка выразить это словами.

Итак, вопрос: почему xmllint пропускает приведенную ниже комбинацию схемы / XML (она говорит, что документ действителен)? Как это исправить, чтобы достичь желаемого ограничения?

Схема:

<?xml version="1.0"?>
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema"
    targetNamespace="test" xmlns="test" elementFormDefault="qualified">

    <xs:element name="foo">
        <xs:complexType>
            <xs:sequence>
                <xs:element name="bar" minOccurs="0" maxOccurs="unbounded">
                    <xs:complexType>
                        <xs:attribute name="id" use="required" type="xs:string" />
                    </xs:complexType>
                </xs:element>
                <xs:element name="batz" minOccurs="0" maxOccurs="unbounded">
                    <xs:complexType>
                        <xs:attribute name="idref" use="required" type="xs:string" />
                    </xs:complexType>
                </xs:element>
            </xs:sequence>
        </xs:complexType>

        <xs:key name="ID">
            <xs:selector xpath="./bar" />
            <xs:field xpath="@id" />
        </xs:key>

        <xs:keyref name="IDREF" refer="ID">
            <xs:selector xpath="./batz" />
            <xs:field xpath="@idref" />
        </xs:keyref>

    </xs:element>
</xs:schema>

Документ:

<?xml version="1.0"?>
<foo xmlns="test">
    <bar id="1" />
    <bar id="2" />
    <batz idref="1" /> <!-- this should succeed because <bar id="1"> exists -->
    <batz idref="3" /> <!-- this should FAIL -->
</foo>

2 ответа

Решение

Ваш XML-документ, как показано, не содержит schemaLocation. Когда документ XML не ссылается на схему или DTD, он может пройти проверку, просто будучи правильно сформированным XML. (Это однажды случилось с коллегой, использующим другой валидатор. Я думаю, что это ошибка, из-за которой валидатор по крайней мере не предупреждал о том, что отсутствует схема или DTD. Но я отвлекся.)

Во всяком случае, это должно быть что-то вроде:

<?xml version="1.0"?>
<foo
  xmlns="test" <!-- This is bad form, by the way... -->
  xsi:schemaLocation="test /path/to/schema/document"
    <bar id="1" />
    <bar id="2" />
    <batz idref="1" /> <!-- this should succeed because <bar id="1"> exists -->
    <batz idref="3" /> <!-- this should FAIL -->
</foo>

Даже с назначенным расположением схемы это не будет работать во всех парсерах.

<?xml version="1.0"?>
<foo xmlns="test"   
     xsi:schemaLocation="test test.xsd"  
     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
    <bar id="1" />
    <bar id="2" />
    <batz idref="1" /> <!-- this should succeed because <bar id="1"> exists -->
    <batz idref="3" /> <!-- this should FAIL -->
</foo>

Это также будет подтверждено, поскольку ключ не ссылается на целевое пространство имен.

Изменения, которые необходимо внести в XSD:

<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema"
    targetNamespace="test" 
    xmlns:t="test"
    xmlns="test" elementFormDefault="qualified">

А также

<xs:key name="ID">
    <xs:selector xpath="./t:bar" />
    <xs:field xpath="@id" />
</xs:key>

<xs:keyref name="IDREF" refer="ID">
    <xs:selector xpath="./t:batz" />
    <xs:field xpath="@idref" />
</xs:keyref>

Для обсуждения относительно этого поведения см. #1545101

Другие вопросы по тегам