Move user data before running the uninstaller

refs #11449
This commit is contained in:
Gunnar Beutner 2016-04-18 15:33:20 +02:00
parent 955df528d7
commit 913bfe4eb2
1 changed files with 45 additions and 21 deletions

View File

@ -167,23 +167,32 @@ static void CollectPaths(std::vector<std::string>& paths, const std::string& pat
paths.push_back(path); paths.push_back(path);
} }
static bool MoveDirectory(const std::string& source, const std::string& destination) static bool CopyDirectory(const std::string& source, const std::string& destination)
{ {
if (!MoveFileEx(source.c_str(), destination.c_str(), MOVEFILE_COPY_ALLOWED | MOVEFILE_WRITE_THROUGH)) { // SHFileOperation requires file names to be terminated with two \0s
// SHFileOperation requires file names to be terminated with two \0s std::string tmpSource = source + std::string(1, '\0');
std::string tmpSource = source + std::string(1, '\0'); std::string tmpDestination = destination + std::string(1, '\0');
std::string tmpDestination = destination + std::string(1, '\0');
SHFILEOPSTRUCT fop; SHFILEOPSTRUCT fop;
fop.wFunc = FO_COPY; fop.wFunc = FO_COPY;
fop.pFrom = tmpSource.c_str(); fop.pFrom = tmpSource.c_str();
fop.pTo = tmpDestination.c_str(); fop.pTo = tmpDestination.c_str();
fop.fFlags = FOF_NO_UI; fop.fFlags = FOF_NO_UI;
if (SHFileOperation(&fop) != 0)
return false;
}
return true; return (SHFileOperation(&fop) == 0);
}
static bool DeleteDirectory(const std::string& dir)
{
// SHFileOperation requires file names to be terminated with two \0s
std::string tmpDir = dir + std::string(1, '\0');
SHFILEOPSTRUCT fop;
fop.wFunc = FO_DELETE;
fop.pFrom = tmpDir.c_str();
fop.fFlags = FOF_NO_UI;
return (SHFileOperation(&fop) == 0);
} }
static int UpgradeNSIS(void) static int UpgradeNSIS(void)
@ -198,24 +207,39 @@ static int UpgradeNSIS(void)
if (!PathExists(uninstallerPath)) if (!PathExists(uninstallerPath))
return 0; return 0;
ExecuteCommand(uninstallerPath, "/S _?=" + installPath);
_unlink(uninstallerPath.c_str());
std::string dataPath = GetIcingaDataPath(); std::string dataPath = GetIcingaDataPath();
if (dataPath.empty())
return 1;
bool moveUserData = !PathExists(dataPath);
/* perform open heart surgery on the user's data dirs - yay */ /* perform open heart surgery on the user's data dirs - yay */
if (!PathExists(dataPath)) { if (moveUserData) {
MkDir(dataPath.c_str()); MkDir(dataPath.c_str());
std::string oldNameEtc = installPath + "\\etc"; std::string oldNameEtc = installPath + "\\etc";
std::string newNameEtc = dataPath + "\\etc"; std::string newNameEtc = dataPath + "\\etc";
if (!MoveDirectory(oldNameEtc, newNameEtc)) if (!CopyDirectory(oldNameEtc, newNameEtc))
return 1; return 1;
std::string oldNameVar = installPath + "\\var"; std::string oldNameVar = installPath + "\\var";
std::string newNameVar = dataPath + "\\var"; std::string newNameVar = dataPath + "\\var";
if (!MoveDirectory(oldNameVar, newNameVar)) if (!CopyDirectory(oldNameVar, newNameVar))
return 1;
}
ExecuteCommand(uninstallerPath, "/S _?=" + installPath);
_unlink(uninstallerPath.c_str());
if (moveUserData) {
std::string oldNameEtc = installPath + "\\etc";
if (!DeleteDirectory(oldNameEtc))
return 1;
std::string oldNameVar = installPath + "\\var";
if (!DeleteDirectory(oldNameVar))
return 1; return 1;
_rmdir(installPath.c_str()); _rmdir(installPath.c_str());