#include "private.h" #include "subsmgrp.h" #include "subitem.h" // Contains implementations of IEnumSubscription and ISubscriptionMgr2 HRESULT SubscriptionItemFromCookie(BOOL fCreateNew, const SUBSCRIPTIONCOOKIE UNALIGNED *pCookie, ISubscriptionItem **ppSubscriptionItem); HRESULT DoGetItemFromURL(LPCTSTR pszURL, ISubscriptionItem **ppSubscriptionItem) { HRESULT hr; SUBSCRIPTIONCOOKIE cookie; if ((NULL == pszURL) || (NULL == ppSubscriptionItem)) { return E_INVALIDARG; } hr = ReadCookieFromInetDB(pszURL, &cookie); if (SUCCEEDED(hr)) { hr = SubscriptionItemFromCookie(FALSE, &cookie, ppSubscriptionItem); } return hr; } HRESULT DoGetItemFromURLW(LPCWSTR pwszURL, ISubscriptionItem **ppSubscriptionItem) { TCHAR szURL[INTERNET_MAX_URL_LENGTH]; if ((NULL == pwszURL) || (NULL == ppSubscriptionItem)) { return E_INVALIDARG; } #ifdef UNICODE StrCpyN(szURL, pwszURL, ARRAYSIZE(szURL)); #else MyOleStrToStrN(szURL, ARRAYSIZE(szURL), pwszURL); #endif return DoGetItemFromURL(szURL, ppSubscriptionItem); } HRESULT DoAbortItems( /* [in] */ DWORD dwNumCookies, /* [size_is][in] */ const SUBSCRIPTIONCOOKIE *pCookies) { HRESULT hr; if ((0 == dwNumCookies) || (NULL == pCookies)) { return E_INVALIDARG; } ISubscriptionThrottler *pst; hr = CoCreateInstance(CLSID_SubscriptionThrottler, NULL, CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER, IID_ISubscriptionThrottler, (void **)&pst); if (SUCCEEDED(hr)) { hr = pst->AbortItems(dwNumCookies, pCookies); pst->Release(); } else { hr = S_FALSE; } return hr; } HRESULT DoCreateSubscriptionItem( /* [in] */ const SUBSCRIPTIONITEMINFO *pSubscriptionItemInfo, /* [out] */ SUBSCRIPTIONCOOKIE *pNewCookie, /* [out] */ ISubscriptionItem **ppSubscriptionItem) { HRESULT hr; ISubscriptionItem *psi; if ((NULL == pSubscriptionItemInfo) || (NULL == pNewCookie) || (NULL == ppSubscriptionItem)) { return E_INVALIDARG; } *ppSubscriptionItem = NULL; CreateCookie(pNewCookie); hr = SubscriptionItemFromCookie(TRUE, pNewCookie, &psi); if (SUCCEEDED(hr)) { ASSERT(NULL != psi); hr = psi->SetSubscriptionItemInfo(pSubscriptionItemInfo); if (SUCCEEDED(hr)) { *ppSubscriptionItem = psi; } else { // Don't leak or leave slop hanging around psi->Release(); DoDeleteSubscriptionItem(pNewCookie, FALSE); } } return hr; } HRESULT DoCloneSubscriptionItem( /* [in] */ ISubscriptionItem *pSubscriptionItem, /* [out] */ SUBSCRIPTIONCOOKIE *pNewCookie, /* [out] */ ISubscriptionItem **ppSubscriptionItem) { HRESULT hr; SUBSCRIPTIONCOOKIE NewCookie; if ((NULL == pSubscriptionItem) || (NULL == ppSubscriptionItem)) { return E_INVALIDARG; } IEnumItemProperties *peip; SUBSCRIPTIONITEMINFO sii; *ppSubscriptionItem = NULL; // First get existing subscription details sii.cbSize = sizeof(SUBSCRIPTIONITEMINFO); hr = pSubscriptionItem->GetSubscriptionItemInfo(&sii); if (SUCCEEDED(hr)) { ISubscriptionItem *psi; // Mark as temp and create a new subscription item sii.dwFlags |= SI_TEMPORARY; hr = DoCreateSubscriptionItem(&sii, &NewCookie, &psi); if (SUCCEEDED(hr)) { if (pNewCookie) *pNewCookie = NewCookie; // Get properties from existing item hr = pSubscriptionItem->EnumProperties(&peip); if (SUCCEEDED(hr)) { ULONG count; ASSERT(NULL != peip); hr = peip->GetCount(&count); if (SUCCEEDED(hr)) { ITEMPROP *pProps = new ITEMPROP[count]; LPWSTR *pNames = new LPWSTR[count]; VARIANT *pVars = new VARIANT[count]; if ((NULL != pProps) && (NULL != pNames) && (NULL != pVars)) { hr = peip->Next(count, pProps, &count); if (SUCCEEDED(hr)) { ULONG i; for (i = 0; i < count; i++) { pNames[i] = pProps[i].pwszName; pVars[i] = pProps[i].variantValue; } hr = psi->WriteProperties(count, pNames, pVars); // clean up from enum for (i = 0; i < count; i++) { if (pProps[i].pwszName) { CoTaskMemFree(pProps[i].pwszName); } VariantClear(&pProps[i].variantValue); } } } else { hr = E_OUTOFMEMORY; } SAFEDELETE(pProps); SAFEDELETE(pNames); SAFEDELETE(pVars); } peip->Release(); } if (SUCCEEDED(hr)) { *ppSubscriptionItem = psi; } else { psi->Release(); } } } return hr; } HRESULT DoDeleteSubscriptionItem( /* [in] */ const SUBSCRIPTIONCOOKIE UNALIGNED *pCookie_ua, /* [in] */ BOOL fAbortItem) { HRESULT hr; TCHAR szKey[MAX_PATH]; SUBSCRIPTIONCOOKIE cookie_buf; SUBSCRIPTIONCOOKIE *pCookie; // // Make an aligned copy of pCookie_ua and set a pointer to it. // if (pCookie_ua != NULL) { cookie_buf = *pCookie_ua; pCookie = &cookie_buf; } else { pCookie = NULL; } if (NULL == pCookie) { return E_INVALIDARG; } if (fAbortItem) { DoAbortItems(1, pCookie); } if (ItemKeyNameFromCookie(pCookie, szKey, ARRAYSIZE(szKey))) { // Notify the agent that it is about to get deleted. ISubscriptionItem *pItem=NULL; if (SUCCEEDED(SubscriptionItemFromCookie(FALSE, pCookie, &pItem))) { SUBSCRIPTIONITEMINFO sii = { sizeof(SUBSCRIPTIONITEMINFO) }; if (SUCCEEDED(pItem->GetSubscriptionItemInfo(&sii))) { ASSERT(!(sii.dwFlags & SI_TEMPORARY)); ISubscriptionAgentControl *psac=NULL; if (SUCCEEDED( CoCreateInstance(sii.clsidAgent, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionAgentControl, (void**)&psac))) { psac->SubscriptionControl(pItem, SUBSCRIPTION_AGENT_DELETE); psac->Release(); } FireSubscriptionEvent(SUBSNOTF_DELETE, pCookie); if (GUID_NULL != sii.ScheduleGroup) { ISyncScheduleMgr *pSyncScheduleMgr; hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncScheduleMgr, (void **)&pSyncScheduleMgr); if (SUCCEEDED(hr)) { pSyncScheduleMgr->RemoveSchedule(&sii.ScheduleGroup); pSyncScheduleMgr->Release(); } } } pItem->Release(); } hr = (SHDeleteKey(HKEY_CURRENT_USER, szKey) == ERROR_SUCCESS) ? S_OK : E_FAIL; } else { TraceMsg(TF_ALWAYS, "Failed to delete subscription item."); hr = E_FAIL; } return hr; } HRESULT AddUpdateSubscription(SUBSCRIPTIONCOOKIE UNALIGNED *pCookie_ua, SUBSCRIPTIONITEMINFO *psii, LPCWSTR pwszURL, ULONG nProps, const LPWSTR rgwszName[], VARIANT rgValue[]) { HRESULT hr = S_OK; SUBSCRIPTIONCOOKIE cookie_buf; SUBSCRIPTIONCOOKIE *pCookie; // // Make an aligned copy of pCookie_ua and set a pointer to it. // cookie_buf = *pCookie_ua; pCookie = &cookie_buf; ASSERT((0 == nProps) || ((NULL != rgwszName) && (NULL != rgValue))); TCHAR szURL[INTERNET_MAX_URL_LENGTH]; ISubscriptionItem *psi = NULL; SUBSCRIPTIONCOOKIE cookie; #ifdef UNICODE StrCpyNW(szURL, pwszURL, ARRAYSIZE(szURL)); #else MyOleStrToStrN(szURL, ARRAYSIZE(szURL), pwszURL); #endif // Try and get the cookie from the inet db otherwise // create a new one. if (*pCookie == CLSID_NULL) { CreateCookie(&cookie); } else { cookie = *pCookie; } // Update the inet db WriteCookieToInetDB(szURL, &cookie, FALSE); hr = SubscriptionItemFromCookie(TRUE, &cookie, &psi); if (SUCCEEDED(hr)) { hr = psi->SetSubscriptionItemInfo(psii); if (SUCCEEDED(hr) && (nProps > 0)) { ASSERT(NULL != psi); hr = psi->WriteProperties(nProps, rgwszName, rgValue); } psi->Release(); if (FAILED(hr)) { DoDeleteSubscriptionItem(&cookie, TRUE); } } *pCookie_ua = cookie_buf; return hr; } HRESULT SubscriptionItemFromCookie(BOOL fCreateNew, const SUBSCRIPTIONCOOKIE UNALIGNED *pCookie_ua, ISubscriptionItem **ppSubscriptionItem) { HRESULT hr; SUBSCRIPTIONCOOKIE cookie_buf; SUBSCRIPTIONCOOKIE *pCookie; // // Make an aligned copy of pCookie_ua and set a pointer to it. // if (pCookie_ua != NULL) { cookie_buf = *pCookie_ua; pCookie = &cookie_buf; } else { pCookie = NULL; } ASSERT((NULL != pCookie) && (NULL != ppSubscriptionItem)); HKEY hkey; if (OpenItemKey(pCookie, fCreateNew ? TRUE : FALSE, KEY_READ | KEY_WRITE, &hkey)) { *ppSubscriptionItem = new CSubscriptionItem(pCookie, hkey); if (NULL != *ppSubscriptionItem) { hr = S_OK; } else { hr = E_OUTOFMEMORY; } RegCloseKey(hkey); } else { *ppSubscriptionItem = NULL; hr = E_FAIL; } return hr; } BOOL ItemKeyNameFromCookie(const SUBSCRIPTIONCOOKIE *pCookie, TCHAR *pszKeyName, DWORD cchKeyName) { WCHAR wszGuid[GUIDSTR_MAX]; ASSERT((NULL != pCookie) && (NULL != pszKeyName) && (cchKeyName >= ARRAYSIZE(WEBCHECK_REGKEY_STORE) + GUIDSTR_MAX)); if (StringFromGUID2(*pCookie, wszGuid, ARRAYSIZE(wszGuid))) { #ifdef UNICODE wnsprintfW(pszKeyName, cchKeyName, L"%s\\%s", c_szRegKeyStore, wszGuid); #else wnsprintfA(pszKeyName, cchKeyName, "%s\\%S", c_szRegKeyStore, wszGuid); #endif return TRUE; } return FALSE; } BOOL OpenItemKey(const SUBSCRIPTIONCOOKIE *pCookie, BOOL fCreateNew, REGSAM samDesired, HKEY *phkey) { TCHAR szKeyName[MAX_PATH]; ASSERT((NULL != pCookie) && (NULL != phkey)); if (ItemKeyNameFromCookie(pCookie, szKeyName, ARRAYSIZE(szKeyName))) { if (fCreateNew) { DWORD dwDisposition; return RegCreateKeyEx(HKEY_CURRENT_USER, szKeyName, 0, NULL, REG_OPTION_NON_VOLATILE, samDesired, NULL, phkey, &dwDisposition) == ERROR_SUCCESS; } else { return RegOpenKeyEx(HKEY_CURRENT_USER, szKeyName, 0, samDesired, phkey) == ERROR_SUCCESS; } } return FALSE; } // ISubscriptionMgr2 members STDMETHODIMP CSubscriptionMgr::GetItemFromURL( /* [in] */ LPCWSTR pwszURL, /* [out] */ ISubscriptionItem **ppSubscriptionItem) { return DoGetItemFromURLW(pwszURL, ppSubscriptionItem); } STDMETHODIMP CSubscriptionMgr::GetItemFromCookie( /* [in] */ const SUBSCRIPTIONCOOKIE *pSubscriptionCookie, /* [out] */ ISubscriptionItem **ppSubscriptionItem) { if ((NULL == pSubscriptionCookie) || (NULL == ppSubscriptionItem)) { return E_INVALIDARG; } return SubscriptionItemFromCookie(FALSE, pSubscriptionCookie, ppSubscriptionItem); } STDMETHODIMP CSubscriptionMgr::GetSubscriptionRunState( /* [in] */ DWORD dwNumCookies, /* [in] */ const SUBSCRIPTIONCOOKIE *pSubscriptionCookies, /* [out] */ DWORD *pdwRunState) { HRESULT hr; if ((0 == dwNumCookies) || (NULL == pSubscriptionCookies) || (NULL == pdwRunState)) { return E_INVALIDARG; } ISubscriptionThrottler *pst; hr = CoCreateInstance(CLSID_SubscriptionThrottler, NULL, CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER, IID_ISubscriptionThrottler, (void **)&pst); if (SUCCEEDED(hr)) { hr = pst->GetSubscriptionRunState(dwNumCookies, pSubscriptionCookies, pdwRunState); pst->Release(); } else { // Couldn't connect to a running throttler so assume nothing // is running. for (DWORD i = 0; i < dwNumCookies; i++) { *pdwRunState++ = 0; } hr = S_FALSE; } return hr; } STDMETHODIMP CSubscriptionMgr::EnumSubscriptions( /* [in] */ DWORD dwFlags, /* [out] */ IEnumSubscription **ppEnumSubscriptions) { HRESULT hr; if ((dwFlags & ~SUBSMGRENUM_MASK) || (NULL == ppEnumSubscriptions)) { return E_INVALIDARG; } CEnumSubscription *pes = new CEnumSubscription; *ppEnumSubscriptions = NULL; if (NULL != pes) { hr = pes->Initialize(dwFlags); if (SUCCEEDED(hr)) { hr = pes->QueryInterface(IID_IEnumSubscription, (void **)ppEnumSubscriptions); } pes->Release(); } else { hr = E_OUTOFMEMORY; } return hr; } STDMETHODIMP CSubscriptionMgr::UpdateItems( /* [in] */ DWORD dwFlags, /* [in] */ DWORD dwNumCookies, /* [size_is][in] */ const SUBSCRIPTIONCOOKIE *pCookies) { HRESULT hr; if ((dwFlags & ~SUBSMGRUPDATE_MASK) || (0 == dwNumCookies) || (NULL == pCookies)) { return E_INVALIDARG; } // // Fail if restrictions are in place. // FEATURE: Should we have a flag parameter to override this? // if (SHRestricted2W(REST_NoManualUpdates, NULL, 0)) { SGMessageBox(NULL, IDS_RESTRICTED, MB_OK); return E_ACCESSDENIED; } ISyncMgrSynchronizeInvoke *pSyncMgrInvoke; hr = CoCreateInstance(CLSID_SyncMgr, NULL, CLSCTX_ALL, IID_ISyncMgrSynchronizeInvoke, (void **)&pSyncMgrInvoke); if (SUCCEEDED(hr)) { DWORD dwInvokeFlags = SYNCMGRINVOKE_STARTSYNC; if (dwFlags & SUBSMGRUPDATE_MINIMIZE) { dwInvokeFlags |= SYNCMGRINVOKE_MINIMIZED; } hr = pSyncMgrInvoke->UpdateItems(dwInvokeFlags, CLSID_WebCheckOfflineSync, dwNumCookies * sizeof(SUBSCRIPTIONCOOKIE), (const BYTE *)pCookies); pSyncMgrInvoke->Release(); } return hr; } STDMETHODIMP CSubscriptionMgr::AbortItems( /* [in] */ DWORD dwNumCookies, /* [size_is][in] */ const SUBSCRIPTIONCOOKIE *pCookies) { return DoAbortItems(dwNumCookies, pCookies); } STDMETHODIMP CSubscriptionMgr::AbortAll() { HRESULT hr; ISubscriptionThrottler *pst; hr = CoCreateInstance(CLSID_SubscriptionThrottler, NULL, CLSCTX_INPROC_SERVER | CLSCTX_LOCAL_SERVER, IID_ISubscriptionThrottler, (void **)&pst); if (SUCCEEDED(hr)) { hr = pst->AbortAll(); pst->Release(); } else { hr = S_FALSE; } return hr; } // ISubscriptionMgrPriv STDMETHODIMP CSubscriptionMgr::CreateSubscriptionItem( /* [in] */ const SUBSCRIPTIONITEMINFO *pSubscriptionItemInfo, /* [out] */ SUBSCRIPTIONCOOKIE *pNewCookie, /* [out] */ ISubscriptionItem **ppSubscriptionItem) { return DoCreateSubscriptionItem(pSubscriptionItemInfo, pNewCookie, ppSubscriptionItem); } STDMETHODIMP CSubscriptionMgr::CloneSubscriptionItem( /* [in] */ ISubscriptionItem *pSubscriptionItem, /* [out] */ SUBSCRIPTIONCOOKIE *pNewCookie, /* [out] */ ISubscriptionItem **ppSubscriptionItem) { return DoCloneSubscriptionItem(pSubscriptionItem, pNewCookie, ppSubscriptionItem); } STDMETHODIMP CSubscriptionMgr::DeleteSubscriptionItem( /* [in] */ const SUBSCRIPTIONCOOKIE *pCookie) { return DoDeleteSubscriptionItem(pCookie, TRUE); } // ** CEnumSubscription ** CEnumSubscription::CEnumSubscription() { ASSERT(NULL == m_pCookies); ASSERT(0 == m_nCurrent); ASSERT(0 == m_nCount); m_cRef = 1; DllAddRef(); } CEnumSubscription::~CEnumSubscription() { if (NULL != m_pCookies) { delete [] m_pCookies; } DllRelease(); } HRESULT CEnumSubscription::Initialize(DWORD dwFlags) { HRESULT hr = E_FAIL; HKEY hkey = NULL; DWORD dwDisposition; ASSERT(0 == m_nCount); if (RegCreateKeyEx(HKEY_CURRENT_USER, c_szRegKeyStore, 0, NULL, REG_OPTION_NON_VOLATILE, KEY_READ | KEY_WRITE, NULL, &hkey, &dwDisposition) == ERROR_SUCCESS) { DWORD nCount; if (RegQueryInfoKey(hkey, NULL, // address of buffer for class string NULL, // address of size of class string buffer NULL, // reserved &nCount, // address of buffer for number of subkeys NULL, // address of buffer for longest subkey name length NULL, // address of buffer for longest class string length NULL, // address of buffer for number of value entries NULL, // address of buffer for longest value name length NULL, // address of buffer for longest value data length NULL, // address of buffer for security descriptor length NULL // address of buffer for last write time ) == ERROR_SUCCESS) { SUBSCRIPTIONCOOKIE Cookie; m_pCookies = new SUBSCRIPTIONCOOKIE[nCount]; if (NULL != m_pCookies) { TCHAR szKeyName[GUIDSTR_MAX]; hr = S_OK; for (ULONG i = 0; (i < nCount) && (S_OK == hr); i++) { if (RegEnumKey(hkey, i, szKeyName, ARRAYSIZE(szKeyName)) == ERROR_SUCCESS) { HRESULT hrConvert; #ifdef UNICODE hrConvert = CLSIDFromString(szKeyName, &Cookie); #else WCHAR wszKeyName[GUIDSTR_MAX]; MultiByteToWideChar(CP_ACP, 0, szKeyName, -1, wszKeyName, ARRAYSIZE(wszKeyName)); hrConvert = CLSIDFromString(wszKeyName, &Cookie); #endif if (SUCCEEDED(hrConvert)) { ISubscriptionItem *psi; m_pCookies[m_nCount] = Cookie; hr = SubscriptionItemFromCookie(FALSE, &Cookie, &psi); if (SUCCEEDED(hr)) { SUBSCRIPTIONITEMINFO sii; sii.cbSize = sizeof(SUBSCRIPTIONITEMINFO); if (SUCCEEDED(psi->GetSubscriptionItemInfo(&sii))) { // Only count this if it's not a temporary // or the caller asked for temporary items. if ((!(sii.dwFlags & SI_TEMPORARY)) || (dwFlags & SUBSMGRENUM_TEMP)) { m_nCount++; } } psi->Release(); } } } } } else { hr = E_OUTOFMEMORY; } } RegCloseKey(hkey); } return hr; } // IUnknown members STDMETHODIMP CEnumSubscription::QueryInterface(REFIID riid, void **ppv) { HRESULT hr; if (NULL == ppv) { return E_INVALIDARG; } if ((IID_IUnknown == riid) || (IID_IEnumSubscription == riid)) { *ppv = (IEnumSubscription *)this; AddRef(); hr = S_OK; } else { *ppv = NULL; hr = E_NOINTERFACE; } return hr; } STDMETHODIMP_(ULONG) CEnumSubscription::AddRef() { return ++m_cRef; } STDMETHODIMP_(ULONG) CEnumSubscription::Release() { if (--m_cRef == 0) { delete this; return 0; } return m_cRef; } HRESULT CEnumSubscription::CopyRange(ULONG nStart, ULONG nCount, SUBSCRIPTIONCOOKIE *pCookies, ULONG *pnCopied) { ULONG nCopied = 0; ASSERT((NULL != pCookies) && (NULL != pnCopied)); if (m_nCurrent < m_nCount) { ULONG nRemaining = m_nCount - m_nCurrent; nCopied = min(nRemaining, nCount); memcpy(pCookies, m_pCookies + m_nCurrent, nCopied * sizeof(SUBSCRIPTIONCOOKIE)); } *pnCopied = nCopied; return (nCopied == nCount) ? S_OK : S_FALSE; } // IEnumSubscription STDMETHODIMP CEnumSubscription::Next( /* [in] */ ULONG celt, /* [length_is][size_is][out] */ SUBSCRIPTIONCOOKIE *rgelt, /* [out] */ ULONG *pceltFetched) { HRESULT hr; if ((0 == celt) || ((celt > 1) && (NULL == pceltFetched)) || (NULL == rgelt)) { return E_INVALIDARG; } DWORD nFetched; hr = CopyRange(m_nCurrent, celt, rgelt, &nFetched); m_nCurrent += nFetched; if (pceltFetched) { *pceltFetched = nFetched; } return hr; } STDMETHODIMP CEnumSubscription::Skip( /* [in] */ ULONG celt) { HRESULT hr; m_nCurrent += celt; if (m_nCurrent > (m_nCount - 1)) { m_nCurrent = m_nCount; // Passed the last one hr = S_FALSE; } else { hr = S_OK; } return hr; } STDMETHODIMP CEnumSubscription::Reset() { m_nCurrent = 0; return S_OK; } STDMETHODIMP CEnumSubscription::Clone( /* [out] */ IEnumSubscription **ppenum) { HRESULT hr = E_OUTOFMEMORY; *ppenum = NULL; CEnumSubscription *pes = new CEnumSubscription; if (NULL != pes) { pes->m_pCookies = new SUBSCRIPTIONCOOKIE[m_nCount]; if (NULL != pes->m_pCookies) { ULONG nFetched; hr = E_FAIL; pes->m_nCount = m_nCount; CopyRange(0, m_nCount, pes->m_pCookies, &nFetched); ASSERT(m_nCount == nFetched); if (m_nCount == nFetched) { hr = pes->QueryInterface(IID_IEnumSubscription, (void **)ppenum); } } pes->Release(); } return hr; } STDMETHODIMP CEnumSubscription::GetCount( /* [out] */ ULONG *pnCount) { if (NULL == pnCount) { return E_INVALIDARG; } *pnCount = m_nCount; return S_OK; }