2020-09-30 16:53:55 +02:00

1016 lines
27 KiB
C++

// DSCache.cpp : implementation file
//
//+-------------------------------------------------------------------------
//
// Microsoft Windows
// Copyright (C) Microsoft Corporation, 1992 - 1999
//
// File: DSCache.cpp
//
// Contents: TBD
//
// History: 31-jan-97 jimharr created
//
//--------------------------------------------------------------------------
#include "stdafx.h"
#include "resource.h"
#include "util.h"
#include "dsutil.h"
#include "dscache.h"
#include "dscookie.h"
#include "newobj.h"
#include "gencreat.h"
#include "querysup.h"
#ifdef _DEBUG
#define new DEBUG_NEW
#undef THIS_FILE
static char THIS_FILE[] = __FILE__;
#endif
//////////////////////////////////////////////////////////////////////////
// helper functions
HRESULT HrVariantToStringList(const VARIANT& refvar, CStringList& refstringlist); // prototype
static CString g_szAllTypesArr[8];
void InitGroupTypeStringTable()
{
AFX_MANAGE_STATE(AfxGetStaticModuleState());
CString szSecTypeArr[2];
szSecTypeArr[0].LoadString(IDS_GROUP_SECURITY);
szSecTypeArr[1].LoadString(IDS_GROUP_DISTRIBUTION);
CString szTypeArr[4];
szTypeArr[0].LoadString(IDS_GROUP_GLOBAL);
szTypeArr[1].LoadString(IDS_GROUP_DOMAIN_LOCAL);
szTypeArr[2].LoadString(IDS_GROUP_UNIVERSAL);
szTypeArr[3].LoadString(IDS_GROUP_BUILTIN_LOCAL);
for (int iSec=0; iSec<2; iSec++)
{
for (int iType=0; iType<4; iType++)
{
int k = (iSec*4)+iType;
g_szAllTypesArr[k] = szSecTypeArr[iSec];
g_szAllTypesArr[k] += szTypeArr[iType];
}
}
}
LPCWSTR GetGroupTypeStringHelper(INT GroupType)
{
// need to map the type into the array index
// first part (2 types)
int iSec = (GroupType & GROUP_TYPE_SECURITY_ENABLED) ? 0 : 1;
//
// second part (4 types)
//
// NOTE : can't use the following switch here because there may be some
// extra bits used in the group type. See bug #90507.
// switch (GroupType & ~GROUP_TYPE_SECURITY_ENABLED)
//
int iType = -1;
if (GroupType & GROUP_TYPE_ACCOUNT_GROUP)
{
iType = 0;
}
else if (GroupType & GROUP_TYPE_RESOURCE_GROUP)
{
iType = 1;
}
else if (GroupType & GROUP_TYPE_UNIVERSAL_GROUP)
{
iType = 2;
}
else if (GroupType & GROUP_TYPE_BUILTIN_LOCAL_GROUP ||
GroupType & GROUP_TYPE_RESOURCE_GROUP)
{
iType = 3;
}
else
{
ASSERT(FALSE); // this should never happen, invalid bit pattern
return NULL;
}
int k = (iSec*4)+iType;
ASSERT((k>=0) && (k<8));
ASSERT(!g_szAllTypesArr[k].IsEmpty());
return g_szAllTypesArr[k];
}
//////////////////////////////////////////////////////////////////////////
// CDSClassCacheItemBase
CDSClassCacheItemBase::~CDSClassCacheItemBase()
{
if (m_pMandPropsList != NULL)
{
delete m_pMandPropsList;
}
if (m_pAdminContextMenu != NULL)
{
delete[] m_pAdminContextMenu;
}
if (m_pAdminPropertyPages != NULL)
{
delete[] m_pAdminPropertyPages ;
}
if (m_pAdminMultiSelectPropertyPages != NULL)
{
delete[] m_pAdminMultiSelectPropertyPages;
}
}
HRESULT CDSClassCacheItemBase::CreateItem(LPCWSTR lpszClass,
IADs* pDSObject,
CDSComponentData* pCD,
CDSClassCacheItemBase** ppItem)
{
ASSERT(ppItem != NULL);
// determine which type of object we have
if (wcscmp(lpszClass, L"user") == 0
#ifdef INETORGPERSON
|| _wcsicmp(lpszClass, L"inetOrgPerson") == 0
#endif
)
{
*ppItem = new CDSClassCacheItemUser;
}
else if(wcscmp(lpszClass,L"group") == 0)
{
*ppItem = new CDSClassCacheItemGroup;
}
else
{
*ppItem = new CDSClassCacheItemGeneric;
}
if (*ppItem == NULL)
return E_OUTOFMEMORY;
HRESULT hr = (*ppItem)->Init(lpszClass, pDSObject, pCD);
if (FAILED(hr))
{
delete *ppItem;
*ppItem = NULL;
}
return hr;
}
HRESULT CDSClassCacheItemBase::Init(LPCWSTR lpszClass, IADs* pDSObject, CDSComponentData *pCD)
{
HRESULT hr = S_OK;
// init to default values
m_bIsContainer = FALSE;
m_GUID = GUID_NULL;
m_szClass = lpszClass;
m_szFriendlyClassName = lpszClass;
m_szNamingAttribute = L"cn";
ASSERT(!m_szClass.IsEmpty());
// get schema path to bind to class object in the schema
CComBSTR bstrSchema;
if (pDSObject != NULL)
{
// we have an ADSI pointer to use
hr = pDSObject->get_Schema(&bstrSchema);
}
else
{
// no object yet (create new case)
CString strSchema;
pCD->GetBasePathsInfo()->GetAbstractSchemaPath(strSchema);
strSchema += L"/";
strSchema += lpszClass;
bstrSchema = (LPCWSTR)strSchema;
}
// bind to the schema object
CComPtr<IADsClass> spDsClass;
hr = DSAdminOpenObject(bstrSchema,
IID_IADsClass,
(LPVOID *)&spDsClass,
TRUE /*bServer*/);
if (SUCCEEDED(hr))
{
// got class info from the schema
// se the container flag
if ((!wcscmp(lpszClass, L"computer")) ||
(!wcscmp(lpszClass, L"user")) ||
#ifdef INETORGPERSON
(!wcscmp(lpszClass, L"inetOrgPerson")) ||
#endif
(!wcscmp(lpszClass,L"group")))
{
// special classes we know about
m_bIsContainer = pCD->ExpandComputers();
}
else
{
// generic class, ask the schema
VARIANT_BOOL bIsContainer;
hr = spDsClass->get_Container(&bIsContainer);
if (SUCCEEDED(hr))
{
if (bIsContainer)
{
m_bIsContainer = TRUE;
}
}
}
// get the class GUID
CComVariant Var;
hr = spDsClass->Get(CComBSTR(L"schemaIDGUID"), &Var);
if (SUCCEEDED(hr))
{
GUID* pgtemp;
pgtemp = (GUID*) (Var.parray->pvData);
m_GUID = *pgtemp;
}
// get the friendly class name
WCHAR wszBuf[120];
hr = pCD->GetBasePathsInfo()->GetFriendlyClassName(lpszClass, wszBuf, 120);
if (SUCCEEDED(hr))
{
m_szFriendlyClassName = wszBuf;
}
// get the naming attribute
Var.Clear();
hr = spDsClass->get_NamingProperties(&Var);
// fill out m_szNamingAttribute here.
if (SUCCEEDED(hr))
{
m_szNamingAttribute = Var.bstrVal;
}
}
else
{
// we failed getting class info from the schema
if (wcscmp(L"Unknown", lpszClass) == 0)
{
m_szFriendlyClassName.LoadString(IDS_DISPLAYTEXT_NONE);
}
}
// locate the column set for this class
// m_pColumnSet = pCD->FindColumnSet(lpszClass);
// ASSERT(m_pColumnSet != NULL);
// set the icon index(es)
SetIconData(pCD);
return S_OK;
}
CMandatoryADsAttributeList*
CDSClassCacheItemBase::GetMandatoryAttributeList(CDSComponentData* pCD)
{
// got it already cached ?
if (m_pMandPropsList != NULL) {
return m_pMandPropsList;
}
// need to build the list
HRESULT hr = S_OK;
CComBSTR bstrSchema;
IADsClass * pDsClass = NULL;
CMandatoryADsAttribute* pNamingAttribute = NULL;
POSITION pos = NULL;
CComVariant MandatoryList;
CStringList Strings;
CString csProp;
LPTSTR pszSyntax;
const LPTSTR pszNameSyntax = L"2.5.5.12";
CDSSearch SchemaSrch(pCD->m_pClassCache, pCD);
CString strPhysSchema;
const int cCols = 2;
LPTSTR pszAttributes[cCols] = {L"ADsPath",
L"attributeSyntax" };
ADS_SEARCH_COLUMN ColumnData;
m_pMandPropsList = new CMandatoryADsAttributeList;
// get the class object from the schema
CString strSchema;
pCD->GetBasePathsInfo()->GetAbstractSchemaPath(strSchema);
strSchema += L"/";
strSchema += GetClassName();
bstrSchema = (LPCWSTR)strSchema;
hr = DSAdminOpenObject(bstrSchema,
IID_IADsClass,
(LPVOID *)&pDsClass,
TRUE /*bServer*/);
if (FAILED(hr))
goto CleanUp;
pCD->GetBasePathsInfo()->GetSchemaPath(strPhysSchema);
SchemaSrch.Init (strPhysSchema);
SchemaSrch.SetSearchScope(ADS_SCOPE_ONELEVEL);
hr = pDsClass->get_MandatoryProperties (&MandatoryList);
if (FAILED(hr))
goto CleanUp;
hr = HrVariantToStringList (IN MandatoryList, OUT Strings);
if (FAILED(hr))
goto CleanUp;
pos = Strings.GetHeadPosition();
TRACE(_T("class: %s\n"), GetClassName());
while (pos != NULL) {
csProp = Strings.GetNext(INOUT pos);
// skip WHAT????
if (!wcscmp(csProp, gsz_objectClass) ||
!wcscmp(csProp, gsz_nTSecurityDescriptor) ||
!wcscmp(csProp, gsz_instanceType) ||
!wcscmp(csProp, gsz_objectCategory) ||
!wcscmp(csProp, gsz_objectSid)) {
continue;
}
TRACE(_T("\tmandatory prop: %s.\n"), csProp);
CString csFilter = CString(L"(&(objectClass=attributeSchema)(lDAPDisplayName=") +
csProp + CString(L"))");
SchemaSrch.SetFilterString((LPTSTR)(LPCTSTR)csFilter);
SchemaSrch.SetAttributeList (pszAttributes, cCols);
hr = SchemaSrch.DoQuery ();
if (SUCCEEDED(hr)) {
hr = SchemaSrch.GetNextRow();
if (SUCCEEDED(hr)) {
hr = SchemaSrch.GetColumn(pszAttributes[cCols - 1],
&ColumnData);
TRACE(_T("\t\tattributeSyntax: %s\n"),
ColumnData.pADsValues->CaseIgnoreString);
pszSyntax = ColumnData.pADsValues->CaseIgnoreString;
CMandatoryADsAttribute* pAttr = new CMandatoryADsAttribute((LPCTSTR)csProp,
NULL,
pszSyntax);
if (wcscmp(csProp, GetNamingAttribute()) == 0)
pNamingAttribute = pAttr;
else
m_pMandPropsList->AddTail(pAttr);
} // if
SchemaSrch.m_pObj->FreeColumn (&ColumnData);
} // if
} // while
// make sure naming attribute is present
if (pNamingAttribute == NULL)
{
pNamingAttribute = new CMandatoryADsAttribute(GetNamingAttribute(),
NULL,
pszNameSyntax);
}
// make sure the naming attribute is the first in the list
m_pMandPropsList->AddHead(pNamingAttribute);
CleanUp:
if (pDsClass) {
pDsClass->Release();
}
return m_pMandPropsList;
}
/*
CDSColumnSet*
CDSClassCacheItemBase::GetColumnSet()
{
return m_pColumnSet;
}
*/
//
// Display Specifier cached accessors
//
GUID* CDSClassCacheItemBase::GetAdminPropertyPages(UINT* pnCount)
{
*pnCount = m_nAdminPPCount;
return m_pAdminPropertyPages;
}
void CDSClassCacheItemBase::SetAdminPropertyPages(UINT nCount, GUID* pGuids)
{
m_nAdminPPCount = nCount;
if (m_pAdminPropertyPages != NULL)
{
delete[] m_pAdminPropertyPages;
}
m_pAdminPropertyPages = pGuids;
}
GUID* CDSClassCacheItemBase::GetAdminContextMenu(UINT* pnCount)
{
*pnCount = m_nAdminCMCount;
return m_pAdminContextMenu;
}
void CDSClassCacheItemBase::SetAdminContextMenu(UINT nCount, GUID* pGuids)
{
m_nAdminCMCount = nCount;
if (m_pAdminContextMenu != NULL)
{
delete[] m_pAdminContextMenu;
}
m_pAdminContextMenu = pGuids;
}
GUID* CDSClassCacheItemBase::GetAdminMultiSelectPropertyPages(UINT* pnCount)
{
*pnCount = m_nAdminMSPPCount;
return m_pAdminMultiSelectPropertyPages;
}
void CDSClassCacheItemBase::SetAdminMultiSelectPropertyPages(UINT nCount, GUID* pGuids)
{
m_nAdminMSPPCount = nCount;
if (m_pAdminMultiSelectPropertyPages != NULL)
{
delete[] m_pAdminMultiSelectPropertyPages;
}
m_pAdminMultiSelectPropertyPages = pGuids;
}
//////////////////////////////////////////////////////////////////////////
// CDSClassIconIndexes
void CDSClassIconIndexes::SetIconData(LPCWSTR lpszClass, BOOL bContainer, CDSComponentData *pCD, int)
{
DWORD dwBaseFlags = DSGIF_GETDEFAULTICON;
if (bContainer)
dwBaseFlags |= DSGIF_DEFAULTISCONTAINER;
int iIconIndex;
// get the generic icon
HRESULT hr = pCD->AddClassIcon(lpszClass, DSGIF_ISNORMAL | dwBaseFlags, &iIconIndex);
m_iIconIndex = SUCCEEDED(hr) ? iIconIndex : -1;
m_iIconIndexOpen = m_iIconIndexDisabled = m_iIconIndex;
// get the open icon
hr = pCD->AddClassIcon(lpszClass, DSGIF_ISOPEN | dwBaseFlags, &iIconIndex);
if (SUCCEEDED(hr))
{
m_iIconIndexOpen = iIconIndex;
}
// get the disabled icon
hr = pCD->AddClassIcon(lpszClass, DSGIF_ISDISABLED | dwBaseFlags, &iIconIndex);
if (SUCCEEDED(hr))
{
m_iIconIndexDisabled = iIconIndex;
}
TRACE(_T("Added icon for class: %s\n"), lpszClass);
TRACE(_T("Index: %d\n"), m_iIconIndex);
TRACE(_T("Open: %d\n"), m_iIconIndexOpen);
TRACE(_T("Disabled: %d\n"), m_iIconIndexDisabled);
}
//////////////////////////////////////////////////////////////////////////
// CDSClassCacheItemGeneric
inline int CDSClassCacheItemGeneric::GetIconIndex(CDSCookie* pCookie, BOOL bOpen)
{
return m_iconIndexesStandard.GetIconIndex(pCookie->IsDisabled(), bOpen);
}
inline void CDSClassCacheItemGeneric::SetIconData(CDSComponentData *pCD)
{
m_iconIndexesStandard.SetIconData(GetClassName(), IsContainer(), pCD,0);
}
//////////////////////////////////////////////////////////////////////////
// CDSClassCacheItemGroup
inline int CDSClassCacheItemGroup::GetIconIndex(CDSCookie* pCookie, BOOL bOpen)
{
CDSClassIconIndexes* pIndexes = &m_iconIndexesStandard;
CDSCookieInfoBase* pExtraInfo = pCookie->GetExtraInfo();
if ( (pExtraInfo != NULL) && (pExtraInfo->GetClass() == CDSCookieInfoBase::group) )
{
if (((((CDSCookieInfoGroup*)pExtraInfo)->m_GroupType) & GROUP_TYPE_SECURITY_ENABLED) != 0)
pIndexes = & m_iconIndexesAlternate;
}
return pIndexes->GetIconIndex(pCookie->IsDisabled(), bOpen);
}
inline void CDSClassCacheItemGroup::SetIconData(CDSComponentData *pCD)
{
LPCWSTR lpszClass = GetClassName();
m_iconIndexesStandard.SetIconData(lpszClass, IsContainer(), pCD,0);
m_iconIndexesAlternate.SetIconData(lpszClass, IsContainer(), pCD,1);
/*
// test just to get load some icons with a fake class and a fake "groupAlt-Display" object
m_iconIndexesAlternate.SetIconData(L"groupAlt", m_bIsContainer, pCD, 1);
*/
}
//////////////////////////////////////////////////////////////////////////
// CDSCache
BOOL CDSCache::ToggleExpandSpecialClasses(BOOL bContainer)
{
_Lock();
BOOL bFound = FALSE;
CDSClassCacheItemBase* pItem;
if (Lookup(L"computer", pItem))
{
pItem->SetContainerFlag(bContainer);
bFound = TRUE;
}
if (Lookup(L"user", pItem))
{
pItem->SetContainerFlag(bContainer);
bFound = TRUE;
}
#ifdef INETORGPERSON
if (Lookup(L"inetOrgPerson", pItem))
{
pItem->SetContainerFlag(bContainer);
bFound = TRUE;
}
#endif
if (Lookup(L"group", pItem))
{
pItem->SetContainerFlag(bContainer);
bFound = TRUE;
}
_Unlock();
return bFound;
}
CDSColumnSet* CDSCache::FindColumnSet(LPCWSTR lpszColumnID)
{
_Lock();
TRACE(L"Entering CDSCache::FindColumnSet\n");
CDSColumnSet* pColumnSet = NULL;
if (_wcsicmp(DEFAULT_COLUMN_SET, lpszColumnID) == 0)
{
//
// return the default column set
//
pColumnSet = dynamic_cast<CDSColumnSet*>(m_ColumnList.GetDefaultColumnSet());
}
else if (_wcsicmp(SPECIAL_COLUMN_SET, lpszColumnID) == 0)
{
//
// return the special column set
//
pColumnSet = dynamic_cast<CDSColumnSet*>(m_ColumnList.GetSpecialColumnSet());
}
else
{
pColumnSet = dynamic_cast<CDSColumnSet*>(m_ColumnList.FindColumnSet(lpszColumnID));
}
_Unlock();
return pColumnSet;
}
CDSClassCacheItemBase* CDSCache::FindClassCacheItem(CDSComponentData* pCD,
LPCWSTR lpszObjectClass,
LPCWSTR lpszObjectLdapPath
)
{
_Lock();
CDSClassCacheItemBase* pDsCacheItem = NULL;
BOOL bFound = m_Map.Lookup(lpszObjectClass, pDsCacheItem);
if (!bFound)
{
// Item not found in cache, create, insert in the cache and return it
TRACE(_T("did not find class <%s> for this item in the Cache.\n"), (LPCWSTR)lpszObjectClass);
// Check to see if the object is a container
CComPtr<IADs> spADsObject = NULL;
if (lpszObjectLdapPath != NULL)
{
DSAdminOpenObject(lpszObjectLdapPath,
IID_IADs,
(LPVOID*)&spADsObject,
TRUE /*bServer*/);
// NOTICE: we might fail to bind here if we do not have read rights
// this will give a NULL spADsObject, that will work just fine on the CreateItem() call below
}
// create object
HRESULT hrCreate = CDSClassCacheItemBase::CreateItem(lpszObjectClass, spADsObject, pCD, &pDsCacheItem);
ASSERT(pDsCacheItem != NULL);
ASSERT(SUCCEEDED(hrCreate));
// set in the cache
m_Map.SetAt(lpszObjectClass, pDsCacheItem);
}
_Unlock();
return pDsCacheItem;
}
#define DS_CACHE_STREAM_VERSION ((DWORD)0x0)
HRESULT CDSCache::Save(IStream* pStm)
{
// save cache version number
HRESULT hr = SaveDWordHelper(pStm, DS_CACHE_STREAM_VERSION);
if (FAILED(hr))
return hr;
// save column list
return m_ColumnList.Save(pStm);
}
HRESULT CDSCache::Load(IStream* pStm)
{
// load cache version number
DWORD dwVersion;
HRESULT hr = LoadDWordHelper(pStm, &dwVersion);
if ( FAILED(hr) ||(dwVersion != DS_CACHE_STREAM_VERSION) )
return E_FAIL;
// load column list
return m_ColumnList.Load(pStm);
}
HRESULT CDSCache::TabCollect_AddMultiSelectPropertyPages(LPPROPERTYSHEETCALLBACK pCall,
LONG_PTR,
LPDATAOBJECT pDataObject,
MyBasePathsInfo* pBasePathsInfo)
{
HRESULT hr = S_OK;
CString szClassName;
CString szDisplayProperty = L"AdminMultiSelectPropertyPages";
GUID* pGuids = NULL;
UINT nCount = 0;
if (IsHomogenousDSSelection(pDataObject, szClassName))
{
//
// Get the guid for the multiselect proppages of the homogenous class selection
//
//
// Check the cache first
//
BOOL bFoundGuids = FALSE;
CDSClassCacheItemBase* pItem = NULL;
BOOL bFoundItem = Lookup(szClassName, pItem);
if (bFoundItem)
{
if (pItem == NULL)
{
ASSERT(FALSE);
bFoundItem = FALSE;
}
else
{
//
// Retrieve guids from cache
//
pGuids = pItem->GetAdminMultiSelectPropertyPages(&nCount);
if (nCount > 0 && pGuids != NULL)
{
bFoundGuids = TRUE;
}
}
}
if (!bFoundGuids)
{
//
// Class cache item did not contain GUID
//
hr = TabCollect_GetDisplayGUIDs(szClassName,
szDisplayProperty,
pBasePathsInfo,
&nCount,
&pGuids);
if (FAILED(hr))
{
//
// Try the default-Display object then
//
hr = TabCollect_GetDisplayGUIDs(L"default",
szDisplayProperty,
pBasePathsInfo,
&nCount,
&pGuids);
if (FAILED(hr))
{
return hr;
}
}
if (bFoundItem)
{
//
// Cache the new guids
//
pItem->SetAdminMultiSelectPropertyPages(nCount, pGuids);
}
}
}
else
{
//
// Get the default multi-select proppages
//
hr = TabCollect_GetDisplayGUIDs(L"default", szDisplayProperty, pBasePathsInfo, &nCount, &pGuids);
//
// Right now there is no default item in the cache so we have to get it each time from
// the DS
//
}
if (SUCCEEDED(hr))
{
if (nCount > 0 && pGuids != NULL)
{
//
// Create all the pages, initialize, and then add them
//
for (UINT nIndex = 0; nIndex < nCount; nIndex++)
{
//
// Create
//
CComPtr<IShellExtInit> spShellInit;
hr = ::CoCreateInstance((pGuids[nIndex]),
NULL,
CLSCTX_INPROC_SERVER,
IID_IShellExtInit,
(PVOID*)&spShellInit);
if (FAILED(hr))
{
continue;
}
//
// Initialize
//
hr = spShellInit->Initialize(NULL, pDataObject, 0);
if (FAILED(hr))
{
continue;
}
//
// Add
//
CComPtr<IShellPropSheetExt> spPropSheetExt;
hr = spShellInit->QueryInterface(IID_IShellPropSheetExt, (PVOID*)&spPropSheetExt);
if (FAILED(hr))
{
continue;
}
hr = spPropSheetExt->AddPages(AddPageProc, (LPARAM)pCall);
if (FAILED(hr))
{
TRACE(TEXT("spPropSheetExt->AddPages failed, hr: 0x%x\n"), hr);
continue;
}
}
}
}
return hr;
}
void CDSCache::_CollectDisplaySettings(MyBasePathsInfo* pBasePathsInfo)
{
static LPCWSTR lpszSettingsObjectClass = L"dsUISettings";
static LPCWSTR lpszSettingsObject = L"cn=DS-UI-Default-Settings";
static LPCWSTR lpszSecurityGroupProperty = L"msDS-Security-Group-Extra-Classes";
static LPCWSTR lpszNonSecurityGroupProperty = L"msDS-Non-Security-Group-Extra-Classes";
static LPCWSTR lpszFilterContainers = L"msDS-FilterContainers";
if (pBasePathsInfo == NULL)
{
return;
}
//
// get the display specifiers locale container (e.g. 409)
//
CComPtr<IADsContainer> spLocaleContainer;
HRESULT hr = pBasePathsInfo->GetDisplaySpecifier(NULL, IID_IADsContainer, (void**)&spLocaleContainer);
if (FAILED(hr))
{
return;
}
//
// bind to the settings object
//
CComPtr<IDispatch> spIDispatchObject;
hr = spLocaleContainer->GetObject(CComBSTR(lpszSettingsObjectClass),
CComBSTR(lpszSettingsObject),
&spIDispatchObject);
if (FAILED(hr))
{
return;
}
CComPtr<IADs> spSettingsObject;
hr = spIDispatchObject->QueryInterface(IID_IADs, (void**)&spSettingsObject);
if (FAILED(hr))
{
return;
}
//
// get the security group extra classes as a CStringList
//
CComVariant var;
hr = spSettingsObject->Get(CComBSTR(lpszSecurityGroupProperty), &var);
if (SUCCEEDED(hr))
{
hr = HrVariantToStringList(var, m_szSecurityGroupExtraClasses);
}
//
// get the non-security group extra classes as a CStringList
//
var.Clear();
hr = spSettingsObject->Get(CComBSTR(lpszNonSecurityGroupProperty), &var);
if (SUCCEEDED(hr))
{
hr = HrVariantToStringList(var, m_szNonSecurityGroupExtraClasses);
}
//
// get the additional filter containers as a CStringList
//
var.Clear();
hr = spSettingsObject->Get(CComBSTR(lpszFilterContainers), &var);
if (SUCCEEDED(hr))
{
CStringList szContainers;
hr = HrVariantToStringList(var, szContainers);
if (SUCCEEDED(hr))
{
//
// Allocate the filter struct element
//
m_pfilterelementDsAdminDrillDown = new FilterElementStruct;
if (m_pfilterelementDsAdminDrillDown != NULL)
{
//
// Allocate the tokens
//
m_pfilterelementDsAdminDrillDown->ppTokens = new FilterTokenStruct*[szContainers.GetCount()];
if (m_pfilterelementDsAdminDrillDown->ppTokens != NULL)
{
//
// Allocate and fill in each token
//
int idx = 0;
POSITION pos = szContainers.GetHeadPosition();
while (pos != NULL)
{
CString szContainerCategory = szContainers.GetNext(pos);
ASSERT(!szContainerCategory.IsEmpty());
m_pfilterelementDsAdminDrillDown->ppTokens[idx] = new FilterTokenStruct;
if (m_pfilterelementDsAdminDrillDown->ppTokens[idx] != NULL)
{
m_pfilterelementDsAdminDrillDown->ppTokens[idx]->nType = TOKEN_TYPE_CATEGORY;
m_pfilterelementDsAdminDrillDown->ppTokens[idx]->lpszString = new WCHAR[szContainerCategory.GetLength() + 1];
if (m_pfilterelementDsAdminDrillDown->ppTokens[idx]->lpszString != NULL)
{
wcscpy(m_pfilterelementDsAdminDrillDown->ppTokens[idx]->lpszString, (LPCWSTR)szContainerCategory);
idx++;
}
}
}
//
// Count only the ones that were added successfully
//
m_pfilterelementDsAdminDrillDown->cNumTokens = idx;
//
// But they all should have been added successfully so assert that
//
ASSERT(idx == szContainers.GetCount());
}
else
{
//
// failed to allocate space for the tokens,
// delete all the other allocated and set the
// global to NULL
//
delete m_pfilterelementDsAdminDrillDown;
m_pfilterelementDsAdminDrillDown = NULL;
}
}
}
}
m_bDisplaySettingsCollected = TRUE;
}
BOOL CDSCache::CanAddToGroup(MyBasePathsInfo* pBasePathsInfo, PCWSTR pszClass, BOOL bSecurity)
{
_Lock();
if (!m_bDisplaySettingsCollected)
{
_CollectDisplaySettings(pBasePathsInfo);
}
BOOL bResult = FALSE;
if (bSecurity)
{
POSITION pos = m_szSecurityGroupExtraClasses.GetHeadPosition();
while (pos != NULL)
{
CString szClass = m_szSecurityGroupExtraClasses.GetNext(pos);
ASSERT(!szClass.IsEmpty());
if (_wcsicmp(szClass, pszClass) == 0)
{
bResult = TRUE;
break;
}
}
}
else
{
POSITION pos = m_szNonSecurityGroupExtraClasses.GetHeadPosition();
while (pos != NULL)
{
CString szClass = m_szNonSecurityGroupExtraClasses.GetNext(pos);
ASSERT(!szClass.IsEmpty());
if (_wcsicmp(szClass, pszClass) == 0)
{
bResult = TRUE;
break;
}
}
}
_Unlock();
return bResult;
}
FilterElementStruct* CDSCache::GetFilterElementStruct(CDSComponentData* pDSComponentData)
{
_Lock();
if (!m_bDisplaySettingsCollected)
{
_CollectDisplaySettings(pDSComponentData->GetBasePathsInfo());
}
_Unlock();
return (SNAPINTYPE_SITE == pDSComponentData->QuerySnapinType()) ?
&g_filterelementSiteReplDrillDown : m_pfilterelementDsAdminDrillDown;
}