.net - XDocument.Validate - 错误元素的预期数据类型

标签 .net xml date datetime xsd-validation

我有一个类根据提供的 XSD 验证提供的 XML 文档。在类中我调用XDocument.Validate 方法来执行验证,并得到以下错误:

The 'http://www.example.com/enrollrequest1:requested-payment-date' element is invalid - The value '2015-05-28T00:00:00' is invalid according to its datatype 'http://www.w3.org/2001/XMLSchema:date' - The string '2015-05-28T00:00:00' is not a valid XsdDateTime value.

元素的值是从 .NET DateTime 变量设置的,该变量最终将其设置为包含时间部分,因为 .NET 中没有等效的 xs:date 类型。

元素的值是从通用模块设置的,因此我无法挑选元素并自定义设置它们的值。开发人员向我发送 .NET DateTime 类型的值,我的程序依次调用 XElemet.SetValue(value) 方法来设置它。

此外,XSD 文件不在我的控制范围内。所以修改 XSD 不是一个选项。

有没有办法知道导致错误的 XElement 的预期类型是什么? 一旦我知道了,我就可以相应地进行类型转换或自定义我的代码。因此,例如在这种情况下,如果我知道预期类型是 xs:date(而不是 xs:datetime),我可以简单地对传入值进行类型转换。

这是我的验证器类,如果有帮助的话:

Option Strict On
Imports System.XML.Schema

Public Class XmlSchemaValidator
    Public ReadOnly Errors As New List(Of String)

    Private XDoc As XDocument
    Private Schemas As XmlSchemaSet

    Public Sub New(ByVal doc As XDocument, ByVal schemaUri As String, ByVal targetNamespace As String)
        Me.XDoc = doc
        Me.Schemas = New XmlSchemaSet
        Me.Schemas.Add(targetNamespace, schemaUri)
    End Sub

    Public Sub Validate()
        Errors.Clear()
        XDoc.Validate(Schemas, AddressOf XsdErrors)
    End Sub

    Private Sub XsdErrors(ByVal sender As Object, ByVal e As ValidationEventArgs)
        Errors.Add (e.Message)
    End Sub
End Class

这是设置 xml 节点值的函数。

Function SetValue(ByVal xmlDoc As XDocument, ByVal keyValues As Dictionary(Of String, Object)) As Boolean
    '' set values
    For Each kvp In keyValues
        Dim xe As XElement = xmlDoc.Root.XPathSelectElement(kvp.Key)

        ''-- this is buggy implementation for handling xs:date vs xs:datetime that needs to be corrected...
        'If TypeOf kvp.Value Is DateTime AndAlso DirectCast(kvp.Value, DateTime).TimeOfDay = TimeSpan.Zero Then
        '    xe.SetValue(DirectCast(kvp.Value, DateTime).ToString("yyyy-MM-dd"))
        'Else
        xe.SetValue(kvp.Value)
        'End If
    Next

    '' validate final document
    Dim schemaValidator As New XmlSchemaValidator(xmlDoc, schemaFile, "")
    schemaValidator.Validate()
    If schemaValidator.Errors.Count > 0 Then
        'Error Logging code goes here...
        Return False
    End If
    Return True
End Function

最佳答案

您在之前的评论中写道:

“我想知道 XSD 对该元素的期望”

然后,请记住,您可以利用验证处理程序的第一个参数“sender”,例如 adapting this MSDN sample例如,

        string xsdMarkup =
 @"<xsd:schema xmlns:xsd='http://www.w3.org/2001/XMLSchema'>
   <xsd:element name='Root'>
    <xsd:complexType>
     <xsd:sequence>
      <xsd:element name='Child1' minOccurs='1' maxOccurs='1'/>
      <xsd:element name='Child2' minOccurs='1' maxOccurs='1'/>
     </xsd:sequence>
    </xsd:complexType>
   </xsd:element>
  </xsd:schema>";
        XmlSchemaSet schemas = new XmlSchemaSet();
        schemas.Add("", XmlReader.Create(new StringReader(xsdMarkup)));

        // (just for debug spying)
        var schemata = new XmlSchema[1];
        schemas.CopyTo(schemata, 0);

        XDocument errorDoc = new XDocument(
            new XElement("Root",
                new XElement("Child1", "content1"),
                new XElement("Child2", "content2"),
                new XElement("Child2", "content3") // (must fail validation on maxOccurs)
            )
        );

        Console.WriteLine();
        Console.WriteLine("Validating errorDoc");
        errorDoc.Validate(schemas, (sender, args) =>
        {
            Console.WriteLine("{0}", args.Message); // (what you're already doing)
            Console.WriteLine();
            // but there's also:
            var xElement = sender as XElement;
            if (xElement != null)
            {
                Console.WriteLine("Element {0} invalid : {1}", xElement, e.Exception.Message);
            }
        });

        Console.ReadKey();

这可以产生一个输出,其中包含关于文档中可识别的罪魁祸首的足够信息,希望:

Validating errorDoc
The element 'Root' has invalid child element 'Child2'.

Element <Child2>content3</Child2> invalid : The element 'Root' has invalid child element 'Child2'.

无论如何,一旦您知道无效文档中的罪魁祸首,您就有更好的机会更可靠地与用于此验证的模式中的相应定义相关联(而不是仅仅依赖于模式验证错误字符串)。

(很抱歉用 C# 语法回答,但我不想用不正确的 VB.NET 编写,到现在为止我已经生疏了)

'希望这有帮助。

关于.net - XDocument.Validate - 错误元素的预期数据类型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30528253/

相关文章:

c# - NHibernate:如何使用 CreateSQLQuery 返回标量值列表(从一列)?

xml - R:将 XML 数据转换为数据框

XML 换行符号替换为回车符

c# - 内部包含内容的元素的 XPath?

javascript - Safari 中的日期解析错误

c# - NHibernate 实体投影

c# - 在 .NET 中创建阻塞 Queue<T>?

c# - Unity - 如何在解析/实例化对象时收到通知?

javascript - 如何隐藏 Bootstrap 日期选择器

有趣的Java日期类