• QtService实现Qt后台服务程序其二_启动外部exe无窗口异常的解决


    ——接上篇——

    启动外部exe无窗口异常的解决,Qt5.9.2亲测通过

    1、背景

    Windows下服务直接启动窗口程序时,在任务管理器中可以看到窗口程序正在运行,但是桌面上并没有显示出窗口。

    这是因为在Windows XP、Windows Server 2003 或早期Windows 系统时代,当第一个用户登录系统后服务和应用程序是在同一个Session 中运行的,也就是Session 0。

    但是这种运行方式提高了系统安全风险,因为服务是通过提升了用户权限运行的,而应用程序往往是那些不具备管理员身份的普通用户运行的,其中的危险显而易见。

    所以从Vista 开始Session 0 中只包含系统服务,其他应用程序则通过分离的Session 运行,将服务与应用程序隔离提高系统的安全性。这样使得Session 0 与其他Session 之间无法进行交互,不能通过服务向桌面用户弹出信息窗口、UI 窗口等信息。

    这个时候如果想让我们的界面程序被服务启动就必须穿透Session 0 隔离。在实际开发过程中,可以通过Process Explorer检查服务或程序处于哪个Session,会不会遇到Session 0 隔离问题。

            下面就是穿透Session 0 隔离及服务启动窗口程序的步骤:

            1、使用OpenProcessToken函数来打开与服务进程相关联的访问令牌;

            2、使用DuplicateTokenEx函数创建一个新的访问令牌来复制一个已经存在的标记;

            3、使用SetTokenInformation函数把服务token的SessionId替换成当前活动的Session;

            4、使用CreateEnvironmentBlock函数创建进程环境块;

            5、使用CreateProcessAsUser函数在活动的Session下创建进程

    2、Demo说明

    加入WtsApi32.lib,Userenv.lib库等,解决QService启动外部进程没有GUI的问题

    .pro加入代码:

    1. #WtsApi32.lib,Userenv.lib库等,解决QService启动外部进程没有GUI的问题
    2. LIBS += \
    3. #-L$$DESTDIR \
    4. -lWtsApi32 \
    5. -lAdvApi32 \
    6. -lUserEnv

    加入ProcessLoader类

    processloader.h代码:

    1. #ifndef PROCESSLOADER_H
    2. #define PROCESSLOADER_H
    3. #include
    4. //解决QService启动外部进程没有GUI的问题
    5. class ProcessLoader
    6. {
    7. public:
    8. ProcessLoader();
    9. ~ProcessLoader();
    10. public:
    11. #ifdef Q_OS_WIN
    12. static bool LaunchAppIntoDifferentSession(std::wstring command);//方式2
    13. #endif
    14. };
    15. #endif // PROCESSLOADER_H

    processloader.cpp代码:

    1. #include "processloader.h"
    2. #ifdef Q_OS_WIN
    3. #include
    4. #include
    5. #include
    6. #include
    7. #include
    8. #endif
    9. ProcessLoader::ProcessLoader()
    10. {}
    11. ProcessLoader::~ProcessLoader()
    12. {}
    13. #ifdef Q_OS_WIN
    14. #include
    15. bool ProcessLoader::LaunchAppIntoDifferentSession(std::wstring command)
    16. {
    17. PROCESS_INFORMATION pi;
    18. STARTUPINFO si;
    19. BOOL bResult = FALSE;
    20. DWORD dwSessionId,winlogonPid;
    21. HANDLE hUserToken=INVALID_HANDLE_VALUE,hUserTokenDup=INVALID_HANDLE_VALUE,
    22. hPToken=INVALID_HANDLE_VALUE,hProcess=INVALID_HANDLE_VALUE;
    23. DWORD dwCreationFlags;
    24. int errorcode;//whl2023-10-18
    25. LPVOID pEnv =NULL;
    26. TCHAR commandLine[MAX_PATH];
    27. // Log the client on to the local computer.
    28. dwSessionId = WTSGetActiveConsoleSessionId();
    29. //
    30. // Find the winlogon process
    31. PROCESSENTRY32 procEntry;
    32. HANDLE hSnap = CreateToolhelp32Snapshot(TH32CS_SNAPPROCESS, 0);
    33. if (hSnap == INVALID_HANDLE_VALUE)
    34. {
    35. return false ;
    36. }
    37. procEntry.dwSize = sizeof(PROCESSENTRY32);
    38. if (!Process32First(hSnap, &procEntry))
    39. {
    40. return false ;
    41. }
    42. do
    43. {
    44. if (_wcsicmp(procEntry.szExeFile, L"winlogon.exe") == 0)
    45. {
    46. // We found a winlogon process...
    47. // make sure it's running in the console session
    48. DWORD winlogonSessId = 0;
    49. if (ProcessIdToSessionId(procEntry.th32ProcessID, &winlogonSessId)
    50. && winlogonSessId == dwSessionId)
    51. {
    52. winlogonPid = procEntry.th32ProcessID;
    53. break;
    54. }
    55. }
    56. } while (Process32Next(hSnap, &procEntry));
    57. WTSQueryUserToken(dwSessionId,&hUserToken);
    58. dwCreationFlags = NORMAL_PRIORITY_CLASS|CREATE_NEW_CONSOLE;
    59. ZeroMemory(&si, sizeof(STARTUPINFO));
    60. si.cb= sizeof(STARTUPINFO);
    61. si.lpDesktop = L"winsta0\\default";
    62. ZeroMemory(&pi, sizeof(pi));
    63. TOKEN_PRIVILEGES tp;
    64. LUID luid;
    65. hProcess = OpenProcess(MAXIMUM_ALLOWED,FALSE,winlogonPid);
    66. if(!::OpenProcessToken(hProcess,TOKEN_ADJUST_PRIVILEGES|TOKEN_QUERY
    67. |TOKEN_DUPLICATE|TOKEN_ASSIGN_PRIMARY|TOKEN_ADJUST_SESSIONID
    68. |TOKEN_READ|TOKEN_WRITE,&hPToken))
    69. {
    70. errorcode = GetLastError();
    71. printf("Process token open Error: %u\n",GetLastError());
    72. //goto ToFree;//whl2023-10-18
    73. }
    74. if (!LookupPrivilegeValue(NULL,SE_DEBUG_NAME,&luid))
    75. {
    76. printf("Lookup Privilege value Error: %u\n",GetLastError());
    77. //goto ToFree;//whl2023-10-18
    78. }
    79. tp.PrivilegeCount =1;
    80. tp.Privileges[0].Luid =luid;
    81. tp.Privileges[0].Attributes =SE_PRIVILEGE_ENABLED;
    82. DuplicateTokenEx(hPToken,MAXIMUM_ALLOWED,NULL,
    83. SecurityIdentification,TokenPrimary,&hUserTokenDup);
    84. errorcode = GetLastError();
    85. // Adjust Token privilege
    86. SetTokenInformation(hUserTokenDup,
    87. TokenSessionId,(void*)dwSessionId,sizeof(DWORD));
    88. if (!AdjustTokenPrivileges(hUserTokenDup,FALSE,&tp,sizeof(TOKEN_PRIVILEGES),
    89. (PTOKEN_PRIVILEGES)NULL,NULL))
    90. {
    91. errorcode = GetLastError();
    92. printf("Adjust Privilege value Error: %u\n",GetLastError());
    93. //goto ToFree;//whl2023-10-18
    94. }
    95. if (GetLastError() == ERROR_NOT_ALL_ASSIGNED)
    96. {
    97. printf("Token does not have the provilege\n");
    98. //goto ToFree;//whl2023-10-18
    99. }
    100. if(CreateEnvironmentBlock(&pEnv,hUserTokenDup,TRUE))
    101. {
    102. dwCreationFlags|=CREATE_UNICODE_ENVIRONMENT;
    103. }
    104. else
    105. pEnv=NULL;
    106. // Launch the process in the client's logon session.
    107. _tcscpy_s(commandLine, MAX_PATH, command.c_str());
    108. bResult = CreateProcessAsUser(
    109. hUserTokenDup, // client's access token
    110. //_T("cmd.exe"), // file to execute
    111. //NULL, // command line
    112. NULL, // file to execute
    113. (LPWSTR)(commandLine), // command line
    114. NULL, // pointer to process SECURITY_ATTRIBUTES
    115. NULL, // pointer to thread SECURITY_ATTRIBUTES
    116. FALSE, // handles are not inheritable
    117. dwCreationFlags, // creation flags
    118. pEnv, // pointer to new environment block
    119. NULL, // name of current directory
    120. &si, // pointer to STARTUPINFO structure
    121. &pi // receives information about new process
    122. );
    123. // End impersonation of client.
    124. // GetLastError Shud be 0
    125. errorcode = GetLastError();
    126. bResult = true;
    127. // Perform All the Close Handles tasks
    128. ToFree:
    129. {
    130. if(hProcess != INVALID_HANDLE_VALUE)
    131. {
    132. CloseHandle(hProcess);
    133. }
    134. if(hUserToken != INVALID_HANDLE_VALUE)
    135. {
    136. CloseHandle(hUserToken);
    137. }
    138. if(hUserTokenDup != INVALID_HANDLE_VALUE)
    139. {
    140. CloseHandle(hUserTokenDup);
    141. }
    142. if(hPToken != INVALID_HANDLE_VALUE)
    143. {
    144. CloseHandle(hPToken);
    145. }
    146. }
    147. return bResult;
    148. }
    149. #endif

    调用方式举例:

    1. //要启动的进程名
    2. #define PROCESS_NAME "EDU_CLIENT.exe"
    3. QString str_app_name = PROCESS_NAME;
    4. #ifdef Q_OS_WIN
    5. std::wstring command = str_app_name.toStdWString();
    6. if (ProcessLoader::LaunchAppIntoDifferentSession(command) == false) {
    7. qDebug() << "Failed to launch " << command.c_str();
    8. }
    9. #endif

    附:

    亲测无效方式1:

    1. bool ProcessLoader::ServerRunWndProcess(std::wstring command)
    2. {
    3. TCHAR commandLine[MAX_PATH];
    4. HANDLE hToken = NULL;
    5. if (!OpenProcessToken(GetCurrentProcess(), TOKEN_ALL_ACCESS, &hToken))
    6. {
    7. return false;
    8. }
    9. HANDLE hTokenDup = NULL;
    10. bool bRet = DuplicateTokenEx(hToken, TOKEN_ALL_ACCESS, NULL, SecurityIdentification, TokenPrimary, &hTokenDup);
    11. if (!bRet || hTokenDup == NULL)
    12. {
    13. CloseHandle(hToken);
    14. return false;
    15. }
    16. DWORD dwSessionId = WTSGetActiveConsoleSessionId();
    17. //把服务hToken的SessionId替换成当前活动的Session(即替换到可与用户交互的winsta0下)
    18. if (!SetTokenInformation(hTokenDup, TokenSessionId, &dwSessionId, sizeof(DWORD)))
    19. {
    20. DWORD nErr = GetLastError();
    21. CloseHandle(hTokenDup);
    22. CloseHandle(hToken);
    23. return false;
    24. }
    25. STARTUPINFO si;
    26. ZeroMemory(&si, sizeof(STARTUPINFO));
    27. si.cb = sizeof(STARTUPINFO);
    28. si.lpDesktop = _T("WinSta0\\Default");
    29. si.wShowWindow = SW_SHOW;
    30. si.dwFlags = STARTF_USESHOWWINDOW /*|STARTF_USESTDHANDLES*/;
    31. //创建进程环境块
    32. LPVOID pEnv = NULL;
    33. bRet = CreateEnvironmentBlock(&pEnv, hTokenDup, FALSE);
    34. if (!bRet)
    35. {
    36. CloseHandle(hTokenDup);
    37. CloseHandle(hToken);
    38. return false;
    39. }
    40. if (pEnv == NULL)
    41. {
    42. CloseHandle(hTokenDup);
    43. CloseHandle(hToken);
    44. return false;
    45. }
    46. //在活动的Session下创建进程
    47. PROCESS_INFORMATION processInfo;
    48. ZeroMemory(&processInfo, sizeof(PROCESS_INFORMATION));
    49. DWORD dwCreationFlag = NORMAL_PRIORITY_CLASS | CREATE_NEW_CONSOLE | CREATE_UNICODE_ENVIRONMENT;
    50. _tcscpy_s(commandLine, MAX_PATH, command.c_str());
    51. if (!CreateProcessAsUser(hTokenDup, NULL, (LPWSTR)(commandLine) , NULL, NULL, FALSE, dwCreationFlag, pEnv, NULL, &si, &processInfo))
    52. {
    53. DWORD nRet = GetLastError();
    54. CloseHandle(hTokenDup);
    55. CloseHandle(hToken);
    56. return false;
    57. }
    58. DestroyEnvironmentBlock(pEnv);
    59. CloseHandle(hTokenDup);
    60. CloseHandle(hToken);
    61. return true;
    62. }

    亲测无效方式2:

    1. bool ProcessLoader::loadWindowsApplication(std::wstring command)
    2. {
    3. BOOL bResult = FALSE;
    4. DWORD dwSessionId = WTSGetActiveConsoleSessionId();
    5. if (dwSessionId == 0xFFFFFFFF)
    6. {
    7. return false;
    8. }
    9. HANDLE hUserToken = NULL;
    10. if (WTSQueryUserToken(dwSessionId, &hUserToken) == FALSE)
    11. {
    12. DWORD error = GetLastError();
    13. return false;
    14. }
    15. HANDLE hTheToken = NULL;
    16. if (DuplicateTokenEx(hUserToken, TOKEN_ASSIGN_PRIMARY | TOKEN_ALL_ACCESS, 0, SecurityImpersonation, TokenPrimary, &hTheToken) == TRUE)
    17. {
    18. if (ImpersonateLoggedOnUser(hTheToken) == TRUE)
    19. {
    20. DWORD dwCreationFlags = HIGH_PRIORITY_CLASS | CREATE_NEW_CONSOLE;
    21. STARTUPINFO si = { sizeof(si) };
    22. PROCESS_INFORMATION pi;
    23. SECURITY_ATTRIBUTES Security1 = { sizeof(Security1) };
    24. SECURITY_ATTRIBUTES Security2 = { sizeof(Security2) };
    25. LPVOID pEnv = NULL;
    26. if (CreateEnvironmentBlock(&pEnv, hTheToken, TRUE) == TRUE)
    27. {
    28. dwCreationFlags |= CREATE_UNICODE_ENVIRONMENT;
    29. }
    30. TCHAR commandLine[MAX_PATH];
    31. _tcscpy_s(commandLine, MAX_PATH, command.c_str());
    32. // Launch the process in the client's logon session.
    33. bResult = CreateProcessAsUser(
    34. hTheToken,
    35. NULL, // (LPWSTR)(path),
    36. (LPWSTR)(commandLine),
    37. &Security1,
    38. &Security2,
    39. FALSE,
    40. dwCreationFlags,
    41. pEnv,
    42. NULL,
    43. &si,
    44. &pi
    45. );
    46. RevertToSelf();
    47. if (pEnv)
    48. {
    49. DestroyEnvironmentBlock(pEnv);
    50. }
    51. }
    52. CloseHandle(hTheToken);
    53. }
    54. CloseHandle(hUserToken);
    55. return bResult==TRUE;
    56. }

  • 相关阅读:
    Unity之ShaderGraph如何实现飘动的红旗
    音视频学习 - Qt6.3.1版本下实现屏幕截图功能
    【Java 基础篇】Java List 使用指南:深入解析列表操作
    Mac苹果电脑分辨率修改管理 安装SwitchResX 完美解决
    人才测评系统在企业招聘中的应用
    Sql Server查数据库job任务
    史上最全英语语法
    【CSS】css常用复杂选择器都有哪些?看这一篇就够了_07
    【Kubernetes】Kubernetes 对象是什么?
    Jumpserver堡垒机使用与二次开发详解
  • 原文地址:https://blog.csdn.net/csdndenglu/article/details/133922483