是的,你可以解决这个问题。
一种选择是创建处理DTD分辨率的自己的解析器。它可以使用任何它喜欢的机制,包括为出站通信使用非默认代理。
var xmlReaderSettings = new XmlReaderSettings
{
ProhibitDtd = false,
ValidationType = ValidationType.DTD,
XmlResolver = new MyCustomDtdResolver()
};
在为MyCustomDtdResolver的代码,你会指定所需的代理设置。它可能因DTD而异。
您没有指定,但是如果您正在解决的DTD是固定且不变的,那么Silverlight和.NET 4.0具有内置的解析器,它不会触及网络(无代理,无任何http通信) 。它被称为XmlPreloadedResolver。它知道如何解析RSS091和XHTML1.0。如果您有其他DTD(包括您自己的自定义DTD),并且它们是固定的或不变的,则可以将它们加载到此解析器中并在运行时使用它,并完全避免HTTP通信和代理复杂化。
More on that.
如果你没有使用.NET 4.0,那么你可以建立一个“无网络”解析器自己。为了避免W3C traffic limit,我建了a custom resolver myself, for XHTML,也许你可以重新使用它。另外,a related link。
例如,以下是ResolveUri在自定义Uri解析器中的代码。
/// <summary>
/// Resolves URIs.
/// </summary>
/// <remarks>
/// <para>
/// The only Uri's supported are those for W3C XHTML 1.0.
/// </para>
/// </remarks>
public override Uri ResolveUri(Uri baseUri, string relativeUri)
{
if (baseUri == null)
{
if (relativeUri.StartsWith("http://"))
{
Trace(" returning {0}", relativeUri);
return new Uri(relativeUri);
}
// throw if Uri scheme is unknown/unhandled
throw new ArgumentException();
}
if (relativeUri == null)
return baseUri;
// both are non-null
var uri = baseUri.AbsoluteUri;
foreach (var key in knownDtds.Keys)
{
// look up the URI in the table of known URIs
var dtdUriRoot = knownDtds[key];
if (uri.StartsWith(dtdUriRoot))
{
string newUri = uri.Substring(0,dtdUriRoot.Length) + relativeUri;
return new Uri(newUri);
}
}
// must throw if Uri is unknown/unhandled
throw new ArgumentException();
}
这里是为GetEntity
/// <summary>
/// Gets the entity associated to the given Uri, role, and
/// Type.
/// </summary>
/// <remarks>
/// <para>
/// The only Type that is supported is the System.IO.Stream.
/// </para>
/// <para>
/// The only Uri's supported are those for W3C XHTML 1.0.
/// </para>
/// </remarks>
public override object GetEntity(Uri absoluteUri, string role, Type t)
{
// only handle streams
if (t != typeof(System.IO.Stream))
throw new ArgumentException();
if (absoluteUri == null)
throw new ArgumentException();
var uri = absoluteUri.AbsoluteUri;
foreach (var key in knownDtds.Keys)
{
if (uri.StartsWith(knownDtds[key]))
{
// Return the stream containing the requested DTD.
// This can be a FileStream, HttpResponseStream, MemoryStream,
// or whatever other stream you like. I used a Resource stream
// myself. If you retrieve the DTDs via HTTP, you could use your
// own IWebProxy here.
var resourceName = GetResourceName(key, uri.Substring(knownDtds[key].Length));
return GetStreamForNamedResource(resourceName);
}
}
throw new ArgumentException();
}
完整的工作代码的代码为我的自定义解析is available。
如果您的解析程序执行网络通信,那么对于一般的解决方案,您可能需要重写凭证属性。
public override System.Net.ICredentials Credentials
{
set { ... }
}
另外,您可能希望公开Proxy属性。或不。正如我上面所说的,您可能想要从DTD URI自动确定要使用的代理。
感谢Cheeso的回复。 你能为MyCustomDtdResolver显示一些代码吗?就像你在http://stackoverflow.com/questions/2558021/an-error-has-occurred-opening-extern-dtd-w3-org-xhtml1-transitional-dtd-503/2559056#2559056中说的那样,文档非常* *很少。我特别感兴趣的是ResolveUri和GetEntity方法应该包含什么以启用通过代理获取dtd。 – Lanceomagnifico 2010-05-05 08:28:51
哦,我没有忘记提到这个解决方案需要在.NET 3.5 – Lanceomagnifico 2010-05-05 08:31:53
中工作好,我会用一些代码更新答案。 – Cheeso 2010-05-05 13:05:21