You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
874 lines
26 KiB
874 lines
26 KiB
//+----------------------------------------------------------------------------
|
|
//
|
|
// Job Scheduler Service
|
|
//
|
|
// Microsoft Windows
|
|
// Copyright (C) Microsoft Corporation, 1992 - 1996.
|
|
//
|
|
// File: sch_itf.cxx
|
|
//
|
|
// Contents: job scheduler service interface impementation
|
|
//
|
|
// Classes: CSchedule
|
|
//
|
|
// Interfaces: ITaskScheduler
|
|
//
|
|
// History: 08-Sep-95 EricB created
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
#include "..\pch\headers.hxx"
|
|
#pragma hdrstop
|
|
#include "Sched.hxx"
|
|
#include <FolderSecurity.h>
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::ITaskScheduler::GetTargetComputer, public
|
|
//
|
|
// Synopsis: Returns the name of the machine towards which the interface is
|
|
// currently targetted.
|
|
//
|
|
// Arguments: [ppwszComputer] - the returned buffer with the machine name
|
|
//
|
|
// Returns: hresults
|
|
//
|
|
// Notes: The string is callee allocated and caller freed with
|
|
// CoTaskMemFree.
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CSchedule::GetTargetComputer(LPWSTR * ppwszComputer)
|
|
{
|
|
TRACE(CSchedule, GetTargetComputer);
|
|
|
|
if (!ppwszComputer)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
HRESULT hr;
|
|
DWORD cch = SA_MAX_COMPUTERNAME_LENGTH + 1;
|
|
TCHAR tszLocalName[SA_MAX_COMPUTERNAME_LENGTH + 3] = TEXT("\\\\");
|
|
TCHAR * ptszTargetMachine;
|
|
WCHAR * pwszTargetMachine;
|
|
|
|
if (m_ptszTargetMachine)
|
|
{
|
|
ptszTargetMachine = m_ptszTargetMachine;
|
|
cch = lstrlen(ptszTargetMachine) + 1;
|
|
}
|
|
else // A NULL m_ptszTargetMachine means that we are targetted locally
|
|
{
|
|
if (!GetComputerName(tszLocalName + 2, &cch))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
ERR_OUT("GetTargetComputer: GetComputerName", hr);
|
|
return hr;
|
|
}
|
|
|
|
ptszTargetMachine = tszLocalName;
|
|
cch += 3; // 2 for the leading slashes + 1 for the NULL
|
|
}
|
|
|
|
pwszTargetMachine = ptszTargetMachine;
|
|
|
|
*ppwszComputer = (LPWSTR)CoTaskMemAlloc(cch * sizeof(WCHAR));
|
|
|
|
if (*ppwszComputer == NULL)
|
|
{
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
if (FAILED(hr = StringCchCopy(*ppwszComputer, cch, pwszTargetMachine)))
|
|
{
|
|
return hr;
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::ITaskScheduler::SetTargetComputer, public
|
|
//
|
|
// Synopsis: Sets the machine towards which subsequent ITaskScheduler
|
|
// calls will be directed
|
|
//
|
|
// Arguments: [pwszComputer] - the machine name string
|
|
//
|
|
// Returns: hresults
|
|
//
|
|
// Notes: The string is Caller allocated and freed. The machine name
|
|
// must include two leading backslashes.
|
|
// The caller may indicate using the local machine in one of two
|
|
// ways: by setting pwszComputer to NULL or to the UNC name of the
|
|
// local machine.
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CSchedule::SetTargetComputer(LPCWSTR pwszComputer)
|
|
{
|
|
TRACE(CSchedule, SetTargetComputer);
|
|
HRESULT hr;
|
|
DWORD cch;
|
|
BOOL fLocal = FALSE;
|
|
//
|
|
// Parameter validation. A null param means to target the local computer.
|
|
//
|
|
if (!pwszComputer)
|
|
{
|
|
fLocal = TRUE;
|
|
}
|
|
|
|
LPCTSTR tszPassedInName = pwszComputer;
|
|
if (!fLocal)
|
|
{
|
|
//
|
|
// Get the local machine name to compare with that passed in.
|
|
//
|
|
TCHAR tszLocalName[SA_MAX_COMPUTERNAME_LENGTH + 1];
|
|
cch = SA_MAX_COMPUTERNAME_LENGTH + 1;
|
|
if (!GetComputerName(tszLocalName, &cch))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
ERR_OUT("SetTargetComputer: GetComputerName", hr);
|
|
return hr;
|
|
}
|
|
|
|
TCHAR tszFQDN[SA_MAX_COMPUTERNAME_LENGTH + 1];
|
|
cch = SA_MAX_COMPUTERNAME_LENGTH + 1;
|
|
if (!GetComputerNameEx(ComputerNamePhysicalDnsFullyQualified, tszFQDN, &cch))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
ERR_OUT("SetTargetComputer: GetComputerNameEx", hr);
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// skip over first two characters ("\\") of tszPassedInName when comparing
|
|
//
|
|
fLocal = (lstrcmpi(tszPassedInName + 2, tszLocalName) == 0) ||
|
|
(lstrcmpi(tszPassedInName + 2, tszFQDN) == 0);
|
|
}
|
|
|
|
//
|
|
// If targeted remotely, get the folder path out of that machine's
|
|
// registry.
|
|
//
|
|
TCHAR tszFolderPath[MAX_PATH + 1];
|
|
if (!fLocal)
|
|
{
|
|
//
|
|
// Open the remote registry.
|
|
//
|
|
long lErr;
|
|
HKEY hRemoteKey, hSchedKey;
|
|
|
|
lErr = RegConnectRegistry(tszPassedInName, HKEY_LOCAL_MACHINE,
|
|
&hRemoteKey);
|
|
if (lErr != ERROR_SUCCESS)
|
|
{
|
|
schDebugOut((DEB_ERROR, "SetTargetComputer: RegConnectRegistry "
|
|
"failed with error %ld\n",
|
|
lErr));
|
|
return(HRESULT_FROM_WIN32(lErr));
|
|
}
|
|
|
|
lErr = RegOpenKeyEx(hRemoteKey, SCH_AGENT_KEY, 0, KEY_READ,
|
|
&hSchedKey);
|
|
if (lErr != ERROR_SUCCESS)
|
|
{
|
|
RegCloseKey(hRemoteKey);
|
|
|
|
if (lErr == ERROR_BADKEY || lErr == ERROR_FILE_NOT_FOUND)
|
|
{
|
|
return SCHED_E_SERVICE_NOT_INSTALLED;
|
|
}
|
|
|
|
schDebugOut((DEB_ERROR, "SetTargetComputer: RegOpenKeyEx "
|
|
"of Scheduler key failed with error %ld\n",
|
|
lErr));
|
|
return HRESULT_FROM_WIN32(lErr);
|
|
}
|
|
//
|
|
// Get the jobs folder location from the remote registry.
|
|
//
|
|
DWORD cb = (MAX_PATH + 1) * sizeof(TCHAR);
|
|
TCHAR tszRegFolderValue[MAX_PATH + 1];
|
|
lErr = RegQueryValueEx(hSchedKey, SCH_FOLDER_VALUE, NULL, NULL,
|
|
(LPBYTE)tszRegFolderValue, &cb);
|
|
if (lErr != ERROR_SUCCESS)
|
|
{
|
|
// use default if value absent
|
|
StringCchCopy(tszRegFolderValue, MAX_PATH + 1, TEXT("%SystemRoot%\\Tasks"));
|
|
}
|
|
RegCloseKey(hSchedKey);
|
|
|
|
//
|
|
// BUGBUG: temporary code to expand %SystemRoot% or %WinDir%
|
|
// The installer will have to write a full path to the registry 'cause
|
|
// expanding arbitrary environment strings remotely is too much work.
|
|
//
|
|
cch = ARRAY_LEN("%SystemRoot%") - 1;
|
|
if (_tcsncicmp(tszRegFolderValue, TEXT("%SystemRoot%"), cch) != 0)
|
|
{
|
|
cch = ARRAY_LEN("%WinDir%") - 1;
|
|
if (_tcsncicmp(tszRegFolderValue, TEXT("%WinDir%"), cch) != 0)
|
|
{
|
|
cch = 0;
|
|
}
|
|
}
|
|
|
|
if (cch != 0)
|
|
{
|
|
HKEY hCurVerKey;
|
|
lErr = RegOpenKeyEx(hRemoteKey,
|
|
TEXT("SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion"),
|
|
0, KEY_ALL_ACCESS,
|
|
&hCurVerKey);
|
|
if (lErr != ERROR_SUCCESS)
|
|
{
|
|
RegCloseKey(hRemoteKey);
|
|
schDebugOut((DEB_ERROR, "SetTargetComputer: RegOpenKeyEx "
|
|
"of CurrentVersion key failed with error %ld\n",
|
|
lErr));
|
|
return HRESULT_FROM_WIN32(lErr);
|
|
}
|
|
TCHAR tszSystemRoot[MAX_PATH + 1];
|
|
cb = (MAX_PATH + 1) * sizeof(TCHAR);
|
|
lErr = RegQueryValueEx(hCurVerKey, TEXT("SystemRoot"), NULL, NULL,
|
|
(LPBYTE)tszSystemRoot, &cb);
|
|
if (lErr != ERROR_SUCCESS)
|
|
{
|
|
RegCloseKey(hCurVerKey);
|
|
RegCloseKey(hRemoteKey);
|
|
schDebugOut((DEB_ERROR, "SetTargetComputer: RegQueryValueEx "
|
|
"of CurrentVersion key failed with error %ld\n",
|
|
lErr));
|
|
return HRESULT_FROM_WIN32(lErr);
|
|
}
|
|
RegCloseKey(hCurVerKey);
|
|
StringCchCopy(tszFolderPath, MAX_PATH + 1, tszSystemRoot);
|
|
StringCchCat(tszFolderPath, MAX_PATH + 1, tszRegFolderValue + cch);
|
|
}
|
|
else
|
|
{
|
|
StringCchCopy(tszFolderPath, MAX_PATH + 1, tszRegFolderValue);
|
|
}
|
|
//
|
|
// end of temporary code to expand %SystemRoot%
|
|
//
|
|
|
|
RegCloseKey(hRemoteKey);
|
|
|
|
//
|
|
// Check the folder path for being a fully qualified path name where
|
|
// the first char is the drive designator and the second char is a
|
|
// colon.
|
|
//
|
|
if (!s_isDriveLetter(tszFolderPath[0]) || tszFolderPath[1] != TEXT(':'))
|
|
{
|
|
ERR_OUT("SetTargetComputer: registry path", ERROR_BAD_PATHNAME);
|
|
return HRESULT_FROM_WIN32(ERROR_BAD_PATHNAME);
|
|
}
|
|
//
|
|
// The UNC path to the job folder will be the result of concatonating
|
|
// the machine name and the expanded folder path. The drive designator
|
|
// in the folder path will be turned in an administrative share name
|
|
// by replacing the colon with a dollar sign and will look like:
|
|
// \\machine\c$\windir\jobs
|
|
// so that the count below includes the slash trailing the machine name
|
|
// plus the terminating null.
|
|
//
|
|
cch = lstrlen(tszPassedInName) + 1 + lstrlen(tszFolderPath) + 1;
|
|
}
|
|
else // Targetted locally.
|
|
{
|
|
//
|
|
// Use the local path. Include one for the null terminator.
|
|
//
|
|
cch = lstrlen(g_TasksFolderInfo.ptszPath) + 1;
|
|
}
|
|
|
|
//
|
|
// Allocate the ITaskScheduler folder path string buffer.
|
|
//
|
|
size_t cchPathBuf = cch;
|
|
TCHAR * ptszPathBuf = new TCHAR[cchPathBuf];
|
|
if (!ptszPathBuf)
|
|
{
|
|
ERR_OUT("SetTargetComputer: Job folder path buffer allocation",
|
|
E_OUTOFMEMORY);
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
//
|
|
// Allocate the ITaskScheduler machine name string buffer.
|
|
//
|
|
size_t cchTargetMachine = 0;
|
|
TCHAR * ptszTargetMachine;
|
|
if (!fLocal)
|
|
{
|
|
cchTargetMachine = lstrlen(tszPassedInName) + 1;
|
|
ptszTargetMachine = new TCHAR[cchTargetMachine];
|
|
if (!ptszTargetMachine)
|
|
{
|
|
ERR_OUT("CSchedule::SetTargetComputer", E_OUTOFMEMORY);
|
|
delete ptszPathBuf;
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Now that all failable operation have completed sucessfully, we can
|
|
// update the machine name and folder path members.
|
|
//
|
|
|
|
if (m_ptszTargetMachine)
|
|
{
|
|
delete m_ptszTargetMachine;
|
|
}
|
|
|
|
if (m_ptszFolderPath)
|
|
{
|
|
delete m_ptszFolderPath;
|
|
}
|
|
|
|
//
|
|
// Save the new machine name.
|
|
//
|
|
|
|
if (fLocal)
|
|
{
|
|
//
|
|
// If we are targetted locally, the machine name member is set to
|
|
// NULL.
|
|
//
|
|
m_ptszTargetMachine = NULL;
|
|
}
|
|
else
|
|
{
|
|
m_ptszTargetMachine = ptszTargetMachine;
|
|
StringCchCopy(m_ptszTargetMachine, cchTargetMachine, tszPassedInName);
|
|
}
|
|
|
|
//
|
|
// Save the folder path name.
|
|
//
|
|
|
|
m_ptszFolderPath = ptszPathBuf;
|
|
|
|
if (fLocal)
|
|
{
|
|
StringCchCopy(m_ptszFolderPath, cchPathBuf, g_TasksFolderInfo.ptszPath);
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Convert the folder location to an UNC path.
|
|
//
|
|
// Turn the drive designator into the admin share by replacing the
|
|
// colon with the dollar sign.
|
|
//
|
|
tszFolderPath[1] = TEXT('$');
|
|
//
|
|
// Compose the UNC path.
|
|
//
|
|
StringCchCopy(m_ptszFolderPath, cchPathBuf, tszPassedInName);
|
|
StringCchCat(m_ptszFolderPath, cchPathBuf, TEXT("\\"));
|
|
StringCchCat(m_ptszFolderPath, cchPathBuf, tszFolderPath);
|
|
}
|
|
|
|
// Now that we have the folder name, check for access
|
|
// only check in the remote case - access checks remotely don't always work
|
|
// due to problems resolving groups local to remote machine
|
|
// this is not a problem, access control will be enforced by the file system on the remote machine.
|
|
if (fLocal && FAILED(hr = CoFolderAccessCheck(m_ptszFolderPath, FILE_READ_DATA)))
|
|
{
|
|
// set back to local machine.
|
|
delete[] m_ptszTargetMachine;
|
|
m_ptszTargetMachine = NULL;
|
|
|
|
// set back to something resembling a default state
|
|
Init();
|
|
|
|
return hr;
|
|
}
|
|
|
|
schDebugOut((DEB_ITRACE,
|
|
"SetTargetComputer: path to sched folder: \"" FMT_TSTR "\"\n",
|
|
m_ptszFolderPath));
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::ITaskScheduler::EnumInternal, public
|
|
// (public member in class for internal use, not public via API)
|
|
//
|
|
// Synopsis: Returns a job/queue object enumerator.
|
|
// only difference between this and the COM version is that
|
|
// this function checks access against the thread/process token
|
|
// rather than the COM call context
|
|
//
|
|
// Arguments: [ppEnumJobs] - a place to return a pointer to the enumerator
|
|
//
|
|
// Returns: hresults
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CSchedule::EnumInternal(IEnumWorkItems ** ppEnumJobs)
|
|
{
|
|
TRACE(CSchedule, Enum);
|
|
HRESULT hr;
|
|
|
|
// only check in the remote case - access checks remotely don't always work
|
|
// due to problems resolving groups local to remote machine
|
|
// this is not a problem, access control will be enforced by the file system on the remote machine.
|
|
if ((NULL == m_ptszTargetMachine) && FAILED(hr = FolderAccessCheckOnThreadToken(m_ptszFolderPath, FILE_READ_DATA)))
|
|
return hr;
|
|
|
|
CEnumJobs * pEnumJobs = new CEnumJobs;
|
|
if (pEnumJobs == NULL)
|
|
{
|
|
*ppEnumJobs = NULL;
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
hr = pEnumJobs->Init(m_ptszFolderPath);
|
|
if (FAILED(hr))
|
|
{
|
|
delete pEnumJobs;
|
|
*ppEnumJobs = NULL;
|
|
}
|
|
|
|
*ppEnumJobs = pEnumJobs;
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::ITaskScheduler::Enum, public
|
|
//
|
|
// Synopsis: Returns a job/queue object enumerator.
|
|
//
|
|
// Arguments: [ppEnumJobs] - a place to return a pointer to the enumerator
|
|
//
|
|
// Returns: hresults
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CSchedule::Enum(IEnumWorkItems ** ppEnumJobs)
|
|
{
|
|
TRACE(CSchedule, Enum);
|
|
HRESULT hr;
|
|
|
|
// only check in the remote case - access checks remotely don't always work
|
|
// due to problems resolving groups local to remote machine
|
|
// this is not a problem, access control will be enforced by the file system on the remote machine.
|
|
if ((NULL == m_ptszTargetMachine) && FAILED(hr = CoFolderAccessCheck(m_ptszFolderPath, FILE_READ_DATA)))
|
|
return hr;
|
|
|
|
if (!ppEnumJobs)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
|
|
CEnumJobs * pEnumJobs = new CEnumJobs;
|
|
if (pEnumJobs == NULL)
|
|
{
|
|
*ppEnumJobs = NULL;
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
hr = pEnumJobs->Init(m_ptszFolderPath);
|
|
if (FAILED(hr))
|
|
{
|
|
delete pEnumJobs;
|
|
*ppEnumJobs = NULL;
|
|
}
|
|
|
|
*ppEnumJobs = pEnumJobs;
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::ITaskScheduler::NewWorkItem, public
|
|
//
|
|
// Synopsis: Create a new job object.
|
|
//
|
|
// Arguments: [pwszJobName] - the name of the new job *REQUIRED*
|
|
// [riid] - the interface desired
|
|
// [ppunk] - a place to return a pointer to the new job object
|
|
//
|
|
// Returns: hresults
|
|
//
|
|
// Notes: ppwszJobName is caller allocated and freed. The CJob::Save
|
|
// method will copy it before returning. The job name must conform
|
|
// to NT file naming conventions but must not include
|
|
// [back]slashes because nesting within the job object folder is
|
|
// not allowed.
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CSchedule::NewWorkItem(LPCWSTR pwszJobName, REFCLSID rclsid,
|
|
REFIID riid, IUnknown ** ppunk)
|
|
{
|
|
TRACE(CSchedule, NewWorkItem);
|
|
HRESULT hr;
|
|
|
|
// only check in the remote case - access checks remotely don't always work
|
|
// due to problems resolving groups local to remote machine
|
|
// this is not a problem, access control will be enforced by the file system on the remote machine.
|
|
if ((NULL == m_ptszTargetMachine) && FAILED(hr = CoFolderAccessCheck(m_ptszFolderPath, FILE_WRITE_DATA)))
|
|
return hr;
|
|
|
|
if (!pwszJobName)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
if (!ppunk)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
*ppunk = NULL;
|
|
|
|
if (!IsEqualCLSID(rclsid, CLSID_CTask))
|
|
{
|
|
return CLASS_E_CLASSNOTAVAILABLE;
|
|
}
|
|
|
|
TCHAR * ptszFullName;
|
|
HANDLE hFile;
|
|
|
|
hr = CheckJobName(pwszJobName, &ptszFullName);
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("CSchedule::NewWorkItem: CheckJobName", hr);
|
|
return hr;
|
|
}
|
|
|
|
CJob * pJob = CJob::Create();
|
|
if (pJob == NULL)
|
|
{
|
|
delete [] ptszFullName;
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
//
|
|
// Do the QI before the CreateFile so that if the caller asks for a non-
|
|
// supported interface, the failure will not result in disk operations.
|
|
//
|
|
hr = pJob->QueryInterface(riid, (void **)ppunk);
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("CSchedule::NewWorkItem: QueryInterface(riid)", hr);
|
|
goto CleanExit;
|
|
}
|
|
// the above QI increased the refcount to 2, so set it back to 1
|
|
pJob->Release();
|
|
|
|
//
|
|
// Per the spec for this method, the file must not already exist.
|
|
//
|
|
hFile = CreateFile(ptszFullName,
|
|
0, // desired access: none
|
|
FILE_SHARE_READ | FILE_SHARE_WRITE,
|
|
// share mode: all
|
|
NULL, // security attributes
|
|
OPEN_EXISTING,
|
|
0,
|
|
NULL);
|
|
|
|
if (hFile == INVALID_HANDLE_VALUE)
|
|
{
|
|
DWORD dwErr = GetLastError();
|
|
|
|
if (dwErr == ERROR_FILE_NOT_FOUND)
|
|
{
|
|
//
|
|
// This is good. Save the new filename.
|
|
//
|
|
pJob->m_ptszFileName = ptszFullName;
|
|
return S_OK;
|
|
}
|
|
else
|
|
{
|
|
hr = HRESULT_FROM_WIN32(dwErr);
|
|
ERR_OUT("CSchedule::NewWorkItem: CreateFile", hr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//
|
|
// Opened successfully - the file exists
|
|
//
|
|
CloseHandle(hFile);
|
|
hr = HRESULT_FROM_WIN32(ERROR_FILE_EXISTS);
|
|
ERR_OUT("CSchedule::NewWorkItem", hr);
|
|
}
|
|
|
|
CleanExit:
|
|
|
|
delete [] ptszFullName;
|
|
delete pJob; // on error, completely destroy the job object
|
|
*ppunk = NULL;
|
|
|
|
return hr;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::ITaskScheduler::AddWorkItem, public
|
|
//
|
|
// Synopsis: Saves the job to the job scheduler folder.
|
|
//
|
|
// Arguments: [pwszJobName] - the name of the job *REQUIRED*
|
|
// [pJob] - pointer to the job object
|
|
//
|
|
// Returns: hresults
|
|
//
|
|
// Notes: Same job name conditions as above.
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CSchedule::AddWorkItem(LPCWSTR pwszJobName,
|
|
IScheduledWorkItem * pWorkItem)
|
|
{
|
|
TRACE(CSchedule, AddWorkItem);
|
|
HRESULT hr;
|
|
|
|
// only check in the remote case - access checks remotely don't always work
|
|
// due to problems resolving groups local to remote machine
|
|
// this is not a problem, access control will be enforced by the file system on the remote machine.
|
|
if ((NULL == m_ptszTargetMachine) && FAILED(hr = CoFolderAccessCheck(m_ptszFolderPath, FILE_WRITE_DATA)))
|
|
return hr;
|
|
|
|
if (!pwszJobName)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
if (!pWorkItem)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
TCHAR * ptszFullName;
|
|
|
|
hr = CheckJobName(pwszJobName, &ptszFullName);
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("CSchedule::AddWorkItem: CheckJobName", hr);
|
|
return hr;
|
|
}
|
|
IPersistFile * pFile;
|
|
|
|
hr = pWorkItem->QueryInterface(IID_IPersistFile, (void **)&pFile);
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("CSchedule::AddWorkItem: QI(IPersistFile)", hr);
|
|
delete [] ptszFullName;
|
|
return hr;
|
|
}
|
|
|
|
WCHAR * pwszName;
|
|
|
|
pwszName = ptszFullName;
|
|
|
|
hr = pFile->Save(pwszName, TRUE);
|
|
pFile->Release();
|
|
delete [] ptszFullName;
|
|
return hr;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::ITaskScheduler::Delete, public
|
|
//
|
|
// Synopsis: Deletes the job/queue.
|
|
//
|
|
// Arguments: [pwszJobName] - indicates the job/queue to delete
|
|
//
|
|
// Returns: hresults
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CSchedule::Delete(LPCWSTR pwszJobName)
|
|
{
|
|
TRACE(CSchedule, Delete);
|
|
|
|
HRESULT hr;
|
|
|
|
// Let's not be redundant, here -
|
|
// if (FAILED(hr = CoFolderAccessCheck(m_ptszFolderPath, FILE_DELETE_CHILD)))
|
|
// return hr;
|
|
|
|
if (!pwszJobName)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
TCHAR * ptszFullName;
|
|
|
|
hr = CheckJobName(pwszJobName, &ptszFullName);
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("CSchedule::Delete: CheckJobName", hr);
|
|
return hr;
|
|
}
|
|
|
|
if (!DeleteFile(ptszFullName))
|
|
{
|
|
hr = HRESULT_FROM_WIN32(GetLastError());
|
|
ERR_OUT("CSchedule::Delete: DeleteFile", hr);
|
|
delete ptszFullName;
|
|
return hr;
|
|
}
|
|
delete ptszFullName;
|
|
return S_OK;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::ITaskScheduler::Activate, public
|
|
//
|
|
// Synopsis: Given a valid name, returns a pointer to the activated job
|
|
// object
|
|
//
|
|
// Arguments: [pwszName] - the name of the job to activate
|
|
// [riid] - the interface to return
|
|
// [ppunk] - a pointer to the job object interface
|
|
//
|
|
// Returns: hresults
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CSchedule::Activate(LPCWSTR pwszName, REFIID riid, IUnknown ** ppunk)
|
|
{
|
|
TRACE(CSchedule, Activate);
|
|
|
|
if (!pwszName)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
if (!ppunk)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
TCHAR * ptszFullName;
|
|
HRESULT hr = CheckJobName(pwszName, &ptszFullName);
|
|
if (FAILED(hr))
|
|
{
|
|
*ppunk = NULL;
|
|
return hr;
|
|
}
|
|
|
|
// only check in the remote case - access checks remotely don't always work
|
|
// due to problems resolving groups local to remote machine
|
|
// this is not a problem, access control will be enforced by the file system on the remote machine.
|
|
if ((NULL == m_ptszTargetMachine) && FAILED(hr = CoFolderAccessCheck(ptszFullName, FILE_READ_DATA)))
|
|
{
|
|
delete[] ptszFullName;
|
|
return hr;
|
|
}
|
|
|
|
CJob * pJob;
|
|
|
|
//
|
|
// CJob is a single-use, in-proc handler, so no need to get OLE in the
|
|
// loop here. Use new (called by CJob::Create) instead of CoCreateInstance.
|
|
//
|
|
pJob = CJob::Create();
|
|
if (pJob == NULL)
|
|
{
|
|
*ppunk = NULL;
|
|
delete [] ptszFullName;
|
|
return E_OUTOFMEMORY;
|
|
}
|
|
|
|
hr = pJob->LoadP(ptszFullName, 0, TRUE, TRUE);
|
|
|
|
delete [] ptszFullName;
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("CSchedule::Activate, Load", hr);
|
|
*ppunk = NULL;
|
|
pJob->Release(); // on error, completely release the job object
|
|
return hr;
|
|
}
|
|
|
|
hr = pJob->QueryInterface(riid, (void **)ppunk);
|
|
if (FAILED(hr))
|
|
{
|
|
ERR_OUT("CSchedule::Activate: QueryInterface(riid)", hr);
|
|
*ppunk = NULL;
|
|
pJob->Release(); // on error, completely release the job object
|
|
return hr;
|
|
}
|
|
|
|
//
|
|
// The above QI increased the refcount to 2, so set it back to 1.
|
|
//
|
|
pJob->Release();
|
|
return S_OK;
|
|
}
|
|
|
|
//+----------------------------------------------------------------------------
|
|
//
|
|
// Member: CSchedule::ITaskScheduler::IsOfType, public
|
|
//
|
|
// Synopsis: Does this object support the desired interface?
|
|
//
|
|
// Arguments: [pwszName] - indicates the object name
|
|
// [riid] - indicates the interface of interest, typically
|
|
// IID_ITask or IID_IScheduledQueue
|
|
//
|
|
// Returns: S_OK if it is, S_FALSE otherwise.
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
STDMETHODIMP
|
|
CSchedule::IsOfType(LPCWSTR pwszName, REFIID riid)
|
|
{
|
|
TRACE(CSchedule, IsOfType);
|
|
|
|
// CODEWORK: A heavyweight implementation for now. It could possibly
|
|
// be optimized by doing the QueryInterface before the LoadP, and
|
|
// doing a lightweight LoadP.
|
|
|
|
if (!pwszName)
|
|
{
|
|
return E_INVALIDARG;
|
|
}
|
|
|
|
IUnknown * punk;
|
|
HRESULT hr = Activate(pwszName, riid, &punk);
|
|
if (SUCCEEDED(hr))
|
|
{
|
|
punk->Release();
|
|
hr = S_OK;
|
|
}
|
|
else
|
|
{
|
|
if (hr == HRESULT_FROM_WIN32(ERROR_INVALID_DATA) ||
|
|
hr == SCHED_E_UNKNOWN_OBJECT_VERSION ||
|
|
hr == E_NOINTERFACE)
|
|
{
|
|
//
|
|
// These errors mean that the object is definitely not of a
|
|
// type that we support. We translate them to S_FALSE.
|
|
// Other errors could include file-not-found, access-denied,
|
|
// invalid-arg, etc. We return those errors unmodified.
|
|
//
|
|
hr = S_FALSE;
|
|
}
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|