Компилировать схемы с включенными файлами 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
):
XmlUrlResolver.ResolveUri(null,"C:\\Temp\\MAIN.XSD")
->{file:///C:/Temp/MAIN.xsd}
XmlUrlResolver.ResolveUri(null,"C:\\Temp\\MAIN.XSD")
->{file:///C:/Temp/MAIN.xsd}
XmlUrlResolver.GetEntity({file:///C:/Temp/MAIN.xsd})
-> Файловый поток в MAIN.xsdXmlUrlResolver.ResolveUri({file:///C:/Temp/MAIN.xsd},"TYPES.XSD")
->{file:///C:/Temp/TYPES.xsd}
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. Тогда все это обрабатывается просто отлично.