Компилировать схемы с включенными файлами xsd

У меня есть статический метод, который я использую для проверки XML-файла по сравнению с XSD-файлом. Это работает нормально, пока не будет XSD-файл, который включает в себя еще один XSD-файл.

Пример, где я получил неприятности:

TYPES.XSD:

<xs:simpleType name="MY_AMOUNT">
    <xs:restriction base="xs:decimal">
        <xs:maxInclusive value="999999999999.99"/>
        <xs:minInclusive value="-999999999999.99"/>
        <xs:totalDigits value="14"/>
        <xs:fractionDigits value="2"/>
    </xs:restriction>
</xs:simpleType>

MAIN.XSD:

<?xml version="1.0" encoding="UTF-8"?>
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema" elementFormDefault="qualified">
    <xs:include schemaLocation="TYPES.xsd"/>
    <xs:element name="ROOT">
        <xs:complexType>
            <xs:sequence>
                <xs:element ref="SOMEREF1"/>
                <xs:element ref="SOMEREF2"/>
                <xs:element name="AMOUNT" type="MY_AMOUNT" minOccurs="0"/>
            </xs:sequence>
        </xs:complexType>
    </xs:element>
</xs:schema>

КОД ПРОВЕРКИ:

public static class XmlUtils
{
    private static string Errors = string.Empty;

    public static bool ValidateAgainstXSD(string xmlFilePath, string xsdFilePath, ref string message)
    {
        try
        {
            var settings = new XmlReaderSettings();

            settings.ValidationType = ValidationType.Schema;
            settings.ValidationFlags = XmlSchemaValidationFlags.ProcessInlineSchema
                | XmlSchemaValidationFlags.ProcessInlineSchema
                | XmlSchemaValidationFlags.ReportValidationWarnings;
            settings.Schemas.Add(null, xsdFilePath);
            settings.Schemas.Compile();

            settings.ValidationEventHandler += (sender, args) =>
            {
                if (args.Severity == XmlSeverityType.Error)
                {
                    Errors += args.Message + "\n";
                }
            };

            using (var reader = XmlReader.Create(xmlFilePath, settings))
            {
                while (reader.Read()) { }
            }

            message = Errors ?? string.Empty;
            return string.IsNullOrEmpty(Errors);
        }
        catch (Exception e)
        {
            message = "# error validating xml file: " + e.Message;
            return false;
        }
    }
}

Почему-то кажется, что мне нужно указать путь к включенному файлу XSD, но я понятия не имею, где.

Ошибка происходит в settings.Schemas.Compile(); где говорится, что тип "MY_AMOUNT" не объявлен. Я читал о пользовательских XmlResolvers, но, честно говоря, у меня не получилось.

Если это важно для ответа: xsd файлы всегда находятся в одном и том же каталоге!

Метод называется вот так:

string msg = string.Empty;
string basedir = @"C:\Temp";
string xml = Path.Combine(basedir, "XML_FILE.xml");
string xsd = Path.Combine(basedir, "MAIN.xsd");

if (XmlUtils.ValidateAgainstXSD(xml, xsd, ref msg))
{
    // do some work
}
else
{
    Console.WriteLine(msg);
}

Console.ReadLine();

Любая помощь высоко ценится - спасибо!

ОБНОВЛЕНИЕ 2016-12-05:

Я написал свой собственный XmlUrlResolver, чтобы посмотреть, что происходит за кулисами:

internal class XUrlResolver : XmlUrlResolver
{
    public override object GetEntity(Uri absoluteUri, string role, Type ofObjectToReturn)
    {
        return base.GetEntity(absoluteUri, role, ofObjectToReturn);
    }

    public override Uri ResolveUri(Uri baseUri, string relativeUri)
    {
        return base.ResolveUri(baseUri, relativeUri);
    }
}

И я просто пытаюсь сделать:

XmlSchemaSet xset = new XmlSchemaSet();
xset.XmlResolver = new XUrlResolver();
xset.Add("", xsdFilePath);
xset.Compile();

Что происходит сейчас (на линии xset.Add):

  1. XmlUrlResolver.ResolveUri(null,"C:\\Temp\\MAIN.XSD") -> {file:///C:/Temp/MAIN.xsd}
  2. XmlUrlResolver.ResolveUri(null,"C:\\Temp\\MAIN.XSD") -> {file:///C:/Temp/MAIN.xsd}
  3. XmlUrlResolver.GetEntity({file:///C:/Temp/MAIN.xsd}) -> Файловый поток в MAIN.xsd
  4. XmlUrlResolver.ResolveUri({file:///C:/Temp/MAIN.xsd},"TYPES.XSD") -> {file:///C:/Temp/TYPES.xsd}
  5. XmlUrlResolver.GetEntity({file:///C:/Temp/TYPES.xsd}) -> Файловый поток в TYPES.xsd

Выглядит хорошо для меня (за исключением того, что первые 2 вызова равны!?!) - путь к TYPES.XSD разрешен, как и должно быть.

тем не менее, xset.Compile() выдает исключение: "Тип MY_AMOUNT не объявлен"

И я понятия не имею, почему:/

2 ответа

Решение

Сначала вам нужно сделать ваши xsd файлы действительными.

Types.xsd (добавлен корневой элемент схемы и пространство имен xs)

<?xml version="1.0" encoding="UTF-8"?>
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema">
<xs:simpleType name="MY_AMOUNT">
    <xs:restriction base="xs:decimal">
        <xs:maxInclusive value="999999999999.99"/>
        <xs:minInclusive value="-999999999999.99"/>
        <xs:totalDigits value="14"/>
        <xs:fractionDigits value="2"/>
    </xs:restriction>
</xs:simpleType>
</xs:schema>

Main.xsd (удалены недействительные ссылки).

<?xml version="1.0" encoding="UTF-8"?>
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema" elementFormDefault="qualified">
    <xs:include schemaLocation="TYPES.xsd"/>
    <xs:element name="ROOT">
        <xs:complexType>
            <xs:sequence>                
                <xs:element name="AMOUNT" type="MY_AMOUNT" minOccurs="0"/>
            </xs:sequence>
        </xs:complexType>
    </xs:element>
</xs:schema>

После этого, учитывая, что оба xsd-файла находятся в одном каталоге, ваши схемы будут хорошо скомпилированы.

Я столкнулся с этой самой проблемой.

Я не предполагаю, что это правильный ответ, но я обошел его, установив Environment.CurrentDirectory свойство быть путем, где были расположены включенные XSD. Тогда все это обрабатывается просто отлично.

Другие вопросы по тегам