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.
1165 lines
36 KiB
1165 lines
36 KiB
// HistoryParser.cpp: implementation of the CHistoryParser class.
|
|
//
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
#include "stdafx.h"
|
|
#include "resource.h"
|
|
#include "HistoryParser.h"
|
|
#include "Filestuff.h"
|
|
|
|
|
|
#ifdef _DEBUG
|
|
#undef THIS_FILE
|
|
static char THIS_FILE[]=__FILE__;
|
|
#define new DEBUG_NEW
|
|
#endif
|
|
|
|
|
|
|
|
extern CMSInfoHistoryCategory catHistorySystemSummary;
|
|
extern CMSInfoHistoryCategory catHistoryResources;
|
|
extern CMSInfoHistoryCategory catHistoryComponents;
|
|
extern CMSInfoHistoryCategory catHistorySWEnv;
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// Construction/Destruction
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
CHistoryParser::CHistoryParser(CComPtr<IXMLDOMDocument> pDoc) : m_pDoc(pDoc)
|
|
{
|
|
|
|
}
|
|
|
|
void CHistoryParser::DeleteAllInstances()
|
|
{
|
|
for(POSITION pos = this->m_listInstances.GetHeadPosition();pos;)
|
|
{
|
|
if (!pos)
|
|
{
|
|
return;
|
|
}
|
|
CInstance* pInci = (CInstance*) m_listInstances.GetNext(pos);
|
|
delete pInci;
|
|
}
|
|
m_listInstances.RemoveAll();
|
|
}
|
|
|
|
CHistoryParser::~CHistoryParser()
|
|
{
|
|
DeleteAllInstances();
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// Construction/Destruction
|
|
// takes a CTime, which comes from timestamp element of the Delta or Snaphot
|
|
// node of which pInstanceNode is a child; an Instance node, and a string
|
|
// containing the WMI class of the Instance
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
CInstance::CInstance(CTime tmstmp, CComPtr<IXMLDOMNode> pInstanceNode,CString strClass) : m_tmstamp(tmstmp), m_strClassName(strClass)
|
|
{
|
|
CComPtr<IXMLDOMNodeList> pPropList;
|
|
HRESULT hr;
|
|
//Get node data, add each PROPERTY name and VALUE to m_mapNameValue
|
|
if (strClass.CompareNoCase(_T("Win32_PNPAllocatedResource")) == 0)
|
|
{
|
|
hr = ProcessPNPAllocatedResource(pInstanceNode);
|
|
ASSERT(SUCCEEDED(hr) && "failed to process Win32_PNPAllocatedResource");
|
|
return;
|
|
}
|
|
hr = pInstanceNode->selectNodes(CComBSTR("PROPERTY"),&pPropList);
|
|
if (FAILED(hr) || !pPropList)
|
|
{
|
|
ASSERT(0 && "could not get property list from Instance node");
|
|
return;
|
|
}
|
|
long lListLen;
|
|
hr = pPropList->get_length(&lListLen);
|
|
CComPtr<IXMLDOMNode> pVNode;
|
|
CComBSTR bstrValue;
|
|
CComVariant varName;
|
|
for(long i = 0; i < lListLen; i++)
|
|
{
|
|
hr = pPropList->nextNode(&pVNode);
|
|
if (FAILED(hr) || !pVNode)
|
|
{
|
|
return;
|
|
}
|
|
CComPtr<IXMLDOMElement> pElement;
|
|
hr = pVNode->QueryInterface(IID_IXMLDOMElement,(void**) &pElement);
|
|
if (FAILED(hr) || !pElement)
|
|
{
|
|
return;
|
|
}
|
|
hr = pElement->getAttribute(L"NAME",&varName);
|
|
ASSERT(SUCCEEDED(hr));
|
|
hr = pVNode->get_text(&bstrValue);
|
|
ASSERT(SUCCEEDED(hr));
|
|
USES_CONVERSION;
|
|
m_mapNameValue.SetAt(OLE2T(varName.bstrVal) ,OLE2T(bstrValue));
|
|
pVNode.Release();
|
|
}
|
|
pPropList.Release();
|
|
return;
|
|
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//Refresh is called for selected category when category selection or delta range changes
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
HRESULT CHistoryParser::Refresh(CMSInfoHistoryCategory* pHistCat,int nDeltasBack)
|
|
{
|
|
nDeltasBack++;
|
|
this->m_fChangeLines = FALSE;// v-stlowe 2/28/2001
|
|
DeleteAllInstances();
|
|
m_pHistCat = pHistCat;
|
|
CComPtr<IXMLDOMNodeList> pDeltaList;
|
|
HRESULT hr;
|
|
hr = this->GetDeltaAndSnapshotNodes(pDeltaList);
|
|
if (FAILED(hr) || !pDeltaList)
|
|
{
|
|
return E_FAIL;
|
|
}
|
|
|
|
if (pHistCat == &catHistoryComponents)
|
|
{
|
|
DeleteAllInstances();
|
|
|
|
hr = ProcessDeltas(pDeltaList,"Win32_DriverVXD",nDeltasBack);
|
|
ASSERT(SUCCEEDED(hr));
|
|
DeleteAllInstances();
|
|
pDeltaList->reset();
|
|
hr = ProcessDeltas(pDeltaList,"Win32_CodecFile",nDeltasBack);
|
|
ASSERT(SUCCEEDED(hr));
|
|
DeleteAllInstances();
|
|
pDeltaList->reset();
|
|
hr = ProcessDeltas(pDeltaList,"Win32_LogicalDisk",nDeltasBack);
|
|
DeleteAllInstances();
|
|
pDeltaList->reset();
|
|
ASSERT(SUCCEEDED(hr));
|
|
hr = ProcessDeltas(pDeltaList,"Win32_NetworkProtocol",nDeltasBack);
|
|
DeleteAllInstances();
|
|
pDeltaList->reset();
|
|
ASSERT(SUCCEEDED(hr));
|
|
hr = ProcessDeltas(pDeltaList,"Win32_Printer",nDeltasBack);
|
|
DeleteAllInstances();
|
|
pDeltaList->reset();
|
|
ASSERT(SUCCEEDED(hr));
|
|
hr = ProcessDeltas(pDeltaList,"Win32_PortResource",nDeltasBack);
|
|
DeleteAllInstances();
|
|
pDeltaList->reset();
|
|
ASSERT(SUCCEEDED(hr));
|
|
hr = ProcessDeltas(pDeltaList,"Win32_PnPEntity",nDeltasBack);
|
|
DeleteAllInstances();
|
|
pDeltaList->reset();
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
else if (pHistCat == &catHistorySystemSummary)
|
|
{
|
|
DeleteAllInstances();
|
|
hr = ProcessDeltas(pDeltaList,"Win32_ComputerSystem",nDeltasBack);
|
|
DeleteAllInstances();
|
|
pDeltaList->reset();
|
|
ASSERT(SUCCEEDED(hr));
|
|
hr = ProcessDeltas(pDeltaList,"Win32_OperatingSystem",nDeltasBack);
|
|
DeleteAllInstances();
|
|
pDeltaList->reset();
|
|
ASSERT(SUCCEEDED(hr));
|
|
//hr = ProcessDeltas(pDeltaList,"Win32_Win32_LogicalMemoryConfiguration",nDeltasBack);
|
|
hr = ProcessDeltas(pDeltaList,"Win32_LogicalMemoryConfiguration",nDeltasBack); //v-stlowe 2/28/2001
|
|
DeleteAllInstances();
|
|
pDeltaList->reset();
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
else if(pHistCat == &catHistoryResources)
|
|
{
|
|
DeleteAllInstances();
|
|
hr = ProcessDeltas(pDeltaList,"Win32_PNPAllocatedResource",nDeltasBack);
|
|
DeleteAllInstances();
|
|
pDeltaList->reset();
|
|
ASSERT(SUCCEEDED(hr));
|
|
}
|
|
else if (pHistCat == &catHistorySWEnv)
|
|
{
|
|
DeleteAllInstances();
|
|
hr = ProcessDeltas(pDeltaList,"Win32_ProgramGroup",nDeltasBack);
|
|
DeleteAllInstances();
|
|
pDeltaList->reset();
|
|
hr = ProcessDeltas(pDeltaList,"Win32_StartupCommand",nDeltasBack);
|
|
DeleteAllInstances();
|
|
pDeltaList->reset();
|
|
|
|
}
|
|
if (!m_fChangeLines)
|
|
{
|
|
#ifdef A_STEPHL2
|
|
::MessageBox(NULL,"!m_fChangeLines)","",MB_OK);
|
|
#endif
|
|
m_fChangeLines = TRUE;
|
|
CString strMSG;
|
|
strMSG.LoadString(IDS_DELTANOCHANGES);//this would be the place to change messaging for situation where summary has no changes
|
|
m_pHistCat->InsertLine(-1, strMSG, _T(""), _T(""), _T(""));
|
|
}
|
|
pDeltaList.Release();
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//Gets the value appropriate to use as a description for the class
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
CString CInstance::GetInstanceDescription()
|
|
{
|
|
CString strDescName = GetDescriptionForClass(m_strClassName);
|
|
CString strInstDesc;
|
|
VERIFY(m_mapNameValue.Lookup(strDescName,strInstDesc));
|
|
return strInstDesc;
|
|
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//Gets the value that can be used to uniquely identify a specific instance of a class
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
CString CInstance::GetInstanceID()
|
|
{
|
|
CString strIDName = GetIDForClass(m_strClassName);
|
|
CString strInstID;
|
|
VERIFY(m_mapNameValue.Lookup(strIDName,strInstID));
|
|
return strInstID;
|
|
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//used to deal with antecedent\dependant relationship classes in Win32_PNPAllocatedResource classes
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
HRESULT CInstance::ProcessPropertyDotReferenceNodes(CComPtr<IXMLDOMNode> pInstanceNameNode,CString* pstrClassName, CString* pstrKeyName,CString* pstrKeyValue)
|
|
{
|
|
USES_CONVERSION;
|
|
HRESULT hr;
|
|
CComPtr<IXMLDOMElement> pNameElement;
|
|
hr = pInstanceNameNode->QueryInterface(IID_IXMLDOMElement,(void**) &pNameElement);
|
|
if (FAILED(hr) | !pNameElement)
|
|
{
|
|
ASSERT(0 && "could not QI pNode for Element");
|
|
return E_FAIL;
|
|
}
|
|
CComVariant varClassName;
|
|
hr = pNameElement->getAttribute(L"CLASSNAME",&varClassName);
|
|
pNameElement.Release();
|
|
if (FAILED(hr))
|
|
{
|
|
ASSERT(0 && "could not get CLASSNAME element");
|
|
}
|
|
*pstrClassName = OLE2T(varClassName.bstrVal);
|
|
CComPtr<IXMLDOMNode> pKeybindingNode;
|
|
hr = pInstanceNameNode->selectSingleNode(CComBSTR("KEYBINDING"),&pKeybindingNode);
|
|
if (FAILED(hr) || !pKeybindingNode)
|
|
{
|
|
ASSERT(0 && "could not get antecedent node");
|
|
}
|
|
CComBSTR bstrKeyValue;
|
|
hr = pKeybindingNode->get_text(&bstrKeyValue);
|
|
ASSERT(SUCCEEDED(hr) && "failed to get keybinding value");
|
|
*pstrKeyValue = OLE2T(bstrKeyValue);
|
|
hr = pKeybindingNode->QueryInterface(IID_IXMLDOMElement,(void**) &pNameElement);
|
|
if (FAILED(hr) | !pNameElement)
|
|
{
|
|
ASSERT(0 && "could not QI pNode for Element");
|
|
return E_FAIL;
|
|
}
|
|
CComVariant varKeybindingName;
|
|
hr = pNameElement->getAttribute(CComBSTR("NAME"),&varKeybindingName);
|
|
if (FAILED(hr))
|
|
{
|
|
ASSERT(0 && "could not get NAME attribute from pNameElement");
|
|
}
|
|
|
|
*pstrKeyName = OLE2T(varKeybindingName.bstrVal);
|
|
return hr;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//used to deal with antecedent\dependant relationship classes in Win32_PNPAllocatedResource classes
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
HRESULT CInstance::ProcessPNPAllocatedResource(CComPtr<IXMLDOMNode> pInstanceNode)
|
|
{
|
|
|
|
HRESULT hr;
|
|
CComPtr<IXMLDOMNodeList> pPropDotRefList;
|
|
hr = pInstanceNode->selectNodes(CComBSTR("PROPERTY.REFERENCE/VALUE.REFERENCE/INSTANCEPATH/INSTANCENAME"),&pPropDotRefList);
|
|
if (FAILED(hr) || !pPropDotRefList)
|
|
{
|
|
ASSERT(0 && "PROPERTY.REFERENCE nodes not found");
|
|
return E_FAIL;
|
|
}
|
|
|
|
//get antecedent node
|
|
CComPtr<IXMLDOMNode> pInstanceNameNode;
|
|
hr = pPropDotRefList->nextNode(&pInstanceNameNode);
|
|
if (FAILED(hr) || !pInstanceNameNode)
|
|
{
|
|
ASSERT(0 && "could not get antecedent node");
|
|
}
|
|
CString strAntecedentName,strResourceName,strResourceValue;
|
|
hr = ProcessPropertyDotReferenceNodes(pInstanceNameNode,&strAntecedentName,&strResourceName,&strResourceValue);
|
|
m_mapNameValue.SetAt(_T("ANTECEDENT"),strAntecedentName);
|
|
m_mapNameValue.SetAt(strResourceName,strResourceValue);
|
|
if (FAILED(hr))
|
|
{
|
|
return hr;
|
|
}
|
|
CString strPNPEntity,strKeyname,strDeviceIDval;
|
|
pInstanceNameNode.Release();
|
|
hr = pPropDotRefList->nextNode(&pInstanceNameNode);
|
|
if (FAILED(hr) || !pInstanceNameNode)
|
|
{
|
|
return hr;
|
|
}
|
|
hr = ProcessPropertyDotReferenceNodes(pInstanceNameNode,&strPNPEntity,&strKeyname,&strDeviceIDval);
|
|
CComPtr<IXMLDOMDocument> pDoc;
|
|
hr = pInstanceNode->get_ownerDocument(&pDoc);
|
|
if (FAILED(hr) || !pDoc)
|
|
{
|
|
ASSERT(0 && "could not get owner doc from pInstanceNode");
|
|
return E_FAIL;
|
|
}
|
|
CString strPNPDeviceName = GetPNPNameByID(pDoc,CComBSTR(strDeviceIDval));
|
|
if (FAILED(hr))
|
|
{
|
|
return hr;
|
|
}
|
|
ASSERT(strPNPEntity.CompareNoCase("Win32_PnPEntity") == 0 && "unexpected value for Dependent classname");
|
|
ASSERT(strKeyname.CompareNoCase("DeviceID") == 0 && "unexpected value for Dependent Keybinding name");
|
|
//we will create an arificial attribute "ASSOCNAME", which will be used to identify this device.
|
|
m_mapNameValue.SetAt(_T("ASSOCNAME"),strAntecedentName + ":" + strDeviceIDval);
|
|
m_mapNameValue.SetAt(_T("DeviceID"),strDeviceIDval);
|
|
m_mapNameValue.SetAt(_T("DeviceName"),strPNPDeviceName);
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//Retrives a value used to select appropriate description value for the class
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
CString CInstance::GetDescriptionForClass(CString strClass)
|
|
{
|
|
//lookup a key which can uniquely identify an instance of a given class
|
|
//for example, DeviceID for Printers
|
|
if (strClass.CompareNoCase(_T("Win32_LogicalDisk")) == 0)
|
|
{
|
|
return "DeviceID";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_CodecFile")) == 0)
|
|
{
|
|
return "Description";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_ComputerSystem")) == 0)
|
|
{
|
|
return "Name";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_OperatingSystem")) == 0)
|
|
{
|
|
return "Caption";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_LogicalMemoryConfiguration")) == 0)
|
|
{
|
|
return "TotalPhysicalMemory";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_PortResource")) == 0)
|
|
{
|
|
return "Name";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_NetworkProtocol")) == 0)
|
|
{
|
|
return "Name";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_Printer")) == 0)
|
|
{
|
|
return "DeviceID";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_PnPEntity")) == 0)
|
|
{
|
|
return "Description";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_StartupCommand")) == 0)
|
|
{
|
|
return "Command";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_ProgramGroup")) == 0)
|
|
{
|
|
return "GroupName";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_PNPAllocatedResource")) == 0)
|
|
{
|
|
//this is an artificial string created in CInstance::ProcessPNPAllocatedResource
|
|
return "DeviceName";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_DriverVXD")) == 0)
|
|
{
|
|
return "Name";
|
|
}
|
|
|
|
return "";
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//used to determine which mapped value to use to ID instances of the clas
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
CString CInstance::GetIDForClass(CString strClass)
|
|
{
|
|
//lookup a key which can uniquely identify an instance of a given class
|
|
//for example, DeviceID for Printers
|
|
if (strClass.CompareNoCase(_T("Win32_LogicalDisk")) == 0)
|
|
{
|
|
return "DeviceID";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_CodecFile")) == 0)
|
|
{
|
|
return "Description";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_OperatingSystem")) == 0)
|
|
{
|
|
return "Caption";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_LogicalMemoryConfiguration")) == 0)
|
|
{
|
|
return "TotalPhysicalMemory";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_ComputerSystem")) == 0)
|
|
{
|
|
return "Name";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_PortResource")) == 0)
|
|
{
|
|
return "Name";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_NetworkProtocol")) == 0)
|
|
{
|
|
return "Name";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_Printer")) == 0)
|
|
{
|
|
return "DeviceID";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_PnPEntity")) == 0)
|
|
{
|
|
return "DeviceID";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_PNPAllocatedResource")) == 0)
|
|
{
|
|
//this is an artificial string created in CInstance::ProcessPNPAllocatedResource
|
|
return "ASSOCNAME";
|
|
}
|
|
|
|
if (strClass.CompareNoCase(_T("Win32_ProgramGroup")) == 0)
|
|
{
|
|
return "GroupName";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_StartupCommand")) == 0)
|
|
{
|
|
return "Command";
|
|
}
|
|
if (strClass.CompareNoCase(_T("Win32_DriverVXD")) == 0)
|
|
{
|
|
return "Name";
|
|
}
|
|
|
|
return "";
|
|
}
|
|
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//used when rolling back through history list, to find previous instance of a given class
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
CInstance* CHistoryParser::FindPreviousInstance(CInstance* pNewInstance)
|
|
{
|
|
//for each existing instance pOld
|
|
for(POSITION pos = m_listInstances.GetHeadPosition( );;)
|
|
{
|
|
if (!pos)
|
|
{
|
|
return NULL;
|
|
}
|
|
CInstance* pOld = (CInstance*) m_listInstances.GetNext(pos);
|
|
if (pOld->GetClassName() == pNewInstance->GetClassName())
|
|
{
|
|
if (pOld->GetInstanceID() == pNewInstance->GetInstanceID())
|
|
{
|
|
return pOld;
|
|
}
|
|
}
|
|
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
void CHistoryParser::CreateChangeStrings(CInstance* pOld, CInstance* pNew)
|
|
{
|
|
CTimeSpan tmsDelta;
|
|
COleDateTime olTime(pNew->m_tmstamp.GetTime());
|
|
if (!pOld )
|
|
{
|
|
ASSERT(pNew );
|
|
tmsDelta = CTime::GetCurrentTime() - pNew->m_tmstamp;
|
|
//change string should be "Delete"
|
|
m_pHistCat->InsertRemoveLine(pNew->m_tmstamp ,pNew->GetClassFriendlyName(),pNew->GetInstanceDescription());
|
|
|
|
m_fChangeLines = TRUE;
|
|
return;
|
|
}
|
|
else if (!pNew)
|
|
{
|
|
ASSERT(pOld);
|
|
tmsDelta = CTime::GetCurrentTime() - pOld->m_tmstamp;
|
|
//change string should be "New"
|
|
m_pHistCat->InsertAddLine(pNew->m_tmstamp,pOld->GetClassFriendlyName(),pOld->GetInstanceDescription());
|
|
//v-stlowe 3/12/2001
|
|
m_fChangeLines = TRUE;
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
|
|
ASSERT(pOld && pNew && "both pointers can't be null");
|
|
tmsDelta = CTime::GetCurrentTime() - pNew->m_tmstamp;
|
|
//for each Name&Value pair, get the name, and then use it to examine
|
|
//the associated value in pCompare's map
|
|
CString strName, strValue,strCompareValue;
|
|
|
|
if (pNew->GetChangeType().CompareNoCase(_T("New")) == 0)
|
|
{
|
|
tmsDelta = CTime::GetCurrentTime() - pNew->m_tmstamp;
|
|
//change string should be "added"
|
|
m_pHistCat->InsertAddLine(pNew->m_tmstamp ,pNew->GetClassFriendlyName(),pNew->GetInstanceDescription());
|
|
m_fChangeLines = TRUE;
|
|
return;
|
|
}
|
|
else if (pNew->GetChangeType().CompareNoCase(_T("Delete")) == 0)
|
|
{
|
|
tmsDelta = CTime::GetCurrentTime() - pNew->m_tmstamp;
|
|
//change string should be "Deleted"
|
|
m_pHistCat->InsertRemoveLine(pNew->m_tmstamp,pNew->GetClassFriendlyName(),pNew->GetInstanceDescription());
|
|
m_fChangeLines = TRUE;
|
|
return;
|
|
}
|
|
|
|
for(POSITION pos = pNew->m_mapNameValue.GetStartPosition();;pNew->m_mapNameValue.GetNextAssoc(pos,strName, strValue))
|
|
{
|
|
strCompareValue = _T("");
|
|
if (!pOld->m_mapNameValue.Lookup(strName,strCompareValue))
|
|
{
|
|
//ASSERT(0 && "value not found in delta");
|
|
//return E_FAIL;
|
|
if (strName.CompareNoCase(_T("Change")) == 0)
|
|
{
|
|
VERIFY(pNew->m_mapNameValue.Lookup(strName,strCompareValue));
|
|
if (strCompareValue.CompareNoCase(_T("New")) == 0)
|
|
{
|
|
m_pHistCat->InsertAddLine(pNew->m_tmstamp,pNew->GetClassFriendlyName(),pNew->GetInstanceDescription());
|
|
m_fChangeLines = TRUE;
|
|
}
|
|
ASSERT(1);
|
|
}
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
pOld->m_mapNameValue.RemoveKey(strName);
|
|
}
|
|
|
|
if (strValue != strCompareValue)
|
|
{
|
|
|
|
m_pHistCat->InsertChangeLine(pNew->m_tmstamp,pNew->GetClassFriendlyName(),pNew->GetInstanceDescription(),strName,strValue,strCompareValue);
|
|
m_fChangeLines = TRUE;
|
|
|
|
}
|
|
if(!pos)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
//handle values that are mapOldInstance, and not the other map
|
|
if (!pOld->m_mapNameValue.IsEmpty())
|
|
{
|
|
for(pos = pOld->m_mapNameValue.GetStartPosition();;pOld->m_mapNameValue.GetNextAssoc(pos,strName, strValue))
|
|
{
|
|
pOld->m_mapNameValue.GetNextAssoc(pos,strName, strValue);
|
|
pNew->m_mapNameValue.SetAt(strName,strValue);
|
|
if (!pos)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
}
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//once the previous instance has been processed, previous instance should be removed and this instance should be added to list
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
void CHistoryParser::ResetInstance(CInstance* pOld, CInstance* pNew)
|
|
{
|
|
POSITION pos = this->m_listInstances.Find(pOld);
|
|
m_listInstances.SetAt(pos,pNew);
|
|
delete pOld;
|
|
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//Used to process a single instance from either history or snapshot
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
void CHistoryParser::ProcessInstance(CInstance* pNewInstance)
|
|
{
|
|
//see if instance is in list of instances
|
|
CInstance* pOld = FindPreviousInstance(pNewInstance);
|
|
if (pOld)
|
|
{
|
|
CreateChangeStrings(pOld,pNewInstance);
|
|
ResetInstance(pOld,pNewInstance);
|
|
}
|
|
//if this is from a Snapshot, just add it
|
|
//if it is from a Delta, it should have a change type of "add", and we
|
|
//want to create a change string for it.
|
|
else
|
|
{
|
|
CString strChange;
|
|
if (pNewInstance->GetValueFromMap(_T("Change"),strChange))
|
|
{
|
|
//we have new Delta instance
|
|
CreateChangeStrings(NULL,pNewInstance);
|
|
m_listInstances.AddTail(pNewInstance);
|
|
}
|
|
else
|
|
{
|
|
//Instance is in snapshot, so we don't generate change lines
|
|
m_listInstances.AddTail(pNewInstance);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**************************************************************************
|
|
returns list of deltas and the snapshot node
|
|
/**************************************************************************/
|
|
HRESULT CHistoryParser::GetDeltaAndSnapshotNodes(CComPtr<IXMLDOMNodeList>& pDeltaList)
|
|
{
|
|
CComPtr<IXMLDOMNode> pDataCollNode;
|
|
HRESULT hr;
|
|
hr = GetDataCollectionNode(m_pDoc,pDataCollNode);
|
|
if (FAILED(hr) || !pDataCollNode)
|
|
{
|
|
//ASSERT(0 && "could not get datacollection node");
|
|
return E_FAIL;
|
|
}
|
|
//all nodes directly under DATACOLLECTION should be either deltas or the snapshot
|
|
hr = pDataCollNode->selectNodes(CComBSTR("*"),&pDeltaList);
|
|
if (FAILED(hr) || !pDeltaList)
|
|
{
|
|
ASSERT(0 && "could not get pDeltaList");
|
|
return E_FAIL;
|
|
}
|
|
#ifndef _DEBUG
|
|
return hr;
|
|
#endif
|
|
long ll;
|
|
hr = pDeltaList->get_length(&ll);
|
|
return hr;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//gets IXMLDOMNodeList of instances from a specific delta or snapshot node
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
HRESULT CHistoryParser::GetInstanceNodeList(CString strClass,CComPtr<IXMLDOMNode> pDeltaNode, CComPtr<IXMLDOMNodeList>& pInstanceList)
|
|
{
|
|
HRESULT hr;
|
|
//CComBSTR bstrQuery;
|
|
// the query will have to be in the form:
|
|
// CIM/DECLARATION/DECLGROUP.WITHPATH/VALUE.OBJECTWITHPATH/INSTANCE[@CLASSNAME $ieq$ "WIN32_CODECFILE"]
|
|
// or
|
|
// CIM/DECLARATION/DECLGROUP.WITHPATH/VALUE.OBJECTWITHPATH/INSTANCE[@CLASSNAME $ieq$ "Win32_ComputerSystem"]
|
|
// because we are querying a node, rather than a document (with which we could get
|
|
// away with specifying only INSTANCE in the query
|
|
|
|
//v-stlowe 1/29/2001 to fix Prefix whistler bug #279519
|
|
//bstrQuery += "CIM/DECLARATION/DECLGROUP.WITHPATH/VALUE.OBJECTWITHPATH/INSTANCE[@CLASSNAME $ieq$ ";
|
|
CComBSTR bstrQuery("CIM/DECLARATION/DECLGROUP.WITHPATH/VALUE.OBJECTWITHPATH/INSTANCE[@CLASSNAME $ieq$ ");
|
|
|
|
//end v-stlowe
|
|
bstrQuery += "\"";
|
|
bstrQuery += CComBSTR(strClass);
|
|
bstrQuery += "\"]";
|
|
hr = pDeltaNode->selectNodes(bstrQuery,&pInstanceList);
|
|
if (FAILED(hr) || !pInstanceList)
|
|
{
|
|
ASSERT(0 && "Could not get node list");
|
|
return E_FAIL;
|
|
}
|
|
|
|
if (FAILED(hr))
|
|
{
|
|
ASSERT(0 && "Could not get node list length");
|
|
|
|
}
|
|
|
|
return hr;
|
|
}
|
|
|
|
|
|
|
|
//for a given Snapshot or Delta node, get all instances of a given class
|
|
HRESULT CHistoryParser::ProcessDeltaNode(CComPtr<IXMLDOMNode> pDeltaNode,CString strClass)
|
|
{
|
|
CString strTime;
|
|
HRESULT hr;
|
|
int nTimeZone;
|
|
hr = GetTimeStampFromFromD_or_SNodeNode(pDeltaNode, &strTime,nTimeZone);
|
|
ASSERT(SUCCEEDED(hr) && "error getting timestamp for node");
|
|
CTime tmDelta = GetDateFromString(strTime,nTimeZone);
|
|
//TD: check for valid time range...
|
|
//get list of all nodes of given class
|
|
CComPtr<IXMLDOMNodeList> pInstanceNodeList;
|
|
hr = GetInstanceNodeList(strClass,pDeltaNode,pInstanceNodeList);
|
|
if (FAILED(hr) | ! pInstanceNodeList)
|
|
{
|
|
ASSERT(0 && "could not get instance list from Delta node");
|
|
return E_FAIL;
|
|
}
|
|
//step through list, getting each instance
|
|
long lListLen;
|
|
hr = pInstanceNodeList->get_length(&lListLen);
|
|
for(long i = 0;i < lListLen;i++)
|
|
{
|
|
CComPtr<IXMLDOMNode> pInstanceNode;
|
|
hr = pInstanceNodeList->nextNode(&pInstanceNode);
|
|
if (FAILED(hr) || ! pInstanceNode)
|
|
{
|
|
ASSERT(0 && "could not get node from instance list");
|
|
return E_FAIL;
|
|
}
|
|
CInstance * pInstance = new CInstance(tmDelta,pInstanceNode,strClass);
|
|
ProcessInstance(pInstance);
|
|
}
|
|
return hr;
|
|
}
|
|
|
|
|
|
//*************************************************************************
|
|
//Takes a list of delta nodes, and the name of a class
|
|
//**************************************************************************
|
|
|
|
HRESULT CHistoryParser::ProcessDeltas(CComPtr<IXMLDOMNodeList> pDeltaList,CString strClassName,int nDeltasBack)
|
|
{
|
|
//for each node in list pNode
|
|
long lListLen;
|
|
HRESULT hr;
|
|
hr = pDeltaList->get_length(&lListLen);
|
|
if (FAILED(hr))
|
|
{
|
|
ASSERT(0 && "couldn't get list length");
|
|
}
|
|
if (0 == lListLen)
|
|
{
|
|
pDeltaList.Release();
|
|
return S_FALSE;
|
|
}
|
|
|
|
for (long i = 0;i < lListLen && i <= nDeltasBack;i++)
|
|
{
|
|
CComPtr<IXMLDOMNode> pNode;
|
|
hr= pDeltaList->nextNode(&pNode);
|
|
if (FAILED(hr) || !pNode)
|
|
{
|
|
ASSERT(0 && "could not get next delta node");
|
|
pDeltaList.Release();
|
|
return E_FAIL;
|
|
}
|
|
|
|
|
|
// here's problem If we're using nDeltasBack method, do we need to compare dates?
|
|
/* CTime tmDelta = GetDeltaTime(pNode);
|
|
if (GetDeltaTime(pNode) >= this->m_tmBack)
|
|
{
|
|
*/
|
|
hr = ProcessDeltaNode(pNode,strClassName);
|
|
if (FAILED(hr))
|
|
{
|
|
pDeltaList.Release();
|
|
return hr;
|
|
}
|
|
// }
|
|
}
|
|
pDeltaList.Release();
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
//*************************************************************************
|
|
//Gets the DATACOLLECTION node, beneath which both the SNAPSHOT and the DELTA nodes reside
|
|
//**************************************************************************
|
|
|
|
|
|
HRESULT GetDataCollectionNode(CComPtr<IXMLDOMDocument> pXMLDoc,CComPtr<IXMLDOMNode>& pDCNode)
|
|
{
|
|
//TD: find a way to do case-insensitive queries.
|
|
HRESULT hr;
|
|
if (!pXMLDoc)
|
|
{
|
|
return S_FALSE;
|
|
}
|
|
CComPtr<IXMLDOMNodeList> pNodeList;
|
|
|
|
//find a change property; that way we know we have a delta
|
|
hr = pXMLDoc->getElementsByTagName(CComBSTR("PROPERTY[@NAME $ieq$ \"CHANGE\"]"),&pNodeList);
|
|
if (FAILED(hr) || !pNodeList)
|
|
{
|
|
ASSERT(0 && "Could not get node list");
|
|
return E_FAIL;
|
|
}
|
|
CComPtr<IXMLDOMNode> pNode;
|
|
hr = pNodeList->nextNode(&pNode);
|
|
if (FAILED(hr) || !pNode)
|
|
{
|
|
// ASSERT(0 && "Could not get node from node list");
|
|
return E_FAIL;
|
|
}
|
|
//loop till we get a node called "DATACOLLECTION"
|
|
CComPtr<IXMLDOMNode> pParentNode;
|
|
for(int i = 0;;i++)
|
|
{
|
|
hr = pNode->get_parentNode(&pParentNode);
|
|
if (FAILED(hr) || !pParentNode)
|
|
{
|
|
ASSERT(0 && "Could not find DATACOLLECTION node");
|
|
pDCNode = NULL;
|
|
return E_FAIL;
|
|
}
|
|
pNode.Release();
|
|
CComBSTR bstrName;
|
|
pParentNode->get_nodeName(&bstrName);
|
|
USES_CONVERSION;
|
|
if (CString(bstrName).CompareNoCase(_T("DATACOLLECTION")) == 0)
|
|
{
|
|
pDCNode = pParentNode;
|
|
return S_OK;
|
|
}
|
|
pNode = pParentNode;
|
|
pParentNode.Release();
|
|
}
|
|
|
|
return S_OK;
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//get timestamp of a delta or snapshot node
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
CTime GetDeltaTime(CComPtr<IXMLDOMNode> pDorSNode)
|
|
{
|
|
CString strTime;
|
|
int nTimeZone;
|
|
GetTimeStampFromFromD_or_SNodeNode(pDorSNode,&strTime,nTimeZone);
|
|
return GetDateFromString(strTime,nTimeZone);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//takes string format used in XML blob, creates a CTime
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
HRESULT GetTimeStampFromFromD_or_SNodeNode(CComPtr<IXMLDOMNode> pDorSNode,CString* pString, int& nTimeZone)
|
|
{
|
|
HRESULT hr;
|
|
CComVariant varTS;
|
|
CComPtr<IXMLDOMElement> pTimestampElement;
|
|
hr = pDorSNode->QueryInterface(IID_IXMLDOMElement,(void**) &pTimestampElement);
|
|
if (FAILED(hr) || !pTimestampElement)
|
|
{
|
|
ASSERT(0 && "could not get attribute element");
|
|
}
|
|
hr = pTimestampElement->getAttribute(L"Timestamp_T0",&varTS);
|
|
if (FAILED(hr) )
|
|
{
|
|
ASSERT(0 && "could not get timestamp value from attribute");
|
|
}
|
|
if (1 == hr)
|
|
{
|
|
//this may be snapshot node...try "Timestamp"
|
|
hr = pTimestampElement->getAttribute(L"Timestamp",&varTS);
|
|
if (FAILED(hr) )
|
|
{
|
|
ASSERT(0 && "could not get timestamp value from attribute");
|
|
}
|
|
}
|
|
CComVariant varTzoneDeltaSeconds;
|
|
hr = pTimestampElement->getAttribute(L"TimeZone",&varTzoneDeltaSeconds);
|
|
if (FAILED(hr) ) //this will happen when loading WinME xml, which has no timezone info
|
|
{
|
|
varTzoneDeltaSeconds = 0;
|
|
}
|
|
//make sure we have an integer type
|
|
hr = varTzoneDeltaSeconds.ChangeType(VT_INT);
|
|
if (FAILED(hr) )
|
|
{
|
|
varTzoneDeltaSeconds = 0;
|
|
}
|
|
nTimeZone = varTzoneDeltaSeconds.intVal;
|
|
USES_CONVERSION;
|
|
pTimestampElement.Release();
|
|
*pString = OLE2T(varTS.bstrVal);
|
|
return hr;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// utility functions
|
|
//////////////////////////////////////////////////////////////////////
|
|
CTime GetDateFromString(const CString& strDate, int nTimeZone)
|
|
{
|
|
//requires linking to Shlwapi.lib
|
|
CString strDateCopy(strDate);
|
|
CString strDateSegment;
|
|
|
|
//year is the 4 leftmost digits of date string
|
|
strDateSegment = strDateCopy.Left(4);
|
|
int nYear;
|
|
VERIFY(StrToIntEx(strDateSegment,STIF_DEFAULT ,&nYear));
|
|
// ASSERT(nYear == 1999 || nYear == 2000);
|
|
strDateCopy = strDateCopy.Right(strDateCopy.GetLength() - 4);
|
|
|
|
//month is now the 2 leftmost digits of remaining date string
|
|
int nMonth;
|
|
strDateSegment = strDateCopy.Left(2);
|
|
VERIFY(StrToIntEx(strDateSegment,STIF_DEFAULT ,&nMonth));
|
|
ASSERT(nMonth >= 1 && nMonth <= 12);
|
|
strDateCopy = strDateCopy.Right(strDateCopy.GetLength() - 2);
|
|
|
|
|
|
//day is now the 2 leftmost digits of remaining date string
|
|
int nDay;
|
|
strDateSegment = strDateCopy.Left(2);
|
|
VERIFY(StrToIntEx(strDateSegment,STIF_DEFAULT ,&nDay));
|
|
ASSERT(nDay >= 1 && nDay <= 31);
|
|
strDateCopy = strDateCopy.Right(strDateCopy.GetLength() - 2);
|
|
|
|
//hour is now the 2 leftmost digits of remaining date string
|
|
int nHour;
|
|
strDateSegment = strDateCopy.Left(2);
|
|
VERIFY(StrToIntEx(strDateSegment,STIF_DEFAULT ,&nHour));
|
|
ASSERT(nHour >= 0 && nHour <= 24);
|
|
strDateCopy = strDateCopy.Right(strDateCopy.GetLength() - 2);
|
|
|
|
//Minute is now the 2 leftmost digits of remaining date string
|
|
int nMin;
|
|
strDateSegment = strDateCopy.Left(2);
|
|
VERIFY(StrToIntEx(strDateSegment,STIF_DEFAULT ,&nMin));
|
|
ASSERT(nMin >= 0 && nMin <= 59);
|
|
strDateCopy = strDateCopy.Right(strDateCopy.GetLength() - 2);
|
|
|
|
|
|
//Minute is now the 2 leftmost digits of remaining date string
|
|
int nSec;
|
|
strDateSegment = strDateCopy.Left(2);
|
|
VERIFY(StrToIntEx(strDateSegment,STIF_DEFAULT ,&nSec));
|
|
ASSERT(nSec >= 0 && nSec <= 59);
|
|
strDateCopy = strDateCopy.Right(strDateCopy.GetLength() - 2);
|
|
|
|
|
|
CTime tmTime(nYear,nMonth,nDay,nHour,nMin,nSec);
|
|
#ifdef _V_STLOWE
|
|
CString strFMT;
|
|
CString strTime;
|
|
strFMT.LoadString(IDS_TIME_FORMAT);
|
|
strTime =tmTime.FormatGmt("%A, %B %d, %Y");
|
|
|
|
#endif
|
|
//Adjust for time zone
|
|
CTimeSpan tspan(0,0,nTimeZone,0);
|
|
tmTime -= tspan;
|
|
|
|
#ifdef _V_STLOWE
|
|
strFMT.LoadString(IDS_TIME_FORMAT);
|
|
strTime =tmTime.FormatGmt("%A, %B %d, %Y");
|
|
|
|
#endif
|
|
return tmTime;
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//finds timestamp string for a given delta or snapshot node
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
CString GetPNPNameByID(CComPtr<IXMLDOMDocument> pDoc,CComBSTR bstrPNPID)
|
|
{
|
|
HRESULT hr;
|
|
CComPtr<IXMLDOMNodeList> pNodeList;
|
|
CComBSTR bstrQuery("INSTANCE[@CLASSNAME $ieq$ \"WIN32_PNPeNTITY\"] /PROPERTY[@NAME $ieq$ \"Description\"]");
|
|
hr = pDoc->getElementsByTagName(bstrQuery,&pNodeList);
|
|
if (FAILED(hr) || !pNodeList)
|
|
{
|
|
ASSERT(0 && "WIN32_PNPeNTITY error getting node list");
|
|
return "";
|
|
}
|
|
|
|
long lListLen;
|
|
hr = pNodeList->get_length(&lListLen);
|
|
ASSERT(lListLen > 0 && "No WIN32_PNPeNTITY nodes found to match query");
|
|
for(long i = 0; i < lListLen;i++)
|
|
{
|
|
CComPtr<IXMLDOMNode> pNode;
|
|
hr = pNodeList->nextNode(&pNode);
|
|
if (FAILED(hr) || !pNode)
|
|
{
|
|
ASSERT(0 && "could not get next node from list");
|
|
return "";
|
|
}
|
|
|
|
USES_CONVERSION;
|
|
CComPtr<IXMLDOMNode> pIDNode;
|
|
hr = pNode->get_nextSibling(&pIDNode);
|
|
if (FAILED(hr) || !pNode)
|
|
{
|
|
ASSERT(0 && "could not get next node from list");
|
|
return "";
|
|
}
|
|
//see if node's DeviceID subnode matches bstrPNPID
|
|
CComBSTR bstrDeviceID;
|
|
hr = pIDNode->get_text(&bstrDeviceID);
|
|
ASSERT(SUCCEEDED(hr) && "could not get text from ID node");
|
|
if (bstrDeviceID == bstrPNPID)
|
|
{
|
|
CComBSTR bstrDeviceDesc;
|
|
hr = pNode->get_text(&bstrDeviceDesc);
|
|
ASSERT(SUCCEEDED(hr) && "could not get text from Desc node");
|
|
return OLE2T(bstrDeviceDesc);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
|
|
return "";
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//returns true if any changes have been entered into CMSInfocategory data
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
BOOL CHistoryParser::AreThereChangeLines()
|
|
{
|
|
return this->m_fChangeLines;
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
//gets (from resources strings) a human-readable name for a the class wrapped the the instance
|
|
//////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
CString CInstance::GetClassFriendlyName()
|
|
{
|
|
CString strClassName = GetClassName();
|
|
if (strClassName.CompareNoCase(_T("Win32_PNPAllocatedResource")) == 0)
|
|
{
|
|
VERIFY(m_mapNameValue.Lookup(_T("ANTECEDENT"),strClassName) && _T("Could not find antecedent"));
|
|
}
|
|
CString strFriendlyName;
|
|
if (strClassName.CompareNoCase(_T("Win32_CodecFile")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_CODEC_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_ComputerSystem")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_COMPUTERSYSTEM_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_LogicalMemoryConfiguration")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_LOGICALMEMEORY_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_LogicalDisk")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_LOGICALDISK_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_IRQResource")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_IRQRESOURCE_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_DriverVXD")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_DRIVERVXD_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_DMAChannel")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_DMACHANNEL_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_DeviceMemoryAddress")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_DEVICEMEMORYADDRESS_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_NetworkProtocol")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_NETWORKPROTOCOL_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_OperatingSystem")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_OPERATINGSYSTEM_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_PNPAllocatedResource")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_PNPALLOCATEDRESOURCE_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_PNPEntity")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_PNPENTITY_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_PortResource")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_PORTRESOURCE_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_Printer")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_PRINTER_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_ProgramGroup")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_PROGRAMGROUP_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
if (strClassName.CompareNoCase(_T("Win32_StartupCommand")) == 0)
|
|
{
|
|
VERIFY(strFriendlyName.LoadString(IDS_STARTUPCOMMAND_DESC) && _T("could not find string resource"));
|
|
return strFriendlyName;
|
|
}
|
|
ASSERT(0 && "Unknown strClassName");
|
|
return "";
|
|
}
|
|
|
|
|
|
|