(PHP 5)
DOMDocument::schemaValidate — Validates a document based on a schema
$filename
[, int $flags
] )
Validates a document based on the given schema file.
filename
The path to the schema.
flags
A bitmask of Libxml schema validation flags. Currently the only supported value is LIBXML_SCHEMA_CREATE. Available since PHP 5.5.2 and Libxml 2.6.14.
Returns TRUE
on success or
FALSE
on failure.
Version | Description |
---|---|
5.5.2 | Added the flags parameter
|
add a note
Initial situation:
- Debian
Lenny
- Apache 2 with PHP 5.2.6
- libxml 2.6.32
Problem: While
trying to validate a manually created (!) DOMDocument against an existing XML
Schema, I ran into a warning like the one below. Validation fails, even though
the document IS valid and the command line tool xmllint confirms this (even with
libxml 2.6.32, so this must be a problem with DOM). The validation works fine
with libxml 2.7.3.
Warning: DOMDocument::schemaValidate()
[domdocument.schemavalidate]: Element 'YourRootElementName': No matching global
declaration available for the validation root. in /var/www/YourFile.php on line
X
Solution: As libxml 2.7.3 is not provided for Debian Lenny yet and this
problem seems to be caused by DOM (s.o.), I currently use the following
workaround on my machines. DOM obviously has some namespace problems with
documents, that have been created manually (i.e. they were not loaded from a
file).
So my workaround is saving the DOMDocument temporarily, re-loading
it and then validating the temporary DOMDocument. Strangely enough the
validation of the same document (= same content) now works. Sure, creating a
temporary file is not a nice solution, but unless this bug is fixed, this
workaround should do just fine.
// Works with libxml 2.7.3 and
higher.public function isValid()
{
return $this->dom->schemaValidate('schema.xsd');
}// Workaround for prior libxml versions, e.g.
2.6.32.public function isValid()
{
// Create temporary file and save manually created
DOMDocument.
$tempFile = time() .
'-' . rand() . '-document.tmp';
$this->dom->save($tempFile);
// Create temporary DOMDocument and re-load content from
file.
$tempDom = new
DOMDocument();
$tempDom->load($tempFile);
// Delete temporary file.
if
(is_file($tempFile))
{
unlink($tempFile);
}
// Validate temporary DOMDocument.
return $tempDom->schemaValidate('schema.xsd');
}?>
In older versions of PHP5 this
function might cause error messages when dealing with namespaces. I had problems
with PHP 5.2.14 which comes with libXML V2.6.16. After switching to PHP 5.3.5
with libXML V2.7.7 I no longer had problems. Took me only about 30 hrs to figure
that one out.
For more detailed feedback from
DOMDocument::schemaValidate, disable libxml errors and fetch error information
yourself. See http://php.net/manual/en/ref.libxml.php for more
info.
example.xml
example.xsd
function libxml_display_error($error)
{
$return = "
\n";
switch
($error->level) {
case LIBXML_ERR_WARNING:
$return .= "Warning $error->code: ";
break;
case LIBXML_ERR_ERROR:
$return .= "Error $error->code: ";
break;
case LIBXML_ERR_FATAL:
$return .= "Fatal Error $error->code: ";
break;
}
$return .= trim($error->message);
if ($error->file)
{
$return .=
" in $error->file";
}
$return .= " on line $error->line\n";
return $return;
}
function libxml_display_errors() {
$errors = libxml_get_errors();
foreach
($errors as $error) {
print libxml_display_error($error);
}
libxml_clear_errors();
}// Enable user error
handlinglibxml_use_internal_errors(true);$xml = new DOMDocument(); $xml->load('example.xml');
if (!$xml->schemaValidate('example.xsd')) {
print 'DOMDocument::schemaValidate() Generated
Errors!';
libxml_display_errors();
}?>
Old
error message:
Warning: DOMDocument::schemaValidate()
[function.schemaValidate]: Element 'child_integer': 'Error condition.' is not a
valid value of the atomic type 'xs:integer'. in example.php on line
40
New error message:
DOMDocument::schemaValidate() Generated
Errors!
Error 1824: Element 'child_integer': 'Error condition.' is not a
valid value of the atomic type 'xs:integer'. in example.xml on line 4
I had a tricky issue using this
method, i thought it was bug but i realized i misundestood something about
namespaces.
When you want to use a schema to describe an xml document, you
basically put the default namespace to a personnal namespace (and you refer to
this namespace in the targetNamespace attribute of your schema).
That xmlns attribute specifies a "default
namespace" and means that the root element and its children are in this
namespace.
What i misunderstood is that, there's no way with the DOM api to
specify a "default namespace" for each children of the root element.
Thus you
may need to use createElementNS() and createAttributeNS() methods for each
created element or attribute in the document specifying the URI of your
namespace each time ("http://my.uri.net").
This only applies if you want to
validate a document built with the API, not with a document loaded from an xml
file or stream.
If write any error detector from xml
file using schema and libxml function watch out on some unreseloved
bug
https://bugs.php.net/bug.php?id=54138
I've seen that trying to validate an
XML with a schema containing keys, they were ignored, and the XML was considered
still valid altough it contained duplicate keys.
In his previous comment, Mike A wrote
about validating documents using an XSD. However, you can validate without one.
In my case, I needed to ensure that the content entered was just valid XML or
not, and I couldn't find an XSD to support that. So I wrote this:
public
static function validate($xml)
{
libxml_use_internal_errors(true);
$doc = new DOMDocument('1.0',
'utf-8');
$doc->loadXML($xml);
$errors =
libxml_get_errors();
if (empty($errors))
{
return
true;
}
$error = $errors[0];
if ($error->level <
3)
{
return true;
}
$lines = explode("\r",
$xml);
$line = $lines[($error->line)-1];
$message =
$error->message.' at line '.$error->line.':
'.htmlentities($line);
return $message;
}
Copyright © 2011 - All Rights Reserved - Softron.in
Template by Softron Technology