首页 > 解决方案 > 如何在 XSLT 遗留代码库中查找死代码

问题描述

我从帮助我分析 XSLT 遗留代码库的BaseX v9 XQuery脚本开始。大约有 100 个 XSL v1.0 文件,这些年来有点乱。

代码需要清理。第一步是找到未使用的代码(模板、变量等)。

我的问题是关于查找不再使用的匹配模板:

 <xsl:template match="blabla">...</xsl:template>

一般来说,找到永远不会执行的匹配模板(有或没有模式属性)的最佳方法是什么?

标签: xmlxsltxquerylegacybasex

解决方案


一个模板规则可能永远不会被执行的原因有两个:要么有另一个模板规则总是被优先选择,要么没有可以想象的源文档的节点与其模式匹配。通过样式表的静态分析来检测这两种情况都不太实际。最好的办法是使用具有代表性的源文档样本(以及其他输入条件,例如样式表参数的值)动态地进行一些代码覆盖率分析。

值得在谷歌上搜索“代码覆盖 XSLT”,尽管它给您的问题多于答案。XSpec 进行代码覆盖,但前提是您拥有一套全面的 XSpec 测试,这似乎不太可能。Saxon 有一个工具 (-TP:profile.html) 可以输出每个模板规则执行的频率,但遗憾的是,它忽略了那些计数为零的规则。将这些数据与一些源代码分析结合起来以找到未出现在列表中的模板规则并不难(而且,将多次运行的输出与不同的源文档结合起来。)

对许多源文档实际执行样式表的替代方法是将匹配模式提取到合成样式表中,该样式表针对每个匹配模式测试每个输入节点。您可以为每个匹配项输出一个元素,然后对输出进行后处理以查找输出中缺少的模式:

第一的:

<xsl:template match="... a sample pattern... ">
  <match id="654321"/>
  <xsl:apply-templates select="@*|node()"/>
</xsl:template>
(repeated for each pattern in the original stylesheet)

然后进行分析:

<xsl:variable name="data" select="."/>
<xsl:key name="k" match="match" use="@id"/>
<xsl:for-each select="1 to max(//match/@id)[not(key('k', ., $data))]">
  No matches for pattern id="{.}"
</xsl:for-each>

但是,对于可能由xsl:apply-importsorxsl:next-match指令触发的模板规则,或匹配多个模板规则(可能在不同模式下)的节点,这将为您提供错误的“不匹配”结果。我相信这个想法可以改进。


推荐阅读