我有一个由WiX的热实用程序生成的WXS文件。我正在尝试使用(现有的)排除.xslt文件对其进行修改,以根据另一个XML文件的内容自动排除某些组件(我将这部分称为.xml)。xslt 文件当前用于从安装程序中删除一些组件/文件/目录,但用于相对静态的列表。
但是,就目前而言,我主要关心的是让它加载并存储参数中的正确路径并将其内容存储在变量中。
这就是转换的应用方式(一旦 WiX 完成将热量收集到文件"before.wxs"):
msxsl.exe "before.wxs" "exclusions.xslt" -o "after.wxs" srcroot="$(SrcRoot)"
我传入 srcroot 作为 XSLT 文件中的一个参数,将用于在用户的机器上查找零件.xml。
要从命令行获取参数并将所有反斜杠替换为正斜杠,请执行以下操作:
<!-- Adapted from http://stackoverflow.com/a/30153713/5605122 and http://geekswithblogs.net/Erik/archive/2008/04/01/120915.aspx-->
<xsl:param name="srcroot" />
<xsl:variable name="filename">
<xsl:call-template name="string-replace-all">
<xsl:with-param name="text" select="concat('file://', $srcroot, '/foo/parts.xml')" />
<xsl:with-param name="replace" select='\' />
<xsl:with-param name="by" select='/' />
</xsl:call-template>
</xsl:variable>
<xsl:variable name="myparts" select="document($filename)" />
我从这里得到的字符串替换是:
<!-- XSL 1.0 string-replace-all, taken from: http://geekswithblogs.net/Erik/archive/2008/04/01/120915.aspx -->
<xsl:template name="string-replace-all">
<xsl:param name="text" />
<xsl:param name="replace" />
<xsl:param name="by" />
<xsl:choose>
<xsl:when test="contains($text, $replace)">
<xsl:value-of select="substring-before($text,$replace)" />
<xsl:value-of select="$by" />
<xsl:call-template name="string-replace-all">
<xsl:with-param name="text"
select="substring-after($text,$replace)" />
<xsl:with-param name="replace" select="$replace" />
<xsl:with-param name="by" select="$by" />
</xsl:call-template>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="$text" />
</xsl:otherwise>
</xsl:choose>
</xsl:template>
我收到一个相当无用的错误:
代码:0x80004005查询字符串中的意外字符。
但是如果我只删除我发布的代码片段,错误就会消失。我不确定如何调试并弄清楚发生了什么,但在浏览它之后,我相信我对我认为应该发生的事情有一个根本性的误解,而不仅仅是拼写错误或语法错误。
最终,我希望零件的内容.xml存储在"myparts"中。我在这里错过了什么?
谢谢!
事实证明,我只需要在单引号中设置字符串文字(并且不需要转义反斜杠)。此外,我的 srcroot 有一个尾随反斜杠,所以我在这里从一开始就省略了它。以下是更新的代码:
<!-- Adapted from http://stackoverflow.com/a/30153713/5605122 and http://geekswithblogs.net/Erik/archive/2008/04/01/120915.aspx-->
<xsl:param name="srcroot" />
<xsl:variable name="filename">
<xsl:call-template name="string-replace-all">
<xsl:with-param name="text" select="concat($srcroot, 'foo/parts.xml')" />
<xsl:with-param name="replace" select="''" />
<xsl:with-param name="by" select="'/'" />
</xsl:call-template>
</xsl:variable>
至于弄清楚如何测试它的问题,我终于通过执行以下操作来检查文件是否存在:
由于我使用的是msxsl,我决定为fileExists函数使用扩展名:
<!-- Adapted from https://www.stylusstudio.com/SSDN/default.asp?action=9&read=2993&fid=48 -->
<msxsl:script language="JScript" implements-prefix="user">
<![CDATA[
fso = new ActiveXObject("Scripting.FileSystemObject");
function fileExists(filename){
return fso.FileExists(filename);
}
// With .NET XSLT processor we need to use the Framework API
//function fileExists(filename){
//return System.IO.File.Exists(filename);
//}
]]>
</msxsl:script>
在模板中,我进行了以下检查:
<xsl:if test="not( function-available('user:fileExists') )">
<xsl:message terminate="yes">
<xsl:text>Required facilities are not available.</xsl:text>
</xsl:message>
</xsl:if>
如果通过,我做了以下事情:
<xsl:choose>
<xsl:when test="user:fileExists(string($filename))">
<xsl:message terminate="yes">
<xsl:text>File found! Here:</xsl:text>
<xsl:copy-of select="$filename"/>
</xsl:message>
</xsl:when>
<xsl:otherwise>
<xsl:message terminate="yes">
<xsl:text>File cannot be found:</xsl:text>
<xsl:copy-of select="$filename"/>
</xsl:message>
</xsl:otherwise>
</xsl:choose>
除非我出于某种原因设置 terminate="yes",否则消息不会输出到控制台,因此这会停止构建,我必须删除它,但它帮助我调试并发现该参数已成功由 XSLT 文件接收,并且我能够检查该路径上的文件是否存在。