mirror of
https://github.com/rustdesk/rustdesk.git
synced 2024-12-19 06:31:02 +08:00
6cd107c3e3
* refact: msi service, shell execute control Signed-off-by: fufesou <shuanglongchen@yeah.net> * fix: msi, skip adding rules when uninstalling Signed-off-by: fufesou <shuanglongchen@yeah.net> * line indentation Signed-off-by: fufesou <shuanglongchen@yeah.net> --------- Signed-off-by: fufesou <shuanglongchen@yeah.net>
874 lines
30 KiB
C++
874 lines
30 KiB
C++
// CustomAction.cpp : Defines the entry point for the custom action.
|
|
#include "pch.h"
|
|
#include <stdlib.h>
|
|
#include <strutil.h>
|
|
#include <shellapi.h>
|
|
#include <tlhelp32.h>
|
|
#include <winternl.h>
|
|
#include <netfw.h>
|
|
#include <shlwapi.h>
|
|
|
|
#include "./Common.h"
|
|
|
|
#pragma comment(lib, "Shlwapi.lib")
|
|
|
|
UINT __stdcall CustomActionHello(
|
|
__in MSIHANDLE hInstall)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD er = ERROR_SUCCESS;
|
|
|
|
hr = WcaInitialize(hInstall, "CustomActionHello");
|
|
ExitOnFailure(hr, "Failed to initialize");
|
|
|
|
WcaLog(LOGMSG_STANDARD, "Initialized.");
|
|
|
|
// TODO: Add your custom action code here.
|
|
WcaLog(LOGMSG_STANDARD, "================= Example CustomAction Hello");
|
|
|
|
LExit:
|
|
er = SUCCEEDED(hr) ? ERROR_SUCCESS : ERROR_INSTALL_FAILURE;
|
|
return WcaFinalize(er);
|
|
}
|
|
|
|
UINT __stdcall RemoveInstallFolder(
|
|
__in MSIHANDLE hInstall)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD er = ERROR_SUCCESS;
|
|
|
|
int nResult = 0;
|
|
LPWSTR installFolder = NULL;
|
|
LPWSTR pwz = NULL;
|
|
LPWSTR pwzData = NULL;
|
|
|
|
hr = WcaInitialize(hInstall, "RemoveInstallFolder");
|
|
ExitOnFailure(hr, "Failed to initialize");
|
|
|
|
hr = WcaGetProperty(L"CustomActionData", &pwzData);
|
|
ExitOnFailure(hr, "failed to get CustomActionData");
|
|
|
|
pwz = pwzData;
|
|
hr = WcaReadStringFromCaData(&pwz, &installFolder);
|
|
ExitOnFailure(hr, "failed to read database key from custom action data: %ls", pwz);
|
|
|
|
SHFILEOPSTRUCTW fileOp;
|
|
ZeroMemory(&fileOp, sizeof(SHFILEOPSTRUCT));
|
|
|
|
fileOp.wFunc = FO_DELETE;
|
|
fileOp.pFrom = installFolder;
|
|
fileOp.fFlags = FOF_NOCONFIRMATION | FOF_SILENT;
|
|
|
|
nResult = SHFileOperationW(&fileOp);
|
|
if (nResult == 0)
|
|
{
|
|
WcaLog(LOGMSG_STANDARD, "The directory \"%ls\" has been deleted.", installFolder);
|
|
}
|
|
else
|
|
{
|
|
WcaLog(LOGMSG_STANDARD, "The directory \"%ls\" has not been deleted, error code: 0x%02X. Please refer to https://learn.microsoft.com/en-us/windows/win32/api/shellapi/nf-shellapi-shfileoperationa for the error codes.", installFolder, nResult);
|
|
}
|
|
|
|
LExit:
|
|
ReleaseStr(pwzData);
|
|
|
|
er = SUCCEEDED(hr) ? ERROR_SUCCESS : ERROR_INSTALL_FAILURE;
|
|
return WcaFinalize(er);
|
|
}
|
|
|
|
// https://learn.microsoft.com/en-us/windows/win32/api/winternl/nf-winternl-ntqueryinformationprocess
|
|
// **NtQueryInformationProcess** may be altered or unavailable in future versions of Windows.
|
|
// Applications should use the alternate functions listed in this topic.
|
|
// But I do not find the alternate functions.
|
|
// https://github.com/heim-rs/heim/issues/105#issuecomment-683647573
|
|
typedef NTSTATUS(NTAPI *pfnNtQueryInformationProcess)(HANDLE, PROCESSINFOCLASS, PVOID, ULONG, PULONG);
|
|
bool TerminateProcessIfNotContainsParam(pfnNtQueryInformationProcess NtQueryInformationProcess, HANDLE process, LPCWSTR excludeParam)
|
|
{
|
|
bool processClosed = false;
|
|
PROCESS_BASIC_INFORMATION processInfo;
|
|
NTSTATUS status = NtQueryInformationProcess(process, ProcessBasicInformation, &processInfo, sizeof(processInfo), NULL);
|
|
if (status == 0 && processInfo.PebBaseAddress != NULL)
|
|
{
|
|
PEB peb;
|
|
SIZE_T dwBytesRead;
|
|
if (ReadProcessMemory(process, processInfo.PebBaseAddress, &peb, sizeof(peb), &dwBytesRead))
|
|
{
|
|
RTL_USER_PROCESS_PARAMETERS pebUpp;
|
|
if (ReadProcessMemory(process,
|
|
peb.ProcessParameters,
|
|
&pebUpp,
|
|
sizeof(RTL_USER_PROCESS_PARAMETERS),
|
|
&dwBytesRead))
|
|
{
|
|
if (pebUpp.CommandLine.Length > 0)
|
|
{
|
|
WCHAR *commandLine = (WCHAR *)malloc(pebUpp.CommandLine.Length);
|
|
if (commandLine != NULL)
|
|
{
|
|
if (ReadProcessMemory(process, pebUpp.CommandLine.Buffer,
|
|
commandLine, pebUpp.CommandLine.Length, &dwBytesRead))
|
|
{
|
|
if (wcsstr(commandLine, excludeParam) == NULL)
|
|
{
|
|
WcaLog(LOGMSG_STANDARD, "Terminate process : %ls", commandLine);
|
|
TerminateProcess(process, 0);
|
|
processClosed = true;
|
|
}
|
|
}
|
|
free(commandLine);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return processClosed;
|
|
}
|
|
|
|
// Terminate processes that do not have parameter [excludeParam]
|
|
// Note. This function relies on "NtQueryInformationProcess",
|
|
// which may not be found.
|
|
// Then all processes of [processName] will be terminated.
|
|
bool TerminateProcessesByNameW(LPCWSTR processName, LPCWSTR excludeParam)
|
|
{
|
|
HMODULE hntdll = GetModuleHandleW(L"ntdll.dll");
|
|
if (hntdll == NULL)
|
|
{
|
|
WcaLog(LOGMSG_STANDARD, "Failed to load ntdll.");
|
|
}
|
|
|
|
pfnNtQueryInformationProcess NtQueryInformationProcess = NULL;
|
|
if (hntdll != NULL)
|
|
{
|
|
NtQueryInformationProcess = (pfnNtQueryInformationProcess)GetProcAddress(
|
|
hntdll, "NtQueryInformationProcess");
|
|
}
|
|
if (NtQueryInformationProcess == NULL)
|
|
{
|
|
WcaLog(LOGMSG_STANDARD, "Failed to get address of NtQueryInformationProcess.");
|
|
}
|
|
|
|
bool processClosed = false;
|
|
// Create a snapshot of the current system processes
|
|
HANDLE snapshot = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
|
|
if (snapshot != INVALID_HANDLE_VALUE)
|
|
{
|
|
PROCESSENTRY32W processEntry;
|
|
processEntry.dwSize = sizeof(PROCESSENTRY32W);
|
|
if (Process32FirstW(snapshot, &processEntry))
|
|
{
|
|
do
|
|
{
|
|
if (lstrcmpW(processName, processEntry.szExeFile) == 0)
|
|
{
|
|
HANDLE process = OpenProcess(PROCESS_TERMINATE | PROCESS_QUERY_INFORMATION | PROCESS_VM_READ, FALSE, processEntry.th32ProcessID);
|
|
if (process != NULL)
|
|
{
|
|
if (NtQueryInformationProcess == NULL)
|
|
{
|
|
WcaLog(LOGMSG_STANDARD, "Terminate process : %ls, while NtQueryInformationProcess is NULL", processName);
|
|
TerminateProcess(process, 0);
|
|
processClosed = true;
|
|
}
|
|
else
|
|
{
|
|
processClosed = TerminateProcessIfNotContainsParam(
|
|
NtQueryInformationProcess,
|
|
process,
|
|
excludeParam);
|
|
}
|
|
CloseHandle(process);
|
|
}
|
|
}
|
|
} while (Process32NextW(snapshot, &processEntry));
|
|
}
|
|
CloseHandle(snapshot);
|
|
}
|
|
if (hntdll != NULL)
|
|
{
|
|
CloseHandle(hntdll);
|
|
}
|
|
return processClosed;
|
|
}
|
|
|
|
UINT __stdcall TerminateProcesses(
|
|
__in MSIHANDLE hInstall)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD er = ERROR_SUCCESS;
|
|
|
|
int nResult = 0;
|
|
wchar_t szProcess[256] = {0};
|
|
DWORD cchProcess = sizeof(szProcess) / sizeof(szProcess[0]);
|
|
|
|
hr = WcaInitialize(hInstall, "TerminateProcesses");
|
|
ExitOnFailure(hr, "Failed to initialize");
|
|
|
|
MsiGetPropertyW(hInstall, L"TerminateProcesses", szProcess, &cchProcess);
|
|
|
|
WcaLog(LOGMSG_STANDARD, "Try terminate processes : %ls", szProcess);
|
|
TerminateProcessesByNameW(szProcess, L"--install");
|
|
|
|
LExit:
|
|
er = SUCCEEDED(hr) ? ERROR_SUCCESS : ERROR_INSTALL_FAILURE;
|
|
return WcaFinalize(er);
|
|
}
|
|
|
|
// No use for now, it can be refer as an example of ShellExecuteW.
|
|
void AddFirewallRuleCmdline(LPWSTR exeName, LPWSTR exeFile, LPCWSTR dir)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
HINSTANCE hi = 0;
|
|
WCHAR cmdline[1024] = { 0, };
|
|
WCHAR rulename[500] = { 0, };
|
|
|
|
StringCchPrintfW(rulename, sizeof(rulename) / sizeof(rulename[0]), L"%ls Service", exeName);
|
|
if (FAILED(hr)) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to make rulename: %ls", exeName);
|
|
return;
|
|
}
|
|
|
|
StringCchPrintfW(cmdline, sizeof(cmdline) / sizeof(cmdline[0]), L"advfirewall firewall add rule name=\"%ls\" dir=%ls action=allow program=\"%ls\" enable=yes", rulename, dir, exeFile);
|
|
if (FAILED(hr)) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to make cmdline: %ls", exeName);
|
|
return;
|
|
}
|
|
|
|
hi = ShellExecuteW(NULL, L"open", L"netsh", cmdline, NULL, SW_HIDE);
|
|
// https://learn.microsoft.com/en-us/windows/win32/api/shellapi/nf-shellapi-shellexecutew
|
|
if ((int)hi <= 32) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to change firewall rule : %d, last error: %d", (int)hi, GetLastError());
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Firewall rule \"%ls\" (%ls) is added", rulename, dir);
|
|
}
|
|
}
|
|
|
|
// No use for now, it can be refer as an example of ShellExecuteW.
|
|
void RemoveFirewallRuleCmdline(LPWSTR exeName)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
HINSTANCE hi = 0;
|
|
WCHAR cmdline[1024] = { 0, };
|
|
WCHAR rulename[500] = { 0, };
|
|
|
|
StringCchPrintfW(rulename, sizeof(rulename) / sizeof(rulename[0]), L"%ls Service", exeName);
|
|
if (FAILED(hr)) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to make rulename: %ls", exeName);
|
|
return;
|
|
}
|
|
|
|
StringCchPrintfW(cmdline, sizeof(cmdline) / sizeof(cmdline[0]), L"advfirewall firewall delete rule name=\"%ls\"", rulename);
|
|
if (FAILED(hr)) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to make cmdline: %ls", exeName);
|
|
return;
|
|
}
|
|
|
|
hi = ShellExecuteW(NULL, L"open", L"netsh", cmdline, NULL, SW_HIDE);
|
|
// https://learn.microsoft.com/en-us/windows/win32/api/shellapi/nf-shellapi-shellexecutew
|
|
if ((int)hi <= 32) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to change firewall rule \"%ls\" : %d, last error: %d", rulename, (int)hi, GetLastError());
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Firewall rule \"%ls\" is removed", rulename);
|
|
}
|
|
}
|
|
|
|
UINT __stdcall AddFirewallRules(
|
|
__in MSIHANDLE hInstall)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD er = ERROR_SUCCESS;
|
|
|
|
int nResult = 0;
|
|
LPWSTR exeFile = NULL;
|
|
LPWSTR exeName = NULL;
|
|
WCHAR exeNameNoExt[500] = { 0, };
|
|
LPWSTR pwz = NULL;
|
|
LPWSTR pwzData = NULL;
|
|
size_t szNameLen = 0;
|
|
|
|
hr = WcaInitialize(hInstall, "AddFirewallRules");
|
|
ExitOnFailure(hr, "Failed to initialize");
|
|
|
|
hr = WcaGetProperty(L"CustomActionData", &pwzData);
|
|
ExitOnFailure(hr, "failed to get CustomActionData");
|
|
|
|
pwz = pwzData;
|
|
hr = WcaReadStringFromCaData(&pwz, &exeFile);
|
|
ExitOnFailure(hr, "failed to read database key from custom action data: %ls", pwz);
|
|
WcaLog(LOGMSG_STANDARD, "Try add firewall exceptions for file : %ls", exeFile);
|
|
|
|
exeName = PathFindFileNameW(exeFile + 1);
|
|
hr = StringCchPrintfW(exeNameNoExt, 500, exeName);
|
|
ExitOnFailure(hr, "Failed to copy exe name: %ls", exeName);
|
|
szNameLen = wcslen(exeNameNoExt);
|
|
if (szNameLen >= 4 && wcscmp(exeNameNoExt + szNameLen - 4, L".exe") == 0) {
|
|
exeNameNoExt[szNameLen - 4] = L'\0';
|
|
}
|
|
|
|
//if (exeFile[0] == L'1') {
|
|
// AddFirewallRuleCmdline(exeNameNoExt, exeFile, L"in");
|
|
// AddFirewallRuleCmdline(exeNameNoExt, exeFile, L"out");
|
|
//}
|
|
//else {
|
|
// RemoveFirewallRuleCmdline(exeNameNoExt);
|
|
//}
|
|
|
|
AddFirewallRule(exeFile[0] == L'1', exeNameNoExt, exeFile + 1);
|
|
|
|
LExit:
|
|
if (pwzData) {
|
|
ReleaseStr(pwzData);
|
|
}
|
|
|
|
er = SUCCEEDED(hr) ? ERROR_SUCCESS : ERROR_INSTALL_FAILURE;
|
|
return WcaFinalize(er);
|
|
}
|
|
|
|
UINT __stdcall SetPropertyIsServiceRunning(__in MSIHANDLE hInstall)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD er = ERROR_SUCCESS;
|
|
|
|
wchar_t szAppName[500] = { 0 };
|
|
DWORD cchAppName = sizeof(szAppName) / sizeof(szAppName[0]);
|
|
wchar_t szPropertyName[500] = { 0 };
|
|
DWORD cchPropertyName = sizeof(szPropertyName) / sizeof(szPropertyName[0]);
|
|
bool isRunning = false;
|
|
|
|
hr = WcaInitialize(hInstall, "SetPropertyIsServiceRunning");
|
|
ExitOnFailure(hr, "Failed to initialize");
|
|
|
|
MsiGetPropertyW(hInstall, L"AppName", szAppName, &cchAppName);
|
|
WcaLog(LOGMSG_STANDARD, "Try query service of : \"%ls\"", szAppName);
|
|
|
|
MsiGetPropertyW(hInstall, L"PropertyName", szPropertyName, &cchPropertyName);
|
|
WcaLog(LOGMSG_STANDARD, "Try set is service running, property name : \"%ls\"", szPropertyName);
|
|
|
|
isRunning = IsServiceRunningW(szAppName);
|
|
MsiSetPropertyW(hInstall, szPropertyName, isRunning ? L"'N'" : L"'Y'");
|
|
|
|
LExit:
|
|
er = SUCCEEDED(hr) ? ERROR_SUCCESS : ERROR_INSTALL_FAILURE;
|
|
return WcaFinalize(er);
|
|
}
|
|
|
|
void TryCreateStartServiceByShell(LPWSTR svcName, LPWSTR svcBinary, LPWSTR szSvcDisplayName);
|
|
UINT __stdcall CreateStartService(__in MSIHANDLE hInstall)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD er = ERROR_SUCCESS;
|
|
|
|
LPWSTR svcParams = NULL;
|
|
LPWSTR pwz = NULL;
|
|
LPWSTR pwzData = NULL;
|
|
LPWSTR svcName = NULL;
|
|
LPWSTR svcBinary = NULL;
|
|
wchar_t szSvcDisplayName[500] = { 0 };
|
|
DWORD cchSvcDisplayName = sizeof(szSvcDisplayName) / sizeof(szSvcDisplayName[0]);
|
|
|
|
hr = WcaInitialize(hInstall, "CreateStartService");
|
|
ExitOnFailure(hr, "Failed to initialize");
|
|
|
|
hr = WcaGetProperty(L"CustomActionData", &pwzData);
|
|
ExitOnFailure(hr, "failed to get CustomActionData");
|
|
|
|
pwz = pwzData;
|
|
hr = WcaReadStringFromCaData(&pwz, &svcParams);
|
|
ExitOnFailure(hr, "failed to read database key from custom action data: %ls", pwz);
|
|
|
|
WcaLog(LOGMSG_STANDARD, "Try create start service : %ls", svcParams);
|
|
|
|
svcName = svcParams;
|
|
svcBinary = wcschr(svcParams, L';');
|
|
if (svcBinary == NULL) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to find binary : %ls", svcParams);
|
|
goto LExit;
|
|
}
|
|
svcBinary[0] = L'\0';
|
|
svcBinary += 1;
|
|
|
|
hr = StringCchPrintfW(szSvcDisplayName, cchSvcDisplayName, L"%ls Service", svcName);
|
|
ExitOnFailure(hr, "Failed to compose a resource identifier string");
|
|
if (MyCreateServiceW(svcName, szSvcDisplayName, svcBinary)) {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is created.", svcName);
|
|
if (MyStartServiceW(svcName)) {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is started.", svcName);
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to start service: \"%ls\"", svcName);
|
|
}
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to create service: \"%ls\"", svcName);
|
|
}
|
|
|
|
if (IsServiceRunningW(svcName)) {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is running.", svcName);
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is not running, try create and start service by shell", svcName);
|
|
TryCreateStartServiceByShell(svcName, svcBinary, szSvcDisplayName);
|
|
}
|
|
|
|
LExit:
|
|
if (pwzData) {
|
|
ReleaseStr(pwzData);
|
|
}
|
|
|
|
er = SUCCEEDED(hr) ? ERROR_SUCCESS : ERROR_INSTALL_FAILURE;
|
|
return WcaFinalize(er);
|
|
}
|
|
|
|
void TryStopDeleteServiceByShell(LPWSTR svcName);
|
|
UINT __stdcall TryStopDeleteService(__in MSIHANDLE hInstall)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD er = ERROR_SUCCESS;
|
|
|
|
int nResult = 0;
|
|
LPWSTR svcName = NULL;
|
|
LPWSTR pwz = NULL;
|
|
LPWSTR pwzData = NULL;
|
|
wchar_t szExeFile[500] = { 0 };
|
|
DWORD cchExeFile = sizeof(szExeFile) / sizeof(szExeFile[0]);
|
|
SERVICE_STATUS_PROCESS svcStatus;
|
|
DWORD lastErrorCode = 0;
|
|
|
|
hr = WcaInitialize(hInstall, "TryStopDeleteService");
|
|
ExitOnFailure(hr, "Failed to initialize");
|
|
|
|
hr = WcaGetProperty(L"CustomActionData", &pwzData);
|
|
ExitOnFailure(hr, "failed to get CustomActionData");
|
|
|
|
pwz = pwzData;
|
|
hr = WcaReadStringFromCaData(&pwz, &svcName);
|
|
ExitOnFailure(hr, "failed to read database key from custom action data: %ls", pwz);
|
|
WcaLog(LOGMSG_STANDARD, "Try stop and delete service : %ls", svcName);
|
|
|
|
if (MyStopServiceW(svcName)) {
|
|
for (int i = 0; i < 10; i++) {
|
|
if (IsServiceRunningW(svcName)) {
|
|
Sleep(100);
|
|
}
|
|
else {
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to stop service: \"%ls\", error: 0x%02X.", svcName, GetLastError());
|
|
}
|
|
|
|
if (IsServiceRunningW(svcName)) {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is not stoped after 1000 ms.", svcName);
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is stoped.", svcName);
|
|
}
|
|
|
|
if (MyDeleteServiceW(svcName)) {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" deletion is completed without errors.", svcName);
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to delete service: \"%ls\", error: 0x%02X.", svcName, GetLastError());
|
|
}
|
|
|
|
if (QueryServiceStatusExW(svcName, &svcStatus)) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to delete service: \"%ls\", current status: %d.", svcName, svcStatus.dwCurrentState);
|
|
TryStopDeleteServiceByShell(svcName);
|
|
}
|
|
else {
|
|
lastErrorCode = GetLastError();
|
|
if (lastErrorCode == ERROR_SERVICE_DOES_NOT_EXIST) {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is deleted.", svcName);
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to query service status: \"%ls\", error: 0x%02X.", svcName, lastErrorCode);
|
|
TryStopDeleteServiceByShell(svcName);
|
|
}
|
|
}
|
|
|
|
// It's really strange that we need sleep here.
|
|
// But the upgrading may be stucked at "copying new files" because the file is in using.
|
|
// Steps to reproduce: Install -> stop service in tray --> start service -> upgrade
|
|
// Sleep(300);
|
|
|
|
// Or we can terminate the process
|
|
hr = StringCchPrintfW(szExeFile, cchExeFile, L"%ls.exe", svcName);
|
|
ExitOnFailure(hr, "Failed to compose a resource identifier string");
|
|
TerminateProcessesByNameW(szExeFile, L"--not-in-use");
|
|
|
|
LExit:
|
|
if (pwzData) {
|
|
ReleaseStr(pwzData);
|
|
}
|
|
|
|
er = SUCCEEDED(hr) ? ERROR_SUCCESS : ERROR_INSTALL_FAILURE;
|
|
return WcaFinalize(er);
|
|
}
|
|
|
|
UINT __stdcall TryDeleteStartupShortcut(__in MSIHANDLE hInstall)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD er = ERROR_SUCCESS;
|
|
|
|
wchar_t szShortcut[500] = { 0 };
|
|
DWORD cchShortcut = sizeof(szShortcut) / sizeof(szShortcut[0]);
|
|
wchar_t szStartupDir[500] = { 0 };
|
|
DWORD cchStartupDir = sizeof(szStartupDir) / sizeof(szStartupDir[0]);
|
|
WCHAR pwszTemp[1024] = L"";
|
|
|
|
hr = WcaInitialize(hInstall, "DeleteStartupShortcut");
|
|
ExitOnFailure(hr, "Failed to initialize");
|
|
|
|
MsiGetPropertyW(hInstall, L"StartupFolder", szStartupDir, &cchStartupDir);
|
|
|
|
MsiGetPropertyW(hInstall, L"ShortcutName", szShortcut, &cchShortcut);
|
|
WcaLog(LOGMSG_STANDARD, "Try delete startup shortcut of : \"%ls\"", szShortcut);
|
|
|
|
hr = StringCchPrintfW(pwszTemp, 1024, L"%ls%ls.lnk", szStartupDir, szShortcut);
|
|
ExitOnFailure(hr, "Failed to compose a resource identifier string");
|
|
|
|
if (DeleteFileW(pwszTemp)) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to delete startup shortcut of : \"%ls\"", pwszTemp);
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Startup shortcut is deleted : \"%ls\"", pwszTemp);
|
|
}
|
|
|
|
LExit:
|
|
er = SUCCEEDED(hr) ? ERROR_SUCCESS : ERROR_INSTALL_FAILURE;
|
|
return WcaFinalize(er);
|
|
}
|
|
|
|
UINT __stdcall SetPropertyFromConfig(__in MSIHANDLE hInstall)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD er = ERROR_SUCCESS;
|
|
|
|
wchar_t szConfigFile[1024] = { 0 };
|
|
DWORD cchConfigFile = sizeof(szConfigFile) / sizeof(szConfigFile[0]);
|
|
wchar_t szConfigKey[500] = { 0 };
|
|
DWORD cchConfigKey = sizeof(szConfigKey) / sizeof(szConfigKey[0]);
|
|
wchar_t szPropertyName[500] = { 0 };
|
|
DWORD cchPropertyName = sizeof(szPropertyName) / sizeof(szPropertyName[0]);
|
|
std::wstring configValue;
|
|
|
|
hr = WcaInitialize(hInstall, "SetPropertyFromConfig");
|
|
ExitOnFailure(hr, "Failed to initialize");
|
|
|
|
MsiGetPropertyW(hInstall, L"ConfigFile", szConfigFile, &cchConfigFile);
|
|
WcaLog(LOGMSG_STANDARD, "Try read config file of : \"%ls\"", szConfigFile);
|
|
|
|
MsiGetPropertyW(hInstall, L"ConfigKey", szConfigKey, &cchConfigKey);
|
|
WcaLog(LOGMSG_STANDARD, "Try read configuration, config key : \"%ls\"", szConfigKey);
|
|
|
|
MsiGetPropertyW(hInstall, L"PropertyName", szPropertyName, &cchPropertyName);
|
|
WcaLog(LOGMSG_STANDARD, "Try read configuration, property name : \"%ls\"", szPropertyName);
|
|
|
|
configValue = ReadConfig(szConfigFile, szConfigKey);
|
|
MsiSetPropertyW(hInstall, szPropertyName, configValue.c_str());
|
|
|
|
LExit:
|
|
er = SUCCEEDED(hr) ? ERROR_SUCCESS : ERROR_INSTALL_FAILURE;
|
|
return WcaFinalize(er);
|
|
}
|
|
|
|
UINT __stdcall AddRegSoftwareSASGeneration(__in MSIHANDLE hInstall)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD er = ERROR_SUCCESS;
|
|
|
|
LSTATUS result = 0;
|
|
HKEY hKey;
|
|
LPCWSTR subKey = L"Software\\Microsoft\\Windows\\CurrentVersion\\Policies\\System";
|
|
LPCWSTR valueName = L"SoftwareSASGeneration";
|
|
DWORD valueType = REG_DWORD;
|
|
DWORD valueData = 1;
|
|
DWORD valueDataSize = sizeof(DWORD);
|
|
|
|
HINSTANCE hi = 0;
|
|
|
|
hr = WcaInitialize(hInstall, "AddRegSoftwareSASGeneration");
|
|
ExitOnFailure(hr, "Failed to initialize");
|
|
|
|
hi = ShellExecuteW(NULL, L"open", L"reg", L" add HKEY_LOCAL_MACHINE\\Software\\Microsoft\\Windows\\CurrentVersion\\Policies\\System /f /v SoftwareSASGeneration /t REG_DWORD /d 1", NULL, SW_HIDE);
|
|
// https://learn.microsoft.com/en-us/windows/win32/api/shellapi/nf-shellapi-shellexecutew
|
|
if ((int)hi <= 32) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to add registry name \"%ls\", %d, %d", valueName, (int)hi, GetLastError());
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Registry name \"%ls\" is added", valueName);
|
|
}
|
|
|
|
// Why RegSetValueExW always return 998?
|
|
//
|
|
result = RegCreateKeyExW(HKEY_LOCAL_MACHINE, subKey, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_WRITE, NULL, &hKey, NULL);
|
|
if (result != ERROR_SUCCESS) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to create or open registry key: %d", result);
|
|
goto LExit;
|
|
}
|
|
|
|
result = RegSetValueExW(hKey, valueName, 0, valueType, reinterpret_cast<const BYTE*>(valueData), valueDataSize);
|
|
if (result != ERROR_SUCCESS) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to set registry value: %d", result);
|
|
RegCloseKey(hKey);
|
|
goto LExit;
|
|
}
|
|
|
|
WcaLog(LOGMSG_STANDARD, "Registry value has been successfully set.");
|
|
RegCloseKey(hKey);
|
|
|
|
LExit:
|
|
er = SUCCEEDED(hr) ? ERROR_SUCCESS : ERROR_INSTALL_FAILURE;
|
|
return WcaFinalize(er);
|
|
}
|
|
|
|
UINT __stdcall RemoveAmyuniIdd(
|
|
__in MSIHANDLE hInstall)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
DWORD er = ERROR_SUCCESS;
|
|
|
|
int nResult = 0;
|
|
LPWSTR installFolder = NULL;
|
|
LPWSTR pwz = NULL;
|
|
LPWSTR pwzData = NULL;
|
|
|
|
WCHAR workDir[1024] = L"";
|
|
DWORD fileAttributes = 0;
|
|
HINSTANCE hi = 0;
|
|
|
|
SYSTEM_INFO si;
|
|
LPCWSTR exe = L"deviceinstaller64.exe";
|
|
WCHAR exePath[1024] = L"";
|
|
|
|
BOOL rebootRequired = FALSE;
|
|
|
|
hr = WcaInitialize(hInstall, "RemoveAmyuniIdd");
|
|
ExitOnFailure(hr, "Failed to initialize");
|
|
|
|
UninstallDriver(L"usbmmidd", rebootRequired);
|
|
|
|
// Only for x86 app on x64
|
|
GetNativeSystemInfo(&si);
|
|
if (si.wProcessorArchitecture != PROCESSOR_ARCHITECTURE_AMD64) {
|
|
goto LExit;
|
|
}
|
|
|
|
hr = WcaGetProperty(L"CustomActionData", &pwzData);
|
|
ExitOnFailure(hr, "failed to get CustomActionData");
|
|
|
|
pwz = pwzData;
|
|
hr = WcaReadStringFromCaData(&pwz, &installFolder);
|
|
ExitOnFailure(hr, "failed to read database key from custom action data: %ls", pwz);
|
|
|
|
hr = StringCchPrintfW(workDir, 1024, L"%lsusbmmidd_v2", installFolder);
|
|
ExitOnFailure(hr, "Failed to compose a resource identifier string");
|
|
fileAttributes = GetFileAttributesW(workDir);
|
|
if (fileAttributes == INVALID_FILE_ATTRIBUTES) {
|
|
WcaLog(LOGMSG_STANDARD, "Amyuni idd dir \"%ls\" is not found, %d", workDir, fileAttributes);
|
|
goto LExit;
|
|
}
|
|
|
|
hr = StringCchPrintfW(exePath, 1024, L"%ls\\%ls", workDir, exe);
|
|
ExitOnFailure(hr, "Failed to compose a resource identifier string");
|
|
fileAttributes = GetFileAttributesW(exePath);
|
|
if (fileAttributes == INVALID_FILE_ATTRIBUTES) {
|
|
goto LExit;
|
|
}
|
|
|
|
WcaLog(LOGMSG_STANDARD, "Remove amyuni idd %ls in %ls", exe, workDir);
|
|
hi = ShellExecuteW(NULL, L"open", exe, L"remove usbmmidd", workDir, SW_HIDE);
|
|
// https://learn.microsoft.com/en-us/windows/win32/api/shellapi/nf-shellapi-shellexecutew
|
|
if ((int)hi <= 32) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to remove amyuni idd : %d, last error: %d", (int)hi, GetLastError());
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Amyuni idd is removed");
|
|
}
|
|
|
|
LExit:
|
|
if (pwzData) {
|
|
ReleaseStr(pwzData);
|
|
}
|
|
|
|
er = SUCCEEDED(hr) ? ERROR_SUCCESS : ERROR_INSTALL_FAILURE;
|
|
return WcaFinalize(er);
|
|
}
|
|
|
|
void TryCreateStartServiceByShell(LPWSTR svcName, LPWSTR svcBinary, LPWSTR szSvcDisplayName)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
HINSTANCE hi = 0;
|
|
wchar_t szNewBin[500] = { 0 };
|
|
DWORD cchNewBin = sizeof(szNewBin) / sizeof(szNewBin[0]);
|
|
wchar_t szCmd[800] = { 0 };
|
|
DWORD cchCmd = sizeof(szCmd) / sizeof(szCmd[0]);
|
|
SERVICE_STATUS_PROCESS svcStatus;
|
|
DWORD lastErrorCode = 0;
|
|
int i = 0;
|
|
int j = 0;
|
|
|
|
WcaLog(LOGMSG_STANDARD, "TryCreateStartServiceByShell, service: %ls", svcName);
|
|
|
|
TryStopDeleteServiceByShell(svcName);
|
|
// Do not check the result here
|
|
|
|
i = 0;
|
|
j = 0;
|
|
// svcBinary is a string with double quotes, we need to escape it for shell arguments.
|
|
// It is orignal used for `CreateServiceW`.
|
|
// eg. "C:\Program Files\MyApp\MyApp.exe" --service -> \"C:\Program Files\MyApp\MyApp.exe\" --service
|
|
while (true) {
|
|
if (svcBinary[j] == L'"') {
|
|
szNewBin[i] = L'\\';
|
|
i += 1;
|
|
if (i >= cchNewBin) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to copy bin for service: %ls, buffer is not enough", svcName);
|
|
return;
|
|
}
|
|
szNewBin[i] = L'"';
|
|
}
|
|
else {
|
|
szNewBin[i] = svcBinary[j];
|
|
}
|
|
if (svcBinary[j] == L'\0') {
|
|
break;
|
|
}
|
|
i += 1;
|
|
j += 1;
|
|
if (i >= cchNewBin) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to copy bin for service: %ls, buffer is not enough", svcName);
|
|
return;
|
|
}
|
|
}
|
|
|
|
hr = StringCchPrintfW(szCmd, cchCmd, L"create %ls binpath= \"%ls\" start= auto DisplayName= \"%ls\"", svcName, szNewBin, szSvcDisplayName);
|
|
if (FAILED(hr)) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to make command: %ls", svcName);
|
|
return;
|
|
}
|
|
hi = ShellExecuteW(NULL, L"open", L"sc", szCmd, NULL, SW_HIDE);
|
|
if ((int)hi <= 32) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to create service with shell : %d, last error: 0x%02X.", (int)hi, GetLastError());
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is created with shell.", svcName);
|
|
}
|
|
|
|
// Query and log if the service is running.
|
|
for (int k = 0; k < 10; ++k) {
|
|
if (!QueryServiceStatusExW(svcName, &svcStatus)) {
|
|
lastErrorCode = GetLastError();
|
|
if (lastErrorCode == ERROR_SERVICE_DOES_NOT_EXIST) {
|
|
if (k == 29) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to query service status: \"%ls\", service is not found.", svcName);
|
|
return;
|
|
}
|
|
else {
|
|
Sleep(100);
|
|
continue;
|
|
}
|
|
}
|
|
// Break if the service exists.
|
|
WcaLog(LOGMSG_STANDARD, "Failed to query service status: \"%ls\", error: 0x%02X.", svcName, lastErrorCode);
|
|
break;
|
|
}
|
|
else {
|
|
if (svcStatus.dwCurrentState == SERVICE_RUNNING) {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is running.", svcName);
|
|
return;
|
|
}
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is not running.", svcName);
|
|
break;
|
|
}
|
|
}
|
|
|
|
hr = StringCchPrintfW(szCmd, cchCmd, L"/c sc start %ls", svcName);
|
|
if (FAILED(hr)) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to make command: %ls", svcName);
|
|
return;
|
|
}
|
|
hi = ShellExecuteW(NULL, L"open", L"cmd.exe", szCmd, NULL, SW_HIDE);
|
|
if ((int)hi <= 32) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to start service with shell : %d, last error: 0x%02X.", (int)hi, GetLastError());
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is started with shell.", svcName);
|
|
}
|
|
}
|
|
|
|
void TryStopDeleteServiceByShell(LPWSTR svcName)
|
|
{
|
|
HRESULT hr = S_OK;
|
|
HINSTANCE hi = 0;
|
|
wchar_t szCmd[800] = { 0 };
|
|
DWORD cchCmd = sizeof(szCmd) / sizeof(szCmd[0]);
|
|
SERVICE_STATUS_PROCESS svcStatus;
|
|
DWORD lastErrorCode = 0;
|
|
|
|
WcaLog(LOGMSG_STANDARD, "TryStopDeleteServiceByShell, service: %ls", svcName);
|
|
|
|
hr = StringCchPrintfW(szCmd, cchCmd, L"/c sc stop %ls", svcName);
|
|
if (FAILED(hr)) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to make command: %ls", svcName);
|
|
return;
|
|
}
|
|
hi = ShellExecuteW(NULL, L"open", L"cmd.exe", szCmd, NULL, SW_HIDE);
|
|
|
|
// Query and log if the service is stopped or deleted.
|
|
for (int k = 0; k < 10; ++k) {
|
|
if (!IsServiceRunningW(svcName)) {
|
|
break;
|
|
}
|
|
Sleep(100);
|
|
}
|
|
if (!QueryServiceStatusExW(svcName, &svcStatus)) {
|
|
if (GetLastError() == ERROR_SERVICE_DOES_NOT_EXIST) {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is already deleted.", svcName);
|
|
return;
|
|
}
|
|
WcaLog(LOGMSG_STANDARD, "Failed to query service status: \"%ls\" with shell, error: 0x%02X.", svcName, lastErrorCode);
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Status of service: \"%ls\" with shell, current status: %d.", svcName, svcStatus.dwCurrentState);
|
|
}
|
|
|
|
hr = StringCchPrintfW(szCmd, cchCmd, L"/c sc delete %ls", svcName);
|
|
if (FAILED(hr)) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to make command: %ls", svcName);
|
|
return;
|
|
}
|
|
hi = ShellExecuteW(NULL, L"open", L"cmd.exe", szCmd, NULL, SW_HIDE);
|
|
if ((int)hi <= 32) {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to delete service with shell : %d, last error: 0x%02X.", (int)hi, GetLastError());
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" deletion is completed without errors with shell,", svcName);
|
|
}
|
|
|
|
// Query and log the status of the service after deletion.
|
|
for (int k = 0; k < 10; ++k) {
|
|
if (!QueryServiceStatusExW(svcName, &svcStatus)) {
|
|
if (GetLastError() == ERROR_SERVICE_DOES_NOT_EXIST) {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is deleted with shell.", svcName);
|
|
return;
|
|
}
|
|
}
|
|
Sleep(100);
|
|
}
|
|
if (!QueryServiceStatusExW(svcName, &svcStatus)) {
|
|
lastErrorCode = GetLastError();
|
|
if (lastErrorCode == ERROR_SERVICE_DOES_NOT_EXIST) {
|
|
WcaLog(LOGMSG_STANDARD, "Service \"%ls\" is deleted with shell.", svcName);
|
|
return;
|
|
}
|
|
WcaLog(LOGMSG_STANDARD, "Failed to query service status: \"%ls\" with shell, error: 0x%02X.", svcName, lastErrorCode);
|
|
}
|
|
else {
|
|
WcaLog(LOGMSG_STANDARD, "Failed to delete service: \"%ls\" with shell, current status: %d.", svcName, svcStatus.dwCurrentState);
|
|
}
|
|
}
|