c++登录任务计划错误:没有完成帐户名和安全id之间的映射



我正在尝试在windows 7上使用c++编写windows登录触发器任务。

我正在遵循这个微软教程。

但是我在保存任务到根文件夹时遇到了问题。:

//  ------------------------------------------------------
    //  Save the task in the root folder.
    IRegisteredTask *pRegisteredTask = NULL;
    hr = pRootFolder->RegisterTaskDefinition(
            _bstr_t( wszTaskName ),
            pTask,
            TASK_CREATE_OR_UPDATE, 
            _variant_t(L"Builtin\Administrators"), 
            _variant_t(), 
            TASK_LOGON_GROUP,
            _variant_t(L""),
            &pRegisteredTask);

hr出错的地方:没有完成账号名和安全id的映射

我还尝试用_variant_t(L"S-1-5-32-544")替换_variant_t(L"Builtin\Administrators")以消除语言硬编码问题,仍然没有运气。

我怎样才能使它工作?

在Windows启动时创建TaskScheduler任务的最终解决方案

(具有管理员权限,适用于Windows 7、8等)请注意,这不会在Windows启动时显示UAC弹出窗口"你确定以Admin权限运行此软件吗?",这就是为什么TaskScheduler方法在这种情况下比老式的HKEY_LOCAL_MACHINE方法更有趣…

CurrentVersion 运行解决方案)

在本教程中有一些东西需要更新要使它工作:

  • _variant_t(L"S-1-5-32-544")代替_variant_t(L"Builtin\Administrators")
  • _CRT_SECURE_NO_WARNINGS
  • 在vc++中,项目属性>配置属性>链接器在清单文件>UAC执行级别>requireAdministrator
  • 删除已经过时的日期边界!
  • hr = pLogonTrigger->put_UserId(_bstr_t(L"DOMAIN\UserName"));替换为硬编码的域用户名,或通过一些域用户名检测代码(我无法使其工作),或只是注释这行,它为我工作!
  • TASK_RUNLEVEL_HIGHEST添加一些代码
  • 添加一些代码来启用任务,即使在笔记本电脑上运行电池(默认为"不运行任务如果电池"),和一些代码来防止。exe在一段时间后被杀死(默认情况下,任务将在开始运行72小时后停止),等等。

然后你会得到著名的:

成功!任务注册成功。


唷!经过每天几个小时和一些编辑,现在这是一个完整的main.cpp:

#define SECURITY_WIN32
#include <windows.h>
#include <iostream>
#include <stdio.h>
#include <comdef.h>
#include <Security.h>
#include <taskschd.h>
#pragma comment(lib, "taskschd.lib")
#pragma comment(lib, "comsupp.lib")
using namespace std;
#define TASKNAME L"Logon Trigger Test Task"
int __cdecl wmain()
{
    //  Get the windows directory and set the path to notepad.exe.
    wstring wstrExecutablePath = _wgetenv(L"WINDIR");
    wstrExecutablePath += L"\SYSTEM32\NOTEPAD.EXE";
    //  Initialize COM  
    HRESULT hr = CoInitializeEx(NULL, COINIT_MULTITHREADED);
    if (FAILED(hr)) return 1;
    //  Set general COM security levels.
    hr = CoInitializeSecurity(NULL, -1, NULL, NULL, RPC_C_AUTHN_LEVEL_PKT_PRIVACY, RPC_C_IMP_LEVEL_IMPERSONATE, NULL, 0, NULL);
    if (FAILED(hr)) goto cleanup0;
    //  Create an instance of the Task Service.     
    ITaskService *pService = NULL;
    hr = CoCreateInstance(CLSID_TaskScheduler, NULL, CLSCTX_INPROC_SERVER, IID_ITaskService, (void**)&pService);
    if (FAILED(hr)) goto cleanup0;
    //  Connect to the task service.    
    hr = pService->Connect(_variant_t(), _variant_t(), _variant_t(), _variant_t());
    if (FAILED(hr)) goto cleanup1;
    //  Get the pointer to the root task folder.  This folder will hold the new task that is registered.
    ITaskFolder *pRootFolder = NULL;
    hr = pService->GetFolder(_bstr_t(L"\"), &pRootFolder);
    if (FAILED(hr)) goto cleanup1;
    //  If the same task exists, remove it. 
    pRootFolder->DeleteTask(_bstr_t(TASKNAME), 0);
    //  Create the task builder object to create the task.  
    ITaskDefinition *pTask = NULL;
    hr = pService->NewTask(0, &pTask);
    // COM clean up.  Pointer is no longer used.    
    pService->Release();
    if (FAILED(hr)) { pRootFolder->Release(); CoUninitialize(); return 1; }
    //  Get the registration info for setting the identification.   
    IRegistrationInfo *pRegInfo = NULL;
    hr = pTask->get_RegistrationInfo(&pRegInfo);
    if (FAILED(hr)) goto cleanup2;
    hr = pRegInfo->put_Author(L"Author Name");
    pRegInfo->Release();
    if (FAILED(hr)) goto cleanup2;
    //  Create the settings for the task
    ITaskSettings *pSettings = NULL;
    hr = pTask->get_Settings(&pSettings);
    if (FAILED(hr)) goto cleanup2;
    
    //  Set setting values for the task. 
    pSettings->put_DisallowStartIfOnBatteries(VARIANT_FALSE);
    pSettings->put_StopIfGoingOnBatteries(VARIANT_FALSE);
    pSettings->put_ExecutionTimeLimit(_bstr_t(L"PT0S"));
    pSettings->Release();
    if (FAILED(hr)) goto cleanup2;
    //  Get the trigger collection to insert the logon trigger.
    ITriggerCollection *pTriggerCollection = NULL;
    hr = pTask->get_Triggers(&pTriggerCollection);
    if (FAILED(hr)) goto cleanup2;
    //  Add the logon trigger to the task.
    ITrigger *pTrigger = NULL;
    hr = pTriggerCollection->Create(TASK_TRIGGER_LOGON, &pTrigger);
    pTriggerCollection->Release();
    if (FAILED(hr)) goto cleanup2;
    ILogonTrigger *pLogonTrigger = NULL;
    hr = pTrigger->QueryInterface(IID_ILogonTrigger, (void**)&pLogonTrigger);
    pTrigger->Release();
    if (FAILED(hr)) goto cleanup2;
    hr = pLogonTrigger->put_Id(_bstr_t(L"Trigger1"));
    if (FAILED(hr)) goto cleanup2;
    //  Define the user.  The task will execute when the user logs on. The specified user must be a user on this computer.  
    //hr = pLogonTrigger->put_UserId(_bstr_t(L"DOMAIN\UserName"));
    pLogonTrigger->Release();
    if (FAILED(hr)) goto cleanup2;
    IPrincipal *pPrincipal;
    hr = pTask->get_Principal(&pPrincipal);
    if (FAILED(hr)) goto cleanup2;
    hr = pPrincipal->put_RunLevel(TASK_RUNLEVEL_HIGHEST);
    if (FAILED(hr)) goto cleanup2;

    //  Add an Action to the task. This task will execute .exe
    IActionCollection *pActionCollection = NULL;
    //  Get the task action collection pointer.
    hr = pTask->get_Actions(&pActionCollection);
    if (FAILED(hr)) goto cleanup2;
    //  Create the action, specifying that it is an executable action.
    IAction *pAction = NULL;
    hr = pActionCollection->Create(TASK_ACTION_EXEC, &pAction);
    pActionCollection->Release();
    if (FAILED(hr)) goto cleanup2;
    //  QI for the executable task pointer.
    IExecAction *pExecAction = NULL;
    hr = pAction->QueryInterface(IID_IExecAction, (void**)&pExecAction);
    pAction->Release();
    if (FAILED(hr)) goto cleanup2;
    //  Set the path of the executable.
    hr = pExecAction->put_Path(_bstr_t(wstrExecutablePath.c_str()));
    pExecAction->Release();
    if (FAILED(hr)) goto cleanup2;
    //  Save the task in the root folder.
    IRegisteredTask *pRegisteredTask = NULL;
    hr = pRootFolder->RegisterTaskDefinition(_bstr_t(TASKNAME), pTask, TASK_CREATE_OR_UPDATE, _variant_t(L"S-1-5-32-544"), _variant_t(), TASK_LOGON_GROUP, _variant_t(L""), &pRegisteredTask);      //_variant_t(L"Builtin\Administrators"),
    if (FAILED(hr)) goto cleanup2;
    printf("Success! Task successfully registered.");
    getchar();
    pRootFolder->Release();
    pTask->Release();
    pRegisteredTask->Release();
    CoUninitialize();
    return 0;
cleanup0:
    CoUninitialize();
    return 1;
cleanup1:
    pService->Release();
    CoUninitialize();
    return 1;
cleanup2:
    pRootFolder->Release();
    pTask->Release();
    CoUninitialize();
    return 1;
}

我怀疑你的演示代码是xp时代的,还没有更新到符合Vista/Win7规则。

我更新了示例,以便在设置登录触发器后设置LUA设置,它似乎可以工作:

    hr = pLogonTrigger->put_UserId(_bstr_t(L"DOMAINusername"));
    if (FAILED(hr))
    {
        printf("nCannot add user ID to logon trigger: %x", hr);
        CoUninitialize();
        return 1;
    }

    //*** NEW**** Set the LUA settings
    CComPtr<IPrincipal>         pPrincipal;
    hr = pTask->get_Principal(&pPrincipal);
    if (SUCCEEDED(hr))
    {
        hr = pPrincipal->put_RunLevel(TASK_RUNLEVEL_LUA);
    }
    if (SUCCEEDED(hr))
    {
        hr = pPrincipal->put_GroupId(_bstr_t(L"Builtin\Administrators"));
    }
    if (FAILED(hr))
    {
        printf("nCannot set runlevel/groupid: %x", hr);
        CoUninitialize();
        return 1;
    }

如果您需要它在XP上运行,那么get_Principal调用很可能会失败,所以让失败通过。

最新更新