I am having difficulty applying templates to an Element which is the result of a selection of Elements by an attribute.
The code is essentially attempting to recurse an Element which contains other elements referenced by a name attribute.
The xml 'design' is not mine to change. The goal is to figure out how many widgets (eg: data fields) it would take to represent all the fields and 'contained' fields in a "page" Element.
Given has_depth.xml:
<?xml version="1.0" encoding="UTF-8"?>
<root>
<page name="Locale" page_type="VIRTUAL">
<datum name="lat"><type name="double" complex='0'/></datum>
<datum name="lon"><type name="double" complex='0'/></datum>
</page>
<page name="how_many_fields_do_i_have" page_type="CONCRETE">
<datum mnemonic="foo"><type name="int32" complex='0'/></datum>
<datum mnemonic="foo_locale"><type name="Locale" complex='1'/></datum>
</page>
</root>
and find_depth.xslt:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="text" indent="yes"/>
<xsl:template match="/root">
<xsl:apply-templates select="//page[@name='how_many_fields_do_i_have']" />
</xsl:template>
<xsl:template match="page" >
<xsl:message><xsl:value-of select="concat('getting depth for page ',@name)"/></xsl:message>
<xsl:variable name="depth">
<!-- ask the Page's first datum for its depth -->
<xsl:apply-templates select="./datum[1]" />
</xsl:variable>
<xsl:message><xsl:value-of select="concat('It would take ',$depth,' widgets to show this page in a gui')"/></xsl:message>
</xsl:template>
<xsl:template match="datum[./type[@complex='0']]" >
<xsl:message> <xsl:value-of select="concat('matched simpletype:',current()/type/@name,':',@mnemonic)"/></xsl:message>
<xsl:variable name="below">
<xsl:choose>
<xsl:when test='current()=last()'><xsl:value-of select='0'/></xsl:when>
<xsl:otherwise><xsl:apply-templates select="following-sibling::datum"/></xsl:otherwise>
</xsl:choose>
</xsl:variable>
<xsl:message> <xsl:value-of select="concat('depth after simple ',@mnemonic,': ', 1 + $below)"/></xsl:message>
<xsl:value-of select='1 + $below'/>
</xsl:template>
<xsl:template match="datum[./type[@complex='1']]" >
<xsl:message> <xsl:value-of select="concat('matched complextype:',./type/@name,':',@mnemonic)"/></xsl:message>
<!-- start depth again recursing into the page[type="'VIRTUAL'"] -->
<xsl:apply-templates select="//page[@name='./type/@name']" />
<xsl:if test='current()!=last()'>
<xsl:apply-templates select="following-sibling::datum"/>
</xsl:if>
</xsl:template>
I get the following when the bash command: xsltproc find_depth.xslt has_depth.xml
getting depth for page how_many_fields_do_i_have
matched simpletype:int32:foo
matched complextype:Locale:foo_locale
depth after simple foo: NaN
It would take NaN widgets to show this page in a gui
Where I would expect the output:
getting depth for page how_many_fields_do_i_have
matched simpletype:int32:foo
matched complextype:Locale:foo_locale
getting depth for page Locale
matched simpletype:double:lat
matched simpletype:double:lon
depth after simple lon: 1
depth after simple lat: 2
depth after simple foo: 3
It would take 3 widgets to show this page in a gui
It looks like things go sour when I try to apply-templates
to Locale
after selecting it by ./type/@name
which evaluations in the message statement too "Locale" (see between colons in the 3rd line of the output statement.) However to be sure I linted using xmllint --xpath "//page[@name='Locale']" has_depth.xml
and found:
xmllint --xpath "//page[@name='Locale']" has_depth.xml
<page name="Locale" page_type="VIRTUAL">
<datum name="lat"><type name="double" complex="0"/></datum>
<datum name="lon"><type name="double" complex="0"/></datum>
</page>
which is as I expect.
Questions is: How can I get this thing to recurse as one would expect and treat the sums as numbers as recusion bubbles up?
I find this very confusing, but maybe the following stylesheet can give you some starting point:
XSLT 1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:key name="page" match="page" use="@name" />
<xsl:template match="/root">
<xsl:variable name="temp">
<xsl:apply-templates select="page[@name='how_many_fields_do_i_have']"/>
</xsl:variable>
<result>
<xsl:value-of select="string-length($temp)"/>
</result>
</xsl:template>
<xsl:template match="type[@complex='0']">x</xsl:template>
<xsl:template match="type[@complex='1']">
<xsl:apply-templates select="key('page', @name)"/>
</xsl:template>
</xsl:stylesheet>
Applied to your input example, this will produce:
Result
<?xml version="1.0" encoding="UTF-8"?>
<result>3</result>
Of course, this is no more than a wild guess intended more to clarify the problem than to solve it.