/* * Test suite for TaskScheduler interface * * Copyright (C) 2008 Google (Roy Shea) * Copyright (C) 2018 Dmitry Timoshkov * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #define COBJMACROS #include "corerror.h" #include "initguid.h" #include "mstask.h" #include "wine/test.h" static ITaskScheduler *test_task_scheduler; static const WCHAR does_not_existW[] = {'\\','\\','d','o','e','s','_','n','o','t','_','e','x','i','s','t',0}; HRESULT taskscheduler_delete(ITaskScheduler *scheduler, const WCHAR *name) { HRESULT hr; int i = 0; while (1) { hr = ITaskScheduler_Delete(scheduler, name); if (hr != HRESULT_FROM_WIN32(ERROR_SHARING_VIOLATION)) break; if (++i == 5) break; /* The task scheduler locked the file to load the task (see * ITaskScheduler_Activate()). It should be done soon. */ trace("Got a sharing violation deleting %s, retrying...\n", wine_dbgstr_w(name)); Sleep(100); } return hr; } static void test_NewWorkItem(void) { HRESULT hres; ITask *task; const WCHAR task_name[] = {'T', 'e', 's', 't', 'i', 'n', 'g', 0}; GUID GUID_BAD; /* Initialize a GUID that will not be a recognized CLSID or a IID */ CoCreateGuid(&GUID_BAD); /* Create TaskScheduler */ hres = CoCreateInstance(&CLSID_CTaskScheduler, NULL, CLSCTX_INPROC_SERVER, &IID_ITaskScheduler, (void **) &test_task_scheduler); ok(hres == S_OK, "CTaskScheduler CoCreateInstance failed: %08x\n", hres); if (hres != S_OK) { skip("Failed to create task scheduler. Skipping tests.\n"); return; } /* Test basic task creation */ hres = ITaskScheduler_NewWorkItem(test_task_scheduler, task_name, &CLSID_CTask, &IID_ITask, (IUnknown**)&task); ok(hres == S_OK, "NewNetworkItem failed: %08x\n", hres); if (hres == S_OK) ITask_Release(task); /* Task creation attempt using invalid work item class ID */ hres = ITaskScheduler_NewWorkItem(test_task_scheduler, task_name, &GUID_BAD, &IID_ITask, (IUnknown**)&task); ok(hres == CLASS_E_CLASSNOTAVAILABLE, "Expected CLASS_E_CLASSNOTAVAILABLE: %08x\n", hres); /* Task creation attempt using invalid interface ID */ hres = ITaskScheduler_NewWorkItem(test_task_scheduler, task_name, &CLSID_CTask, &GUID_BAD, (IUnknown**)&task); ok(hres == E_NOINTERFACE, "Expected E_NOINTERFACE: %08x\n", hres); /* Task creation attempt using invalid work item class and interface ID */ hres = ITaskScheduler_NewWorkItem(test_task_scheduler, task_name, &GUID_BAD, &GUID_BAD, (IUnknown**)&task); ok(hres == CLASS_E_CLASSNOTAVAILABLE, "Expected CLASS_E_CLASSNOTAVAILABLE: %08x\n", hres); ITaskScheduler_Release(test_task_scheduler); return; } static void test_Activate(void) { HRESULT hres; ITask *task = NULL; const WCHAR not_task_name[] = {'N', 'o', 'S', 'u', 'c', 'h', 'T', 'a', 's', 'k', 0}; /* Create TaskScheduler */ hres = CoCreateInstance(&CLSID_CTaskScheduler, NULL, CLSCTX_INPROC_SERVER, &IID_ITaskScheduler, (void **) &test_task_scheduler); ok(hres == S_OK, "CTaskScheduler CoCreateInstance failed: %08x\n", hres); if (hres != S_OK) { skip("Failed to create task scheduler. Skipping tests.\n"); return; } /* Attempt to activate a nonexistent task */ hres = ITaskScheduler_Activate(test_task_scheduler, not_task_name, &IID_ITask, (IUnknown**)&task); ok(hres == COR_E_FILENOTFOUND, "Expected COR_E_FILENOTFOUND: %08x\n", hres); ITaskScheduler_Release(test_task_scheduler); return; } static void test_GetTargetComputer(void) { HRESULT hres; WCHAR *oldname; /* Create TaskScheduler */ hres = CoCreateInstance(&CLSID_CTaskScheduler, NULL, CLSCTX_INPROC_SERVER, &IID_ITaskScheduler, (void **) &test_task_scheduler); ok(hres == S_OK, "CTaskScheduler CoCreateInstance failed: %08x\n", hres); if (hres != S_OK) { skip("Failed to create task scheduler.\n"); return; } if (0) { /* This crashes on w2k */ hres = ITaskScheduler_GetTargetComputer(test_task_scheduler, NULL); ok(hres == E_INVALIDARG, "got 0x%x (expected E_INVALIDARG)\n", hres); } hres = ITaskScheduler_GetTargetComputer(test_task_scheduler, &oldname); ok((hres == S_OK) && oldname && oldname[0] == '\\' && oldname[1] == '\\' && oldname[2], "got 0x%x and %s (expected S_OK and an unc name)\n", hres, wine_dbgstr_w(oldname)); CoTaskMemFree(oldname); ITaskScheduler_Release(test_task_scheduler); return; } static void test_SetTargetComputer(void) { WCHAR buffer[MAX_COMPUTERNAME_LENGTH + 3]; /* extra space for two '\' and a zero */ DWORD len = MAX_COMPUTERNAME_LENGTH + 1; /* extra space for a zero */ WCHAR *oldname = NULL; WCHAR *name = NULL; HRESULT hres; buffer[0] = '\\'; buffer[1] = '\\'; if (!GetComputerNameW(buffer + 2, &len)) return; /* Create TaskScheduler */ hres = CoCreateInstance(&CLSID_CTaskScheduler, NULL, CLSCTX_INPROC_SERVER, &IID_ITaskScheduler, (void **) &test_task_scheduler); ok(hres == S_OK, "CTaskScheduler CoCreateInstance failed: %08x\n", hres); if (hres != S_OK) { skip("Failed to create task scheduler. Skipping tests.\n"); return; } hres = ITaskScheduler_GetTargetComputer(test_task_scheduler, &oldname); ok(hres == S_OK, "got 0x%x and %s (expected S_OK)\n", hres, wine_dbgstr_w(oldname)); /* NULL is an alias for the local computer */ hres = ITaskScheduler_SetTargetComputer(test_task_scheduler, NULL); ok(hres == S_OK, "got 0x%x (expected S_OK)\n", hres); hres = ITaskScheduler_GetTargetComputer(test_task_scheduler, &name); ok((hres == S_OK && !lstrcmpiW(name, buffer)), "got 0x%x with %s (expected S_OK and %s)\n", hres, wine_dbgstr_w(name), wine_dbgstr_w(buffer)); CoTaskMemFree(name); /* The name must be valid */ hres = ITaskScheduler_SetTargetComputer(test_task_scheduler, does_not_existW); ok(hres == HRESULT_FROM_WIN32(ERROR_BAD_NETPATH), "got 0x%x (expected 0x80070035)\n", hres); /* the name of the target computer is unchanged */ hres = ITaskScheduler_GetTargetComputer(test_task_scheduler, &name); ok((hres == S_OK && !lstrcmpiW(name, buffer)), "got 0x%x with %s (expected S_OK and %s)\n", hres, wine_dbgstr_w(name), wine_dbgstr_w(buffer)); CoTaskMemFree(name); /* the two backslashes are optional */ hres = ITaskScheduler_SetTargetComputer(test_task_scheduler, oldname + 2); if (hres == E_ACCESSDENIED) { skip("SetTargetComputer failed with E_ACCESSDENIED (needs admin rights)\n"); goto done; } ok(hres == S_OK, "got 0x%x (expected S_OK)\n", hres); /* the case is ignored */ CharUpperW(buffer); hres = ITaskScheduler_SetTargetComputer(test_task_scheduler, buffer); ok(hres == S_OK, "got 0x%x (expected S_OK)\n", hres); CharLowerW(buffer); hres = ITaskScheduler_SetTargetComputer(test_task_scheduler, buffer); ok(hres == S_OK, "got 0x%x (expected S_OK)\n", hres); /* cleanup */ hres = ITaskScheduler_SetTargetComputer(test_task_scheduler, oldname); ok(hres == S_OK, "got 0x%x (expected S_OK)\n", hres); done: CoTaskMemFree(oldname); ITaskScheduler_Release(test_task_scheduler); return; } static void test_Enum(void) { static const WCHAR Task1[] = { 'w','i','n','e','t','a','s','k','1',0 }; ITaskScheduler *scheduler; ITask *task; IEnumWorkItems *tasks; WCHAR **names; ULONG fetched; HRESULT hr; hr = CoCreateInstance(&CLSID_CTaskScheduler, NULL, CLSCTX_INPROC_SERVER, &IID_ITaskScheduler, (void **)&scheduler); ok(hr == S_OK, "got %#x\n", hr); /* cleanup after previous runs */ taskscheduler_delete(scheduler, Task1); hr = ITaskScheduler_NewWorkItem(scheduler, Task1, &CLSID_CTask, &IID_ITask, (IUnknown **)&task); ok(hr == S_OK, "got %#x\n", hr); hr = ITaskScheduler_AddWorkItem(scheduler, Task1, (IScheduledWorkItem *)task); ok(hr == S_OK, "got %#x\n", hr); ITask_Release(task); hr = ITaskScheduler_Enum(scheduler, NULL); ok(hr == E_INVALIDARG, "got %#x\n", hr); hr = ITaskScheduler_Enum(scheduler, &tasks); ok(hr == S_OK, "got %#x\n", hr); names = (void *)0xdeadbeef; fetched = 0xdeadbeef; hr = IEnumWorkItems_Next(tasks, 0, &names, &fetched); ok(hr == E_INVALIDARG, "got %#x\n", hr); ok(names == (void *)0xdeadbeef, "got %p\n", names); ok(fetched == 0xdeadbeef, "got %#x\n", fetched); hr = IEnumWorkItems_Next(tasks, 1, NULL, NULL); ok(hr == E_INVALIDARG, "got %#x\n", hr); names = NULL; hr = IEnumWorkItems_Next(tasks, 1, &names, NULL); ok(hr == S_OK, "got %#x\n", hr); ok(names != NULL, "got NULL\n"); ok(names[0] != NULL, "got NULL\n"); CoTaskMemFree(names[0]); CoTaskMemFree(names); names = (void *)0xdeadbeef; hr = IEnumWorkItems_Next(tasks, 2, &names, NULL); ok(hr == E_INVALIDARG, "got %#x\n", hr); ok(names == (void *)0xdeadbeef, "got %p\n", names); hr = IEnumWorkItems_Reset(tasks); ok(hr == S_OK, "got %#x\n", hr); names = NULL; fetched = 0xdeadbeef; hr = IEnumWorkItems_Next(tasks, 1, &names, &fetched); ok(hr == S_OK, "got %#x\n", hr); ok(names != NULL, "got NULL\n"); ok(names[0] != NULL, "got NULL\n"); ok(fetched == 1, "got %u\n", fetched); CoTaskMemFree(names[0]); CoTaskMemFree(names); while (IEnumWorkItems_Skip(tasks, 1) == S_OK) /* do nothing*/; hr = IEnumWorkItems_Skip(tasks, 1); ok(hr == S_FALSE, "got %#x\n", hr); names = (void *)0xdeadbeef; fetched = 0xdeadbeef; hr = IEnumWorkItems_Next(tasks, 1, &names, &fetched); ok(hr == S_FALSE, "got %#x\n", hr); ok(names == NULL, "got %p\n", names); ok(fetched == 0, "got %u\n", fetched); IEnumWorkItems_Release(tasks); hr = taskscheduler_delete(scheduler, Task1); ok(hr == S_OK, "got %#x\n", hr); ITaskScheduler_Release(scheduler); } static BOOL file_exists(const WCHAR *name) { return GetFileAttributesW(name) != INVALID_FILE_ATTRIBUTES; } static void test_save_task_curfile(ITask *task) { HRESULT hr; IPersistFile *pfile; WCHAR *curfile; hr = ITask_QueryInterface(task, &IID_IPersistFile, (void **)&pfile); ok(hr == S_OK, "QueryInterface error %#x\n", hr); curfile = NULL; hr = IPersistFile_GetCurFile(pfile, &curfile); ok(hr == S_OK, "GetCurFile error %#x\n", hr); ok(curfile && curfile[0], "curfile should not be NULL\n"); ok(file_exists(curfile), "curfile should exist\n"); hr = IPersistFile_Save(pfile, curfile, FALSE); ok(hr == HRESULT_FROM_WIN32(ERROR_FILE_EXISTS), "wrong error %#x\n", hr); hr = IPersistFile_Save(pfile, curfile, TRUE); ok(hr == HRESULT_FROM_WIN32(ERROR_FILE_EXISTS), "wrong error %#x\n", hr); hr = IPersistFile_Save(pfile, NULL, FALSE); ok(hr == S_OK, "Save error %#x\n", hr); hr = IPersistFile_Save(pfile, NULL, TRUE); ok(hr == S_OK, "Save error %#x\n", hr); CoTaskMemFree(curfile); curfile = NULL; hr = IPersistFile_GetCurFile(pfile, &curfile); ok(hr == S_OK, "GetCurFile error %#x\n", hr); ok(curfile && curfile[0] , "curfile should not be NULL\n"); CoTaskMemFree(curfile); IPersistFile_Release(pfile); } static WCHAR *get_task_curfile(ITask *task, BOOL should_exist, BOOL is_dirty, int line) { HRESULT hr; IPersistFile *pfile; WCHAR *curfile; CLSID clsid; hr = ITask_QueryInterface(task, &IID_IPersistFile, (void **)&pfile); ok_(__FILE__, line)(hr == S_OK, "QueryInterface error %#x\n", hr); hr = IPersistFile_IsDirty(pfile); ok_(__FILE__, line)(hr == is_dirty ? S_OK : S_FALSE, "got %#x\n", hr); curfile = NULL; hr = IPersistFile_GetCurFile(pfile, &curfile); ok_(__FILE__, line)(hr == S_OK, "GetCurFile error %#x\n", hr); ok_(__FILE__, line)(curfile && curfile[0] , "curfile should not be NULL\n"); hr = IPersistFile_Load(pfile, curfile, STGM_READ); if (should_exist) ok_(__FILE__, line)(hr == S_OK, "Load error %#x\n", hr); else ok_(__FILE__, line)(hr == HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND), "wrong error %#x\n", hr); if (0) /* crashes under Windows */ hr = IPersistFile_GetClassID(pfile, NULL); hr = IPersistFile_GetClassID(pfile, &clsid); ok_(__FILE__, line)(hr == S_OK, "GetClassID error %#x\n", hr); ok_(__FILE__, line)(IsEqualCLSID(&clsid, &CLSID_CTask), "got %s\n", wine_dbgstr_guid(&clsid)); IPersistFile_Release(pfile); return curfile; } static void test_task_storage(void) { static const WCHAR Task1[] = { 'w','i','n','e','t','a','s','k','1',0 }; static const WCHAR Task2[] = { 'w','i','n','e','t','a','s','k','2',0 }; static const WCHAR Task3[] = { 'w','i','n','e','t','a','s','k','3',0 }; static const WCHAR Task1_ext[] = { 'w','i','n','e','t','a','s','k','.','e','x','t',0 }; static const WCHAR Task1_job[] = { '\\','T','a','s','k','s','\\','w','i','n','e','t','a','s','k','1','.','j','o','b',0 }; static const WCHAR Task2_job[] = { '\\','T','a','s','k','s','\\','w','i','n','e','t','a','s','k','2','.','j','o','b',0 }; static const WCHAR Task3_job[] = { '\\','T','a','s','k','s','\\','w','i','n','e','t','a','s','k','3','.','j','o','b',0 }; WCHAR task1_full_name[MAX_PATH], task2_full_name[MAX_PATH], task3_full_name[MAX_PATH]; HRESULT hr; ITaskScheduler *scheduler; ITask *task, *task2; WCHAR *curfile, *curfile2; GetWindowsDirectoryW(task1_full_name, MAX_PATH); lstrcatW(task1_full_name, Task1_job); GetWindowsDirectoryW(task2_full_name, MAX_PATH); lstrcatW(task2_full_name, Task2_job); GetWindowsDirectoryW(task3_full_name, MAX_PATH); lstrcatW(task3_full_name, Task3_job); /* cleanup after previous runs */ DeleteFileW(task1_full_name); DeleteFileW(task2_full_name); DeleteFileW(task3_full_name); hr = CoCreateInstance(&CLSID_CTaskScheduler, NULL, CLSCTX_INPROC_SERVER, &IID_ITaskScheduler, (void **)&scheduler); if (hr != S_OK) { win_skip("CoCreateInstance(CLSID_CTaskScheduler) error %#x\n", hr); return; } hr = ITaskScheduler_Delete(scheduler, Task1_ext); ok(hr == E_INVALIDARG, "wrong error %#x\n", hr); hr = ITaskScheduler_Delete(scheduler, Task1); ok(hr == HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND), "wrong error %#x\n", hr); hr = ITaskScheduler_NewWorkItem(scheduler, Task1_ext, &CLSID_CTask, &IID_ITask, (IUnknown **)&task); ok(hr == E_INVALIDARG, "wrong error %#x\n", hr); hr = ITaskScheduler_NewWorkItem(scheduler, Task1, &CLSID_CTask, &IID_ITask, (IUnknown **)&task); ok(hr == S_OK, "NewWorkItem error %#x\n", hr); curfile = get_task_curfile(task, FALSE, FALSE, __LINE__); ok(!file_exists(curfile), "curfile should not exist\n"); ok(!lstrcmpW(curfile, task1_full_name), "name is wrong %s\n", wine_dbgstr_w(curfile)); hr = ITask_SetComment(task, Task1); ok(hr == S_OK, "got %#x\n", hr); curfile2 = get_task_curfile(task, FALSE, TRUE, __LINE__); ok(!file_exists(curfile2), "curfile should not exist\n"); ok(!lstrcmpW(curfile2, task1_full_name), "name is wrong %s\n", wine_dbgstr_w(curfile2)); CoTaskMemFree(curfile2); hr = ITaskScheduler_NewWorkItem(scheduler, Task1, &CLSID_CTask, &IID_ITask, (IUnknown **)&task2); ok(hr == S_OK, "NewWorkItem error %#x\n", hr); ok(task2 != task, "tasks should not be equal\n"); curfile2 = get_task_curfile(task2, FALSE, FALSE, __LINE__); ok(!file_exists(curfile2), "curfile2 should not exist\n"); ok(!lstrcmpW(curfile2, task1_full_name), "name is wrong %s\n", wine_dbgstr_w(curfile2)); CoTaskMemFree(curfile); CoTaskMemFree(curfile2); ITask_Release(task2); task2 = (ITask *)0xdeadbeef; hr = ITaskScheduler_Activate(scheduler, Task1, &IID_ITask, (IUnknown **)&task2); ok(hr == HRESULT_FROM_WIN32(ERROR_FILE_NOT_FOUND), "wrong error %#x\n", hr); ok(task2 == (ITask *)0xdeadbeef, "task should not be set to NULL\n"); hr = ITaskScheduler_AddWorkItem(scheduler, Task2, (IScheduledWorkItem *)task); ok(hr == S_OK, "AddWorkItem error %#x\n", hr); curfile = get_task_curfile(task, TRUE, FALSE, __LINE__); ok(file_exists(curfile), "curfile should exist\n"); ok(!lstrcmpW(curfile, task2_full_name), "name is wrong %s\n", wine_dbgstr_w(curfile)); CoTaskMemFree(curfile); hr = ITaskScheduler_AddWorkItem(scheduler, Task3, (IScheduledWorkItem *)task); ok(hr == S_OK, "AddWorkItem error %#x\n", hr); curfile = get_task_curfile(task, TRUE, FALSE, __LINE__); ok(file_exists(curfile), "curfile should exist\n"); ok(!lstrcmpW(curfile, task3_full_name), "name is wrong %s\n", wine_dbgstr_w(curfile)); CoTaskMemFree(curfile); hr = ITaskScheduler_AddWorkItem(scheduler, Task1, (IScheduledWorkItem *)task); ok(hr == S_OK, "AddWorkItem error %#x\n", hr); curfile = get_task_curfile(task, TRUE, FALSE, __LINE__); ok(file_exists(curfile), "curfile should exist\n"); ok(!lstrcmpW(curfile, task1_full_name), "name is wrong %s\n", wine_dbgstr_w(curfile)); CoTaskMemFree(curfile); hr = ITaskScheduler_AddWorkItem(scheduler, Task1, (IScheduledWorkItem *)task); ok(hr == HRESULT_FROM_WIN32(ERROR_FILE_EXISTS), "wrong error %#x\n", hr); curfile = get_task_curfile(task, TRUE, FALSE, __LINE__); ok(file_exists(curfile), "curfile should exist\n"); CoTaskMemFree(curfile); ITask_Release(task); task = NULL; hr = ITaskScheduler_Activate(scheduler, Task1, &IID_ITask, (IUnknown **)&task); ok(hr == S_OK, "Activate error %#x\n", hr); ok(task != NULL, "task should not be set to NULL\n"); curfile = get_task_curfile(task, TRUE, FALSE, __LINE__); ok(file_exists(curfile), "curfile2 should exist\n"); ok(!lstrcmpW(curfile, task1_full_name), "name is wrong %s\n", wine_dbgstr_w(curfile)); CoTaskMemFree(curfile); test_save_task_curfile(task); hr = taskscheduler_delete(scheduler, Task1); ok(hr == S_OK, "got %#x\n", hr); hr = taskscheduler_delete(scheduler, Task2); ok(hr == S_OK, "got %#x\n", hr); hr = taskscheduler_delete(scheduler, Task3); ok(hr == S_OK, "got %#x\n", hr); ITask_Release(task); ITaskScheduler_Release(scheduler); } START_TEST(task_scheduler) { CoInitialize(NULL); test_task_storage(); test_NewWorkItem(); test_Activate(); test_GetTargetComputer(); test_SetTargetComputer(); test_Enum(); CoUninitialize(); }