我怎样才能使用 xslt 获得所有 xml 项匹配复杂条件的计数并在终止语句中使用它?
How can I can get a count of all xml items matching complex conditions using xslt and use it in a terminate statement?
我需要过滤项目列表,并且只保留日期元素比另一个日期元素大一天以上的项目。然后我还需要知道还有多少筛选项,如果有 none,则终止整个过程。我已经简化了我的数据,但它或多或少是这样的:
<data>
<person>
<name>Tyler</name>
</person>
<items>
<item>
<title>A</title>
<start_date>10/31/2021</start_date>
<end_date>11/01/2021</end_date>
</item>
<item>
<title>B</title>
<start_date>08/05/2021</start_date>
<end_date>08/10/2021</end_date>
</item>
<item>
<title>C</title>
<start_date>09/04/2021</start_date>
<end_date>09/05/2021</end_date>
</item>
</items>
</data>
因此,在该示例中,将只保留 B 并发送消息。但是如果 B 是
<item>
<title>B</title>
<start_date>08/05/2021</start_date>
<end_date>08/06/2021</end_date>
</item>
邮件无法发送。
到目前为止,我已经找到了一种使用建议的方法 here 来转换文本日期的方法。这适用于过滤列表,但我不知道如何确定结果列表中是否包含任何元素,然后如何在终止语句中使用它。任何帮助将不胜感激,并在此先感谢您!这是我在 xsl 上的位置:
<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="/data">
<xsl:if test="???? Whatever can figure out the number of elements left ????">
<xsl:message terminate="yes">There are no items left</xsl:message>
</xsl:if>
<html>
<head>
<title></title>
</head>
<body>
<p>
<xsl:text>Name: </xsl:text>
<xsl:value-of select="person/name"/>
</p>
<table>
<thead>
<tr><th>Title</th></tr>
</thead>
<xsl:for-each select="items/item">
<xsl:variable name="JDN_start_date">
<xsl:call-template name="JDN">
<xsl:with-param name="date" select="start_date" />
</xsl:call-template>
</xsl:variable>
<xsl:variable name="JDN_end_date">
<xsl:call-template name="JDN">
<xsl:with-param name="date" select="end_date" />
</xsl:call-template>
</xsl:variable>
<xsl:if test="($JDN_end_date - $JDN_start_date) > 1">
<tr>
<td><xsl:value-of select="title"/></td>
</tr>
</xsl:if>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
<xsl:template name="JDN"> <!-- Date string to Julian day number -->
<xsl:param name="date"/>
<xsl:param name="year" select="substring($date, 7, 4)"/>
<xsl:param name="month" select="substring($date, 1, 2)"/>
<xsl:param name="day" select="substring($date, 4, 2)"/>
<xsl:param name="a" select="floor((14 - $month) div 12)"/>
<xsl:param name="y" select="$year + 4800 - $a"/>
<xsl:param name="m" select="$month + 12*$a - 3"/>
<xsl:value-of select="$day + floor((153*$m + 2) div 5) + 365*$y + floor($y div 4) - floor($y div 100) + floor($y div 400) - 32045" />
</xsl:template>
</xsl:stylesheet>
考虑以下示例:
XSLT 1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:exsl="http://exslt.org/common"
extension-element-prefixes="exsl">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:template match="/data">
<!-- 1. calculate durations -->
<xsl:variable name="items">
<xsl:for-each select="items/item">
<xsl:copy>
<xsl:attribute name="duration">
<xsl:call-template name="date-difference">
<xsl:with-param name="date1" select="start_date" />
<xsl:with-param name="date2" select="end_date" />
</xsl:call-template>
</xsl:attribute>
<xsl:copy-of select="*"/>
</xsl:copy>
</xsl:for-each>
</xsl:variable>
<!-- 2. find items with duration > 1 -->
<xsl:variable name="pass-items" select="exsl:node-set($items)/item[@duration > 1]" />
<xsl:if test="not($pass-items)">
<xsl:message terminate="yes">no items pass</xsl:message>
</xsl:if>
<!-- 3. output -->
<output>
<xsl:copy-of select="$pass-items"/>
</output>
</xsl:template>
<xsl:template name="date-difference">
<xsl:param name="date1"/>
<xsl:param name="date2"/>
<xsl:param name="JDN1">
<xsl:call-template name="JDN">
<xsl:with-param name="date" select="$date1" />
</xsl:call-template>
</xsl:param>
<xsl:param name="JDN2">
<xsl:call-template name="JDN">
<xsl:with-param name="date" select="$date2" />
</xsl:call-template>
</xsl:param>
<xsl:value-of select="$JDN2 - $JDN1"/>
</xsl:template>
<xsl:template name="JDN">
<xsl:param name="date"/>
<xsl:param name="year" select="substring($date, 7, 4)"/>
<xsl:param name="month" select="substring($date, 1, 2)"/>
<xsl:param name="day" select="substring($date, 4, 2)"/>
<xsl:param name="a" select="floor((14 - $month) div 12)"/>
<xsl:param name="y" select="$year + 4800 - $a"/>
<xsl:param name="m" select="$month + 12*$a - 3"/>
<xsl:value-of select="$day + floor((153*$m + 2) div 5) + 365*$y + floor($y div 4) - floor($y div 100) + floor($y div 400) - 32045" />
</xsl:template>
</xsl:stylesheet>
我需要过滤项目列表,并且只保留日期元素比另一个日期元素大一天以上的项目。然后我还需要知道还有多少筛选项,如果有 none,则终止整个过程。我已经简化了我的数据,但它或多或少是这样的:
<data>
<person>
<name>Tyler</name>
</person>
<items>
<item>
<title>A</title>
<start_date>10/31/2021</start_date>
<end_date>11/01/2021</end_date>
</item>
<item>
<title>B</title>
<start_date>08/05/2021</start_date>
<end_date>08/10/2021</end_date>
</item>
<item>
<title>C</title>
<start_date>09/04/2021</start_date>
<end_date>09/05/2021</end_date>
</item>
</items>
</data>
因此,在该示例中,将只保留 B 并发送消息。但是如果 B 是
<item>
<title>B</title>
<start_date>08/05/2021</start_date>
<end_date>08/06/2021</end_date>
</item>
邮件无法发送。
到目前为止,我已经找到了一种使用建议的方法 here 来转换文本日期的方法。这适用于过滤列表,但我不知道如何确定结果列表中是否包含任何元素,然后如何在终止语句中使用它。任何帮助将不胜感激,并在此先感谢您!这是我在 xsl 上的位置:
<?xml version="1.0" encoding="utf-8"?>
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="/data">
<xsl:if test="???? Whatever can figure out the number of elements left ????">
<xsl:message terminate="yes">There are no items left</xsl:message>
</xsl:if>
<html>
<head>
<title></title>
</head>
<body>
<p>
<xsl:text>Name: </xsl:text>
<xsl:value-of select="person/name"/>
</p>
<table>
<thead>
<tr><th>Title</th></tr>
</thead>
<xsl:for-each select="items/item">
<xsl:variable name="JDN_start_date">
<xsl:call-template name="JDN">
<xsl:with-param name="date" select="start_date" />
</xsl:call-template>
</xsl:variable>
<xsl:variable name="JDN_end_date">
<xsl:call-template name="JDN">
<xsl:with-param name="date" select="end_date" />
</xsl:call-template>
</xsl:variable>
<xsl:if test="($JDN_end_date - $JDN_start_date) > 1">
<tr>
<td><xsl:value-of select="title"/></td>
</tr>
</xsl:if>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
<xsl:template name="JDN"> <!-- Date string to Julian day number -->
<xsl:param name="date"/>
<xsl:param name="year" select="substring($date, 7, 4)"/>
<xsl:param name="month" select="substring($date, 1, 2)"/>
<xsl:param name="day" select="substring($date, 4, 2)"/>
<xsl:param name="a" select="floor((14 - $month) div 12)"/>
<xsl:param name="y" select="$year + 4800 - $a"/>
<xsl:param name="m" select="$month + 12*$a - 3"/>
<xsl:value-of select="$day + floor((153*$m + 2) div 5) + 365*$y + floor($y div 4) - floor($y div 100) + floor($y div 400) - 32045" />
</xsl:template>
</xsl:stylesheet>
考虑以下示例:
XSLT 1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:exsl="http://exslt.org/common"
extension-element-prefixes="exsl">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:template match="/data">
<!-- 1. calculate durations -->
<xsl:variable name="items">
<xsl:for-each select="items/item">
<xsl:copy>
<xsl:attribute name="duration">
<xsl:call-template name="date-difference">
<xsl:with-param name="date1" select="start_date" />
<xsl:with-param name="date2" select="end_date" />
</xsl:call-template>
</xsl:attribute>
<xsl:copy-of select="*"/>
</xsl:copy>
</xsl:for-each>
</xsl:variable>
<!-- 2. find items with duration > 1 -->
<xsl:variable name="pass-items" select="exsl:node-set($items)/item[@duration > 1]" />
<xsl:if test="not($pass-items)">
<xsl:message terminate="yes">no items pass</xsl:message>
</xsl:if>
<!-- 3. output -->
<output>
<xsl:copy-of select="$pass-items"/>
</output>
</xsl:template>
<xsl:template name="date-difference">
<xsl:param name="date1"/>
<xsl:param name="date2"/>
<xsl:param name="JDN1">
<xsl:call-template name="JDN">
<xsl:with-param name="date" select="$date1" />
</xsl:call-template>
</xsl:param>
<xsl:param name="JDN2">
<xsl:call-template name="JDN">
<xsl:with-param name="date" select="$date2" />
</xsl:call-template>
</xsl:param>
<xsl:value-of select="$JDN2 - $JDN1"/>
</xsl:template>
<xsl:template name="JDN">
<xsl:param name="date"/>
<xsl:param name="year" select="substring($date, 7, 4)"/>
<xsl:param name="month" select="substring($date, 1, 2)"/>
<xsl:param name="day" select="substring($date, 4, 2)"/>
<xsl:param name="a" select="floor((14 - $month) div 12)"/>
<xsl:param name="y" select="$year + 4800 - $a"/>
<xsl:param name="m" select="$month + 12*$a - 3"/>
<xsl:value-of select="$day + floor((153*$m + 2) div 5) + 365*$y + floor($y div 4) - floor($y div 100) + floor($y div 400) - 32045" />
</xsl:template>
</xsl:stylesheet>