2012-11-04 22:02:06 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2012 Detlef Riekenberg
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
|
2018-01-05 22:12:39 +00:00
|
|
|
#define COBJMACROS
|
|
|
|
|
|
|
|
#include "initguid.h"
|
|
|
|
#include "taskschd.h"
|
|
|
|
|
2012-11-04 22:02:06 +00:00
|
|
|
#include "wine/debug.h"
|
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(schtasks);
|
|
|
|
|
2018-01-19 18:16:20 +00:00
|
|
|
static ITaskFolder *get_tasks_root_folder(void)
|
2018-01-05 22:12:39 +00:00
|
|
|
{
|
|
|
|
ITaskService *service;
|
2018-01-19 18:16:20 +00:00
|
|
|
ITaskFolder *root;
|
2018-01-05 22:12:39 +00:00
|
|
|
VARIANT empty;
|
|
|
|
HRESULT hres;
|
|
|
|
|
|
|
|
hres = CoCreateInstance(&CLSID_TaskScheduler, NULL, CLSCTX_INPROC_SERVER,
|
|
|
|
&IID_ITaskService, (void**)&service);
|
|
|
|
if (FAILED(hres))
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
V_VT(&empty) = VT_EMPTY;
|
|
|
|
hres = ITaskService_Connect(service, empty, empty, empty, empty);
|
|
|
|
if (FAILED(hres)) {
|
2022-02-09 08:24:55 +00:00
|
|
|
FIXME("Connect failed: %08lx\n", hres);
|
2018-01-05 22:12:39 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-01-19 18:16:20 +00:00
|
|
|
hres = ITaskService_GetFolder(service, NULL, &root);
|
2018-01-05 22:12:39 +00:00
|
|
|
ITaskService_Release(service);
|
|
|
|
if (FAILED(hres)) {
|
2022-02-09 08:24:55 +00:00
|
|
|
FIXME("GetFolder failed: %08lx\n", hres);
|
2018-01-05 22:12:39 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2018-01-19 18:16:20 +00:00
|
|
|
return root;
|
|
|
|
}
|
|
|
|
|
|
|
|
static IRegisteredTask *get_registered_task(const WCHAR *name)
|
|
|
|
{
|
|
|
|
IRegisteredTask *registered_task;
|
|
|
|
ITaskFolder *root;
|
|
|
|
BSTR str;
|
|
|
|
HRESULT hres;
|
|
|
|
|
|
|
|
root = get_tasks_root_folder();
|
|
|
|
if (!root)
|
|
|
|
return NULL;
|
|
|
|
|
2018-01-05 22:12:39 +00:00
|
|
|
str = SysAllocString(name);
|
2018-01-19 18:16:20 +00:00
|
|
|
hres = ITaskFolder_GetTask(root, str, ®istered_task);
|
2018-01-05 22:12:39 +00:00
|
|
|
SysFreeString(str);
|
2018-01-19 18:16:20 +00:00
|
|
|
ITaskFolder_Release(root);
|
2018-01-05 22:12:39 +00:00
|
|
|
if (FAILED(hres)) {
|
2022-02-09 08:24:55 +00:00
|
|
|
FIXME("GetTask failed: %08lx\n", hres);
|
2018-01-05 22:12:39 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return registered_task;
|
|
|
|
}
|
|
|
|
|
2018-01-19 18:16:29 +00:00
|
|
|
static BSTR read_file_to_bstr(const WCHAR *file_name)
|
|
|
|
{
|
|
|
|
LARGE_INTEGER file_size;
|
|
|
|
DWORD read_size, size;
|
|
|
|
unsigned char *data;
|
|
|
|
HANDLE file;
|
|
|
|
BOOL r = FALSE;
|
|
|
|
BSTR ret;
|
|
|
|
|
|
|
|
file = CreateFileW(file_name, GENERIC_READ, FILE_SHARE_READ, NULL,
|
|
|
|
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
|
|
|
|
if (file == INVALID_HANDLE_VALUE) {
|
|
|
|
FIXME("Could not open file\n");
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!GetFileSizeEx(file, &file_size) || !file_size.QuadPart) {
|
|
|
|
FIXME("Empty file\n");
|
|
|
|
CloseHandle(file);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
data = HeapAlloc(GetProcessHeap(), 0, file_size.QuadPart);
|
|
|
|
if (data)
|
|
|
|
r = ReadFile(file, data, file_size.QuadPart, &read_size, NULL);
|
|
|
|
CloseHandle(file);
|
|
|
|
if (!r) {
|
2021-12-21 14:40:48 +00:00
|
|
|
FIXME("Read failed\n");
|
2018-01-19 18:16:29 +00:00
|
|
|
HeapFree(GetProcessHeap(), 0, data);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (read_size > 2 && data[0] == 0xff && data[1] == 0xfe) { /* UTF-16 BOM */
|
|
|
|
ret = SysAllocStringLen((const WCHAR *)(data + 2), (read_size - 2) / sizeof(WCHAR));
|
|
|
|
}else {
|
|
|
|
size = MultiByteToWideChar(CP_ACP, 0, (const char *)data, read_size, NULL, 0);
|
|
|
|
ret = SysAllocStringLen(NULL, size);
|
|
|
|
if (ret)
|
|
|
|
MultiByteToWideChar(CP_ACP, 0, (const char *)data, read_size, ret, size);
|
|
|
|
}
|
|
|
|
HeapFree(GetProcessHeap(), 0, data);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2018-01-05 22:12:39 +00:00
|
|
|
static int change_command(int argc, WCHAR *argv[])
|
|
|
|
{
|
2018-01-23 16:11:20 +00:00
|
|
|
BOOL have_option = FALSE, enable = FALSE;
|
|
|
|
const WCHAR *task_name = NULL;
|
2018-01-05 22:12:39 +00:00
|
|
|
IRegisteredTask *task;
|
|
|
|
HRESULT hres;
|
|
|
|
|
2018-01-23 16:11:20 +00:00
|
|
|
while (argc) {
|
2020-10-05 21:22:54 +00:00
|
|
|
if (!wcsicmp(argv[0], L"/tn")) {
|
2018-01-23 16:11:20 +00:00
|
|
|
if (argc < 2) {
|
|
|
|
FIXME("Missing /tn value\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (task_name) {
|
|
|
|
FIXME("Duplicated /tn argument\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
task_name = argv[1];
|
|
|
|
argc -= 2;
|
|
|
|
argv += 2;
|
2020-10-05 21:22:54 +00:00
|
|
|
} else if (!wcsicmp(argv[0], L"/enable")) {
|
2018-01-23 16:11:20 +00:00
|
|
|
enable = TRUE;
|
|
|
|
have_option = TRUE;
|
|
|
|
argc--;
|
|
|
|
argv++;
|
2020-10-05 21:22:54 +00:00
|
|
|
} else if (!wcsicmp(argv[0], L"/tr")) {
|
2018-01-23 16:11:20 +00:00
|
|
|
if (argc < 2) {
|
|
|
|
FIXME("Missing /tr value\n");
|
|
|
|
return 1;
|
|
|
|
}
|
2018-01-05 22:12:39 +00:00
|
|
|
|
2018-01-23 16:11:20 +00:00
|
|
|
FIXME("Unsupported /tr option %s\n", debugstr_w(argv[1]));
|
|
|
|
have_option = TRUE;
|
|
|
|
argc -= 2;
|
|
|
|
argv += 2;
|
|
|
|
}else {
|
|
|
|
FIXME("Unsupported arguments %s\n", debugstr_w(argv[0]));
|
|
|
|
return 1;
|
|
|
|
}
|
2018-01-05 22:12:39 +00:00
|
|
|
}
|
|
|
|
|
2018-01-23 16:11:20 +00:00
|
|
|
if (!task_name) {
|
|
|
|
FIXME("Missing /tn option\n");
|
2018-01-05 22:12:39 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2018-01-23 16:11:20 +00:00
|
|
|
if (!have_option) {
|
2018-01-05 22:12:39 +00:00
|
|
|
FIXME("Missing change options\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
task = get_registered_task(task_name);
|
|
|
|
if (!task)
|
|
|
|
return 1;
|
|
|
|
|
2018-01-23 16:11:20 +00:00
|
|
|
if (enable) {
|
|
|
|
hres = IRegisteredTask_put_Enabled(task, VARIANT_TRUE);
|
|
|
|
if (FAILED(hres)) {
|
|
|
|
IRegisteredTask_Release(task);
|
2022-02-09 08:24:55 +00:00
|
|
|
FIXME("put_Enabled failed: %08lx\n", hres);
|
2018-01-23 16:11:20 +00:00
|
|
|
return 1;
|
|
|
|
}
|
2018-01-05 22:12:39 +00:00
|
|
|
}
|
|
|
|
|
2018-01-23 16:11:20 +00:00
|
|
|
IRegisteredTask_Release(task);
|
2018-01-05 22:12:39 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-01-19 18:16:29 +00:00
|
|
|
static int create_command(int argc, WCHAR *argv[])
|
|
|
|
{
|
|
|
|
const WCHAR *task_name = NULL, *xml_file = NULL;
|
|
|
|
ITaskFolder *root = NULL;
|
2018-01-22 16:35:55 +00:00
|
|
|
LONG flags = TASK_CREATE;
|
2018-01-19 18:16:29 +00:00
|
|
|
IRegisteredTask *task;
|
|
|
|
VARIANT empty;
|
|
|
|
BSTR str, xml;
|
|
|
|
HRESULT hres;
|
|
|
|
|
|
|
|
while (argc) {
|
2020-10-05 21:22:54 +00:00
|
|
|
if (!wcsicmp(argv[0], L"/xml")) {
|
2018-01-19 18:16:29 +00:00
|
|
|
if (argc < 2) {
|
|
|
|
FIXME("Missing /xml value\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (xml_file) {
|
|
|
|
FIXME("Duplicated /xml argument\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
xml_file = argv[1];
|
|
|
|
argc -= 2;
|
|
|
|
argv += 2;
|
2020-10-05 21:22:54 +00:00
|
|
|
} else if (!wcsicmp(argv[0], L"/tn")) {
|
2018-01-19 18:16:29 +00:00
|
|
|
if (argc < 2) {
|
|
|
|
FIXME("Missing /tn value\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (task_name) {
|
|
|
|
FIXME("Duplicated /tn argument\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
task_name = argv[1];
|
|
|
|
argc -= 2;
|
|
|
|
argv += 2;
|
2020-10-05 21:22:54 +00:00
|
|
|
} else if (!wcsicmp(argv[0], L"/f")) {
|
2018-01-22 16:35:55 +00:00
|
|
|
flags = TASK_CREATE_OR_UPDATE;
|
|
|
|
argc--;
|
|
|
|
argv++;
|
2020-10-05 21:22:54 +00:00
|
|
|
} else if (!wcsicmp(argv[0], L"/ru")) {
|
2018-01-23 16:11:20 +00:00
|
|
|
if (argc < 2) {
|
|
|
|
FIXME("Missing /ru value\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
FIXME("Unsupported /ru option %s\n", debugstr_w(argv[1]));
|
|
|
|
argc -= 2;
|
|
|
|
argv += 2;
|
2018-01-19 18:16:29 +00:00
|
|
|
}else {
|
|
|
|
FIXME("Unsupported argument %s\n", debugstr_w(argv[0]));
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!task_name) {
|
|
|
|
FIXME("Missing /tn argument\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!xml_file) {
|
|
|
|
FIXME("Missing /xml argument\n");
|
|
|
|
return E_FAIL;
|
|
|
|
}
|
|
|
|
|
|
|
|
xml = read_file_to_bstr(xml_file);
|
|
|
|
if (!xml)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
root = get_tasks_root_folder();
|
|
|
|
if (!root) {
|
|
|
|
SysFreeString(xml);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
V_VT(&empty) = VT_EMPTY;
|
|
|
|
str = SysAllocString(task_name);
|
2018-01-22 16:35:55 +00:00
|
|
|
hres = ITaskFolder_RegisterTask(root, str, xml, flags, empty, empty,
|
2018-01-19 18:16:29 +00:00
|
|
|
TASK_LOGON_NONE, empty, &task);
|
|
|
|
SysFreeString(str);
|
|
|
|
SysFreeString(xml);
|
|
|
|
ITaskFolder_Release(root);
|
|
|
|
if (FAILED(hres))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
IRegisteredTask_Release(task);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2018-01-19 18:16:20 +00:00
|
|
|
static int delete_command(int argc, WCHAR *argv[])
|
|
|
|
{
|
|
|
|
const WCHAR *task_name = NULL;
|
|
|
|
ITaskFolder *root = NULL;
|
|
|
|
BSTR str;
|
|
|
|
HRESULT hres;
|
|
|
|
|
|
|
|
while (argc) {
|
2020-10-05 21:22:54 +00:00
|
|
|
if (!wcsicmp(argv[0], L"/f")) {
|
2018-01-19 18:16:20 +00:00
|
|
|
TRACE("force opt\n");
|
|
|
|
argc--;
|
|
|
|
argv++;
|
2020-10-05 21:22:54 +00:00
|
|
|
} else if (!wcsicmp(argv[0], L"/tn")) {
|
2018-01-19 18:16:20 +00:00
|
|
|
if (argc < 2) {
|
|
|
|
FIXME("Missing /tn value\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (task_name) {
|
|
|
|
FIXME("Duplicated /tn argument\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
task_name = argv[1];
|
|
|
|
argc -= 2;
|
|
|
|
argv += 2;
|
|
|
|
}else {
|
|
|
|
FIXME("Unsupported argument %s\n", debugstr_w(argv[0]));
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!task_name) {
|
|
|
|
FIXME("Missing /tn argument\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
root = get_tasks_root_folder();
|
|
|
|
if (!root)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
str = SysAllocString(task_name);
|
|
|
|
hres = ITaskFolder_DeleteTask(root, str, 0);
|
|
|
|
SysFreeString(str);
|
|
|
|
ITaskFolder_Release(root);
|
|
|
|
if (FAILED(hres))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2019-10-17 15:05:47 +00:00
|
|
|
int __cdecl wmain(int argc, WCHAR *argv[])
|
2012-11-04 22:02:06 +00:00
|
|
|
{
|
2018-01-05 22:12:39 +00:00
|
|
|
int i, ret = 0;
|
2012-11-04 22:02:06 +00:00
|
|
|
|
|
|
|
for (i = 0; i < argc; i++)
|
2018-01-05 22:12:39 +00:00
|
|
|
TRACE(" %s", wine_dbgstr_w(argv[i]));
|
|
|
|
TRACE("\n");
|
2012-11-04 22:02:06 +00:00
|
|
|
|
2018-01-05 22:12:39 +00:00
|
|
|
CoInitialize(NULL);
|
|
|
|
|
|
|
|
if (argc < 2)
|
|
|
|
FIXME("Print current tasks state\n");
|
2020-10-05 21:22:54 +00:00
|
|
|
else if (!wcsicmp(argv[1], L"/change"))
|
2018-01-05 22:12:39 +00:00
|
|
|
ret = change_command(argc - 2, argv + 2);
|
2020-10-05 21:22:54 +00:00
|
|
|
else if (!wcsicmp(argv[1], L"/create"))
|
2018-01-19 18:16:29 +00:00
|
|
|
ret = create_command(argc - 2, argv + 2);
|
2020-10-05 21:22:54 +00:00
|
|
|
else if (!wcsicmp(argv[1], L"/delete"))
|
2018-01-19 18:16:20 +00:00
|
|
|
ret = delete_command(argc - 2, argv + 2);
|
2018-01-05 22:12:39 +00:00
|
|
|
else
|
|
|
|
FIXME("Unsupported command %s\n", debugstr_w(argv[1]));
|
|
|
|
|
|
|
|
CoUninitialize();
|
|
|
|
return ret;
|
2012-11-04 22:02:06 +00:00
|
|
|
}
|