Validación de esquemas con XPathNavigator
Con la clase XmlDocument, puede validar el contenido XML de un objeto XmlDocument de dos formas. La primera forma sirve para validar el contenido XML utilizando un objeto XmlReader de validación y, la segunda, consiste en utilizar el método Validate de la clase XmlDocument. También se puede realizar una validación de solo lectura del contenido XML utilizando la clase XPathDocument.
Validación de datos XML
La clase XmlDocument no valida un documento XML utilizando una DTD o una validación de esquema del lenguaje de definición de esquemas XML (XSD). Solo comprueba que el documento XML sea correcto.
La primera forma de validar un documento XML consiste en validarlo a medida que se carga en un objeto XmlDocument utilizando un objeto XmlReader de validación. La segunda forma consiste en validar un documento XML que previamente no tenga información de tipos utilizando el método Validate de la clase XmlDocument. En ambos casos, se pueden volver a validar los cambios realizados en el documento XML utilizando el método Validate de la clase XmlDocument.
Validación de un documento a medida que se carga
Para crear un objeto XmlReader de validación, se pasa un objeto XmlReaderSettings al método Create de la clase XmlReader que toma un objeto XmlReaderSettings como parámetro. El objeto XmlReaderSettings que se pasa como parámetro tiene la propiedad ValidationType establecida en Schema
y un esquema XML para el documento XML contenido en el objeto XmlDocument que se ha agregado a su propiedad Schemas. A continuación, el objeto XmlReader de validación se utiliza para crear el objeto XmlDocument.
En el siguiente ejemplo se valida el archivo contosoBooks.xml
a medida que se carga en el objeto XmlDocument creando el objeto XmlDocument utilizando un objeto XmlReader de validación. Dado que el documento XML es válido de acuerdo con su esquema, no se genera ninguna advertencia ni error de validación del esquema.
Imports System
Imports System.Xml
Imports System.Xml.Schema
Imports System.Xml.XPath
Class ValidatingReaderExample
Shared Sub Main(ByVal args() As String)
Try
Dim settings As XmlReaderSettings = New XmlReaderSettings()
settings.Schemas.Add("http://www.contoso.com/books", "contosoBooks.xsd")
settings.ValidationType = ValidationType.Schema
Dim reader As XmlReader = XmlReader.Create("contosoBooks.xml", settings)
Dim document As XmlDocument = New XmlDocument()
document.Load(reader)
Dim navigator As XPathNavigator = document.CreateNavigator()
Catch e As Exception
Console.WriteLine("ValidatingReaderExample.Exception: {0}", e.Message)
End Try
End Sub
Shared Sub SchemaValidationHandler(ByVal sender As Object, ByVal e As ValidationEventArgs)
Select Case e.Severity
Case XmlSeverityType.Error
Console.WriteLine("Schema Validation Error: {0}", e.Message)
Exit Sub
Case XmlSeverityType.Warning
Console.WriteLine("Schema Validation Warning: {0}", e.Message)
Exit Sub
End Select
End Sub
End Class
using System;
using System.Xml;
using System.Xml.Schema;
using System.Xml.XPath;
class ValidatingReaderExample
{
static void Main(string[] args)
{
try
{
XmlReaderSettings settings = new XmlReaderSettings();
settings.Schemas.Add("http://www.contoso.com/books", "contosoBooks.xsd");
settings.ValidationType = ValidationType.Schema;
XmlReader reader = XmlReader.Create("contosoBooks.xml", settings);
XmlDocument document = new XmlDocument();
document.Load(reader);
XPathNavigator navigator = document.CreateNavigator();
}
catch (Exception e)
{
Console.WriteLine("ValidatingReaderExample.Exception: {0}", e.Message);
}
}
static void SchemaValidationHandler(object sender, ValidationEventArgs e)
{
switch (e.Severity)
{
case XmlSeverityType.Error:
Console.WriteLine("Schema Validation Error: {0}", e.Message);
break;
case XmlSeverityType.Warning:
Console.WriteLine("Schema Validation Warning: {0}", e.Message);
break;
}
}
}
En el ejemplo se toma como entrada el archivo contosoBooks.xml
.
<?xml version="1.0" encoding="utf-8" ?>
<bookstore xmlns="http://www.contoso.com/books">
<book genre="autobiography" publicationdate="1981-03-22" ISBN="1-861003-11-0">
<title>The Autobiography of Benjamin Franklin</title>
<author>
<first-name>Benjamin</first-name>
<last-name>Franklin</last-name>
</author>
<price>8.99</price>
</book>
<book genre="novel" publicationdate="1967-11-17" ISBN="0-201-63361-2">
<title>The Confidence Man</title>
<author>
<first-name>Herman</first-name>
<last-name>Melville</last-name>
</author>
<price>11.99</price>
</book>
<book genre="philosophy" publicationdate="1991-02-15" ISBN="1-861001-57-6">
<title>The Gorgias</title>
<author>
<name>Plato</name>
</author>
<price>9.99</price>
</book>
</bookstore>
En el ejemplo también se toma como entrada el archivo contosoBooks.xsd
.
<?xml version="1.0" encoding="utf-8"?>
<xs:schema attributeFormDefault="unqualified" elementFormDefault="qualified" targetNamespace="http://www.contoso.com/books" xmlns:xs="http://www.w3.org/2001/XMLSchema">
<xs:element name="bookstore">
<xs:complexType>
<xs:sequence>
<xs:element maxOccurs="unbounded" name="book">
<xs:complexType>
<xs:sequence>
<xs:element name="title" type="xs:string" />
<xs:element name="author">
<xs:complexType>
<xs:sequence>
<xs:element minOccurs="0" name="name" type="xs:string" />
<xs:element minOccurs="0" name="first-name" type="xs:string" />
<xs:element minOccurs="0" name="last-name" type="xs:string" />
</xs:sequence>
</xs:complexType>
</xs:element>
<xs:element name="price" type="xs:decimal" />
</xs:sequence>
<xs:attribute name="genre" type="xs:string" use="required" />
<xs:attribute name="publicationdate" type="xs:date" use="required" />
<xs:attribute name="ISBN" type="xs:string" use="required" />
</xs:complexType>
</xs:element>
</xs:sequence>
</xs:complexType>
</xs:element>
</xs:schema>
En el ejemplo anterior, si un atributo o tipo de elemento no se corresponde con el tipo especificado en el esquema de validación, se producirá una XmlSchemaValidationException cuando se llame a Load. Si se establece un ValidationEventHandler en la validación de XmlReader, se llamará al ValidationEventHandler siempre que se encuentre un tipo no válido.
Se producirá una XmlSchemaException siempre que el TypedValue obtenga acceso a un atributo o elemento con invalid
establecido en XPathNavigator.
La propiedad Validity se puede utilizar para determinar si un elemento o atributo en particular es válido cuando se obtiene acceso a atributos o elementos con el XPathNavigator.
Nota
Cuando se carga un documento XML en un objeto XmlDocument con un esquema asociado que define valores predeterminados, el objeto XmlDocument trata estos valores predeterminados como si aparecieran en el documento XML. Esto significa que la propiedad IsEmptyElement siempre devuelve false
para un elemento que utilice los valores predeterminados de un esquema, incluso aunque el documento XML se hubiera escrito como un elemento vacío.
Validación de un documento con el método Validate
El método Validate de la clase XmlDocument valida el documento XML contenido en un objeto XmlDocument con los esquemas que se especifican en la propiedad XmlDocument del objeto Schemas y aumenta el conjunto de información. El resultado es que se reemplaza un documento XML que previamente no tiene información de tipos del objeto XmlDocument por un documento con información de tipos.
El objeto XmlDocument notifica los errores y advertencias de validación de esquemas utilizando el delegado ValidationEventHandler que se pasa como parámetro al método Validate.
En el siguiente ejemplo se valida el archivo contosoBooks.xml
contenido en el objeto XmlDocument con el esquema contosoBooks.xsd
contenido en la propiedad XmlDocument del objeto Schemas.
Imports System
Imports System.Xml
Imports System.Xml.Schema
Imports System.Xml.XPath
Class ValidateExample
Shared Sub Main(ByVal args() As String)
Dim document As XmlDocument = New XmlDocument()
document.Load("contosoBooks.xml")
Dim navigator As XPathNavigator = document.CreateNavigator()
document.Schemas.Add("http://www.contoso.com/books", "contosoBooks.xsd")
Dim validation As ValidationEventHandler = New ValidationEventHandler(AddressOf SchemaValidationHandler)
document.Validate(validation)
End Sub
Shared Sub SchemaValidationHandler(ByVal sender As Object, ByVal e As ValidationEventArgs)
Select Case e.Severity
Case XmlSeverityType.Error
Console.WriteLine("Schema Validation Error: {0}", e.Message)
Exit Sub
Case XmlSeverityType.Warning
Console.WriteLine("Schema Validation Warning: {0}", e.Message)
Exit Sub
End Select
End Sub
End Class
using System;
using System.Xml;
using System.Xml.Schema;
using System.Xml.XPath;
class ValidateExample
{
static void Main(string[] args)
{
XmlDocument document = new XmlDocument();
document.Load("contosoBooks.xml");
XPathNavigator navigator = document.CreateNavigator();
document.Schemas.Add("http://www.contoso.com/books", "contosoBooks.xsd");
ValidationEventHandler validation = new ValidationEventHandler(SchemaValidationHandler);
document.Validate(validation);
}
static void SchemaValidationHandler(object sender, ValidationEventArgs e)
{
switch (e.Severity)
{
case XmlSeverityType.Error:
Console.WriteLine("Schema Validation Error: {0}", e.Message);
break;
case XmlSeverityType.Warning:
Console.WriteLine("Schema Validation Warning: {0}", e.Message);
break;
}
}
}
En el ejemplo se toma como entrada el archivo contosoBooks.xml
.
<?xml version="1.0" encoding="utf-8" ?>
<bookstore xmlns="http://www.contoso.com/books">
<book genre="autobiography" publicationdate="1981-03-22" ISBN="1-861003-11-0">
<title>The Autobiography of Benjamin Franklin</title>
<author>
<first-name>Benjamin</first-name>
<last-name>Franklin</last-name>
</author>
<price>8.99</price>
</book>
<book genre="novel" publicationdate="1967-11-17" ISBN="0-201-63361-2">
<title>The Confidence Man</title>
<author>
<first-name>Herman</first-name>
<last-name>Melville</last-name>
</author>
<price>11.99</price>
</book>
<book genre="philosophy" publicationdate="1991-02-15" ISBN="1-861001-57-6">
<title>The Gorgias</title>
<author>
<name>Plato</name>
</author>
<price>9.99</price>
</book>
</bookstore>
En el ejemplo también se toma como entrada el archivo contosoBooks.xsd
.
<?xml version="1.0" encoding="utf-8"?>
<xs:schema attributeFormDefault="unqualified" elementFormDefault="qualified" targetNamespace="http://www.contoso.com/books" xmlns:xs="http://www.w3.org/2001/XMLSchema">
<xs:element name="bookstore">
<xs:complexType>
<xs:sequence>
<xs:element maxOccurs="unbounded" name="book">
<xs:complexType>
<xs:sequence>
<xs:element name="title" type="xs:string" />
<xs:element name="author">
<xs:complexType>
<xs:sequence>
<xs:element minOccurs="0" name="name" type="xs:string" />
<xs:element minOccurs="0" name="first-name" type="xs:string" />
<xs:element minOccurs="0" name="last-name" type="xs:string" />
</xs:sequence>
</xs:complexType>
</xs:element>
<xs:element name="price" type="xs:decimal" />
</xs:sequence>
<xs:attribute name="genre" type="xs:string" use="required" />
<xs:attribute name="publicationdate" type="xs:date" use="required" />
<xs:attribute name="ISBN" type="xs:string" use="required" />
</xs:complexType>
</xs:element>
</xs:sequence>
</xs:complexType>
</xs:element>
</xs:schema>
Validación de modificaciones
Después de realizar modificaciones en un documento XML, puede validarlas con el esquema del documento XML utilizando el método Validate de la clase XmlDocument.
En el siguiente ejemplo se valida el archivo contosoBooks.xml
a medida que se carga en el objeto XmlDocument creando el objeto XmlDocument utilizando un objeto XmlReader de validación. El documento XML se valida correctamente a medida que se carga sin generar ningún error ni advertencia de validación del esquema. A continuación, el ejemplo realiza dos modificaciones en el documento XML que no son válidas de acuerdo con el esquema contosoBooks.xsd
. La primera modificación inserta un elemento secundario no válido que produce un error de validación del esquema, y la segunda modificación establece el valor de un nodo con información de tipos en un valor que no es válido de acuerdo con el tipo de nodo, lo cual produce una excepción.
Imports System
Imports System.Xml
Imports System.Xml.Schema
Imports System.Xml.XPath
Class ValidatingReaderExample
Shared Sub Main(ByVal args() As String)
Try
Dim settings As XmlReaderSettings = New XmlReaderSettings()
settings.Schemas.Add("http://www.contoso.com/books", "contosoBooks.xsd")
settings.ValidationType = ValidationType.Schema
Dim reader As XmlReader = XmlReader.Create("contosoBooks.xml", settings)
Dim document As XmlDocument = New XmlDocument()
document.Load(reader)
Dim navigator As XPathNavigator = document.CreateNavigator()
Dim validation As ValidationEventHandler = New ValidationEventHandler(AddressOf SchemaValidationHandler)
navigator.MoveToChild("bookstore", "http://www.contoso.com/books")
navigator.MoveToChild("book", "http://www.contoso.com/books")
navigator.MoveToChild("author", "http://www.contoso.com/books")
navigator.AppendChild("<title>Book Title</title>")
document.Validate(validation)
navigator.MoveToParent()
navigator.MoveToChild("price", "http://www.contoso.com/books")
navigator.SetTypedValue(DateTime.Now)
Catch e As Exception
Console.WriteLine("ValidatingReaderExample.Exception: {0}", e.Message)
End Try
End Sub
Shared Sub SchemaValidationHandler(ByVal sender As Object, ByVal e As ValidationEventArgs)
Select Case e.Severity
Case XmlSeverityType.Error
Console.WriteLine("Schema Validation Error: {0}", e.Message)
Exit Sub
Case XmlSeverityType.Warning
Console.WriteLine("Schema Validation Warning: {0}", e.Message)
Exit Sub
End Select
End Sub
End Class
using System;
using System.Xml;
using System.Xml.Schema;
using System.Xml.XPath;
class ValidatingReaderExample
{
static void Main(string[] args)
{
try
{
XmlReaderSettings settings = new XmlReaderSettings();
settings.Schemas.Add("http://www.contoso.com/books", "contosoBooks.xsd");
settings.ValidationType = ValidationType.Schema;
XmlReader reader = XmlReader.Create("contosoBooks.xml", settings);
XmlDocument document = new XmlDocument();
document.Load(reader);
XPathNavigator navigator = document.CreateNavigator();
ValidationEventHandler validation = new ValidationEventHandler(SchemaValidationHandler);
navigator.MoveToChild("bookstore", "http://www.contoso.com/books");
navigator.MoveToChild("book", "http://www.contoso.com/books");
navigator.MoveToChild("author", "http://www.contoso.com/books");
navigator.AppendChild("<title>Book Title</title>");
document.Validate(validation);
navigator.MoveToParent();
navigator.MoveToChild("price", "http://www.contoso.com/books");
navigator.SetTypedValue(DateTime.Now);
}
catch (Exception e)
{
Console.WriteLine("ValidatingReaderExample.Exception: {0}", e.Message);
}
}
static void SchemaValidationHandler(object sender, ValidationEventArgs e)
{
switch (e.Severity)
{
case XmlSeverityType.Error:
Console.WriteLine("Schema Validation Error: {0}", e.Message);
break;
case XmlSeverityType.Warning:
Console.WriteLine("Schema Validation Warning: {0}", e.Message);
break;
}
}
}
En el ejemplo se toma como entrada el archivo contosoBooks.xml
.
<?xml version="1.0" encoding="utf-8" ?>
<bookstore xmlns="http://www.contoso.com/books">
<book genre="autobiography" publicationdate="1981-03-22" ISBN="1-861003-11-0">
<title>The Autobiography of Benjamin Franklin</title>
<author>
<first-name>Benjamin</first-name>
<last-name>Franklin</last-name>
</author>
<price>8.99</price>
</book>
<book genre="novel" publicationdate="1967-11-17" ISBN="0-201-63361-2">
<title>The Confidence Man</title>
<author>
<first-name>Herman</first-name>
<last-name>Melville</last-name>
</author>
<price>11.99</price>
</book>
<book genre="philosophy" publicationdate="1991-02-15" ISBN="1-861001-57-6">
<title>The Gorgias</title>
<author>
<name>Plato</name>
</author>
<price>9.99</price>
</book>
</bookstore>
En el ejemplo también se toma como entrada el archivo contosoBooks.xsd
.
<?xml version="1.0" encoding="utf-8"?>
<xs:schema attributeFormDefault="unqualified" elementFormDefault="qualified" targetNamespace="http://www.contoso.com/books" xmlns:xs="http://www.w3.org/2001/XMLSchema">
<xs:element name="bookstore">
<xs:complexType>
<xs:sequence>
<xs:element maxOccurs="unbounded" name="book">
<xs:complexType>
<xs:sequence>
<xs:element name="title" type="xs:string" />
<xs:element name="author">
<xs:complexType>
<xs:sequence>
<xs:element minOccurs="0" name="name" type="xs:string" />
<xs:element minOccurs="0" name="first-name" type="xs:string" />
<xs:element minOccurs="0" name="last-name" type="xs:string" />
</xs:sequence>
</xs:complexType>
</xs:element>
<xs:element name="price" type="xs:decimal" />
</xs:sequence>
<xs:attribute name="genre" type="xs:string" use="required" />
<xs:attribute name="publicationdate" type="xs:date" use="required" />
<xs:attribute name="ISBN" type="xs:string" use="required" />
</xs:complexType>
</xs:element>
</xs:sequence>
</xs:complexType>
</xs:element>
</xs:schema>
En el ejemplo anterior se realizan dos modificaciones en el documento XML contenido en el objeto XmlDocument. A medida que se fuera cargando el documento XML, el método del controlador de eventos de validación controlaría cualquier error de validación del esquema que se encontrara y lo escribiría en la consola.
En este ejemplo los errores de validación se introdujeron después de cargar el documento XML y se encontraron utilizando el método Validate de la clase XmlDocument.
Las modificaciones realizadas con el método SetTypedValue de la clase XPathNavigator iniciaron una InvalidCastException porque el nuevo valor no era válido de acuerdo con el tipo de esquema del nodo.
Para obtener más información sobre la modificación de valores con el método SetTypedValue, vea el tema Modificación de datos XML con XPathNavigator.
Validación de solo lectura
La clase XPathDocument es una representación en memoria de solo lectura de un documento XML. Tanto la clase XPathDocument como la clase XmlDocument crean objetos XPathNavigator para navegar por documentos XML y editarlos. Dado que la clase XPathDocument es de solo lectura, los objetos XPathNavigator que devuelven los objetos XPathDocument no pueden editar el documento XML contenido en el objeto XPathDocument.
En caso de validación, puede crear un objeto XPathDocument igual que crea un objeto XmlDocument utilizando un objeto XmlReader de validación, tal y como se ha descrito anteriormente en este tema. El objeto XPathDocument valida el documento XML a medida que se carga, pero dado que no se pueden editar los datos XML en el objeto XPathDocument, no se puede volver a validar el documento XML.
Para más información sobre objetos XPathNavigator editables y de solo lectura, vea el tema Lectura de datos XML con XPathDocument y XmlDocument.