Come aggiungere orari in XSLT?
Domanda
Ho alcuni output da software di terze parti:
XML di esempio dal software:
<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>
Devo scrivere un XSLT che lo trasforma nel seguente:
<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>
E ci sono quasi arrivato, ad eccezione dell'attributo time dell'elemento testuite . Invece di ottenere il totale, sto ottenendo NaN . L'espressione XPath che sto usando per ottenere questo è sum(//testsuite/@time)
Si noti che l'errore non si verifica quando tutte le volte sono < 1000. Questo probabilmente perché XSLT non analizza il numero quando incontra virgole. (Non riesco a liberarmi di queste virgole dall'input perché proviene da software di terze parti.)
Quindi, come posso sommare questi valori per tempo? È possibile modificare la somma (// testuite / @ time) in modo che sia in grado di rimuovere le virgole al volo?
Grazie!
Soluzione
Usa questo modello ricorsivo per aggiungere tutti gli attributi del tempo:
<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>
E poi nell'elemento testuite, effettua una chiamata al modello sopra
<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>
Piuttosto complicato, ma se sei limitato a XSLT 1.0, credo che questa sarebbe la strada da percorrere.
Il credito per il modello dovrebbe andare a Jelovirt, in quanto è solo una leggera mod dalla sua risposta a un'altra domanda: stackoverflow.com/questions/647991/summing-numbers-with-comma-as-decimal-separator-in-xslt
Altri suggerimenti
Usa la translate
funzione di XSLT in questo modo:
sum(translate(//testsuite/@time,',',''))