XSLT Дублирование узла и всех дочерних узлов и атрибутов с новыми значениями

Добрый день .... Я пытаюсь дублировать узлы с обновленным / новым текстом элемента и / или значениями атрибутов.

Мой входной XML-файл:

<?xml version="1.0"?>
<products author="Jesper">
  <product id="p1">
    <name>Delta</name>
    <price>800</price>
    <stock>4</stock>
    <country>Denmark</country>
  </product>
</products>

Желаемый вывод XML:

<?xml version="1.0" encoding="utf-8"?>
<products author="Jesper">
  <product id="p1">
    <name>Delta</name>
    <price>800</price>
    <stock>4</stock>
    <country>Denmark</country>
  </product>
  <product id="NEW_p1">
    <name>NEW_Delta</name>
    <price>NEW_800</price>
    <stock>NEW_4</stock>
    <country>NEW_Denmark</country>
  </product>
</products>

Через некоторое время XSLT, который у меня сейчас есть, выглядит следующим образом:

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
                xmlns:msxsl="urn:schemas-microsoft-com:xslt" 
                exclude-result-prefixes="msxsl">
  <xsl:output method="xml" indent="yes"/>
  <xsl:template match="@*|node()">
    <xsl:copy>
      <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
  </xsl:template>
  <xsl:template match ="product">
    <xsl:copy>
      <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
    <product>
      <xsl:attribute name ="id">
        <xsl:value-of select ="concat('NEW_',@id"/>
      </xsl:attribute>
      <xsl:copy>
        <xsl:apply-templates select="node()"/>
      </xsl:copy>
    </product>
  </xsl:template>

However, using the above transform, I get the following XML output:

<?xml version="1.0" encoding="utf-8"?>
<products author="Jesper">
  <product id="p1">
    <name>Delta</name>
    <price>800</price>
    <stock>4</stock>
    <country>Denmark</country>
  </product>
  <product id="NEW_p1"><product>
    <name>Delta</name>
    <price>800</price>
    <stock>4</stock>
    <country>Denmark</country>
  </product></product>
</products>

Как видите, элемент product добавлен, в то время как я объявил новый элемент product с новым значением @id. Поскольку я использую для обработки дочерних узлов, я считаю, что это снова обрабатывает элемент product.

Кроме того, мне нужна помощь в обновлении значений дочернего узла (добавление «NEW_» к каждому значению). Изучая обширные вопросы на этом сайте, я считаю, что мне нужен такой шаблон:

<xsl:template match="*">
  <xsl:element name ="{local-name()}">
    <!--for all attributes-->
    <xsl:copy-of select ="@*"/>
    <xsl:value-of select = "."/>
  </xsl:element>
</xsl:template>

Заранее благодарим вас за любые предложения / идеи по моей проблеме.

ОБНОВЛЕНО Спасибо @Mathias за ответ на мой первоначальный вопрос. Полученный ответ вызвал еще один вопрос, связанный с рекурсией на более глубокие уровни структуры XML.

Входной XML-файл:

    <products author="Jesper">
      <product id="p1">
        <name>Delta
          <innerName>MiddleDelta
            <baseName>FinalDelta</baseName>
          </innerName>
        </name>
        <price>800</price>
        <stock>4</stock>
        <country>Denmark
          <city>Copenhagen</city>
        </country>
      </product>
    </products>

И обновленный выходной файл желания выглядит следующим образом:

<?xml version="1.0" encoding="utf-8"?>
<products author="Jesper">
  <product id="p1">
    <name>Delta
      <innerName>MiddleDelta
        <baseName>FinalDelta</baseName>
      </innerName>
    </name>
    <price>800</price>
    <stock>4</stock>
    <country>Denmark
      <city>Copenhagen</city>
    </country>
  </product>
  <product id="NEW_p1">
    <name>NEW_Delta
      <innerName>NEW_MiddleDelta
        <baseName>NEW_FinalDelta</baseName>
      </innerName>
    </name>
    <price>NEW_800</price>
    <stock>NEW_4</stock>
    <country>NEW_Denmark
      <city>NEW_Copenhagen</city>
    </country>
  </product>
</products>

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


person Lorentz    schedule 14.04.2015    source источник
comment
Вы получили ответ на более глубокую рекурсию?   -  person Daniel Haley    schedule 14.04.2015
comment
@DanielHaley Нет, не видел.   -  person Lorentz    schedule 14.04.2015


Ответы (2)


Это ответ на ваш обновленный вопрос (который, ИМХО, должен был быть задан как новый вопрос):

XSLT 1.0

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>

<xsl:template match="/products">
    <xsl:copy>
        <xsl:copy-of select="product"/>
        <xsl:apply-templates select="product"/>
    </xsl:copy>
</xsl:template>

<!-- modified identity transform -->
<xsl:template match="*">
    <xsl:copy>
        <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
</xsl:template>

<xsl:template match="@*">
    <xsl:attribute name="{name()}">
        <xsl:value-of select="concat('NEW_', .)"/>
    </xsl:attribute>
</xsl:template>

<xsl:template match="text()">
    <xsl:value-of select="concat('NEW_', .)"/>
</xsl:template>

</xsl:stylesheet>
person michael.hor257k    schedule 14.04.2015
comment
Я бы также добавил приоритет к шаблону, который соответствует text(), чтобы вы не получали возможных неоднозначных предупреждений о совпадении шаблона (с шаблоном, который соответствует node()). - person Daniel Haley; 14.04.2015
comment
@DanielHaley Спасибо. Вы правильно указали на проблему. Однако я выбрал несколько иное решение. - person michael.hor257k; 14.04.2015
comment
Спасибо @ michael.hor257k. Одна вещь, которую я получаю, после каждого закрывающего тега ‹/ ...› у меня есть текст NEW_. Итак, для обновленного входного файла, приведенного выше, у меня есть ... ‹/name› NEW_ ... ‹ / страна ›NEW_ - person Lorentz; 15.04.2015
comment
В вашей таблице стилей есть <xsl:strip-space elements="*"/> на верхнем уровне? - person michael.hor257k; 15.04.2015
comment
@ michael.hor257k нет, это не так. Отлично работает - еще раз спасибо! - person Lorentz; 15.04.2015

Вы не за горами, но есть две основные проблемы:

  • вы используете xsl:copy внутри второго, буквального product элемента, что приводит к дополнительному product элементу на выходе
  • вам нужен способ найти все дочерние элементы элемента product, вывести их снова и добавить «NEW_» к их текстовому содержанию.

Вы уверены, что атрибут version должен иметь значение «2.0»? Кроме того, я не уверен, в чем суть этого упражнения ...

Таблица стилей XSLT

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">

  <xsl:output method="xml" indent="yes"/>

  <xsl:template match="@*|node()">
    <xsl:copy>
      <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
  </xsl:template>

  <xsl:template match ="product">
    <xsl:copy>
      <xsl:apply-templates select="@*|node()"/>
    </xsl:copy>
    <product id="{concat('NEW_',@id)}">
      <xsl:for-each select="*">
          <xsl:copy>
              <xsl:value-of select="concat('NEW_',.)"/>
          </xsl:copy>
      </xsl:for-each>
    </product>
  </xsl:template>

</xsl:stylesheet>

Вывод XML

<?xml version="1.0" encoding="UTF-8"?>
<products author="Jesper">
  <product id="p1">
      <name>Delta</name>
      <price>800</price>
      <stock>4</stock>
      <country>Denmark</country>
  </product>
   <product id="NEW_p1">
      <name>NEW_Delta</name>
      <price>NEW_800</price>
      <stock>NEW_4</stock>
      <country>NEW_Denmark</country>
   </product>
</products>
person Mathias Müller    schedule 14.04.2015
comment
Спасибо за ответ. Я узнал, что использование шаблонов было способом преобразования XML - я вижу использование синтаксиса for-each. - person Lorentz; 14.04.2015
comment
@Lorentz И for-each, и шаблоны имеют свое место - важно, чтобы вы не использовали одно, когда другое было бы лучше. И это правда, что xsl:for-each - это конструкция, которую часто неправильно используют, в основном новички. Не забудьте принять этот ответ, если он решил вашу проблему. Спасибо! - person Mathias Müller; 14.04.2015
comment
Должным образом отмечено использование слова «для каждого» - я буду учитывать ваши комментарии по мере продвижения вперед. Это упражнение является частью более крупного проекта - мне просто нужно было разобраться в сути дублирования узлов с новыми данными. Будет ли это преобразование допускать рекурсию на более глубокие уровни во входном XML-файле? Заранее спасибо. - person Lorentz; 14.04.2015
comment
@Lorentz Нет, в настоящее время не будет. В общем, люди в Stackoverflow могут работать только с показанным им XML. Вам нужно будет показать эти глубокие структуры и объяснить, каким образом вы хотите, чтобы ваш код вел себя рекурсивно. - person Mathias Müller; 14.04.2015
comment
Еще раз спасибо за ваш ответ / ответ. - person Lorentz; 15.04.2015