XML в CSV со всеми детьми в одной строке - PullRequest
0 голосов
/ 14 января 2019

Как я могу экспортировать следующий XML в CSV и иметь дочерние элементы в одной строке:

<?xml version="1.0" encoding="UTF-8"?>
<NetworkEntity xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:ns="3.0">
    <Location>
        <SureName>9143146669213334465</SureName>
        <Type>Location</Type>
        <SubType>Rack</SubType>
        <DisplayName>NAME/O2OR_HUIB_04 / O2OR_HUIN_04 - IP DSLAM Type 5 - Model B</DisplayName>
        <Category>INFRASTRUCTURE</Category>
        <Features>
            <Feature>
                <Name>Site</Name>
                <Value>NAME</Value>
                <DisplayName>Site</DisplayName>
                <Discovered>true</Discovered>
            </Feature>
        </Features>
        <Associations>
            <Association>
                <Target xsi:type="Location">
                    <SureName>9143101104413609244</SureName>
                </Target>
                <Type>
                    <Type>ASSOCIATES_WITH</Type>
                </Type>
            </Association>
        </Associations>
    </Location>
</NetworkEntity>

Мне удалось экспортировать его следующим образом, но я не могу добавить остальные:

SureName,Type,SubType,DisplayName,Category,Name,Value,Discovered
9143146669213334465,Location,Rack,NAME/O2OR_HUIB_04 / O2OR_HUIN_04 - IP DSLAM Type 5 - Model B,INFRASTRUCTURE,,,
9143147170713655610,Location,Rack,NAME/12PT_HUIB_06 / 12PT_HUIN_06 - IP DSLAM Type 5 - Model B,INFRASTRUCTURE,,,
9143147288813732088,Location,Rack,NAME/74GW_HUIB_04 / 74GW_HUIN_04 - IP DSLAM Type 5 - Model B,INFRASTRUCTURE,,,
9143155460813423214,Location,Rack,NAME/O2HX_HUIB_02 / O2HX_HUIN_02 - IP DSLAM Type 5 - Model B,INFRASTRUCTURE,,,

XSLT:

<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="text" />
<xsl:variable name="delimiter" select="','" />
<!-- define an array containing the fields we are interested in -->

<xsl:variable name="mainAttributesArray">
    <field>SureName</field>
    <field>Type</field>
    <field>SubType</field>
    <field>DisplayName</field>
    <field>Category</field>
    <field>Name</field> 
    <field>Value</field>    
    <field>Discovered</field>
</xsl:variable>

<xsl:param name="fields" select="document('')/*/xsl:variable[@name='mainAttributesArray']/*" />

<xsl:template match="/">
  <!-- output the header row -->
  <xsl:for-each select="$fields">
    <xsl:if test="position() != 1">
      <xsl:value-of select="$delimiter"/>
    </xsl:if>
    <xsl:value-of select="." />
  </xsl:for-each>
  <!-- output newline -->
  <xsl:text>
</xsl:text>
  <xsl:apply-templates select=".//NetworkEntity/Location"/>
</xsl:template>

<xsl:template match="Location">
  <xsl:variable name="currNode" select="." />
  <!-- output the data row -->
  <!-- loop over the field names and find the value of each one in the xml -->
  <xsl:for-each select="$fields">
    <xsl:if test="position() != 1">
      <xsl:value-of select="$delimiter"/>
    </xsl:if>
    <xsl:value-of select="$currNode/*[name() = current()]" />
    <xsl:for-each select="Features/Feature">
    <xsl:text>test</xsl:text>
        <xsl:value-of select="name()" />
    </xsl:for-each>

  </xsl:for-each>
<!-- output newline -->
  <xsl:text>&#xa;</xsl:text>
</xsl:template>
</xsl:stylesheet>

Ожидаемый результат:

SureName,Type,SubType,DisplayName,Category,Name,Value,Discovered,AssociationTarget,AssociationSureName,AssociationType
9143146669213334465,Location,Rack,NAME/O2OR_HUIB_04 / O2OR_HUIN_04 - IP DSLAM Type 5 - Model B,INFRASTRUCTURE,Site,NAME,Site,True,Location,9143101104413609244,ASSOCIATES_WITH

Спасибо!

1 Ответ

0 голосов
/ 15 января 2019

Почему бы вам просто не сделать:

XSLT 1.0

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance">
<xsl:output method="text"/>

<xsl:template match="/NetworkEntity">
    <!-- header (Location + Features + Associations) -->
    <xsl:text>SureName,Type,SubType,DisplayName,Category,</xsl:text>
    <xsl:text>Name,Value,Discovered,</xsl:text>
    <xsl:text>AssociationTarget,AssociationSureName,AssociationType&#10;</xsl:text>
    <!-- data -->
    <xsl:for-each select="Location">
        <!-- Location -->
        <xsl:value-of select="SureName"/>
        <xsl:text>,</xsl:text>
        <xsl:value-of select="Type"/>
        <xsl:text>,</xsl:text>
        <xsl:value-of select="SubType"/>
        <xsl:text>,</xsl:text>
        <xsl:value-of select="DisplayName"/>
        <xsl:text>,</xsl:text>
        <xsl:value-of select="Category"/>
        <xsl:text>,</xsl:text>
        <!-- Features -->       
        <xsl:value-of select="Features/Feature/Name"/>
        <xsl:text>,</xsl:text>
        <xsl:value-of select="Features/Feature/Value"/>
        <xsl:text>,</xsl:text>
        <xsl:value-of select="Features/Feature/Discovered"/>
        <xsl:text>,</xsl:text>
        <!-- Associations -->
        <xsl:value-of select="Associations/Association/Target/@xsi:type"/>
        <xsl:text>,</xsl:text>
        <xsl:value-of select="Associations/Association/Target/SureName"/>
        <xsl:text>,</xsl:text>
        <xsl:value-of select="Associations/Association/Type/Type"/>
        <xsl:text>&#10;</xsl:text>
    </xsl:for-each>
</xsl:template>

</xsl:stylesheet>

Обратите внимание, что здесь мы предполагаем, что у каждого Location будет не более одного Feature и одного Association - но структура вашего XML допускает несколько экземпляров каждого.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...