wix - 如何在重大升级期间使用 Wix 工具集显式删除 dll
问题描述
我们正在尝试为我们的产品部署更新,其中包含更新的 dll。这个 dll 也有一个版本号,所以通常安装程序应该能够看到这个并替换旧的 dll。
Windows 安装程序在检查版本号时只关心前三个数字。但是,在这种情况下,更新后的版本号看起来与第四个数字相同。因此,例如,如果前一个版本为 1.0.0.12,则此版本为 1.0.0.20,因此安装程序会将它们视为相同的版本,并且不会替换文件。在这种情况下,我无法控制版本编号,并且由于我们包含很多 dll,将来可能会再次发生这种情况。
无论版本号是否相同,如何让安装程序替换文件?
在搜索了不同的解决方案后,我试图告诉安装程序在安装过程中删除该文件。如果没有必要,我不想删除所有文件。到目前为止,我已经尝试了包含的代码片段中显示的解决方案。removefile 元素在此版本的安装程序中是新的。
<Component Id="SomeComponent" Win64="yes" Guid="*">
<File Id="SomeFile" KeyPath="yes" Source="$(var.app.TargetDir)\some.dll" />
<RemoveFile Id="RemoveSomeFile" Name="some.dll" On="install"/>
</Component>
我期望的结果是旧的 dll 被新的 dll 替换,但是安装后旧的 dll 仍然存在并且没有新的。
解决方案
配套文件:如何在WiX中使用配套文件- 只是一个片段(调试后发现 OP 存在需要降级文件的问题,而不是版本位数的问题 -非常常见的问题):
<..>
<Component Id="MyFile.exe" Feature="Main">
<File Id="MyFile.exe" Source="MyFile.exe"></File>
</Component>
<Component Id="MyFile_2.exe" Guid="{0EBDFD64-017B-428F-BB67-3D82EA2A99AF}" Feature="Main">
<File Source="MyFile_2.exe" CompanionFile="MyFile.exe"></File>
</Component>
<..>
单行摘要:在第二个组件中,我们指向第一个组件的文件,以便在安装 MyFile.exe 时安装 MyFile_2.exe - 无论版本控制问题如何。
下面留下较旧的答案以供参考和可用于测试的 WiX 源。
MSI 版本:Windows 安装程序在检查版本号时只关心前三个数字。
文件版本与产品版本:上述陈述通常是正确的,但据我所知(并基于快速冒烟测试),此 3 位数限制仅适用于 MSIProductVersion
(MSI 本身的版本),而不适用到实际file version numbers
。
文件版本以 4 位数表示,与 MSI 的 3 位数限制
ProductVersion
(MSI 本身的版本)相反。请自己进行测试以确定。在下面执行此操作的示例 WiX 标记。
REINSTALLMODE:文件覆盖修饰符机制REINSTALLMODE可用于强制覆盖文件,无论版本如何。不得使用此机制,因为它可能会导致许多问题:在早期版本的 Windows 中,不必要的重新启动提示、系统共享文件的降级、导致某些文件升级而其他文件不升级(新旧软件包安装顺序不正确)由于尝试降级受保护的文件等而崩溃...
WiX Mockup 测试源:我将在此处转储一个简单的测试 WiX 源以帮助您快速测试(例如,在不同的操作系统版本上,我在 Windows 10 上测试过):
顺便说一句,由于此示例的某些固有设计特征,它还说明了当您仅增加 4 位版本号的最后一位时,主要升级会失败,但文件覆盖将起作用。当您安装版本 2 时,您会在添加/删除程序中找到两个产品条目。这是基于样本设计的预期。只需卸载两者。
<?define MyVersion = "1.0.0.0" ?>
<?define MyReleasePath = "$(sys.SOURCEFILEDIR)_Files\$(var.MyVersion)\" ?>
<Wix xmlns="http://schemas.microsoft.com/wix/2006/wi">
<Product Id="*" Name="FileVersioning" Language="1033" Version="$(var.MyVersion)"
Manufacturer="FileVersioning" UpgradeCode="{4864AA4A-EA1D-4367-8427-85D296B0A2A6}">
<Package InstallerVersion="200" Compressed="yes" InstallScope="perMachine" />
<MajorUpgrade DowngradeErrorMessage="A newer version of [ProductName] is already installed." />
<MediaTemplate EmbedCab="yes" />
<Feature Id="Main" Title="FileVersioning" Level="1" />
<Directory Id="TARGETDIR" Name="SourceDir">
<Directory Id="ProgramFilesFolder">
<Directory Id="INSTALLFOLDER" Name="FileVersioning">
<Component Feature="Main">
<File Source="$(var.MyReleasePath)MyTestFile.exe"></File>
</Component>
</Directory>
</Directory>
</Directory>
</Product>
</Wix>
用法:
- 创建 WiX 项目,用上面的标记替换模板内容。
- 在 WiX 项目文件夹中创建一个名为的
_Files
子文件夹,并在其中创建两个另外的子文件夹,其中包含相同文件的两个版本,如下图所示。
- 快速方法:在 Visual Studio 中,右键单击 WiX 项目并选择
Open Folder in File Explorer
快速进入 WiX 项目文件夹。 - 现在粘贴到两个文件夹中或创建它们。
- 您可以在 Visual Studio 中打开 EXE 以“破解”版本号,以便在具有不同版本号的两个文件夹中使用相同的 EXE 或 DLL。有关信息,请参阅下面的部分。
- 将 MyVersion 定义更改为您要构建的版本号。例如:
<?define MyVersion = "23.4.5.2" ?>
。这也会影响示例期望源文件在磁盘上的位置。 - 更新 File 元素中的文件名以匹配版本化文件的名称 (
<File Source="$(var.MyReleasePath)MyTestFile.exe"></File>
)。 - 编译安装程序,安装它,检查已安装文件的版本号。
- 更改 WiX 版本并编译下一个设置 (
<?define MyVersion = "23.4.5.3" ?>
)。 - 在现有安装之上安装。检查是否已安装新版本的文件。
版本化文件的文件夹结构(在主项目文件夹中创建):
_Files
23.4.5.2\MyTestFile.exe
23.4.5.3\MyTestFile.exe
打开 EXE 作为资源:在 Visual Studio 中试试这个:
File
=>Open
=>File
- 浏览版本化文件 (
EXE
,DLL
,etc...
) - 单击版本控制文件一次(且仅一次)
- 单击按钮的向下箭头
Open
=>Open With...
- 选择
Resource Editor
并打开文件。 - 找到
Version section
,打开并双击条目。 - 更改版本号,保存新文件版本(不在现有文件之上)。
链接:
推荐阅读
- python - 打字机错误:必须在散列之前对 Unicode 对象进行编码
- google-chrome - Chrome 音频问题 - Javascript
- python-3.x - 是否有使用 Gdk 查找特定窗口的功能?
- airflow - Airflow 2 错误,不推荐在插件中注册操作员或传感器
- c - 递归调用的 C 函数因 strtol_l_internal SIGSEGV 而失败
- performance - VueJS 性能问题
- php - 迁移和更改目录后缺少 PHP 脚本图像
- apache-spark - 如何从火花中的笛卡尔积中删除重复项
- python - 如何使用 tkinter 将文本变量添加到列表中
- python - 如何匹配/搜索特定模式但没有另一个特定模式?