Таблица стилей группировки Muenchian, возвращающая дубликаты несгруппированных узлов

Я пытаюсь использовать Apache Xalan для группировки списка вариантов продукта по их productId. Вот пример ввода:

***input_1.xml***
<?xml version="1.0" encoding="utf-8"?>
<root>
    <variant>
        <productId>1</productId>
        <price>100</price>
        <stock unit="item">10</stock>
        <attributes>
            <attribute name="color" value="red" />
        </attributes>
    </variant>
    <variant>
        <productId>1</productId>
        <price>100</price>
        <stock unit="item">8</stock>
        <attributes>
            <attribute name="color" value="blue" />
        </attributes>
    </variant>
    <variant>
        <productId>1</productId>
        <price>150</price>
        <stock unit="item">12</stock>
        <attributes>
            <attribute name="color" value="green" />
        </attributes>
    </variant>
    <variant>
        <productId>2</productId>
        <price>200</price>
        <stock unit="item">7</stock>
        <attributes>
            <attribute name="color" value="purple" />
            <attribute name="material" value="faux-leather" />
        </attributes>
    </variant>
    <variant>
        <productId>3</productId>
        <price>190</price>
        <stock unit="item">22</stock>
        <attributes>
            <attribute name="color" value="yellow" />
            <attribute name="size" value="XL" />
        </attributes>
    </variant>
    <variant>
        <productId>3</productId>
        <price>180</price>
        <stock unit="item">13</stock>
        <attributes>
            <attribute name="color" value="yellow" />
            <attribute name="size" value="L" />
        </attributes>
    </variant>
    <variant>
        <productId>3</productId>
        <price>170</price>
        <stock unit="item">5</stock>
        <attributes>
            <attribute name="color" value="yellow" />
            <attribute name="size" value="M" />
        </attributes>
    </variant>
    <variant>
        <productId>3</productId>
        <price>170</price>
        <stock unit="item">7</stock>
        <attributes>
            <attribute name="color" value="yellow" />
            <attribute name="size" value="S" />
        </attributes>
    </variant>
    <variant>
        <productId>3</productId>
        <price>180</price>
        <stock unit="item">12</stock>
        <attributes>
            <attribute name="color" value="yellow" />
            <attribute name="size" value="XS" />
        </attributes>
    </variant>
</root>

Затем я использую следующую команду из оболочки:

xalan -in input_1.xml -xsl muenchian_1.xsl -out output_1.xml -indent 4

Чтобы преобразовать ввод с помощью следующей таблицы стилей:

***muenchian_1.xml***
<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0"
    xmlns:xsl="http://www.w3.org/1999/XSL/Transform"   >
    <xsl:strip-space elements="*" />
    <xsl:output method="xml" indent="yes"/>
    <xsl:key name="variants-by-productId" match="/root/variant" use="productId"/>
    <xsl:template match="@*|node()" name="identity">
        <xsl:copy>
            <xsl:apply-templates select="@*|node()"/>
        </xsl:copy>
    </xsl:template>
    <xsl:template match="/root/variant[productId][generate-id() =
         generate-id(key('variants-by-productId', productId)[1])]" priority="1">
        <product-listing-group>
            <productId>
                <xsl:value-of select="productId"/>
            </productId>
            <xsl:for-each select="key('variants-by-productId', productId)">
                <xsl:call-template name="identity" />
            </xsl:for-each>
        </product-listing-group>
    </xsl:template>
</xsl:stylesheet>

Ожидается получить следующий результат:

***expected_1.xml***
<?xml version="1.0" encoding="utf-8"?>
<root>
    <product>
        <productId>1</productId>
        <variant>
            <price>100</price>
            <stock unit="item">10</stock>
            <attributes>
                <attribute name="color" value="red" />
            </attributes>
        </variant>
        <variant>
            <price>100</price>
            <stock unit="item">8</stock>
            <attributes>
                <attribute name="color" value="blue" />
            </attributes>
        </variant>
        <variant>
            <price>150</price>
            <stock unit="item">12</stock>
            <attributes>
                <attribute name="color" value="green" />
            </attributes>
        </variant>
    </product>
    <product>
        <productId>2</productId>
        <variant>
            <price>200</price>
            <stock unit="item">7</stock>
            <attributes>
                <attribute name="color" value="purple" />
                <attribute name="material" value="faux-leather" />
            </attributes>
        </variant>
    </product>
    <product>
        <productId>3</productId>
        <variant>
            <price>190</price>
            <stock unit="item">22</stock>
            <attributes>
                <attribute name="color" value="yellow" />
                <attribute name="size" value="XL" />
            </attributes>
        </variant>
        <variant>
            <price>180</price>
            <stock unit="item">13</stock>
            <attributes>
                <attribute name="color" value="L" />
            </attributes>
        </variant>
        <variant>
            <price>170</price>
            <stock unit="item">5</stock>
            <attributes>
                <attribute name="color" value="M" />
            </attributes>
        </variant>
        <variant>
            <price>170</price>
            <stock unit="item">7</stock>
            <attributes>
                <attribute name="color" value="S" />
            </attributes>
        </variant>
        <variant>
            <price>180</price>
            <stock unit="item">12</stock>
            <attributes>
                <attribute name="color" value="XS" />
            </attributes>
        </variant>
    </product>
</root>

но вместо этого я получаю:

***output_1.xml***
<?xml version="1.0" encoding="UTF-8"?>
<root>
    <product-listing-group>
        <productId>1</productId>
        <variant>
            <productId>1</productId>
            <price>100</price>
            <stock unit="item">10</stock>
            <attributes>
                <attribute name="color" value="red"/>
            </attributes>
        </variant>
        <variant>
            <productId>1</productId>
            <price>100</price>
            <stock unit="item">8</stock>
            <attributes>
                <attribute name="color" value="blue"/>
            </attributes>
        </variant>
        <variant>
            <productId>1</productId>
            <price>150</price>
            <stock unit="item">12</stock>
            <attributes>
                <attribute name="color" value="green"/>
            </attributes>
        </variant>
    </product-listing-group>
    <variant>
        <productId>1</productId>
        <price>100</price>
        <stock unit="item">8</stock>
        <attributes>
            <attribute name="color" value="blue"/>
        </attributes>
    </variant>
    <variant>
        <productId>1</productId>
        <price>150</price>
        <stock unit="item">12</stock>
        <attributes>
            <attribute name="color" value="green"/>
        </attributes>
    </variant>
    <product-listing-group>
        <productId>2</productId>
        <variant>
            <productId>2</productId>
            <price>200</price>
            <stock unit="item">7</stock>
            <attributes>
                <attribute name="color" value="purple"/>
                <attribute name="material" value="faux-leather"/>
            </attributes>
        </variant>
    </product-listing-group>
    <product-listing-group>
        <productId>3</productId>
        <variant>
            <productId>3</productId>
            <price>190</price>
            <stock unit="item">22</stock>
            <attributes>
                <attribute name="color" value="yellow"/>
                <attribute name="size" value="XL"/>
            </attributes>
        </variant>
        <variant>
            <productId>3</productId>
            <price>180</price>
            <stock unit="item">13</stock>
            <attributes>
                <attribute name="color" value="yellow"/>
                <attribute name="size" value="L"/>
            </attributes>
        </variant>
        <variant>
            <productId>3</productId>
            <price>170</price>
            <stock unit="item">5</stock>
            <attributes>
                <attribute name="color" value="yellow"/>
                <attribute name="size" value="M"/>
            </attributes>
        </variant>
        <variant>
            <productId>3</productId>
            <price>170</price>
            <stock unit="item">7</stock>
            <attributes>
                <attribute name="color" value="yellow"/>
                <attribute name="size" value="S"/>
            </attributes>
        </variant>
        <variant>
            <productId>3</productId>
            <price>180</price>
            <stock unit="item">12</stock>
            <attributes>
                <attribute name="color" value="yellow"/>
                <attribute name="size" value="XS"/>
            </attributes>
        </variant>
    </product-listing-group>
    <variant>
        <productId>3</productId>
        <price>180</price>
        <stock unit="item">13</stock>
        <attributes>
            <attribute name="color" value="yellow"/>
            <attribute name="size" value="L"/>
        </attributes>
    </variant>
    <variant>
        <productId>3</productId>
        <price>170</price>
        <stock unit="item">5</stock>
        <attributes>
            <attribute name="color" value="yellow"/>
            <attribute name="size" value="M"/>
        </attributes>
    </variant>
    <variant>
        <productId>3</productId>
        <price>170</price>
        <stock unit="item">7</stock>
        <attributes>
            <attribute name="color" value="yellow"/>
            <attribute name="size" value="S"/>
        </attributes>
    </variant>
    <variant>
        <productId>3</productId>
        <price>180</price>
        <stock unit="item">12</stock>
        <attributes>
            <attribute name="color" value="yellow"/>
            <attribute name="size" value="XS"/>
        </attributes>
    </variant>
</root>

Как вы можете видеть, хотя варианты сгруппированы правильно, все варианты, кроме первого в соответствующих группах, повторяются дважды, один раз внутри группы и один раз вне ее.

Почему это? И как я могу это исправить?


person kaan_a    schedule 03.07.2020    source источник


Ответы (1)


Вам нужно заблокировать обработку для второго, третьего, четвертого, .. варианта в группе, иначе преобразование идентичности по умолчанию скопирует их:

<xsl:template match="/root/variant[productId][not(generate-id() = generate-id(key('variants-by-productId', productId)[1]))]"/>
person Martin Honnen    schedule 03.07.2020
comment
Не могли бы вы также сказать мне, почему преобразование личности не копирует первое? - person kaan_a; 05.07.2020
comment
Что ж, для первого у вас уже есть шаблон, настроенный с помощью xsl:template match="/root/variant[productId][generate-id() = generate-id(key('variants-by-productId', productId)[1])]" для вывода оболочки product-listing-group, поэтому, очевидно, преобразование идентичности не срабатывает, поскольку match="@* | node()" (где здесь может совпадать node()) имеет более низкий приоритет (по умолчанию) как любой match="variant" или match="/root/variant...". Но предикат generate-id() = generate-id(key('variants-by-productId', productId)[1]) в основном является способом XSLT 1 для . is key('...', ...)[1]. - person Martin Honnen; 05.07.2020