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