У меня есть служба, и мне нужно запустить приложение графического интерфейса пользователя с привилегиями текущего пользователя из этой службы. Это мой код, и он всегда возвращает GetLastError с функцией 1305 CreateProcessAsUser. Как я могу это исправить или, возможно, мой код неверен, и вы можете посоветовать мне что-нибудь полезное. Спасибо.
void ConnectionManager::LaunchDialer ()
{
HANDLE currentToken;
HANDLE primaryToken;
int dwSessionId = 0;
PHANDLE hUserToken = 0;
PHANDLE hTokenDup = 0;
PWTS_SESSION_INFO pSessionInfo = 0;
DWORD dwCount = 0;
// Get the list of all terminal sessions
WTSEnumerateSessions (WTS_CURRENT_SERVER_HANDLE, 0, 1,
&pSessionInfo, &dwCount);
int dataSize = sizeof (WTS_SESSION_INFO);
// look over obtained list in search of the active session
for (DWORD i = 0; i < dwCount; ++i)
{
WTS_SESSION_INFO si = pSessionInfo [i];
if (WTSActive == si.State)
{
// If the current session is active – store its ID
dwSessionId = si.SessionId;
break;
}
}
WTSFreeMemory (pSessionInfo);
// Get token of the logged in user by the active session ID
BOOL bRet = WTSQueryUserToken (dwSessionId, ¤tToken);
if (!bRet)
{
ModemDetectorService::instance ()->logMessage (QString ("WTSQueryUserToken: %1")
.arg (GetLastError ()));
return;
}
bRet = DuplicateTokenEx (currentToken,
TOKEN_ASSIGN_PRIMARY | TOKEN_ALL_ACCESS,
0,
SecurityImpersonation,
TokenPrimary,
&primaryToken);
if (!bRet)
{
ModemDetectorService::instance ()->logMessage (QString ("DuplicateTokenEx: %1")
.arg (GetLastError ()));
return;
}
if (!primaryToken)
{
ModemDetectorService::instance ()->logMessage ("Invalid user token");
return;
}
STARTUPINFO StartupInfo;
PROCESS_INFORMATION processInfo;
ZeroMemory(&StartupInfo, sizeof(STARTUPINFO));
StartupInfo.cb= sizeof(STARTUPINFO);
StartupInfo.lpDesktop = TEXT("winsta0\\default");
SECURITY_ATTRIBUTES Security1;
SECURITY_ATTRIBUTES Security2;
QSettings settings ("HKEY_LOCAL_MACHINE\\Software\\Olive\\OliveDialer",
QSettings::NativeFormat);
const QString path = QDir::toNativeSeparators (settings.value ("InstallationDirectory").toString ());
QByteArray command = ("\"" + path + "\\" +
ApplicationInfo::Olive::ShortApplicationName + ".exe" + "\"").toUtf8 ();
void* lpEnvironment = NULL;
// Get all necessary environment variables of logged in user
// to pass them to the process
BOOL resultEnv = CreateEnvironmentBlock (&lpEnvironment,
primaryToken,
FALSE);
if (!resultEnv)
{
long nError = GetLastError ();
ModemDetectorService::instance ()->logMessage (QString ("CreateEnvironmentBlock failed with: %1")
.arg (nError));
}
// Start the process on behalf of the current user
BOOL result = CreateProcessAsUser (primaryToken, 0,
(LPSTR)(command.data ()),
&Security1,
&Security2,
FALSE,
NORMAL_PRIORITY_CLASS | CREATE_UNICODE_ENVIRONMENT,
lpEnvironment,
NULL,
&StartupInfo,
&processInfo);
if (!result)
{
DWORD errorCode = GetLastError ();
ModemDetectorService::instance ()->logMessage (QString ("Application start failed: %1 %2")
.arg (errorCode)
.arg (command.data ()));
}
else
ModemDetectorService::instance ()->logMessage ("Application started successfully");
DestroyEnvironmentBlock (lpEnvironment);
CloseHandle (primaryToken);
}
Ошибка 1305 ERROR_UNKNOWN_REVISION
(«Уровень редакции неизвестен»), который обычно относится к объектам безопасности. Действительно, вы проходите два SECURITY_ATTRIBUTES
структуры (Security1
а также Security2
), которые никогда не были инициализированы.
Вам нужно либо пройти NULL
вместо &Security1
а также &Security2
или правильно инициализировать структуры.
Попробуйте установить lpCurrentDirectory что-то вроде C: \ Windows
Или загрузив профиль пользователя LoadUserProfile