|
|
//+----------------------------------------------------------------------------
//
// 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; }
|