Я создал \ DomDocument в PHP, который я хотел бы проверить по XSD-файлу. Я зарегистрировал больше онлайн-валидатора XML-XSD, и мой XML прошел валидацию по всем из них. Что я делаю неправильно? Почему мой XML передает другие валидаторы, но не при вызове schemaValidate для самого DomDocument?
Это фрагмент кода PHP, который генерирует XML:
$xmlDoc = new \DOMDocument('1.0', 'UTF-8');
$rootElem = $xmlDoc->createElementNS('http://fip.loginet.hu', 'allatok');
/** @var RearingAnimal $animal */
foreach($this->animals as $animal){
$animalElem = $xmlDoc->createElement('allat');
$animalElem->appendChild($xmlDoc->createElement('fulszam', $animal->getEarNumber()));
$animalElem->appendChild($xmlDoc->createElement('tenyeszet', $animal->getRearingCode()));
$animalElem->appendChild($xmlDoc->createElement('szuletesi_ido', $animal->getBirthDate()));
$animalElem->appendChild($xmlDoc->createElement('fajta', $animal->getBreed()));
$animalElem->appendChild($xmlDoc->createElement('ivar', $animal->getSex()));
$rootElem->appendChild($animalElem);
}
$xmlDoc->appendChild($rootElem);
if(!$xmlDoc->schemaValidate($this->getXsdFileName())){
throw new \Exception("XML Socument validation failed!");
}
XSD:
<schema
attributeFormDefault="unqualified"elementFormDefault="qualified"targetNamespace="http://fip.loginet.hu"xmlns="http://www.w3.org/2001/XMLSchema"xmlns:tns="http://fip.loginet.hu">
<element name="allatok">
<complexType>
<sequence>
<element name="allat" minOccurs="0" maxOccurs="unbounded" type="tns:Allat"/>
</sequence>
</complexType>
</element>
<complexType name="Allat">
<sequence>
<element name="fulszam" type="string"/>
<element name="tenyeszet" type="integer"/>
<element name="szuletesi_ido" type="date"/>
<element name="fajta" type="integer"/>
<element name="ivar" type="tns:Ivar"/>
</sequence>
</complexType>
<simpleType name="Ivar">
<restriction base="string">
<enumeration value="m"/>
<enumeration value="f"/>
</restriction>
</simpleType>
И XML, который я хотел бы проверить:
<?xml version="1.0" encoding="UTF-8"?>
<allatok xmlns="http://fip.loginet.hu">
<allat>
<fulszam>HU 30966 0259 0</fulszam>
<tenyeszet>4737016</tenyeszet>
<szuletesi_ido>2016-09-03</szuletesi_ido>
<fajta>1</fajta>
<ivar>m</ivar>
</allat>
<allat>
<fulszam>HU 31342 0375 1</fulszam>
<tenyeszet>4737016</tenyeszet>
<szuletesi_ido>2016-03-21</szuletesi_ido>
<fajta>2</fajta>
<ivar>m</ivar>
</allat>
<allat>
<fulszam>HU 31342 4595 1</fulszam>
<tenyeszet>4737016</tenyeszet>
<szuletesi_ido>2016-03-21</szuletesi_ido>
<fajta>2</fajta>
<ivar>m</ivar>
</allat>
</allatok>
Ошибка, которую я получаю:
Warning: DOMDocument::schemaValidate(): Element 'allat': This element is not expected. Expected is ( {http://fip.loginet.hu}allat )
ОБНОВИТЬ:
Я понял, что предупреждение можно исправить, если я использую $xmlDoc->createElementNS
и передать пространство имен индивидуально везде, вместо того, чтобы использовать $xmlDoc->createElement
, Однако вывод обоих — одна и та же строка XML. Тем не менее xmlns
определение 'allatok'
элемент должен применяться ко всем элементам-потомкам до тех пор, пока не будет указано другое … так что я решил это, но мне все еще любопытно, может кто-нибудь объяснить это поведение?
Методы, поддерживающие пространство имен (например, DOMDocument::createElementNS()
) создать узел в предоставленном пространстве имен. Они устанавливают namespaceURI
свойство, другие методы нет.
Таким образом, узел не находится в ожидаемом пространстве имен, а XML не соответствует схеме. Вот небольшая демонстрация:
$document = new \DOMDocument();
$foo = $document->appendChild($document->createElementNS('urn:foo', 'f:foo'));
$foo->appendChild($document->createElementNS('urn:foo', 'f:bar'));
$foo->appendChild($document->createElement('f:bar'));
echo "after create\n";
foreach ($document->documentElement->childNodes as $bar) {
echo '{'.$bar->namespaceURI.'}'.$bar->localName, "\n";
}
Выход:
after create
{urn:foo}bar
{}f:bar
Используя методы, не поддерживающие пространство имен, вы можете создать DOM, который будет сериализован в одну и ту же строку XML — даже если узел был создан без пространств имен.
echo $xml = $document->saveXML();
Выход:
<?xml version="1.0"?>
<f:foo xmlns:f="urn:foo"><f:bar/><f:bar/></f:foo>
Теперь, если этот документ загружен, анализатор разрешит пространства имен и оба bar
узлы находятся в ожидаемом пространстве имен.
$document->loadXML($xml);
echo "after load\n";
foreach ($document->documentElement->childNodes as $bar) {
echo '{'.$bar->namespaceURI.'}'.$bar->localName, "\n";
}
Выход:
after load
{urn:foo}bar
{urn:foo}bar
Я предлагаю использовать методы, учитывающие пространство имен, если вы создаете XML с пространствами имен. Что касается того, почему узел не может наследовать пространство имен родителя. Он существует до этого (после создания) и получает родителя, только если вы добавляете / вставляете его.
Других решений пока нет …