#include "private.h" #include #include #include #include "cdfagent.h" #include "cdlabsc.h" #include "cdlagent.h" #include #include #include "subsmgrp.h" #include HRESULT GetXMLAttribute(IXMLElement *pItem, LPCWSTR pwszAttribute, VARIANT *pvRet); HRESULT GetNextChildTag(IXMLElement *pRoot, LPCWSTR szTag, IXMLElement **ppChildReq, int &nLastChild) { BSTR bstrTag = NULL; IXMLElementCollection * pChildren = NULL; HRESULT hr = S_FALSE; // assume not found. IXMLElement * pChild = NULL; // Find the children if they exist if (SUCCEEDED(pRoot->get_children(&pChildren)) && pChildren) { long length = 0; if (SUCCEEDED(pChildren->get_length(&length)) && length > 0) { VARIANT vIndex, vEmpty; vIndex.vt = VT_I4; vEmpty.vt = VT_EMPTY; nLastChild++; for (long i=nLastChild; iitem(vIndex, vEmpty, &pDispItem))) { if (SUCCEEDED(pDispItem->QueryInterface(IID_IXMLElement, (void **)&pChild))) { // look for first SoftDist tag pChild->get_tagName(&bstrTag); if (StrCmpIW(bstrTag, szTag) == 0) { nLastChild = i; hr = S_OK; goto Exit; } SAFEFREEBSTR(bstrTag); SAFERELEASE(pChild); } pDispItem->Release(); } } } } else { hr = E_FAIL; } Exit: *ppChildReq = pChild; if (pChildren) SAFERELEASE(pChildren); SAFEFREEBSTR(bstrTag); return hr; } HRESULT GetFirstChildTag(IXMLElement *pRoot, LPCWSTR szTag, IXMLElement **ppChildReq) { int nLastChild = -1; // first child, never seen any before this one return GetNextChildTag(pRoot, szTag, ppChildReq, nLastChild); } CCDLAgent::CCDLAgent() : m_pCCDLAgentBSC(NULL) , m_szCDF(NULL) , m_bAcceptSoftware(FALSE) { m_sdi.cbSize = sizeof(SOFTDISTINFO); m_bSilentMode = TRUE; } CCDLAgent::~CCDLAgent() { SAFERELEASE(m_pSoftDistElement); SAFERELEASE(m_pSoftDistExt); CRunDeliveryAgent::SafeRelease(m_pAgent); SAFEFREEOLESTR(m_szCDF); SAFEFREEBSTR(m_szErrorText); SAFEDELETE(m_sdi.szAbstract); SAFEDELETE(m_sdi.szTitle); SAFEDELETE(m_sdi.szHREF); SAFEFREEOLESTR(m_szDistUnit); } HRESULT CCDLAgent::StartOperation() { HRESULT hr = S_OK, hr2; // unknown pointers IUnknown *punk = NULL; IServiceProvider *pSP; m_pSoftDistElement = NULL; if (FAILED(ReadOLESTR(m_pSubscriptionItem, c_szPropURL, &m_szURL))) { hr = E_INVALIDARG; goto Failed; } hr2 = E_FAIL; if (SUCCEEDED(m_pAgentEvents->QueryInterface(IID_IServiceProvider, (void **)&pSP)) && pSP) { hr2 = pSP->QueryService(CLSID_XMLDocument, IID_IXMLElement, (void **)&punk); pSP->Release(); } if (FAILED(hr2) || !punk) { // We are processing a request to pull a CAB, probably from Web Crawler agent. if (FAILED(ReadOLESTR(m_pSubscriptionItem, L"DistUnit", &m_szDistUnit)) || FAILED(ReadDWORD(m_pSubscriptionItem, L"VersionMS",&m_dwVersionMS)) || FAILED(ReadDWORD(m_pSubscriptionItem, L"VersionLS", &m_dwVersionLS))) { hr = E_INVALIDARG; goto Failed; } m_pSoftDistElement = NULL; } else { if (FAILED(punk->QueryInterface(IID_IXMLElement, (void **)&m_pSoftDistElement))) { SAFERELEASE(punk); hr = E_INVALIDARG; goto Failed; } SAFERELEASE(punk); Assert(m_pSoftDistElement); } ReadDWORD(m_pSubscriptionItem, c_szPropCrawlMaxSize, &m_dwMaxSizeKB); ReadDWORD(m_pSubscriptionItem, c_szPropChannelFlags, &m_dwChannelFlags); ReadDWORD(m_pSubscriptionItem, c_szPropAgentFlags, &m_dwAgentFlags); hr = CDeliveryAgent::StartOperation(); return hr; Failed: SetEndStatus(hr); SendUpdateNone(); return hr; } HRESULT CCDLAgent::StartDownload() { IBindCtx *pbc = NULL; HRESULT hr = S_OK; LPWSTR szCodeBase; DWORD dwSize; BOOL bCleanUpNow = FALSE; DWORD dwPolicy = 0; DWORD dwContext = 0; IInternetSecurityManager * pism = NULL; if (FAILED(GetEndStatus())) { hr = GetEndStatus(); goto Exit; } hr = CoCreateInstance(CLSID_SoftDistExt, NULL, CLSCTX_INPROC_SERVER, IID_ISoftDistExt, (void **)&m_pSoftDistExt); if (FAILED(hr)) goto Exit; // Process SOFTDIST tag structure if present. if (m_pSoftDistElement != NULL) { dwPolicy = 0xFFFF0000; if (FAILED(CoCreateInstance(CLSID_InternetSecurityManager, NULL, CLSCTX_INPROC_SERVER, IID_IInternetSecurityManager, (void**)&pism)) || !pism) { hr = E_ACCESSDENIED; goto Exit; } hr = pism->ProcessUrlAction(m_szURL, URLACTION_CHANNEL_SOFTDIST_PERMISSIONS, (BYTE *)&dwPolicy, sizeof(dwPolicy), (BYTE *)&dwContext, sizeof(dwContext), PUAF_NOUI, 0); pism->Release(); if (FAILED(hr)) { goto Exit; } dwPolicy &= 0xFFFF0000; if (dwPolicy != URLPOLICY_CHANNEL_SOFTDIST_PROHIBIT && dwPolicy != URLPOLICY_CHANNEL_SOFTDIST_PRECACHE && dwPolicy != URLPOLICY_CHANNEL_SOFTDIST_AUTOINSTALL) { hr = E_INVALIDARG; goto Exit; } if (dwPolicy == URLPOLICY_CHANNEL_SOFTDIST_PROHIBIT) { hr = E_ACCESSDENIED; goto Exit; } hr = m_pSoftDistExt->ProcessSoftDist(m_szCDF, m_pSoftDistElement, &m_sdi); if (m_sdi.dwFlags & SOFTDIST_FLAG_DELETE_SUBSCRIPTION) { ISubscriptionMgr *pSubMgr = NULL; hr = CoCreateInstance(CLSID_SubscriptionMgr, NULL, CLSCTX_INPROC_SERVER, IID_ISubscriptionMgr, (void**)&pSubMgr); if (SUCCEEDED(hr)) { hr = pSubMgr->DeleteSubscription(m_szURL,NULL); pSubMgr->Release(); } hr = S_FALSE; } // Send email & update software? if (hr == S_OK) { if (m_sdi.dwFlags) { m_bSendEmail = TRUE; } else { // no usage flag and no restriction implies no email. m_bSendEmail = FALSE; } if (m_sdi.dwFlags & SOFTDIST_FLAG_USAGE_AUTOINSTALL) { m_bAcceptSoftware = (dwPolicy == URLPOLICY_CHANNEL_SOFTDIST_AUTOINSTALL) ? TRUE : FALSE; m_bSilentMode = FALSE; } else if (m_sdi.dwFlags & SOFTDIST_FLAG_USAGE_PRECACHE) { // to get here, we must have precache or autoinstall policy permissions m_bAcceptSoftware = TRUE; } else { m_bAcceptSoftware = FALSE; } } else { m_bSendEmail = FALSE; m_bAcceptSoftware = FALSE; bCleanUpNow = TRUE; } // Do only code download from here on. if (!m_bAcceptSoftware || !((m_dwChannelFlags & CHANNEL_AGENT_PRECACHE_SOME) || (m_dwChannelFlags & CHANNEL_AGENT_PRECACHE_ALL)) ) { // No caching allowed, return immediately. bCleanUpNow = TRUE; goto Exit; } else { if (m_dwChannelFlags & CHANNEL_AGENT_PRECACHE_ALL) { m_dwMaxSizeKB = 0; } } } m_pCCDLAgentBSC = new CDLAgentBSC(this, m_dwMaxSizeKB, m_bSilentMode, m_szCDF); if (m_pCCDLAgentBSC == NULL) { hr = E_OUTOFMEMORY; goto Exit; } // attempt to use AsyncInstallDistributionUnit hr = CreateBindCtx(0, &pbc); if (FAILED(hr)) { goto Exit; } hr = RegisterBindStatusCallback(pbc, m_pCCDLAgentBSC, NULL, 0); if (FAILED(hr)) { goto Exit; } if (m_pSoftDistElement != NULL) { hr = m_pSoftDistExt->AsyncInstallDistributionUnit(pbc, NULL, 0, NULL); if (hr == S_OK) { SendUpdateNone(); } } else { CODEBASEHOLD *pcbh = new CODEBASEHOLD; if (pcbh == NULL) { hr = E_OUTOFMEMORY; goto Exit; } pcbh->cbSize = sizeof(CODEBASEHOLD); pcbh->szDistUnit = m_szDistUnit; pcbh->szCodeBase = m_szURL; pcbh->dwVersionMS = m_dwVersionMS; pcbh->dwVersionLS = m_dwVersionLS; pcbh->dwStyle = 0; // Since notification is likely from web crawler and we only support MSICD we // don't fire a notification back. hr = m_pSoftDistExt->AsyncInstallDistributionUnit(pbc, NULL, 0, pcbh); if (hr == S_OK) { SendUpdateNone(); } SAFEDELETE(pcbh); goto Exit; } if (hr != E_NOTIMPL) { // May have succeeded or failed, either way, we are out of here. goto Exit; } hr = m_pSoftDistExt->GetFirstCodeBase(&szCodeBase, &dwSize); if (SUCCEEDED(hr) && szCodeBase) { hr = StartNextDownload(szCodeBase,dwSize); SAFEDELETE(szCodeBase); } else { // no CODEBASE, return OK bCleanUpNow = TRUE; hr = S_OK; } Exit: // In case of SOFTDIST tag we work asychronously and send an END_REPORT back immediately. If we were called // to install a particular CAB then CleanUp is called by CDLABSC::OnStopBinding and report is sent back then. SAFERELEASE(pbc); if (FAILED(hr) || bCleanUpNow) { SetEndStatus(hr); CleanUp(); } return hr; } HRESULT CCDLAgent::StartNextDownload(LPWSTR wzCodeBase, DWORD dwSize) { HRESULT hr = E_FAIL; DWORD dwTemp = 0; ISubscriptionItem *pItem; if (m_dwMaxSizeKB && (dwSize > m_dwMaxSizeKB)) { hr = INET_E_AGENT_MAX_SIZE_EXCEEDED; goto Exit; } else { // Any other type of INSTALL protocol. // Send notification to WebCrawl agent to crawl the codebase. This should force it in the // case. Only do this if there is any chance the DL will not overflow the cache. // Note this will only download the CAB file and not any dependencies inside the CAB. They // should be included as separate CONFIG entries. if (m_dwMaxSizeKB && ((m_dwCurSize>>10) > m_dwMaxSizeKB)) { // We've exceeded our maximum download KB limit and can't continue. hr = INET_E_AGENT_MAX_SIZE_EXCEEDED; goto Exit; } if (FAILED(hr = DoCloneSubscriptionItem(m_pSubscriptionItem, NULL, &pItem)) || !pItem) { goto Exit; } dwTemp = DELIVERY_AGENT_FLAG_NO_BROADCAST; WriteDWORD(pItem, c_szPropAgentFlags, dwTemp); WriteOLESTR(pItem, c_szPropURL, wzCodeBase); if (m_dwMaxSizeKB) { // KB limit for us to pull. WriteDWORD(pItem, c_szPropCrawlMaxSize, m_dwMaxSizeKB - (m_dwCurSize>>10)); } WriteDWORD(pItem, c_szPropCrawlLevels, 0); m_dwCurSize += dwSize; m_pAgent = new CRunDeliveryAgent(); if (m_pAgent) hr = m_pAgent->Init((CRunDeliveryAgentSink *)this, pItem, CLSID_WebCrawlerAgent); pItem->Release(); if (m_pAgent && SUCCEEDED(hr)) { hr = m_pAgent->StartAgent(); if (hr == E_PENDING) { hr = S_OK; } else { DBG_WARN("StartNextDownload in CDL agent failed!"); hr = E_FAIL; } } else { hr = E_OUTOFMEMORY; } } Exit: return hr; } HRESULT CCDLAgent::OnAgentEnd(const SUBSCRIPTIONCOOKIE *pSubscriptionCookie, long lSizeDownloaded, HRESULT hrResult, LPCWSTR wszResult, BOOL fSynchronous) { HRESULT hr = S_OK; BOOL fDone = FALSE; LPWSTR wzCodeBase = NULL; DWORD dwSize; ASSERT(m_pAgent != NULL); if (fSynchronous) { // We must have failed. Let StartNextDownload return failure. return S_OK; } CRunDeliveryAgent::SafeRelease(m_pAgent); if (SUCCEEDED(hrResult)) { hr = m_pSoftDistExt->GetNextCodeBase(&wzCodeBase, &dwSize); if (SUCCEEDED(hr) && wzCodeBase) { hr = StartNextDownload(wzCodeBase, dwSize); SAFEDELETE(wzCodeBase); if (FAILED(hr)) { // we are done fDone = TRUE; } } else { // no more codebases to crawl hr = S_OK; fDone = TRUE; } } else { hr = hrResult; fDone = TRUE; } if (fDone) { SetEndStatus(hr); CleanUp(); } return hr; } void CCDLAgent::CleanUp() { if (m_pCCDLAgentBSC != NULL) { m_pCCDLAgentBSC->Release(); } m_pCCDLAgentBSC = NULL; CDeliveryAgent::CleanUp(); } void CCDLAgent::SetErrorEndText(LPCWSTR szErrorText) { if (szErrorText) m_szErrorText = SysAllocString(szErrorText); } HRESULT CCDLAgent::AgentAbort(DWORD dwFlags) { HRESULT hr = S_OK; if (m_pCCDLAgentBSC != NULL ) { hr = m_pCCDLAgentBSC->Abort(); } return hr; } HRESULT CCDLAgent::AgentPause(DWORD dwFlags) { HRESULT hr = S_OK; if (m_pCCDLAgentBSC != NULL ) { hr = m_pCCDLAgentBSC->Pause(); } return hr; } HRESULT CCDLAgent::AgentResume(DWORD dwFlags) { HRESULT hr = S_OK; if (m_pCCDLAgentBSC != NULL ) { hr = m_pCCDLAgentBSC->Resume(); } return hr; } HRESULT CCDLAgent::ModifyUpdateEnd(ISubscriptionItem *pEndItem, UINT *puiRes) { VARIANT vHref; ASSERT(pEndItem); // The END_REPORT is sent for both functionalities of CDL agent (SOFTDIST and Pull single CAB). // customize our end status string switch (GetEndStatus()) { case E_OUTOFMEMORY : *puiRes = IDS_AGNT_STATUS_SIZELIMIT; break; case E_FAIL : *puiRes = IDS_CRAWL_STATUS_NOT_OK; break; case S_FALSE : *puiRes = IDS_CRAWL_STATUS_UNCHANGED; break; case INET_S_AGENT_PART_FAIL : *puiRes = IDS_CRAWL_STATUS_MOSTLYOK; break; // This is actually a success code from URLMON case HRESULT_FROM_WIN32(ERROR_CANCELLED) : SetEndStatus(S_OK); *puiRes = IDS_CRAWL_STATUS_OK; break; case TRUST_E_FAIL : SetEndStatus(TRUST_E_SUBJECT_NOT_TRUSTED); case TRUST_E_SUBJECT_NOT_TRUSTED : case HRESULT_FROM_WIN32(ERROR_IO_INCOMPLETE) : SetEndStatus(S_OK); // fall through case S_OK : *puiRes = IDS_CRAWL_STATUS_OK; break; default : *puiRes = IDS_CRAWL_STATUS_NOT_OK; break; break; } // force gleam on this channel if we got S_OK on precaching bits if (SUCCEEDED(GetEndStatus()) && (GetEndStatus() != S_FALSE)) { WriteDWORD(pEndItem, c_szPropEnableShortcutGleam, 1); } // If we are sending email the status must be S_OK, we incorporate the error // message into the text body for reporting. if (m_bSendEmail) { VariantInit(&vHref); WriteDWORD(pEndItem, c_szPropEmailFlags, MAILAGENT_FLAG_CUSTOM_MSG); // This must exist or m_bSendEmail would never have been set in first place. GetXMLAttribute(m_pSoftDistElement, L"HREF", &vHref); WriteOLESTR(pEndItem, c_szPropURL, vHref.bstrVal); VariantClear(&vHref); if (m_sdi.szTitle) { BSTR bstrTitle = SysAllocString(m_sdi.szTitle); if (bstrTitle) WriteOLESTR(pEndItem, c_szPropEmailTitle, m_sdi.szTitle); SAFEFREEBSTR(bstrTitle); } if (FAILED(GetEndStatus()) && !m_szErrorText) { m_szErrorText = GetErrorMessage(GetEndStatus()); } if (m_sdi.szAbstract) { BSTR bstrAbstract = SysAllocString(m_sdi.szAbstract); if (bstrAbstract != NULL) { if (m_szErrorText) { //This is wrecking havoc with the email message, some resource strings //have a 'CR/LF' tacked on the end. We kill any that exist. DWORD dwLen = lstrlenW(m_szErrorText)-1; while (dwLen > 0 && (m_szErrorText[dwLen] == 0x0a || m_szErrorText[dwLen] == 0x0d || m_szErrorText[dwLen] == L'.')) { m_szErrorText[dwLen] = L'\0'; dwLen--; } // BUGBUG - needs cleanup! CHAR szPrefixMsg[MAX_PATH], szFormattedPrefixMsg[MAX_PATH*2]; if (MLLoadStringA(IDS_CDLAGENT_ERROR_EMAIL, szPrefixMsg, ARRAYSIZE(szPrefixMsg))>0) { LPWSTR wszNewAbstract = NULL; LPSTR szNewAbstract = NULL; wnsprintfA(szFormattedPrefixMsg, ARRAYSIZE(szFormattedPrefixMsg), szPrefixMsg, m_szErrorText); DWORD dwNewLen = lstrlenA(szFormattedPrefixMsg) + lstrlenW(bstrAbstract) + 4; szNewAbstract = (LPSTR)LocalAlloc(0,dwNewLen*sizeof(CHAR)); if (szNewAbstract) { wnsprintfA(szNewAbstract, dwNewLen*sizeof(CHAR), "%s%ws", szFormattedPrefixMsg, bstrAbstract); dwNewLen = lstrlenA(szNewAbstract) + 1; wszNewAbstract = (LPWSTR)LocalAlloc(0,dwNewLen*sizeof(WCHAR)); if (wszNewAbstract && (MultiByteToWideChar(CP_ACP, 0, szNewAbstract, -1, wszNewAbstract, dwNewLen)>0)) { SAFEFREEBSTR(bstrAbstract); bstrAbstract = SysAllocString(wszNewAbstract); } if (wszNewAbstract) LocalFree(wszNewAbstract); LocalFree(szNewAbstract); } } } WriteOLESTR(pEndItem, c_szPropEmailAbstract, bstrAbstract); SAFEFREEBSTR(bstrAbstract); } } // because user is notified of error we don't pass it on anywhere else SetEndStatus(S_OK); WriteSCODE(pEndItem, c_szPropStatusCode, S_OK); } ClearAgentFlag(DELIVERY_AGENT_FLAG_NO_BROADCAST); return CDeliveryAgent::ModifyUpdateEnd(pEndItem, puiRes); } LPWSTR CCDLAgent::GetErrorMessage(HRESULT hr) { LPSTR szBuf = NULL; LPWSTR wszBuf = NULL; DWORD dwLen; DWORD dwResource = 0; if (SUCCEEDED(hr)) return NULL; dwLen = FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM, NULL, hr, 0, (LPTSTR)&szBuf, 0, NULL); if (!dwLen) { // NOTE: If out of memory we return NULL. if (SUCCEEDED(hr)) dwResource = IDS_CDLAGENT_SUCCESS; else if (hr == TRUST_E_SUBJECT_NOT_TRUSTED) dwResource = IDS_CDLAGENT_TRUST_ERROR; else dwResource = IDS_CDLAGENT_FAILURE; // We know strings will fit into max_path WCHAR szTmp[MAX_PATH]; if (MLLoadStringW(dwResource, szTmp, MAX_PATH)>0) { wszBuf = SysAllocString(szTmp); } } else { WCHAR wszTemp[MAX_PATH]; if (MultiByteToWideChar(CP_ACP, 0, szBuf, -1, wszTemp, MAX_PATH)>0) { wszBuf = SysAllocString(wszTemp); } else wszBuf = NULL; SAFEDELETE(szBuf); } return wszBuf; }