Вопрос XSD/keyref для начинающих

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

Атрибут idref для элементов тип <batz> должно быть разрешено иметь значение, которое соответствует атрибут 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>

Ответ 1

Ваш 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>

Ответ 2

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

<?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