Saxon-js 是否对 xsl:param 执行 XML 语法检查?

Does Saxon-js perform XML syntax checking on a xsl:param?

我 运行 这个 XSL 脚本通过 Saxon-js。它使用在使用 stylesheetParams 的转换调用中收到的 XML 更新主输入 XHTML 上的成本字段。都好。问题是没有对 param-XML 进行语法检查(您可以在注释掉的行中看到它的样子)。它在 XHTML 上并且转换会产生错误,但不会在 param-XML 上产生错误。它只是允许它进入,然后按键功能就不会更新 XHTML。有没有办法在同一个转换调用中检查格式正确的 XML 参数,或者我是否必须使用 2 个转换调用:调用 param-XSL 上的转换进行语法检查,然后调用这个主转换更新 XHTML?

<xsl:stylesheet version="3.0"
  xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
  xmlns:xs="http://www.w3.org/2001/XMLSchema"
  exclude-result-prefixes="#all"
  expand-text="yes">
  <xsl:output method="html" omit-xml-declaration="yes" encoding="UTF-8" include-content-type="no"/>
  <xsl:param name="cost-data"/>
    <!-- <supplier><product><key>3</key><pcost uptype="1"><key>21341</key><cost>12.99</cost></pcost></product></supplier> -->
    <!-- </xsl:param> -->
  
  <xsl:key name="cost" match="product/pcost[@uptype = 1]/cost" use="'cost' || ancestor::product/key"/>
  
  <xsl:mode on-no-match="shallow-copy"/>
        
  <xsl:template match="td[@name][key('cost', @name, fn:parse-xml($cost-data))]/text()">{key('cost', ../@name, fn:parse-xml($cost-data))}%</xsl:template>

<xsl:template match="/" name="xsl:initial-template">
  <xsl:next-match/>
</xsl:template>
</xsl:stylesheet>

如果您的样式表代码是

<xsl:param name="cost-data">
    <supplier><product><key>3</key><pcost uptype="1"><key>21341</key>     
       <cost>12.99</cost></pcost></product></supplier>
</xsl:param>

那么格式正确XML,不应报告任何错误。

然而,fn:parse-xml($cost-data)是错误的。参数的值是节点树,而不是字符串,fn:parse-xml() 期望字符串中的词法 XML。在此节点树上调用 fn:parse-xml() 的效果将首先原子化该节点,生成无类型的原子值“32134112.99”,然后尝试将此字符串“32134112.99”解析为词法 XML,这应该失败。

为避免这样的混淆,最好始终声明参数的预期类型,例如 as="xs:string"as="document-node()"

如果你想让$cost-data的默认值是一个包含词法XML的字符串,试试

<xsl:param name="cost-data" as="xs:string"><![CDATA[
    <supplier><product><key>3</key><pcost uptype="1"><key>21341</key>     
       <cost>12.99</cost></pcost></product></supplier>
]]></xsl:param>

全部在一行中,没有空格(或者如果您需要空格以提高可读性,请将其放在“>”分隔符之前)。

注意:您问题的字面答案是:否。Saxon-JS 不执行此检查。 XML 解析器会在 Saxon-JS 看到数据之前执行它。