.NET 2.0/VS2005Jak rozwiązać XSL zawiera transformacja, która ładuje XSL z ciągu?
Próbuję użyć klasy XslCompiledTransform
do przeprowadzenia transformacji XSL. Mam dwa pliki XSL, z których pierwsza zawiera odniesienie do innego w formie oświadczenia <xsl:include>
:
Main.xsl:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:include href="Included.xsl" />
...
...
</xsl:stylesheet>
Teraz, jeśli mogę załadować „Main .xsl”złożyć się jako URI, mój kod transformacja będzie tak proste, jak:
// This is a function that works. For demo only.
private string Transform(string xslFileURI)
{
XslCompiledTransform xslt = new XslCompiledTransform();
// This load works just fine, if I provide the path to "Main.xsl".
// The xsl:include is automatically resolved.
xslTransform.Load(xslFileURI);
StringWriter sw = new StringWriter();
xslt.Transform(Server.MapPath("~/XML/input.xml"), null, sw);
return sw.ToString();
}
problemem jest to, że mogę otrzymać zawartość pliku Main.xsl jako ciąg i trzeba załadować ciąg w postaci XmlReader/IXpathNavigable
. Jest to niezbędne ograniczenie w tej chwili. Kiedy próbuję zrobić to samo używając XmlReader/XpathDocument
, to się nie powiedzie, ponieważ kod wyszukuje "Included.xsl" w folderze C:\Program Files\Microsoft Visual Studio 8\Common7\IDE\
! Oczywiście, XmlResolver
nie jest w stanie rozwiązać względnego adresu URL, ponieważ odbiera tylko ciąg znaków jako wejściowy XSL.
Moje wysiłki w tym kierunku wyglądać następująco:
// This doesn't work! Halp!
private string Transform(string xslContents)
{
XslCompiledTransform xslt = new XslCompiledTransform();
XmlUrlResolver resolver = new XmlUrlResolver();
resolver.Credentials = CredentialCache.DefaultCredentials;
//METHOD 1: This method does not work.
XmlReaderSettings settings = new XmlReaderSettings();
settings.XmlResolver = resolver;
XmlReader xR = XmlReader.Create(new StringReader(xslContents), settings);
xslt.Load(xR); // fails
// METHOD 2: Does not work either.
XPathDocument xpDoc = new XPathDocument(new StringReader(xslContents));
xslt.Load(xpDoc, new XsltSettings(true, true), resolver); //fails.
StringWriter sw = new StringWriter();
xslt.Transform(Server.MapPath("~/XML/input.xml"), null, sw);
return sw.ToString();
}
Próbowałem użyć sposobu XmlUrlResolver ResolveUri
do uzyskania Stream
przedstawieniu plik XSL, które należy uwzględnić, ale jestem zdezorientowany, jak korzystać z tego strumienia. IOW, jak mogę powiedzieć obiekt XslCompiledTransform
używać tego strumienia oprócz Main.xsl XmlReader:
Uri mainURI = new Uri(Request.PhysicalApplicationPath + "Main.xsl");
Uri uri = resolver.ResolveUri(mainURI, "Included.xsl");
// I can verify that the Included.xsl file loads in the Stream below.
Stream s = resolver.GetEntity(uri, null, typeof(Stream)) as Stream;
// How do I use this Stream in the function above??
Każda pomoc jest mile widziana. Przepraszamy za długi post!
Dla odniesienia, Wyjątek StackTrace wygląda następująco:
[FileNotFoundException: Could not find file 'C:\Program Files\Microsoft Visual Studio 8\Common7\IDE\Included.xsl'.]
System.IO.__Error.WinIOError(Int32 errorCode, String maybeFullPath) +328
System.IO.FileStream.Init(String path, FileMode mode, FileAccess access, Int32 rights, Boolean useRights, FileShare share, Int32 bufferSize, FileOptions options, SECURITY_ATTRIBUTES secAttrs, String msgPath, Boolean bFromProxy) +1038
System.IO.FileStream..ctor(String path, FileMode mode, FileAccess access, FileShare share, Int32 bufferSize) +113
System.Xml.XmlDownloadManager.GetStream(Uri uri, ICredentials credentials) +78
System.Xml.XmlUrlResolver.GetEntity(Uri absoluteUri, String role, Type ofObjectToReturn) +51
System.Xml.Xsl.Xslt.XsltLoader.CreateReader(Uri uri, XmlResolver xmlResolver) +22
System.Xml.Xsl.Xslt.XsltLoader.LoadStylesheet(Uri uri, Boolean include) +33
System.Xml.Xsl.Xslt.XsltLoader.LoadInclude() +349
System.Xml.Xsl.Xslt.XsltLoader.LoadRealStylesheet() +704
System.Xml.Xsl.Xslt.XsltLoader.LoadDocument() +293
System.Xml.Xsl.Xslt.XsltLoader.LoadStylesheet(XmlReader reader, Boolean include) +173
Pracuję nad czymś podobnym do tego, czego wymaga twoje pytanie i znalazłem artykuł MSDN - [Resolving the Unknown: Building Custom XmlResolvers w .NET Framework] (http://msdn.microsoft.com/en-us/ library/aa302284.aspx) - wydaje się to być bardzo obiecującym rozwiązaniem. –