Объединение двух XPathDocuments с использованием XmlCompiledTransform - PullRequest
2 голосов
/ 02 марта 2010

Я не могу быть первым человеком, который сделает это, похоже, что было бы такой обычной практикой объединять два документа с использованием XSLT. Тем не менее, я не могу найти ни одного примера на старой сети.

У меня есть два XML-документа, которые извлекаются как строки XML из SQL Server. Я хочу использовать XslCompiledTransform для объединения двух документов. Я знаю, что XslCompiledTransform по умолчанию отключает функцию XSL document(). Я включил это, используя XsltSettings, когда создаю свой объект XslCompiledTransform.

Мое понимание того, как "добавить" второй документ к преобразованию, состоит в том, чтобы использовать XsltArgumentList, использовать метод AddParam() и добавить объект XPathNavigator:

XsltArgumentList xsltArgs = new XsltArgumentList();
xsltArgs.AddParam(
  (string)e.UserState + "s", "http://www.myuri.com/tabledata", 
  dataXmlDoc.CreateNavigator()
);

Однако любые попытки доступа к добавленному документу приводят либо к ошибке, либо ничего не возвращается. & Mdash; C #:

XslCompiledTransform fieldToXhtmlTransform = new XslCompiledTransform(true);
try
{
  UriBuilder xsltUri = new UriBuilder(
    Request.Url.Scheme, Request.Url.Host, 
    Request.Url.Port, this.ResolveUrl("Transforms/address1.xslt")
  );

  XmlSecureResolver resolver = new XmlSecureResolver(
    new XmlUrlResolver(), new PermissionSet(PermissionState.Unrestricted)
  );
  fieldToXhtmlTransform.Load(
    xsltUri.ToString(), new XsltSettings(true, false), resolver
  );
}
catch
{
  //TODO: do something useful here. 
}

XPathDocument fieldSchemaXmlDoc = null;

using (MemoryStream fieldMemoryStream = new MemoryStream(
  Encoding.UTF8.GetBytes(e.Result.TableMetaDataXml)
))
{
  fieldSchemaXmlDoc = new XPathDocument(fieldMemoryStream);
}

XPathDocument dataXmlDoc = null;

using (MemoryStream dataMemoryStream = new MemoryStream(
  Encoding.UTF8.GetBytes(e.Result.DataXml)
))
{
  dataXmlDoc = new XPathDocument(dataMemoryStream);
}

StringBuilder output = new StringBuilder();

XmlWriterSettings writerSettings = new XmlWriterSettings();
writerSettings.OmitXmlDeclaration = true;
writerSettings.Encoding = Encoding.UTF8;

XsltArgumentList xsltArgs = new XsltArgumentList();
xsltArgs.AddParam(
  (string)e.UserState + "s", "http://www.myuri.com/tabledata",
  dataXmlDoc.CreateNavigator()
);    
XmlWriter transformedDataWriter = XmlWriter.Create(output, writerSettings);
fieldToXhtmlTransform.Transform(
  fieldSchemaXmlDoc, xsltArgs, transformedDataWriter
);

XSLT - Доступ только к добавленному документу, но не к документу, загруженному с преобразованием.

<xsl:stylesheet 
  version="1.0" 
  xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
  xmlns:msxsl="urn:schemas-microsoft-com:xslt"
  xmlns:hlsschema="http://www.myuri.com/tableschema"
  xmlns:hlsdata="http://www.myuri.com/tabledata"
  exclude-result-prefixes="msxsl hlsschema hlsdata xsl"
>
  <xsl:output method="html" indent="yes"/>

  <p>
  <xsl:template match="hlsdata:Address1s">
    <xsl:for-each select="hlsdata:Address1">
      <p>
        <xsl:value-of select="hlsdata:dr_id"/>
      </p>
    </xsl:for-each>
  </xsl:template>
  </p>

</xsl:stylesheet>

XML

<hlsdata:Address1s 
  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
  xmlns:hlsdata="http://www.myuri.com/tabledata"
>
  <hlsdata:Address1>
    <hlsdata:dr_id>12345678</hlsdata:dr_id>
  </hlsdata:Address1>
</hlsdata:Address1s>

Я знаю, что упускаю что-то очевидное, но это выходит за рамки разочарования. Я знаю, что документ добавляется в качестве параметра, но я не могу найти пример доступа к документу, загруженному в качестве параметра.

Любая помощь будет принята с благодарностью. Имейте в виду, что приведенный выше код находится в стадии разработки и занимает от двух до сотен попыток заставить его работать, поэтому, если что-то выглядит немного странно, это, вероятно, потому, что между попытками.

1 Ответ

4 голосов
/ 03 марта 2010

Вам необходимо определить параметр в таблице стилей, а затем использовать этот параметр.Вот простой пример, таблица стилей выглядит следующим образом:

<xsl:stylesheet
  version="1.0"
  xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
  xmlns:docs="http://example.com/2010/docs"
  exclude-result-prefixes="docs"
>
  <xsl:param name="docs:doc1" select="/.."/>

  <xsl:template match="/">
    <xsl:apply-templates select="$docs:doc1/node()"/>
  </xsl:template>

  <xsl:template match="root">
    <ul>
      <xsl:apply-templates/>
    </ul>
  </xsl:template>

  <xsl:template match="foo">
    <li>
      <xsl:apply-templates/>
    </li>
  </xsl:template>
</xsl:stylesheet>

Код C # выглядит следующим образом:

    string xml = "<root><foo>1</foo><foo>2</foo></root>";
    XPathDocument doc = new XPathDocument(new StringReader(xml));

    XslCompiledTransform proc = new XslCompiledTransform();
    proc.Load(@"..\..\XSLTFile1.xslt");

    XsltArgumentList xsltArgs = new XsltArgumentList();
    xsltArgs.AddParam("doc1", "http://example.com/2010/docs", doc.CreateNavigator());

    proc.Transform(XmlReader.Create(new StringReader("<dummy/>")), xsltArgs, Console.Out);

Это консольное приложение, которое для простоты пишет в Console.Outно вы, конечно, можете использовать другие выходные данные, которые допускает метод Transform.

В этом примере записывается <ul><li>1</li><li>2</li></ul>, поэтому входной параметр обработан.

Так что это должно показать вам, как передатьпараметр, который XslCompiledTransform видит как набор узлов, который вы можете обработать с помощью XSLT.

Что касается написания таблицы стилей, которая объединяет два документа, пожалуйста, опубликуйте два входных образца и соответствующий пример результата, который вы хотите создать, если у вас возникли проблемы при записичто XSLT.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...