windows - 从以本地系统帐户启动的服务,以登录用户身份运行程序

标签 windows networking permissions local-system-account

tl/dr: 我正在寻找一种方法,以当前从作为 本地系统帐户 启动的服务登录的用户身份运行程序.


长版:

跟进来源:Get current username from a program started as Local System Account

我的程序是从作为本地系统帐户运行的服务启动的。

此服务在 Windows 启动时启动,基本上为某些硬件按钮提供功能,例如显示屏幕键盘。我的程序也分配给其中一个按钮,但它仅在实际用户登录时可用。

它所做的是使用硬件相机拍摄照片并将其存储在文件系统的某个位置,这工作正常,但我不能将照片存储在网络路径上,这是可以理解的,因为这样做需要身份验证。

如果程序以登录用户的身份直接启动,则此身份验证可用并且在网络路径上存储文件可以正常工作。

有什么办法可以解决这个问题吗?

我的首选解决方案是以当前登录的用户身份启动程序,而不在某处存储密码和用户名。

在不同站点上找到的工作解决方案: http://chabster.blogspot.com/2008/01/run-as-interactive-user-from-service.html

stdafx.h:

#include <WtsApi32.h>
#pragma comment(lib, "WtsApi32.lib")

#include <Userenv.h>
#pragma comment(lib, "Userenv.lib")

RunAsInteractiveUser 函数:

BOOL bRet;
HRESULT hr;

HANDLE processToken = NULL;
TOKEN_PRIVILEGES oldTokenPrivileges = { 0 };

HANDLE impersonationToken = NULL;
HANDLE userToken = NULL;

LPVOID pEnvironment = NULL;
PROCESS_INFORMATION processInformation = { 0 };

__try {
    bRet = OpenProcessToken(GetCurrentProcess(), 
       TOKEN_ADJUST_PRIVILEGES | TOKEN_QUERY, &processToken);
    if (!bRet) {
        hr = GetLastError();
        return hr;
    }

    // This step might not be necessary because 
    // SeTcbPrivilege is enabled by default for Local System
    LUID luid;
    bRet = LookupPrivilegeValue(NULL, _T("SeTcbPrivilege"), &luid);
    if (!bRet) {
        hr = GetLastError();
        return hr;
    }

    TOKEN_PRIVILEGES adjTokenPrivileges = { 0 };
    adjTokenPrivileges.PrivilegeCount = 1;
    adjTokenPrivileges.Privileges[0].Luid = luid;
    adjTokenPrivileges.Privileges[0].Attributes = SE_PRIVILEGE_ENABLED;

    DWORD dwOldTPLen;
    bRet = AdjustTokenPrivileges(processToken, FALSE, 
       &adjTokenPrivileges, sizeof(TOKEN_PRIVILEGES), 
       &oldTokenPrivileges, &dwOldTPLen);

    if (bRet) {
        hr = GetLastError();
        if (hr == ERROR_SUCCESS);
        else if (hr == ERROR_NOT_ALL_ASSIGNED) {
            // Enabled by default
        }
    }
    else {
        hr = GetLastError();
        return hr;
    }

    DWORD conSessId = WTSGetActiveConsoleSessionId();
    if (conSessId == 0xFFFFFFFF) {
        // There is no session attached to the console
        return ERROR_SUCCESS;
    }

    bRet = WTSQueryUserToken(conSessId, &impersonationToken);
    if (!bRet) {
        hr = GetLastError();
        return hr;
    }

    bRet = DuplicateTokenEx(impersonationToken, MAXIMUM_ALLOWED, NULL,
       SecurityIdentification, TokenPrimary, &userToken);
    if (!bRet) {
        hr = GetLastError();
        return hr;
    }

    STARTUPINFO si = { 0 };
    si.cb = sizeof(STARTUPINFO);
    si.lpDesktop = _T("winsta0\\default");

    bRet = CreateEnvironmentBlock(&pEnvironment, userToken, TRUE);
    if (!bRet) {
        hr = GetLastError();
        return hr;
    }

    bRet = CreateProcessAsUser(userToken, _T("C:\\Windows\\notepad.exe"), 
       NULL, NULL, NULL, FALSE, CREATE_UNICODE_ENVIRONMENT, 
       pEnvironment, NULL, &si, &processInformation);

    if (!bRet) {
        hr = GetLastError();
        return hr;
    }
}
__finally {
    if (processInformation.hThread) {
        CloseHandle(processInformation.hThread);
    }
    if (processInformation.hProcess) {
        CloseHandle(processInformation.hProcess);
    }
    if (pEnvironment) {
        bRet = DestroyEnvironmentBlock(pEnvironment);
    }
    if (userToken) {
        CloseHandle(userToken);
    }
    if (impersonationToken) {
        CloseHandle(impersonationToken);
    }
    if (processToken) {
        bRet = AdjustTokenPrivileges(processToken, 
           FALSE, &oldTokenPrivileges, sizeof(TOKEN_PRIVILEGES), NULL, NULL);
        CloseHandle(processToken);
    }
}

最佳答案

您必须使用 CreateProcessAsUser。可以找到指南 herehere .希望这会有所帮助。

关于windows - 从以本地系统帐户启动的服务,以登录用户身份运行程序,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/5592797/

相关文章:

linux - 使用linux系统调用在基本TCP服务器中获取http header

ios - 使用 NSURLSessionDownloadTask 只下载文件的一部分

android - 网络更改广播接收器未执行

mysql - 需要我的 MySQL 用户的所有权限

ruby-on-rails - rmagick gem 安装在 Windows 上但捆绑失败

windows - 通过 powershell 脚本自定义协议(protocol)处理程序

java - 我的文件无法在 cmd 中打开

c - 如何捕获tcp/ip数据包

android - Oreo (API 26) - drawOverlay + 绘制状态栏

windows - 是否可以获得 XInput 设备的名称、产品 ID、供应商 ID 或其他类型的唯一标识符?