XSLT pour transformer un XML avec des fractures répétées en un fichier plat
-
29-10-2019 - |
Question
J'ai un XML comme ci-dessous:
<?xml version="1.0" encoding="utf-8"?>
<GetSavedReportResponse>
<ResponseType>Success</ResponseType>
<FileModifiedDateTime>2012-01-03T17:05:04</FileModifiedDateTime>
<FileSizeBytes>7816</FileSizeBytes>
<FileDataFormat>XML</FileDataFormat>
<FileData>
<Zthes>
<term>
<termId>49555</termId>
<termUpdate>add</termUpdate>
<termName>Active Personnel</termName>
<termVocabulary>People Status Global</termVocabulary>
<termVocabulary>Global People Status</termVocabulary>
<termCategory>PDA</termCategory>
<termCategory>PDI</termCategory>
<termCategory>GLB</termCategory>
<relation weight="100">
<termId>49556</termId>
<relationType>EQ</relationType>
<termName>term name</termName>
<termVocabulary>term vocabulary</termVocabulary>
</relation>
<relation weight="100">
<termId>49557</termId>
<relationType>BT</relationType>
<termName>General Active Personnel</termName>
<termVocabulary>People Status Global Updated</termVocabulary>
</relation>
</term>
<term>
<termId>49556</termId>
<termUpdate>add</termUpdate>
<termName>Leave of Absence Personnel</termName>
<termVocabulary>People Status Global</termVocabulary>
<termCategory>GLB</termCategory>
<termCategory>PDI</termCategory>
<relation weight="100">
<relationType>BT</relationType>
<termId>49554</termId>
<termName>General Non-Active Personnel</termName>
<termVocabulary>People Status Global</termVocabulary>
</relation>
</term>
</Zthes>
</FileData>
</GetSavedReportResponse>
J'ai besoin de le transformer en fichier plat. Pour cela, j'ai écrit le XSL suivant
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
<xsl:output method="text" />
<xsl:template match="Zthes">
<xsl:text> </xsl:text>
<xsl:for-each select="term">
<xsl:text>"</xsl:text>
<xsl:text>GL</xsl:text>
<xsl:text>"</xsl:text>
<xsl:text>;</xsl:text>
<xsl:text>"</xsl:text>
<xsl:for-each select="termCategory">
<xsl:value-of select="." />
</xsl:for-each>
<xsl:text>"</xsl:text>
<xsl:text>;</xsl:text>
<xsl:text>"</xsl:text>
<xsl:for-each select="termVocabulary">
<xsl:value-of select="." />
</xsl:for-each>
<xsl:text>"</xsl:text>
<xsl:text>;</xsl:text>
<xsl:text>"</xsl:text>
<xsl:for-each select="relation/termVocabulary">
<xsl:value-of select="." />
</xsl:for-each>
<xsl:text>"</xsl:text>
<xsl:text> </xsl:text>
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>
Donc, la sortie doit être
"HDR"; "texte"; "20120112045620"; "F"
"GL"; "PDA"; "Statut de People Global"; "Term Vocabulary"
"GL"; "PDA"; "Statut des gens Global"; "Statut des gens Global Met à jour"
"GL"; "PDA"; "Global People Status"; "Terme Vocabulary"
"GL"; "PDA"; "Global People Status"; "Statut People Global Mise à jour"
"GL"; "PDI"; "Statut de People Global"; "Term Vocabulary"
"GL"; "PDI"; "Statut des gens Global"; "Statut des gens Global Met à jour"
"GL"; "PDI"; "Global People Status"; "Term Vocabulary"
"GL"; "PDI"; "Global People Status"; "Statut People Global Mise à jour"
"GL"; "GLB"; "Statut de People Global"; "Term Vocabulary"
"GL"; "GLB"; "Statut des gens Global"; "Statut des gens Global Met à jour"
"GL"; "GLB"; "Global People Status"; "Terme Vocabulary"
"GL"; "GLB"; "Global People Status"; "Statut People Global Mise à jour"
"Ftr"; 12
Avec mon XSL, je suis une ligne unique:
"GL"; "PdapDiglb"; "Statut des gens GlobalGlobal People Status"; "Term VocabularyPeople Status Global Met à jour"
Et une rangée d'en-tête:
"HDR"; "Piglsstd"; "20120112045620"; "F":
devrait être ajouté au début, avec une ligne de pied de page
"Ftr";
au fond.
La solution
Tu veux quelque chose comme ça:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:my="my:my">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>
<my:defaults>
<termCat/>
<termVocab/>
</my:defaults>
<xsl:variable name="vDefaults" select="document('')/*/my:defaults"/>
<xsl:variable name="vQ">"</xsl:variable>
<xsl:template match="term">
<xsl:variable name="vTerm" select="."/>
<xsl:variable name="vRow1" select="'
"GL";'"/>
<xsl:for-each select=
"termCategory
|
$vDefaults/termCat[not($vTerm/termCategory)]">
<xsl:variable name="vRow2" select=
"concat($vRow1, $vQ, ., $vQ, ';')"/>
<xsl:for-each select=
"$vTerm/termVocabulary
|
$vDefaults/termCat[not($vTerm/termVocabulary)]
">
<xsl:variable name="vRow3" select=
"concat($vRow2, $vQ, ., $vQ, ';')"/>
<xsl:for-each select=
"$vTerm/relation/termVocabulary
|
$vDefaults/termCat[not($vTerm/relation/termVocabulary)]
">
<xsl:value-of select="concat($vRow3, $vQ, ., $vQ, ';')"/>
</xsl:for-each>
</xsl:for-each>
</xsl:for-each>
</xsl:template>
<xsl:template match="text()"/>
</xsl:stylesheet>
Lorsque cette transformation est appliquée sur le document XML fourni:
<GetSavedReportResponse>
<ResponseType>Success</ResponseType>
<FileModifiedDateTime>2012-01-03T17:05:04</FileModifiedDateTime>
<FileSizeBytes>7816</FileSizeBytes>
<FileDataFormat>XML</FileDataFormat>
<FileData>
<Zthes>
<term>
<termId>49555</termId>
<termUpdate>add</termUpdate>
<termName>Active Personnel</termName>
<termVocabulary>People Status Global</termVocabulary>
<termVocabulary>Global People Status</termVocabulary>
<termCategory>PDA</termCategory>
<termCategory>PDI</termCategory>
<termCategory>GLB</termCategory>
<relation weight="100">
<termId>49556</termId>
<relationType>EQ</relationType>
<termName>term name</termName>
<termVocabulary>term vocabulary</termVocabulary>
</relation>
<relation weight="100">
<termId>49557</termId>
<relationType>BT</relationType>
<termName>General Active Personnel</termName>
<termVocabulary>People Status Global Updated</termVocabulary>
</relation>
</term>
<term>
<termId>49556</termId>
<termUpdate>add</termUpdate>
<termName>Leave of Absence Personnel</termName>
<termVocabulary>People Status Global</termVocabulary>
<termCategory>GLB</termCategory>
<termCategory>PDI</termCategory>
<relation weight="100">
<relationType>BT</relationType>
<termId>49554</termId>
<termName>General Non-Active Personnel</termName>
<termVocabulary>People Status Global</termVocabulary>
</relation>
</term>
</Zthes>
</FileData>
</GetSavedReportResponse>
Le résultat bon et le résultat est produit:
"GL";"PDA";"People Status Global";"term vocabulary";
"GL";"PDA";"People Status Global";"People Status Global Updated";
"GL";"PDA";"Global People Status";"term vocabulary";
"GL";"PDA";"Global People Status";"People Status Global Updated";
"GL";"PDI";"People Status Global";"term vocabulary";
"GL";"PDI";"People Status Global";"People Status Global Updated";
"GL";"PDI";"Global People Status";"term vocabulary";
"GL";"PDI";"Global People Status";"People Status Global Updated";
"GL";"GLB";"People Status Global";"term vocabulary";
"GL";"GLB";"People Status Global";"People Status Global Updated";
"GL";"GLB";"Global People Status";"term vocabulary";
"GL";"GLB";"Global People Status";"People Status Global Updated";
"GL";"GLB";"People Status Global";"People Status Global";
"GL";"PDI";"People Status Global";"People Status Global";
Explication: Vous voulez faire la sortie uniquement lorsqu'une ligne complète a été formée - pas avant cela.
Mise à jour: L'OP fonctionne dans un environnement, où le document()
La fonction est désactivée. Il veut également une en-tête et un pied de page.
Dans ce cas, une transformation légèrement modifiée peut être utilisée (en utilisant le exslt:node-set()
fonction d'extension) maintenant:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:ext="http://exslt.org/common">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:variable name="vrtfDefaults">
<termCat/>
<termVocab/>
</xsl:variable>
<xsl:variable name="vDefaults" select=
"ext:node-set($vrtfDefaults)"/>
<xsl:variable name="vQ">"</xsl:variable>
<xsl:template match="Zthes">
<xsl:text>HDR";"PIGLSSTD";"20120112045620";"F":</xsl:text>
<xsl:apply-templates/>
<xsl:text>
FTR</xsl:text>
</xsl:template>
<xsl:template match="term">
<xsl:variable name="vTerm" select="."/>
<xsl:variable name="vRow1" select="'
"GL";'"/>
<xsl:for-each select=
"termCategory
|
$vDefaults/termCat[not($vTerm/termCategory)]">
<xsl:variable name="vRow2" select=
"concat($vRow1, $vQ, ., $vQ, ';')"/>
<xsl:for-each select=
"$vTerm/termVocabulary
|
$vDefaults/termCat[not($vTerm/termVocabulary)]
">
<xsl:variable name="vRow3" select=
"concat($vRow2, $vQ, ., $vQ, ';')"/>
<xsl:for-each select=
"$vTerm/relation/termVocabulary
|
$vDefaults/termCat[not($vTerm/relation/termVocabulary)]
">
<xsl:value-of select="concat($vRow3, $vQ, ., $vQ, ';')"/>
</xsl:for-each>
</xsl:for-each>
</xsl:for-each>
</xsl:template>
<xsl:template match="text()"/>
</xsl:stylesheet>