XSLT 1.0 - не указывать дубликаты и условия - PullRequest
0 голосов
/ 02 ноября 2018

с помощью другого форума мне удалось удалить дубликаты в XML с помощью XSLT, к сожалению, я не могу получить здесь больше.

Справочная информация. В моем XML дубликаты часто выводятся под узлом HeaderInformation в HeaderText. Нам удалось решить эту проблему, но в настоящее время узел HeaderText ДОЛЖЕН всегда присутствовать в HeaderInformation. - что не всегда так ...

Я хотел бы добавить расширение (когда ?, если?, Иначе?, ...), чтобы это решение XSLT выполнялось только в том случае, если узел HeaderText фактически присутствует в XML, в противном случае все должно остаться без изменений.

Мой XML с HeaderText

<?xml version="1.0" encoding="utf-8"?><SALESINVOICE>
  <Interchange>
    <Recipient></Recipient>
    <Sender></Sender>
    <CreationDate></CreationDate>
    <Test></Test>
    <Interchange_Control_Number></Interchange_Control_Number>
    <HeaderInformation>
	<HeaderText>
        <Qualifier>AAI</Qualifier>
        <Text>AAI10854</Text>
      </HeaderText><HeaderText>
        <Qualifier>DEL</Qualifier>
        <Text>DEL0854</Text>
      </HeaderText>
	  <HeaderText>
        <Qualifier>AAI</Qualifier>
        <Text>AAI10854</Text>
      </HeaderText><HeaderText>
        <Qualifier>DEL</Qualifier>
        <Text>DEL0854</Text>
      </HeaderText>
      <OrigInvoiceNumber></OrigInvoiceNumber>
      <InvoiceType></InvoiceType>
      <InvoiceDate></InvoiceDate>
      <InvoiceNumber></InvoiceNumber>      
      <CustomerOrderReference></CustomerOrderReference>
      <SalesOrderReference></SalesOrderReference>
      <SalesOrderDate></SalesOrderDate>
      <PackingslipId></PackingslipId>
      <DeliveryDate></DeliveryDate>
      <Currency></Currency>
      <TaxExempt></TaxExempt>      
      <Contact>
        <Contact></Contact>        
      </Contact>
      <TermsOfPayment>
        <PaymTermDescription></PaymTermDescription>
        <DueDate></DueDate>
      </TermsOfPayment>
      <CashDiscountConditions>        
      </CashDiscountConditions>
      <WeightAndVolume>
        <NetWeight></NetWeight>
        <UnifOfWeight></UnifOfWeight>
      </WeightAndVolume>
      <SupplierAddressInformation>
        <GLN></GLN>
        <Name></Name>
        <Street></Street>
        <ZipCode></ZipCode>
        <City></City>
        <Country></Country>        
        <VATNum></VATNum>        
      </SupplierAddressInformation>
      <BuyerAddressInformation>
        <GLN></GLN>
        <Name></Name>
        <Street></Street>
        <ZipCode></ZipCode>
        <City></City>
        <Country></Country>
        <VATNum></VATNum>
        <InternalNumber></InternalNumber>
      </BuyerAddressInformation>
      <InvoiceAddressInformation>
        <GLN></GLN>
        <Name></Name>
        <Street></Street>
        <ZipCode></ZipCode>
        <City></City>
        <Country></Country>
        <VATNum></VATNum>
        <InternalNumber></InternalNumber>
      </InvoiceAddressInformation>
      <DeliveryAddressInformation>
        <GLN></GLN>
        <Name></Name>
        <Street></Street>
        <ZipCode></ZipCode>
        <City></City>
        <Country></Country>
        <State></State>
        <VATNum></VATNum>
        <InternalNumber></InternalNumber>
      </DeliveryAddressInformation>
      <TransportDetails>        
        <DeliveryTerms></DeliveryTerms>        
      </TransportDetails>
      <LineItem>
        <Lines>
          <OrderLineNum>1</OrderLineNum>          
          <GTIN></GTIN>
          <GTINDescription></GTINDescription>
          <SupplierArticleNumber></SupplierArticleNumber>          
          <Quantity></Quantity>
          <PriceUnit></PriceUnit>
          <FixedPrice></FixedPrice>
          <SalesPrice></SalesPrice>          
          <DiscAmount></DiscAmount>
          <DiscPercent></DiscPercent>
          <SalesLinePercent1></SalesLinePercent1>
          <SalesLinePercent2></SalesLinePercent2>
          <MultiLnDisc></MultiLnDisc>
          <MultiLnPercent></MultiLnPercent>
          <LineAmount></LineAmount>
          <SumLineDisc></SumLineDisc>
          <DeliveredQuantity></DeliveredQuantity>
          <VatBaseAmount></VatBaseAmount>
          <VatPercentage></VatPercentage>
          <Measure_Unit></Measure_Unit>
		  <LineTextt>
			<Qualifier>LIN</Qualifier>
			<Text>2534</Text>
		  </LineTextt>
        </Lines>
        <Lines>
          <OrderLineNum>2</OrderLineNum>          
          <GTIN></GTIN>
          <GTINDescription></GTINDescription>
          <SupplierArticleNumber></SupplierArticleNumber>          
          <Quantity></Quantity>
          <PriceUnit></PriceUnit>
          <FixedPrice></FixedPrice>
          <SalesPrice></SalesPrice>          
          <DiscAmount></DiscAmount>
          <DiscPercent></DiscPercent>
          <SalesLinePercent1></SalesLinePercent1>
          <SalesLinePercent2></SalesLinePercent2>
          <MultiLnDisc></MultiLnDisc>
          <MultiLnPercent></MultiLnPercent>
          <LineAmount></LineAmount>
          <SumLineDisc></SumLineDisc>
          <DeliveredQuantity></DeliveredQuantity>
          <VatBaseAmount></VatBaseAmount>
          <VatPercentage></VatPercentage>
          <Measure_Unit></Measure_Unit>
		  <LineTextt>
			<Qualifier>LIN</Qualifier>
			<Text>234</Text>
			</LineTextt>
        </Lines>
      </LineItem>
      <Totals>
        <InvoiceTotal></InvoiceTotal>
        <NetAmount></NetAmount>
        <VATAmount></VATAmount>
        <DiscountAmount></DiscountAmount>
        <AllowanceOrChargeAmount></AllowanceOrChargeAmount>
        <DueDate></DueDate>
        <VAT>
          <VATS>
            <Percentage></Percentage>
            <Type></Type>
            <BaseAmount></BaseAmount>
            <VATAmount></VATAmount>
          </VATS>
        </VAT>
      </Totals>
    </HeaderInformation>
  </Interchange>
</SALESINVOICE>

Мой XML без HeaderText

<?xml version="1.0" encoding="utf-8"?><SALESINVOICE>
  <Interchange>
    <Recipient></Recipient>
    <Sender></Sender>
    <CreationDate></CreationDate>
    <Test></Test>
    <Interchange_Control_Number></Interchange_Control_Number>
    <HeaderInformation>	
      <OrigInvoiceNumber></OrigInvoiceNumber>
      <InvoiceType></InvoiceType>
      <InvoiceDate></InvoiceDate>
      <InvoiceNumber></InvoiceNumber>      
      <CustomerOrderReference></CustomerOrderReference>
      <SalesOrderReference></SalesOrderReference>
      <SalesOrderDate></SalesOrderDate>
      <PackingslipId></PackingslipId>
      <DeliveryDate></DeliveryDate>
      <Currency></Currency>
      <TaxExempt></TaxExempt>      
      <Contact>
        <Contact></Contact>        
      </Contact>
      <TermsOfPayment>
        <PaymTermDescription></PaymTermDescription>
        <DueDate></DueDate>
      </TermsOfPayment>
      <CashDiscountConditions>        
      </CashDiscountConditions>
      <WeightAndVolume>
        <NetWeight></NetWeight>
        <UnifOfWeight></UnifOfWeight>
      </WeightAndVolume>
      <SupplierAddressInformation>
        <GLN></GLN>
        <Name></Name>
        <Street></Street>
        <ZipCode></ZipCode>
        <City></City>
        <Country></Country>        
        <VATNum></VATNum>        
      </SupplierAddressInformation>
      <BuyerAddressInformation>
        <GLN></GLN>
        <Name></Name>
        <Street></Street>
        <ZipCode></ZipCode>
        <City></City>
        <Country></Country>
        <VATNum></VATNum>
        <InternalNumber></InternalNumber>
      </BuyerAddressInformation>
      <InvoiceAddressInformation>
        <GLN></GLN>
        <Name></Name>
        <Street></Street>
        <ZipCode></ZipCode>
        <City></City>
        <Country></Country>
        <VATNum></VATNum>
        <InternalNumber></InternalNumber>
      </InvoiceAddressInformation>
      <DeliveryAddressInformation>
        <GLN></GLN>
        <Name></Name>
        <Street></Street>
        <ZipCode></ZipCode>
        <City></City>
        <Country></Country>
        <State></State>
        <VATNum></VATNum>
        <InternalNumber></InternalNumber>
      </DeliveryAddressInformation>
      <TransportDetails>        
        <DeliveryTerms></DeliveryTerms>        
      </TransportDetails>
      <LineItem>
        <Lines>
          <OrderLineNum>1</OrderLineNum>          
          <GTIN></GTIN>
          <GTINDescription></GTINDescription>
          <SupplierArticleNumber></SupplierArticleNumber>          
          <Quantity></Quantity>
          <PriceUnit></PriceUnit>
          <FixedPrice></FixedPrice>
          <SalesPrice></SalesPrice>          
          <DiscAmount></DiscAmount>
          <DiscPercent></DiscPercent>
          <SalesLinePercent1></SalesLinePercent1>
          <SalesLinePercent2></SalesLinePercent2>
          <MultiLnDisc></MultiLnDisc>
          <MultiLnPercent></MultiLnPercent>
          <LineAmount></LineAmount>
          <SumLineDisc></SumLineDisc>
          <DeliveredQuantity></DeliveredQuantity>
          <VatBaseAmount></VatBaseAmount>
          <VatPercentage></VatPercentage>
          <Measure_Unit></Measure_Unit>		  
        </Lines>
        <Lines>
          <OrderLineNum>2</OrderLineNum>          
          <GTIN></GTIN>
          <GTINDescription></GTINDescription>
          <SupplierArticleNumber></SupplierArticleNumber>          
          <Quantity></Quantity>
          <PriceUnit></PriceUnit>
          <FixedPrice></FixedPrice>
          <SalesPrice></SalesPrice>          
          <DiscAmount></DiscAmount>
          <DiscPercent></DiscPercent>
          <SalesLinePercent1></SalesLinePercent1>
          <SalesLinePercent2></SalesLinePercent2>
          <MultiLnDisc></MultiLnDisc>
          <MultiLnPercent></MultiLnPercent>
          <LineAmount></LineAmount>
          <SumLineDisc></SumLineDisc>
          <DeliveredQuantity></DeliveredQuantity>
          <VatBaseAmount></VatBaseAmount>
          <VatPercentage></VatPercentage>
          <Measure_Unit></Measure_Unit>		  
        </Lines>
      </LineItem>
      <Totals>
        <InvoiceTotal></InvoiceTotal>
        <NetAmount></NetAmount>
        <VATAmount></VATAmount>
        <DiscountAmount></DiscountAmount>
        <AllowanceOrChargeAmount></AllowanceOrChargeAmount>
        <DueDate></DueDate>
        <VAT>
          <VATS>
            <Percentage></Percentage>
            <Type></Type>
            <BaseAmount></BaseAmount>
            <VATAmount></VATAmount>
          </VATS>
        </VAT>
      </Totals>
    </HeaderInformation>
  </Interchange>
</SALESINVOICE>

Мой 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="xml" version="1.0" encoding="UTF-8" standalone="yes" indent="yes"/>

  <xsl:key name="header_text" match="HeaderText" use="Text"/>
  <xsl:key name="line_text" match="LineText" use="Text"/>

  <!-- Identity-Template für die nicht explizit benannten Elemente -->
  <xsl:template match="@* | node()">
    <xsl:copy>
      <xsl:apply-templates select="@* | node()"/>
    </xsl:copy>
  </xsl:template>

  <xsl:template match="HeaderInformation">
    <HeaderInformation>
      <xsl:copy-of select="HeaderText[1]/preceding-sibling::*"/>
      <xsl:for-each select="HeaderText[generate-id() =
                            generate-id(key('header_text', Text)[1])]">
        <xsl:variable name="akt_key" select="Text"/>
        <xsl:variable name="akt_group" select="key('header_text', $akt_key)"/>
        <xsl:for-each select="$akt_group">
          <xsl:if test="position() = 1">
            <xsl:copy-of select="."/>
          </xsl:if>
        </xsl:for-each>
      </xsl:for-each>
      <xsl:copy-of select="HeaderText[last()]/following-sibling::*[following-sibling::LineItem]"/>
      <xsl:apply-templates select="LineItem"/>
      <xsl:copy-of select="LineItem/following-sibling::*"/>
    </HeaderInformation>
  </xsl:template>

</xsl:stylesheet>

Та же проблема в «Lines» -> «LineText».

Я надеюсь, что вы можете помочь мне здесь, пожалуйста.

СПАСИБО

LG Julian

1 Ответ

0 голосов
/ 02 ноября 2018

Я думаю, что все, что вам нужно сделать, это изменить свой шаблон соответствия, чтобы проверить, существует ли HeaderText узел

 <xsl:template match="HeaderInformation[HeaderText]">

Если HeaderInformation не имеет HeaderText, тогда шаблон идентификации будет соответствовать ему, и тогда все останется неизменным.

Обратите внимание, меня смущают эти три строки

  <xsl:copy-of select="HeaderText[last()]/following-sibling::*[following-sibling::LineItem]"/>
  <xsl:apply-templates select="LineItem"/>
  <xsl:copy-of select="LineItem/following-sibling::*"/>

Похоже, вы пытаетесь выбрать все после HeaderText, но до LineItem. Затем вы выбираете LineItem, а затем все после этого. Но это действительно то же самое, что и сделать это:

 <xsl:apply-templates select="HeaderText[last()]/following-sibling::*"/>

Поскольку у вас есть шаблон идентификации, выполнение xsl:apply-templates здесь аналогично выполнению xsl:copy-of. (И я предполагаю, что для LineItem у вас действительно есть соответствующий шаблон, так что он будет иметь приоритет над шаблоном идентификации).

Кроме того, ваше <xsl:for-each select="HeaderText[generate-id() = generate-id(key('header_text', Text)[1])]"> утверждение, включая его тело, на самом деле можно просто упростить до этого ...

<xsl:copy-of select="HeaderText[generate-id() = generate-id(key('header_text', Text)[1])]" />

Попробуйте это XSLT

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
  <xsl:output method="xml" version="1.0" encoding="UTF-8" standalone="yes" indent="yes"/>

  <xsl:key name="header_text" match="HeaderText" use="Text"/>
  <xsl:key name="line_text" match="LineText" use="Text"/>

  <!-- Identity-Template für die nicht explizit benannten Elemente -->
  <xsl:template match="@* | node()">
    <xsl:copy>
      <xsl:apply-templates select="@* | node()"/>
    </xsl:copy>
  </xsl:template>

  <xsl:template match="HeaderInformation[HeaderText]">
    <HeaderInformation>
      <xsl:apply-templates select="HeaderText[1]/preceding-sibling::*"/>
      <xsl:copy-of select="HeaderText[generate-id() = generate-id(key('header_text', Text)[1])]" />
      <xsl:apply-templates select="HeaderText[last()]/following-sibling::*"/>
    </HeaderInformation>
  </xsl:template>
</xsl:stylesheet>

Сказав это, есть еще более простой подход, который вы можете использовать, и вместо того, чтобы явно копировать отдельные элементы, подумайте об удалении дублирующих элементов.

Попробуйте тоже XSLT

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
  <xsl:output method="xml" version="1.0" encoding="UTF-8" standalone="yes" indent="yes"/>

  <xsl:key name="header_text" match="HeaderText" use="Text"/>
  <xsl:key name="line_text" match="LineText" use="Text"/>

  <!-- Identity-Template für die nicht explizit benannten Elemente -->
  <xsl:template match="@* | node()">
    <xsl:copy>
      <xsl:apply-templates select="@* | node()"/>
    </xsl:copy>
  </xsl:template>

  <xsl:template match="HeaderText[generate-id() != generate-id(key('header_text', Text)[1])]" />
</xsl:stylesheet>

У вас был бы аналогичный шаблон для LineText и здесь.

...