Вставка XQuery XML

У меня есть целевые данные на основе XML:

<myTargetData>
    <myMap>
        <pairs>
            <key>KEY_1</key>
            <value>
                <myMap>
                    <pairs>
                        <key>INNER_KEY_1</key>
                        <value/>
                    </pairs>
                    <pairs>
                        <key>INNER_KEY_2</key>
                        <value/>
                    </pairs>
                    <pairs>
                        <key>INNER_KEY_3</key>
                        <value>
                            <myMap>
                                <pairs>
                                    <key>INNER_INNER_KEY_1</key>
                                    <value/>
                                </pairs>
                                <pairs>
                                    <key>INNER_INNER_KEY_2</key>
                                    <value/>
                                </pairs>
                            </myMap>
                        </value>
                    </pairs>
                </myMap>
            </value>
        </pairs>
        <pairs>
            <key>KEY_2</key>
            <value>
                <myString>test string 2</myString>
            </value>
        </pairs>
        <pairs>
            <key>KEY_3</key>
            <value>
                <myString>test string 3</myString>
            </value>
        </pairs>
        <pairs>
            <key>KEY_4</key>
            <value>
                <myString>test string 4</myString>
            </value>
        </pairs>
    </myMap>
</myTargetData>

Список полей конфигурации:

<myFieldData>
    <field>KEY_1</field>
    <field>INNER_KEY_2</field>
    <field>INNER_INNER_KEY_1</field>
    <field>KEY_3</field>
<myFieldData>

Значение поля конфигурации означает, что целевой XML должен быть дополнен следующими двумя парами:

<pairs>
    <key>{key name}_appendix1</key>
    <value><myString>smth text</myString></value>
</pairs>
<pairs>
    <key>{key name}_appendix2</key>
    <value><myString>smth text</myString></value>
</pairs>

Результат после преобразования должен быть:

<myTargetData>
    <myMap>
        <pairs>
            <key>KEY_1</key>
            <value>
                <myMap>
                    <pairs>
                        <key>INNER_KEY_1</key>
                        <value/>
                    </pairs>
                    <pairs>
                        <key>INNER_KEY_2</key>
                        <value/>
                    </pairs>
                    <pairs>
                        <key>INNER_KEY_2_appendix1</key>
                        <value><myString>smth text</myString></value>
                    </pairs>
                    <pairs>
                        <key>INNER_KEY_2_appendix2</key>
                        <value><myString>smth text</myString></value>
                    </pairs>
                    <pairs>
                        <key>INNER_KEY_3</key>
                        <value>
                            <myMap>
                                <pairs>
                                    <key>INNER_INNER_KEY_1</key>
                                    <value/>
                                </pairs>
                                <pairs>
                                    <key>INNER_INNER_KEY_1_appendix1</key>
                                    <value><myString>smth text</myString></value>
                                </pairs>
                                <pairs>
                                    <key>INNER_INNER_KEY_1_appendix2</key>
                                    <value><myString>smth text</myString></value>
                                </pairs>    
                                <pairs>
                                    <key>INNER_INNER_KEY_2</key>
                                    <value/>
                                </pairs>
                            </myMap>
                        </value>
                    </pairs>
                </myMap>
            </value>
        </pairs>
        <pairs>
            <key>KEY_1_appendix1</key>
            <value><myString>smth text</myString></value>
        </pairs>
        <pairs>
            <key>KEY_1_appendix2</key>
            <value><myString>smth text</myString></value>
        </pairs>
        <pairs>
            <key>KEY_2</key>
            <value>
                <myString>test string 2</myString>
            </value>
        </pairs>
        <pairs>
            <key>KEY_3</key>
            <value>
                <myString>test string 3</myString>
            </value>
        </pairs>
        <pairs>
            <key>KEY_3_appendix1</key>
            <value><myString>smth text</myString></value>
        </pairs>
        <pairs>
            <key>KEY_3_appendix2</key>
            <value><myString>smth text</myString></value>
        </pairs>
        <pairs>
            <key>KEY_4</key>
            <value>
                <myString>test string 4</myString>
            </value>
        </pairs>
    </myMap>
</myTargetData>

Можно ли это сделать с помощью XQuery? Или лучше использовать XSLT? Если можно - как это сделать? Может рекурсивная функция? Преобразование будет использоваться в Oracle Service Bus.


person Martynas    schedule 15.06.2015    source источник
comment
Предоставляет ли OSB процессор XSLT 2.0?   -  person Florent Georges    schedule 16.06.2015
comment
Нет. OSB поддерживает только XSLT 1.0.   -  person Martynas    schedule 16.06.2015


Ответы (1)


Вам нужно сделать рекурсивную копию (см. шаблон Identity Template), и если текущий узел является совпадением (является элементом pairs с соответствующим key), то вам также нужно сгенерировать 2 элемента после. Эта таблица стилей делает именно это:

<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">

   <xsl:variable name="keys" select="document('so-001-keys.xml')/myFieldData"/>

   <xsl:template match="node()">
      <xsl:copy>
         <xsl:copy-of select="@*"/>
         <xsl:apply-templates select="node()"/>
      </xsl:copy>
      <!-- if it's a match, add new elements -->
      <xsl:if test="self::pairs[key = $keys/field]">
         <pairs>
            <key>
               <xsl:value-of select="key"/>
               <xsl:text>_appendix1</xsl:text>
            </key>
            <value><myString>smth text</myString></value>
         </pairs>
         <pairs>
            <key>
               <xsl:value-of select="key"/>
               <xsl:text>_appendix2</xsl:text>
            </key>
            <value><myString>smth text</myString></value>
         </pairs>
      </xsl:if>
   </xsl:template>

</xsl:stylesheet>
person Florent Georges    schedule 16.06.2015
comment
Я нашел одну ошибку в этой строке: ‹xsl:if test=self::pairs[key = $keys/field/]› пропустил символ косой черты после поля, но все равно - спасибо. оно работает. - person Martynas; 17.06.2015
comment
Действительно? Если это правда, поздравляю, вы только что нашли ошибку в компиляторе ;-) - person Florent Georges; 17.06.2015