首页 > 解决方案 > 如何在重大升级期间使用 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 仍然存在并且没有新的。

标签: wixwindows-installerinstallation

解决方案


类似答案未安装新组件的文件,因为旧组件具有相同的文件


配套文件:如何在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>

用法

  1. 创建 WiX 项目,用上面的标记替换模板内容。
  2. 在 WiX 项目文件夹中创建一个名为的_Files子文件夹,并在其中创建两个另外的子文件夹,其中包含相同文件的两个版本,如下图所示。
  • 快速方法:在 Visual Studio 中,右键单击 WiX 项目并选择Open Folder in File Explorer快速进入 WiX 项目文件夹。
  • 现在粘贴到两个文件夹中或创建它们。
  • 您可以在 Visual Studio 中打开 EXE 以“破解”版本号,以便在具有不同版本号的两个文件夹中使用相同的 EXE 或 DLL。有关信息,请参阅下面的部分。
  1. 将 MyVersion 定义更改为您要构建的版本号。例如:<?define MyVersion = "23.4.5.2" ?>。这也会影响示例期望源文件在磁盘上的位置。
  2. 更新 File 元素中的文件名以匹配版本化文件的名称 ( <File Source="$(var.MyReleasePath)MyTestFile.exe"></File>)。
  3. 编译安装程序,安装它,检查已安装文件的版本号。
  4. 更改 WiX 版本并编译下一个设置 ( <?define MyVersion = "23.4.5.3" ?>)。
  5. 在现有安装之上安装。检查是否已安装新版本的文件。

版本化文件的文件夹结构(在主项目文件夹中创建):

_Files
   23.4.5.2\MyTestFile.exe
   23.4.5.3\MyTestFile.exe

打开 EXE 作为资源:在 Visual Studio 中试试这个:

  1. File=> Open=>File
  2. 浏览版本化文件 ( EXE, DLL, etc...)
  3. 单击版本控制文件一次(且仅一次)
  4. 单击按钮的向下箭头Open=>Open With...
  5. 选择Resource Editor并打开文件。
  6. 找到Version section,打开并双击条目。
  7. 更改版本号,保存新文件版本(不在现有文件之上)。

链接


推荐阅读