Windows
安装
- 为了确定安装目标,在安装前后调用了
get_reg_items
1 2 3 4 |
reg_helper reg; auto res1 = reg.get_reg_items(); auto res = create_process(path); auto res2 = reg.get_reg_items(); |
get_reg_items
1 2 3 4 5 6 7 8 9 10 |
std::map<std::wstring, std::map<std::wstring, std::wstring>> reg_helper::get_reg_items() { std::map<std::wstring, std::map<std::wstring, std::wstring>> res; for (auto item : programKeys) { enum_sub_key(HKEY_LOCAL_MACHINE, item, res); } for (auto item : programKeysUser) { enum_sub_key(HKEY_CURRENT_USER, item, res); } return res; } |
- 上买的
Windows
实现是通过枚举注册表- 把几个目标路径的注册表存到一起
- 然后根据安装前后这些路径下的注册表的项数,来找出新增的那个
1 2 3 4 5 6 7 |
std::vector<std::wstring> programKeys = { L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Uninstall", L"SOFTWARE\\WOW6432Node\\Microsoft\\Windows\\CurrentVersion\\Uninstall", }; std::vector<std::wstring> programKeysUser = { L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Uninstall", }; |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 |
void reg_helper::enum_sub_key(HKEY key, const std::wstring & path, std::map<std::wstring, std::map<std::wstring, std::wstring>>&res) { HKEY hKey; if (RegOpenKeyEx(key, path.c_str(), 0, KEY_READ | KEY_WOW64_64KEY, &hKey) != ERROR_SUCCESS) { DWORD dwError = GetLastError(); LPVOID lpMsgBuf; DWORD len = FormatMessageW( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, dwError, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR)&lpMsgBuf, 0, NULL); if (len) { LPWSTR lpMsgStr = (LPWSTR)lpMsgBuf; std::wstring result(lpMsgStr, lpMsgStr + len); LocalFree(lpMsgBuf); } return; } DWORD subkeys = 0; DWORD maxValueNameLen = 0; if (RegQueryInfoKey(hKey, NULL, NULL, NULL, &subkeys, NULL, NULL, NULL, &maxValueNameLen, NULL, NULL, NULL) != ERROR_SUCCESS) { RegCloseKey(hKey); return; } TCHAR subkeyName[256]; DWORD subkeyNameSize; for (DWORD i = 0; i < subkeys; i++) { subkeyNameSize = sizeof(subkeyName) / sizeof(TCHAR); if (RegEnumKeyEx(hKey, i, subkeyName, &subkeyNameSize, NULL, NULL, NULL, NULL) == ERROR_SUCCESS) { std::wstring tmp(subkeyName); auto mp = res.emplace(tmp, std::map<std::wstring, std::wstring>()); if (!mp.second) { continue; } HKEY hSubKey; if (RegOpenKeyEx(hKey, subkeyName, 0, KEY_READ, &hSubKey) == ERROR_SUCCESS) { read_reg_values(hSubKey, res, tmp); RegCloseKey(hSubKey); } } } RegCloseKey(hKey); } |
- 通过比较,确定新增项,然后记录下新安装软件的卸载位置,为了安装后的卸载
- 下面的
set_db_record
就是把下载的软件的编号和对应的卸载路径存起来
- 下面的
1 2 3 4 5 6 |
if (res2.size() > res1.size()) { auto str = get_goal_reg(res2, res1); if (!str.empty()) { set_db_record(appid, str); } } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
std::wstring installer_helper::get_goal_reg(std::map<std::wstring, std::map<std::wstring, std::wstring>>&vec1, std::map<std::wstring, std::map<std::wstring, std::wstring>>&vec2) { std::wstring res; for (const auto& item : vec1) { if (vec2.find(item.first) == vec2.end()) { auto tmp = item.second; auto ite = tmp.find(L"UninstallString"); if (ite == tmp.end()) { continue; } res = ite->second; break; } } return res; } |
卸载
- 根据编号去获取存储的软件卸载路径
1 |
std::wstring goal_path = m_app->get_unist_goal(appid); |
- 设置路径参数
1 2 3 4 |
auto helper = std::make_unique<uninstaller_helper>(this); helper->set_goal_path(appid, goal_path.c_str()); helper->set_group(task_group::em_tg_low_group); helper->check_and_start(); |
- 根据这个路径开始卸载
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 |
void uninstaller_helper::start_uninstall() { #if defined(WIN32) || defined(__WIN32) || defined(__WIN32__) || defined(__NT__) DWORD exitCode; sync_status syst(sync_status::em_syst_install_default); SHELLEXECUTEINFO execInfo = { 0 }; execInfo.cbSize = sizeof(SHELLEXECUTEINFO); execInfo.fMask = SEE_MASK_NOCLOSEPROCESS; execInfo.hwnd = NULL; execInfo.lpVerb = L"runas"; execInfo.lpFile = m_path; execInfo.lpParameters = NULL; execInfo.lpDirectory = NULL; execInfo.nShow = SW_SHOW; execInfo.hInstApp = NULL; execInfo.hProcess = NULL; if (!ShellExecuteEx(&execInfo)) { DWORD dwError = GetLastError(); LPVOID lpMsgBuf; DWORD len = FormatMessageW( FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, dwError, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR)&lpMsgBuf, 0, NULL); if (len) { LPWSTR lpMsgStr = (LPWSTR)lpMsgBuf; std::wstring result(lpMsgStr, lpMsgStr + len); LocalFree(lpMsgBuf); } if (m_unist && m_appid) { m_unist->uninstall_status(install_type::em_it_uninstalled_failed, m_appid); } return; } if (execInfo.hProcess == NULL || execInfo.hProcess == INVALID_HANDLE_VALUE) { if (m_unist && m_appid) { m_unist->uninstall_status(install_type::em_it_uninstalled_failed, m_appid); } return; } WaitForSingleObject(execInfo.hProcess, INFINITE); if (GetExitCodeProcess(execInfo.hProcess, &exitCode)) { syst = (exitCode == 0) ? sync_status::em_syst_uninstall_success : sync_status::em_syst_uninstall_failed; } CloseHandle(execInfo.hProcess); if (syst == sync_status::em_syst_uninstall_success) { if (m_unist && m_appid) { m_unist->uninstall_status(install_type::em_it_uninstalled_success, m_appid); } } else if (syst == sync_status::em_syst_uninstall_failed) { if (m_unist && m_appid) { m_unist->uninstall_status(install_type::em_it_uninstalled_failed, m_appid); } } } |
Linux
安装
linux
同样也是在安装前后,把一些文件做了保存,然后根据这些文件来确定,具体安装的是哪一个软件- 这里选择保存的是
linux
下软件的list
文件
- 这里选择保存的是
1 2 3 4 |
reg_helper reg; auto res1 = reg.get_reg_items(); auto res = create_process(path); auto res2 = reg.get_reg_items(); |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
std::vector<std::string> reg_helper::get_reg_items() { std::vector<std::string> res; try { std::string dPkgPath("/var/lib/dpkg/info/"); if (!boost::filesystem::exists(dPkgPath)) { dPkgPath = "/usr/lib/dpkg-db/info/"; } boost::filesystem::path dirPath(dPkgPath); for (boost::filesystem::directory_iterator it(dirPath); it != boost::filesystem::directory_iterator(); ++it) { if (it->path().extension() == ".list") { auto file = it->path().string(); res.push_back(file); } } } catch(const boost::filesystem::filesystem_error& err) { return {}; } return res; } |
- 具体安装的方法暂时
linux
和apple
都使用了system
调命令的方法
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
#elif defined(__APPLE__) || defined(__linux__) std::string tmpStr = to_str(path); if (tmpStr.empty()) { return sync_status::em_syst_create_process_failed; } #if defined(__APPLE__) std::string tmp("/System/Library/CoreServices/Installer.app/Contents/MacOS/Installer "); #elif defined(__linux__) std::string tmp("pkexec dpkg -i "); #endif std::string cmdStr("\""); cmdStr.append(tmpStr).append("\""); tmp.append(cmdStr); int return_code = system(tmp.c_str()); if (return_code == 0) { return sync_status::em_syst_install_success; } else { return sync_status::em_syst_install_failed; } #endif |
- 不同之处在于,单步过程中发现个别软件表现不一致,比如一些软件卸载完成后就删掉了
list
文件,而有一些则是清空list
文件但不删除- 这会导致部分软件在第二次安装的时候,比
list
数量时,发现是一样的 - 所以,暂时想了个策略,就是在安装前记录下一个时间戳,用以对比,如下:
- 这会导致部分软件在第二次安装的时候,比
1 2 3 4 5 6 7 8 |
#elif defined(__APPLE__) || defined(__linux__) reg_helper reg; auto res1 = reg.get_reg_items(); #if defined(__linux__) auto now = std::chrono::system_clock::now(); #endif auto res = create_process(path); auto res2 = reg.get_reg_items(); |
- 当安装完后,发现成功了,当发现
list
数量安装后比安装前多,就获取目标软件的路径- 可以看到,这里是用上了上面记录的时间戳,进程函数后,又创建了一个时间戳,并计算出了一个时间间隔
- 然后获取每个文件的最后修改时间,和后面的时间戳作间隔处理,并和上面的对比
- 如果后面的间隔小于等于前面两个时间戳的间隔,则认为这个文件是目标软件安装过程中被修改过的
- 所以把它保存起来,并且,这里我的做法是,找了一个这样的路径后,就不再查找了
- 这样的做法,从理论上来看,是做不到完全的准确的
1 2 3 4 5 6 7 8 9 10 11 12 |
auto str = get_goal_reg(res2, res1, now); if (str.empty()) { type = install_type::em_it_installed_failed; } else { auto tmp_str = to_wstr(str.c_str()); if (tmp_str.empty()) { type = install_type::em_it_installed_failed; } else { set_db_record(appid, tmp_str); type = install_type::em_it_installed_success; } } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 |
std::string installer_helper::get_goal_reg(std::vector<std::string>& vec1, std::vector<std::string>& vec2, std::chrono::system_clock::time_point clock) { auto now = std::chrono::system_clock::now(); auto interval = std::chrono::duration_cast<std::chrono::seconds>(now - clock); std::vector<std::string> tmp; for (auto item : vec1) { if (std::find(vec2.begin(), vec2.end(), item) == vec2.end()) { tmp.push_back(item); } } if (tmp.empty()) { return ""; } std::string res_p; for (auto items : tmp) { try { boost::filesystem::path file(items); auto last_time = boost::filesystem::last_write_time(file); auto time = std::chrono::system_clock::from_time_t(last_time); auto file_interval = std::chrono::duration_cast<std::chrono::seconds>(now - time); if (file_interval <= interval) { res_p = items; break; } } catch(const boost::filesystem::filesystem_error&) { continue ; } } return res_p; } |
- 还有一种情况,就是上面说的
list
数量一样的情况,同样也做了类似处理- 根据时间间隔来处理,考虑到这样的文件可能会有多个,所以在实现时,只使用了
res[0]
- 并对路径进行了保存
- 根据时间间隔来处理,考虑到这样的文件可能会有多个,所以在实现时,只使用了
1 2 3 4 5 6 7 8 9 10 11 12 13 |
if (res2.size() > res1.size()) { // ... } else if (res2.size() == res1.size()) { auto str = get_goal_second_linux(res2, now); if (str.empty()) { type = install_type::em_it_installed_failed; } else { std::wstring wstr = to_wstr(str.c_str()); set_db_record(appid, wstr); type = install_type::em_it_installed_success; } } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 |
std::string installer_helper::get_goal_second_linux(std::vector<std::string>& res, std::chrono::system_clock::time_point clock) { std::vector<std::string> tmpres; auto now = std::chrono::system_clock::now(); auto interval = std::chrono::duration_cast<std::chrono::seconds>(now - clock); try { for (auto item : res) { boost::filesystem::path file(item); if (!boost::filesystem::exists(file)) { continue ; } std::time_t lastModified = boost::filesystem::last_write_time(file); auto fileTimePoint = std::chrono::system_clock::from_time_t(lastModified); auto duration = std::chrono::duration_cast<std::chrono::seconds>(now - fileTimePoint); if (duration.count() <= interval.count()) { tmpres.push_back(file.string()); } } }catch(boost::filesystem::filesystem_error&) {} if (tmpres.empty()) { return ""; } auto resstr = tmpres.at(0); return resstr; } |
卸载
linux
的卸载,同样还是调用了system
命令
1 2 3 4 5 6 7 8 9 10 11 12 13 |
boost::filesystem::path path(m_path); std::string programName = path.stem().string(); std::string uninstallCmd = "pkexec dpkg -P --force-all " + programName; int result = system(uninstallCmd.c_str()); if (result == 0) { if (m_unist && m_appid) { m_unist->uninstall_status(install_type::em_it_uninstalled_success, m_appid); } } else { if (m_unist && m_appid) { m_unist->uninstall_status(install_type::em_it_uninstalled_failed, m_appid); } } |
- 关于上面代码中的路径,使用了
list
文件的名字来拼接了卸载命令
1 |
std::string programName = path.stem().string(); |
macOS
安装
macos
下面的安装,和上面讲的linux
也有类似之处- 同样是通过文件数量来定位安装的是哪个文件,不过
macos
用的bom
文件
- 同样是通过文件数量来定位安装的是哪个文件,不过
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
std::unordered_set<std::string> reg_helper::get_reg_items() { std::unordered_set<std::string> res; try { boost::filesystem::path dirPath("/var/db/receipts"); for (boost::filesystem::directory_iterator it(dirPath); it != boost::filesystem::directory_iterator(); ++it) { if (it->path().extension() == ".bom") { auto file = it->path().string(); res.insert(file); } } } catch (const boost::filesystem::filesystem_error& err) { return {}; } } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
auto str = get_goal_reg(res2, res1); std::string installer_helper::get_goal_reg(std::unordered_set<std::string>& res1, std::unordered_set<std::string>& res2) { std::vector<std::string> tmp; for (auto item : res1) { if (res2.find(item) != res2.end()) { continue; } tmp.push_back(item); } if (tmp.empty()) { return ""; } return tmp[0]; } |
- 安装
1 2 |
auto res = create_process(path); auto res2 = reg.get_reg_items(); |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 |
// create_process std::string tmpStr = to_str(path); if (tmpStr.empty()) { return sync_status::em_syst_create_process_failed; } #if defined(__APPLE__) std::string tmp("/System/Library/CoreServices/Installer.app/Contents/MacOS/Installer "); #elif defined(__linux__) std::string tmp("pkexec dpkg -i "); #endif std::string cmdStr("\""); cmdStr.append(tmpStr).append("\""); tmp.append(cmdStr); int return_code = system(tmp.c_str()); if (return_code == 0) { return sync_status::em_syst_install_success; } else { return sync_status::em_syst_install_failed; } #endif } |
- 至于路径的确定,
macos
这里使用了不一样的方法- 可以看到调用了一个
get_goal_second_path
- 具体就是,定位到这个
bom
文件,用apple
的工具lsbom
去输出它的内容,并把内容重定位到一个临时文件里面 - 然后再冲文件里面去逐行读取
- 把第一个找到
.app
字样的行进行处理,从中拿出一个路径,类似/Applications/xx.app
- 可以看到调用了一个
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
auto str = get_goal_reg(res2, res1); if (!str.empty()) { auto res_path = get_goal_second_path(str); if (res_path.empty()) { type = install_type::em_it_installed_failed; break; } size_t len = std::mbstowcs(nullptr, res_path.c_str(), 0); std::wstring wstr_to(len, L'\0'); std::mbstowcs(&wstr_to[0], str.c_str(), len); set_db_record(appid, wstr_to); type = install_type::em_it_installed_success; } else { type = install_type::em_it_installed_failed; } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 |
std::string installer_helper::get_goal_second_path(std::string & respath) { std::string res_str; #if defined(__APPLE__) auto path = get_ues_path(); if (path.empty()) { return ""; } struct stat buff; const char* log_path = path.c_str(); if (stat(log_path, &buff) != 0) { auto mask_tmp = umask(0); mode_t mode = 0777; ::mkdir(log_path, mode); umask(mask_tmp); } path.append("/tmpbom"); auto tmp_bomm = respath; std::string command = "lsbom " + tmp_bomm + " > " + path; int result = system(command.c_str()); if (result != 0) { return ""; } std::ifstream infile(path); if (!infile.is_open()) { return ""; } std::string line; while (std::getline(infile, line)) { if (line.size() >= 4) { auto pos = line.find(".app"); if (pos != std::string::npos) { res_str = line.substr(1, pos + 3); break; } } } infile.close(); std::remove(path.c_str()); #endif return res_str; } |
- 同样的,上面的
bom
的文件数量也有一样的时候,因为这个bom
文件是apple
操作系统写的,所以,软件卸载的时候不会去删掉它- 也就是,第二次安装的时候,这个软件对应的
bom
文件其实已经存在的 - 所以,还是需要和上面
linux
一样的方法来确定到底是哪个bom
文件 - 不同的是,
apple
这里的实现,我当时写死了,就是判断30秒以内的
- 也就是,第二次安装的时候,这个软件对应的
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 |
std::string installer_helper::get_goal_second() { std::vector<std::string> tmpres; auto now = std::chrono::system_clock::now(); std::string res; try { boost::filesystem::path dirPath("/var/db/receipts"); for (boost::filesystem::directory_iterator it(dirPath); it != boost::filesystem::directory_iterator(); ++it) { auto file = it->path(); if (file.extension() != ".bom") { continue; } std::time_t lastModified = boost::filesystem::last_write_time(file); auto fileTimePoint = std::chrono::system_clock::from_time_t(lastModified); auto duration = std::chrono::duration_cast<std::chrono::seconds>(now - fileTimePoint); if (duration.count() <= 30) { tmpres.push_back(file.string()); } } } catch (const boost::filesystem::filesystem_error& err) { } if (tmpres.empty()) { return ""; } std::string res_str; res_str = get_goal_second_path(tmpres[0]); return res_str; } |
- 然后存储的时候,就是把软件的队友编号,和上面解析出来的那个
/Applications/xx.app
这个路径对照起来,然后存起来
卸载
- 卸载的话,首先从数据库里面获取了软件编号对应的应用路径,如
/Applications/xx.app
- 然后,为了达到阻塞的效果,使用了
oc
的方法,如下:- 具体的效果,就是用这个路径,拼了一个苹果脚本
- 这个脚本的效果,就是把这个路径对应的应用放到废纸篓(涉及权限)
- 为了阻塞的去执行这个任务,用
oc
的方法创建了一个进程
1 |
int result = unist_task(tmpStr.c_str()); |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 |
int unist_task(const char* path) { @autoreleasepool { OSStatus status; AuthorizationRef authorizationRef; status = AuthorizationCreate(nullptr, kAuthorizationEmptyEnvironment, kAuthorizationFlagDefaults, &authorizationRef); if (status != errAuthorizationSuccess) { return -1; } NSString *osascriptCmd = [NSString stringWithFormat:@"tell app \"Finder\" to move POSIX file \"%s\" to trash", path]; const char *cmd = [osascriptCmd UTF8String]; char *osascriptArgs[] = { "-e", (char*)cmd, nullptr }; FILE *pipe = nullptr; status = AuthorizationExecuteWithPrivileges(authorizationRef, "/usr/bin/osascript", kAuthorizationFlagDefaults, osascriptArgs, &pipe); if (status != errAuthorizationSuccess) { return -1; } char buffer[1024]; while (fgets(buffer, sizeof(buffer), pipe) != nullptr) { } bool flag = false; do { int exitStatus = pclose(pipe); if (WIFEXITED(exitStatus)) { int returnCode = WEXITSTATUS(exitStatus); if (returnCode == 0) { flag = true; break; } } } while(false); AuthorizationFree(authorizationRef, kAuthorizationFlagDestroyRights); if (flag) { return 0; } return -1; } } |
本文为原创文章,版权归Aet所有,欢迎分享本文,转载请保留出处!
你可能也喜欢
- ♥ 【Manjaro】Vmware分辨率不能修改03/22
- ♥ Linux 线程等待&&取消&&终止03/31
- ♥ Linux_ 命令大全 磁盘管理03/16
- ♥ Linux 基于文件描述符的文件操作(非缓冲)03/23
- ♥ X86_64汇编学习记述四08/09
- ♥ Dump分析:堆内存泄露03/29