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!

È stato utile?

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

scroll top