- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
在我开始之前,让我注意到,我在这里的所有相关问题,等等,都没有找到解决办法。所以这个问题的目标和以前一样:
如何正确删除在应用程序使用wix工具工作时创建的临时文件夹和杂项文件?
到目前为止,我想出了以下方法:
使用customaction(例如,用c编写)只需删除所有文件和文件夹(这很好,但在我看来,此解决方案无效,因为不支持从msi侧回滚)。
使用util:removeFolderEx标记从WixUtilExtension。我做不到。
使用customaction(再次用c编写)在卸载之前填充msi db的表(目录和removefile)。这将迫使msi通过removefiles默认操作以其自己的方式正确卸载所有枚举的文件和文件夹(理论上它应该这样做)。
我把重点放在第三条道路上,在这里请你帮忙。
关于应用程序布局的几点注意事项
我构建的msi可以很好地处理程序文件、快捷方式、注册表和所有这些东西。但在安装过程中,我将一些配置文件放入C:\ProgramData\MyApp\
文件夹(它们也被删除,没有任何问题)。
但是,当应用程序工作时,它会在C:\ProgramData\MyApp\
中生成其他文件和目录,这些文件和目录用于在新版本可用时更新应用程序。假设用户在更新过程中关闭应用程序并希望卸载该应用程序。以下是我们目前在C:\ProgramData\MyApp
文件夹中的内容:
C:\程序数据\MyApp\
C:\程序数据\MyApp\Temp\
C:\程序数据\myapp\temp\tool.exe
C:\ProgramData\MyApp\Temp\SomeLib.dll
C:\ProgramData\MyApp\Temp\
C:\ProgramData\MyApp\Temp\
C:\ProgramData\MyApp\
文件夹。如果没有创建临时目录/文件,我可以看到这一点。
请注意,我不知道放入
C:\ProgramData\MyApp\Temp\
文件夹的文件夹和文件的名称,因为最终的文件夹名称是使用guid自动生成的。
让我专注于项目中最重要的部分,并向您展示我迄今为止为完成任务所做的工作(请记住,我选择了第三种方式:通过customaction):
Main MyApp.wxs file
<Product Id=...>
...
<!-- Defines a DLL contains the RemoveUpdatesAction function -->
<Binary Id="RemoveUpdatesAction.CA.dll" src="RemoveUpdatesAction.CA.dll" />
<CustomAction Id="RemoveUpdatesAction"
Return="check"
Execute="immediate"
BinaryKey="RemoveUpdatesAction.CA.dll"
DllEntry="RemoveUpdatesAction" />
...
<InstallExecuteSequence>
<!-- Perform custom CleanUp only on 'UnInstall' action - see condition -->
<Custom Action='RemoveUpdatesAction' Before='RemoveFiles'>
(NOT UPGRADINGPRODUCTCODE) AND (REMOVE="ALL")
</Custom>
</InstallExecuteSequence>
...
</Product>
Folder Layout
:
<Fragment>
<Directory Id="TARGETDIR" Name="SourceDir">
....
<!-- ...\Program Files\MyApp\ -->
<Directory Id="ProgramFilesFolder">
<Directory Id="APPINSTALLFOLDER" Name="MyApp" />
</Directory>
....
<!-- This is it! ...\ProgramData\MyApp\ -->
<Directory Id="CommonAppDataFolder">
<Directory Id="SETTINGSINSTALLFOLDER" Name="MyApp" />
</Directory>
....
</Directory>
</Fragment>
... a lot of different stuff here ...
<Fragment>
<Component Id="AddConfigurationFilesFolder" ...>
...
<!-- This component just serves as a bound point - see below -->
...
</Fragment>
RemoveUpdatesAction.cs file
包含自定义操作:
public class CustomActions
{
[CustomAction]
public static ActionResult RemoveUpdatesAction(Session session)
{
try
{
// Provide unique IDs
int indexFile = 1;
int indexDir = 1;
// Begin work
session.Log("Begin RemoveUpdatesAction");
// Bind to the component that for sure will be uninstalled during UnInstall action
// You can see this component mentioned above
const string componentId = "AddConfigurationFilesFolder";
// Get '..\{ProgramData}\MyApp' folder
// This property (SETTINGSINSTALLFOLDER) is mentioned too
string appDataFolder = session["SETTINGSINSTALLFOLDER"];
// Populate RemoveFile table in MSI database with all files
// created in '..\{ProgramData}\MyApp\*.*' folder - pls see notes at the beginning
if (!Directory.Exists(appDataFolder))
{
session.Log("End RemoveUpdatesAction");
return ActionResult.Success;
}
foreach (var directory in Directory.GetDirectories(appDataFolder, "*", SearchOption.AllDirectories))
{
session.Log("Processing Subdirectory {0}", directory);
foreach (var file in Directory.EnumerateFiles(directory))
{
session.Log("Processing file {0}", file);
string keyFile = string.Format("CLEANFILE_{0}", indexFile);
// Set values for columns in RemoveFile table:
// {1}: FileKey => just unique ID for the row
// {2}: Component_ => reference to a component existed in Component table
// In our case it is already mentioned 'AddConfigurationFilesFolder'
// {3}: FileName => localizable name of the file to be removed (with ext.)
// {4}: DirProperty => reference to a full dir path
// {5}: InstallMode => 3 means remove on Install/Remove stage
var fieldsForFiles = new object[] { keyFile, componentId, Path.GetFileName(file), directory, 3 };
// The following files will be processed:
// 1. '..\ProgramData\MyApp\Temp\tool.exe'
// 2. '..\ProgramData\MyApp\Temp\somelib.dll'
// 3. '..\ProgramData\MyApp\Temp\<UniqueFolderNameBasedOnGUID>\someliba.dll'
// 4. '..\ProgramData\MyApp\Temp\<UniqueFolderNameBasedOnGUID>\somelibb.dll'
InsertRecord(session, "RemoveFile", fieldsForFiles);
indexFile++;
}
string keyDir = string.Format("CLEANDIR_{0}", indexDir);
// Empty quotes mean we we want to delete the folder itself
var fieldsForDir = new object[] { keyDir, componentId, "", directory, 3 };
// The following paths will be processed:
// 1. '..\ProgramData\MyApp\Temp\'
// 2. '..\ProgramData\MyApp\Temp\<UniqueFolderNameBasedOnGUID>\'
InsertRecord(session, "RemoveFile", fieldsForDir);
indexDir++;
}
session.Log("End RemoveUpdatesAction");
return ActionResult.Success;
}
catch (Exception exception)
{
session.Log("RemoveUpdatesAction EXCEPTION:" + exception.Message);
return ActionResult.Failure;
}
}
// Took completely from another SO question, but is accoring to MSDN docs
private static void InsertRecord(Session session, string tableName, Object[] objects)
{
Database db = session.Database;
string sqlInsertSring = db.Tables[tableName].SqlInsertString + " TEMPORARY";
View view = db.OpenView(sqlInsertSring);
view.Execute(new Record(objects));
view.Close();
}
}
(NOT UPGRADINGPRODUCTCODE) AND (REMOVE="ALL")
)是我从
read中获取的。
..\ProgramData\MyApp\Temp\...
中的文件和文件夹都会保留下来。
Directory.GetDirectories
找到的temp文件夹的记录。
..\ProgramData\MyApp\Temp\
..\ProgramData\MyApp\Temp\<UniqueFolderNameBasedOnGUID>\
MSI (s) (B4:28) [05:28:39:427]: Doing action: RemoveUpdatesAction
....
MSI (s) (B4:4C) [05:28:39:450]: Invoking remote custom action. DLL: C:\WINDOWS\Installer\MSI7643.tmp, Entrypoint: RemoveUpdatesAction
....
Action start 5:28:39: RemoveUpdatesAction.
SFXCA: Extracting custom action to temporary directory: C:\WINDOWS\Installer\MSI7643.tmp-\
SFXCA: Binding to CLR version v4.0.30319
Calling custom action RemoveUpdatesAction!RemoveUpdatesAction.CustomActions.RemoveUpdatesAction
Begin RemoveUpdatesAction
Processing Subdirectory C:\ProgramData\MyApp\Temp
Processing file C:\ProgramData\MyApp\Temp\somelib.dll
Processing file C:\ProgramData\MyApp\Temp\tool.exe
Processing Subdirectory C:\ProgramData\MyApp\Temp\48574917-4351-4d4c-a36c-381f3ceb2e56
Processing file C:\ProgramData\MyApp\Temp\48574917-4351-4d4c-a36c-381f3ceb2e56\someliba.dll
Processing file C:\ProgramData\MyApp\Temp\48574917-4351-4d4c-a36c-381f3ceb2e56\somelibb.dll
End RemoveUpdatesAction
MSI (s) (B4:28) [05:28:39:602]: Doing action: RemoveFiles
MSI (s) (B4:28) [05:28:39:602]: Note: 1: 2205 2: 3: ActionText
Action ended 5:28:39: RemoveUpdatesAction. Return value 1.
Action start 5:28:39: RemoveFiles.
MSI (s) (B4:28) [05:28:39:607]: Note: 1: 2727 2: C:\ProgramData\MyApp\Temp
MSI (s) (B4:28) [05:28:39:607]: Note: 1: 2727 2: C:\ProgramData\MyApp\Temp\48574917-4351-4d4c-a36c-381f3ceb2e56
MSI (s) (B4:28) [05:28:39:607]: Counted 2 foreign folders to be removed.
MSI (s) (B4:28) [05:28:39:607]: Removing foreign folder: C:\ProgramData\Microsoft\Windows\Start Menu\Programs\Compass Mobile\
MSI (s) (B4:28) [05:28:39:607]: Removing foreign folder: C:\ProgramData\MyApp\
MSI (s) (B4:28) [05:28:39:607]: Doing action: RemoveFolders
MSI (s) (B4:28) [05:28:39:607]: Note: 1: 2205 2: 3: ActionText
Action ended 5:28:39: RemoveFiles. Return value 1.
Action start 5:28:39: RemoveFolders.
最佳答案
据我所知,您没有正确填充removefile表。如果您记录了确切的sql insert字符串来查看真正存在的内容,这将有所帮助。我认为您得到了错误2727,因为插入的第四个内容应该是引用msi文件中目录表的目录属性。它不是一个字面上的目录名——它应该是MSI文件目录表的一个键——据我所知,它是一个实际的目录,而不是目录表中的值。
关于c# - WiX:在卸载时正确删除非空的临时文件和文件夹,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25433576/
#include using namespace std; class C{ private: int value; public: C(){ value = 0;
这个问题已经有答案了: What is the difference between char a[] = ?string?; and char *p = ?string?;? (8 个回答) 已关闭
关闭。此题需要details or clarity 。目前不接受答案。 想要改进这个问题吗?通过 editing this post 添加详细信息并澄清问题. 已关闭 7 年前。 此帖子已于 8 个月
除了调试之外,是否有任何针对 c、c++ 或 c# 的测试工具,其工作原理类似于将独立函数复制粘贴到某个文本框,然后在其他文本框中输入参数? 最佳答案 也许您会考虑单元测试。我推荐你谷歌测试和谷歌模拟
我想在第二台显示器中移动一个窗口 (HWND)。问题是我尝试了很多方法,例如将分辨率加倍或输入负值,但它永远无法将窗口放在我的第二台显示器上。 关于如何在 C/C++/c# 中执行此操作的任何线索 最
我正在寻找 C/C++/C## 中不同类型 DES 的现有实现。我的运行平台是Windows XP/Vista/7。 我正在尝试编写一个 C# 程序,它将使用 DES 算法进行加密和解密。我需要一些实
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
有没有办法强制将另一个 窗口置于顶部? 不是应用程序的窗口,而是另一个已经在系统上运行的窗口。 (Windows, C/C++/C#) 最佳答案 SetWindowPos(that_window_ha
假设您可以在 C/C++ 或 Csharp 之间做出选择,并且您打算在 Windows 和 Linux 服务器上运行同一服务器的多个实例,那么构建套接字服务器应用程序的最明智选择是什么? 最佳答案 如
你们能告诉我它们之间的区别吗? 顺便问一下,有什么叫C++库或C库的吗? 最佳答案 C++ 标准库 和 C 标准库 是 C++ 和 C 标准定义的库,提供给 C++ 和 C 程序使用。那是那些词的共同
下面的测试代码,我将输出信息放在注释中。我使用的是 gcc 4.8.5 和 Centos 7.2。 #include #include class C { public:
很难说出这里问的是什么。这个问题是含糊的、模糊的、不完整的、过于宽泛的或修辞性的,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开它,visit the help center 。 已关
我的客户将使用名为 annoucement 的结构/类与客户通信。我想我会用 C++ 编写服务器。会有很多不同的类继承annoucement。我的问题是通过网络将这些类发送给客户端 我想也许我应该使用
我在 C# 中有以下函数: public Matrix ConcatDescriptors(IList> descriptors) { int cols = descriptors[0].Co
我有一个项目要编写一个函数来对某些数据执行某些操作。我可以用 C/C++ 编写代码,但我不想与雇主共享该函数的代码。相反,我只想让他有权在他自己的代码中调用该函数。是否可以?我想到了这两种方法 - 在
我使用的是编写糟糕的第 3 方 (C/C++) Api。我从托管代码(C++/CLI)中使用它。有时会出现“访问冲突错误”。这使整个应用程序崩溃。我知道我无法处理这些错误[如果指针访问非法内存位置等,
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 7 年前。
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 要求我们推荐或查找工具、库或最喜欢的场外资源的问题对于 Stack Overflow 来说是偏离主题的,因为
我有一些 C 代码,将使用 P/Invoke 从 C# 调用。我正在尝试为这个 C 函数定义一个 C# 等效项。 SomeData* DoSomething(); struct SomeData {
这个问题已经有答案了: Why are these constructs using pre and post-increment undefined behavior? (14 个回答) 已关闭 6
我是一名优秀的程序员,十分优秀!