質問
サードパーティのソフトウェアからの出力があります:
ソフトウェアからのサンプルXML:
<testsuite name="XYZ">
<testcase name="ABC" status="0" time="12.001">Some stuff</testcase>
<testcase name="DEF" status="0" time="12,345.001">Some stuff</testcase>
<testcase name="GHI" status="0" time="4,321.001">Some stuff</testcase>
</testsuite>
これを次のように変換するXSLTを作成する必要があります。
<testsuite name="XYZ" time="16678.003">
<testcase name="ABC" time="12.001">Some stuff</testcase>
<testcase name="DEF" time="12,345.001">Some stuff</testcase>
<testcase name="GHI" time="4,321.001">Some stuff</testcase>
</testsuite>
そして testsuite 要素の time 属性を除いて、ほとんどそこに到達しました。合計を取得する代わりに、 NaN を取得しています。 これを取得するために使用しているXPath式は、 sum(// testsuite / @ time)
です。すべての時間が<!> ltの場合、エラーは発生しません。 1000。これはおそらく、XSLTがコンマを検出したときに数値を解析しないためです。 (入力からこれらのコンマを取り除くことはできません。なぜなら、それはサードパーティのソフトウェアから来ているからです。)
では、これらの値を時間で合計するにはどうすればよいですか?すぐにカンマを削除できるように、 sum(// testsuite / @ time)を修正する余地はありますか?
ありがとう!
解決
この再帰的なテンプレートを使用して、すべての時間属性を一緒に追加します。
<xsl:template name="sumoftime">
<xsl:param name="node"/>
<xsl:param name="sum" />
<xsl:choose>
<xsl:when test="$node">
<xsl:call-template name="sumoftime">
<xsl:with-param name="node" select="$node/following-sibling::testcase[1]"/>
<xsl:with-param name="sum" select="$sum + number(translate($node/@time, ',', ''))"/>
</xsl:call-template>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="$sum"/>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
そしてtestsuite要素で、上記のテンプレートを呼び出します
<xsl:element name="testsuite">
...
<xsl:attribute name="{'time'}">
<xsl:call-template name="sumoftime">
<xsl:with-param name="node" select="testcase[1]"/>
<xsl:with-param name="sum" select="0" />
</xsl:call-template>
</xsl:attribute>
</xsl:element>
やや複雑ですが、XSLT 1.0に制限されている場合は、これが道のりになると思います。
テンプレートのクレジットはJelovirtに送信する必要があります。これは、別の質問への回答からのわずかな変更に過ぎないためです。 stackoverflow.com/questions/647991/summing-numbers-with-comma-as-decimal-separator-in-xslt
他のヒント
XSLTの translate
関数を次のように使用します:
sum(translate(//testsuite/@time,',',''))